diff --git a/.github/workflows/dlc.json b/.github/workflows/dlc.json index ca6020cf00..be27700cfb 100644 --- a/.github/workflows/dlc.json +++ b/.github/workflows/dlc.json @@ -1,9 +1,4 @@ { - "ignorePatterns": [ - { - "pattern": "https://github.com/WeBankFinTech/DataSphereStudio-Doc" - } - ], "timeout": "10s", "retryOn429": true, "retryCount": 10, @@ -13,5 +8,4 @@ 401, 403 ] -} - +} \ No newline at end of file diff --git a/.gitignore b/.gitignore index 7bc2f236d6..aa5f9b2526 100644 --- a/.gitignore +++ b/.gitignore @@ -1,33 +1,115 @@ +.idea +.idea/* + .DS_Store -.cache -# for ide +assembly/target +assembly/dss-package/target + + +# dss-commons +dss-commons/dss-sender-service/target + +# dss-appconn +dss-appconn/appconns/dss-datachecker-appconn/target +dss-appconn/appconns/dss-eventchecker-appconn/target +dss-appconn/appconns/dss-orchestrator-framework-appconn/target +dss-appconn/appconns/dss-schedulis-appconn/target +dss-appconn/appconns/dss-sendemail-appconn/esb-email-support/target +dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/target +dss-appconn/appconns/dss-workflow-appconn/target +dss-appconn/appconns/dss-sso-appconn/target +dss-appconn/appconns/dss-scriptis-appconn-appconn/target +dss-appconn/dss-appconn-core/target +dss-appconn/dss-appconn-loader/target +dss-appconn/dss-schedule-appconn-core/target +dss-appconn/linkis-appconn-engineplugin/target +dss-appconn/appconns/dss-dolphinscheduler-appconn/target +dss-appconn/dss-appconn-manager/dss-appconn-manager-client/target +dss-appconn/dss-appconn-manager/dss-appconn-manager-core/target +dss-appconn/dss-scheduler-appconn/target +dss-appconn/appconns/dss-scriptis-appconn/target/ + +# dss-apps +dss-apps/dss-apiservice-server/target +dss-apps/dss-scriptis-server/target +dss-apps/dss-apps-server/target +dss-server/target + +#dss-guide +dss-apps/dss-user-guide/dss-user-guide-server/src/main/resources/ +dss-apps/dss-user-guide/dss-user-guide-server/target + +# dss-commons +dss-commons/dss-common/target +dss-commons/dss-contextservice/target +dss-commons/dss-bmlservice/target + +# dss-framework +dss-framework/dss-appconn-framework/target +dss-framework/dss-framework-common/target +dss-framework/dss-framework-orchestrator-server/target +dss-framework/dss-framework-project-server/target +dss-framework/dss-framework-workspace-server/target +dss-framework/framework-plugins/dss-framework-orchestrator-publish/target +dss-framework/framework-plugins/dss-framework-release-server/target +dss-framework/framework-plugins/dss-framework-migrate-server/target +dss-framework/dss-framework-admin/target +dss-framework/dss-framework-sql-template/target +dss-framework/dss-framework-dbapi-server/target +dss-framework/dss-framework-admin-service/target +dss-framework/dss-framework-proxy-user-service/target + + +# dss-orchestrator +dss-orchestrator/dss-orchestrator-common/target +dss-orchestrator/dss-orchestrator-core/target +dss-orchestrator/dss-orchestrator-db/target +dss-orchestrator/dss-orchestrator-loader/target +dss-orchestrator/orchestrators/dss-workflow/dss-flow-execution-server/target +dss-orchestrator/orchestrators/dss-workflow/dss-linkis-node-execution/target +dss-orchestrator/orchestrators/dss-workflow/dss-workflow-common/target +dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/target +dss-orchestrator/dss-orchestrator-conversion-standard/target +dss-orchestrator/orchestrators/dss-workflow/dss-workflow-conversion-standard/target +dss-orchestrator/orchestrators/dss-workflow/dss-workflow-sdk/target + +# dss-standars +dss-standard/dss-standard-common/target +dss-standard/sso-standard/origin-sso-integration-standard/target +dss-standard/sso-standard/spring-origin-sso-integration-plugin/target +dss-standard/sso-standard/sso-integration-standard/target +dss-standard/development-standard/development-process-standard/target +dss-standard/development-standard/development-process-standard-execution/target +dss-standard/structure-standard/dss-project-plugin/target +dss-standard/structure-standard/dss-role-plugin/target +dss-standard/structure-standard/dss-structure-integration-standard/target +dss-standard/structure-standard/spring-origin-dss-project-plugin/target + +#dss-data-api +dss-apps/dss-data-api/dss-data-api-server/src/main/resources +dss-apps/dss-data-api/dss-api-sql-template/target +dss-apps/dss-data-api/dss-data-api-server/target + +#dss-data-governance +dss-apps/dss-data-governance/dss-data-governance-server/src/main/resources +dss-apps/dss-data-governance/dss-data-governance-server/target +dss-apps/dss-data-governance/dss-data-governance-common/target +dss-apps/dss-data-governance/dss-data-asset-server/target +dss-apps/dss-data-governance/dss-data-classification-server/target +dss-apps/dss-data-governance/dss-data-warehouse-dao/target +dss-apps/dss-data-governance/dss-data-warehouse-service/target +dss-apps/dss-data-governance/dss-data-warehouse-server/target + + +# plugins +plugins/azkaban/linkis-jobtype/target +plugins/linkis/dss-gateway-support/target + +logs +logs/* + *.iml -*.ipr -*.iws -*.pyc -*.pyo -*.swp -.idea/ -.idea_modules/ -.project -.pydevproject -.scala_dependencies -.settings -.classpath - -# For SBT -.jvmopts - -# For Node.js -node_modules/ - -# generated file -.mvn/wrapper/maven-wrapper.jar -dist/ -out/ -target/ - -# log folder -logs/ -*.log \ No newline at end of file +/plugins/dolphinscheduler/dss-dolphinscheduler-client/target +/plugins/dolphinscheduler/dolphinscheduler-prod-metrics/target +/plugins/dolphinscheduler/dss-dolphinscheduler-token/target \ No newline at end of file diff --git a/assembly/bin/appconn-uninstall.sh b/assembly/bin/appconn-uninstall.sh new file mode 100644 index 0000000000..d34cbbbee1 --- /dev/null +++ b/assembly/bin/appconn-uninstall.sh @@ -0,0 +1,107 @@ +#!/bin/sh +#Actively load user env +source ~/.bashrc +shellDir=`dirname $0` +workDir=`cd ${shellDir}/..;pwd` + +SOURCE_ROOT=${workDir} + +#load config +source ${SOURCE_ROOT}/conf/config.sh +source ${SOURCE_ROOT}/conf/db.sh + +APPCONN_NAME='' + +LOCAL_IP="`ifconfig | grep 'inet' | grep -v '127.0.0.1' | cut -d: -f2 | awk '{ print $2}'`" + +function isSuccess(){ + if [ $? -ne 0 ]; then + echo "Failed to " $1 + exit 1 + else + echo "Succeed to " $1 + fi +} + + +PROC_NAME=DSSProjectServerApplication +ProcNumber=`ps -ef | grep -w $PROC_NAME | grep -v grep | wc -l` +if [ $ProcNumber -le 0 ];then + echo "${PROC_NAME} is not running, please ensure whether DSS is installed and started." + exit 10 +else + echo "Begin to uninstall AppConn plugin..." +fi + + +function getUninstallAppConn() { + echo "Please input the name of uninstallation AppConn, e.g: schedulis." + read -p "Please input the AppConn name:" idx + if [[ 'exit' = "$idx" ]];then + echo "exit!" + exit 1 + else + APPCONN_NAME=$idx + fi + echo "Current uninstallation AppConn is ${APPCONN_NAME}" + echo "" + echo -e "\e[1;31m Are you sure you want to uninstall AppConn ${APPCONN_NAME}, the workflow associated with the AppConn will not be available after uninstallation.\e[0m" + echo -e "\e[1;31mIf you want to uninstall AppConn ${APPCONN_NAME}, please enter 1, otherwise enter 0.\e[0m" + echo "" + read -p "Please input your choice:" idx + + if [[ '0' = "$idx" ]]; then + echo "exit!" + exit 1 + fi +} + +##choose execute mysql mode +function executeSQL() { + TEMP_DB_DML_PATH=${SOURCE_ROOT}/dss-appconns/${APPCONN_NAME}/db + DB_DML_PATH=$TEMP_DB_DML_PATH/uninstall.sql + mysql -h$MYSQL_HOST -P$MYSQL_PORT -u$MYSQL_USER -p$MYSQL_PASSWORD -D$MYSQL_DB --default-character-set=utf8 -e "source $DB_DML_PATH" + isSuccess "complete the cleanup of the database $DB_DML_PATH" + echo "Implement the $TEMP_DB_DML_PATH/uninstall.sql for $APPCONN_NAME succeed." +} + + +function deleteDML() { + echo "" + echo -e "\e[1;31m If you want to delete AppConn ${APPCONN_NAME} file, please enter 1, otherwise enter 0.\e[0m" + echo "" + read -p "Please input your choice:" idx + if [[ '1' = "$idx" ]];then + DML_PATH=${SOURCE_ROOT}/dss-appconns/${APPCONN_NAME} + suffix=.zip + rm -rf $DML_PATH$suffix + rm -rf $DML_PATH + isSuccess "complete the cleanup of the ${APPCONN_NAME} file." + fi +} + + +echo "" +echo "Step1: Get the uninstall AppConn name." +getUninstallAppConn +echo "" + +echo "Step2: Delete AppConn $APPCONN_NAME database info." +executeSQL +echo "" + +echo "Step3: Clear the plugin of $APPCONN_NAME AppConn in DSS." +deleteDML +echo "" +echo "Now try to delete the plugin of ${APPCONN_NAME} AppConn in all DSS micro-services." +echo "The following 2 ways can take effect:" +echo "1. Restart DSS, we will use sh $SOURCE_ROOT/sbin/dss-start-all.sh to restart, it will spend 1 minute." +echo "2. Do nothing, just wait for 5 minutes. Since the DSS micro-services will refresh all the AppConn plugins every 10 minutes." +echo "" +read -p "Please input the choise: " choise +if [[ '1' = "$choise" ]]; then + echo "You chose to restart dss-framework-project micro-services, now try to restart ..." + sh $SOURCE_ROOT/sbin/dss-start-all.sh +else + echo "You chose to wait for 5 minutes." +fi \ No newline at end of file diff --git a/assembly/bin/install.sh b/assembly/bin/install.sh index d3b02f36cd..e1a7f66cf5 100644 --- a/assembly/bin/install.sh +++ b/assembly/bin/install.sh @@ -13,8 +13,7 @@ SERVER_IP="" SERVER_HOME="" local_host="`hostname --fqdn`" -LOCAL_IP=$(hostname -I) -LOCAL_IP=${LOCAL_IP// /} +LOCAL_IP=$(hostname -I | awk '{print $1}') #To be compatible with MacOS and Linux txt="" @@ -228,19 +227,12 @@ fi ##Install dss projects function installDssProject() { echo "step2:update config" -# if [ "$DSS_INSTALL_HOME" != "" ] -# then -# rm -rf $DSS_INSTALL_HOME -# fi - #echo "" - #echo "-----------------DSS install start--------------------" SERVER_HOME=$DSS_INSTALL_HOME if [ "$SERVER_HOME" == "" ] then export SERVER_HOME=${workDir}/DSSInstall fi if [ -d $SERVER_HOME ] && [ "$SERVER_HOME" != "$workDir" ]; then - rm -r $SERVER_HOME-bak echo "mv $SERVER_HOME $SERVER_HOME-bak" mv $SERVER_HOME $SERVER_HOME-bak fi diff --git a/assembly/config/config.sh b/assembly/config/config.sh index d36b2acb9a..5a27e82eca 100644 --- a/assembly/config/config.sh +++ b/assembly/config/config.sh @@ -7,7 +7,7 @@ SERVER_HEAP_SIZE="512M" ### The install home path of DSS,Must provided DSS_INSTALL_HOME=/appcom/Install/DSSInstall -DSS_VERSION=1.1.0 +DSS_VERSION=1.1.2 DSS_FILE_NAME="dss-$DSS_VERSION" @@ -33,35 +33,13 @@ GATEWAY_PORT=9001 ### DSS_SERVER ### This service is used to provide dss-server capability. +### dss-server +DSS_SERVER_INSTALL_IP=127.0.0.1 +DSS_SERVER_PORT=9043 -### project-server -DSS_FRAMEWORK_PROJECT_SERVER_INSTALL_IP=127.0.0.1 -DSS_FRAMEWORK_PROJECT_SERVER_PORT=9002 -### orchestrator-server -DSS_FRAMEWORK_ORCHESTRATOR_SERVER_INSTALL_IP=127.0.0.1 -DSS_FRAMEWORK_ORCHESTRATOR_SERVER_PORT=9003 -### apiservice-server -DSS_APISERVICE_SERVER_INSTALL_IP=127.0.0.1 -DSS_APISERVICE_SERVER_PORT=9004 -### dss-workflow-server -DSS_WORKFLOW_SERVER_INSTALL_IP=127.0.0.1 -DSS_WORKFLOW_SERVER_PORT=9005 -### dss-flow-execution-server -DSS_FLOW_EXECUTION_SERVER_INSTALL_IP=127.0.0.1 -DSS_FLOW_EXECUTION_SERVER_PORT=9006 -###dss-scriptis-server -DSS_SCRIPTIS_SERVER_INSTALL_IP=127.0.0.1 -DSS_SCRIPTIS_SERVER_PORT=9008 - -###dss-data-api-server -DSS_DATA_API_SERVER_INSTALL_IP=127.0.0.1 -DSS_DATA_API_SERVER_PORT=9208 -###dss-data-governance-server -DSS_DATA_GOVERNANCE_SERVER_INSTALL_IP=127.0.0.1 -DSS_DATA_GOVERNANCE_SERVER_PORT=9209 -###dss-guide-server -DSS_GUIDE_SERVER_INSTALL_IP=127.0.0.1 -DSS_GUIDE_SERVER_PORT=9210 +### dss-apps-server +DSS_APPS_SERVER_INSTALL_IP=127.0.0.1 +DSS_APPS_SERVER_PORT=9044 ############## ############## dss_appconn_instance configuration start ############## ############## ####eventchecker表的地址,一般就是dss数据库 diff --git a/assembly/dss-package/pom.xml b/assembly/dss-package/pom.xml index 7b1402215c..d264e5c0a3 100644 --- a/assembly/dss-package/pom.xml +++ b/assembly/dss-package/pom.xml @@ -21,7 +21,8 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 + ../../pom.xml 4.0.0 dss-package @@ -93,6 +94,12 @@ org.apache.linkis linkis-hadoop-common ${linkis.version} + + + netty + io.netty + + com.fasterxml.jackson.core @@ -116,11 +123,39 @@ org.apache.linkis linkis-mybatis ${linkis.version} + + + org.springframework.boot + spring-boot-starter + + + org.springframework.boot + spring-boot-autoconfigure + + + org.springframework + spring-beans + + + org.springframework + spring-jdbc + + + com.zaxxer + HikariCP + + org.apache.linkis linkis-storage ${linkis.version} + + + commons-text + org.apache.commons + + @@ -132,7 +167,7 @@ com.fasterxml classmate - 1.3.4 + 1.5.1 diff --git a/assembly/dss-package/src/main/assembly/distribution.xml b/assembly/dss-package/src/main/assembly/distribution.xml index 09fce1e059..79f2eeee62 100644 --- a/assembly/dss-package/src/main/assembly/distribution.xml +++ b/assembly/dss-package/src/main/assembly/distribution.xml @@ -69,70 +69,59 @@ - - - ${basedir}/../../dss-framework/dss-framework-project-server/target/out/dss-framework-project-server/lib/ - - lib/dss-framework/dss-framework-project-server - - **/* - - + + + + + + + + + - - - ${basedir}/../../dss-framework/framework-plugins/dss-framework-migrate-server/target/ - - lib/dss-framework/dss-framework-project-server - - *${dss.version}.jar - - + + + + + + + + + + + + + + + + + + + - ${basedir}/../../dss-framework/dss-framework-orchestrator-server/target/out/dss-framework-orchestrator-server/lib/ - - lib/dss-framework/dss-framework-orchestrator-server/ - - **/* - - - - - - - ${basedir}/../../dss-apps/dss-apiservice-server/target/out/dss-apiservice-server/lib/ + ${basedir}/../../dss-apps/dss-apps-server/target/out/dss-apps-server/lib/ - lib/dss-apps/dss-apiservice-server + lib/dss-apps-server/ **/* - + - ${basedir}/../../dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/target/out/dss-workflow-server/lib/ + ${basedir}/../../dss-server/target/out/dss-server/lib/ - lib/dss-orchestrator/dss-workflow-server/ + lib/dss-server/ **/* - - - - ${basedir}/../../dss-orchestrator/orchestrators/dss-workflow/dss-flow-execution-server/target/out/dss-flow-execution-server/lib/ - - lib/dss-orchestrator/dss-flow-execution-server/ - - **/* - - @@ -156,17 +145,6 @@ - - - - ${basedir}/../../dss-appconn/appconns/dss-orchestrator-framework-appconn/target/out/ - - dss-appconns - - **/* - - - @@ -211,17 +189,6 @@ - - - @@ -273,48 +240,5 @@ - - - - ${basedir}/../../dss-apps/dss-scriptis-server/target/out/dss-scriptis-server/lib/ - - lib/dss-apps/dss-scriptis-server - - **/* - - - - - - - ${basedir}/../../dss-data-api/dss-data-api-server/target/out/dss-data-api-server/lib/ - - lib/dss-data-api/dss-data-api-server - - **/* - - - - - - - ${basedir}/../../dss-data-governance/dss-data-governance-server/target/out/dss-data-governance-server/lib/ - - lib/dss-data-governance/dss-data-governance-server - - **/* - - - - - - - ${basedir}/../../dss-apps/dss-user-guide/dss-user-guide-server/target/out/dss-user-guide-server/lib/ - - lib/dss-guide/dss-guide-server - - **/* - - diff --git a/assembly/pom.xml b/assembly/pom.xml index 2e41028b04..09f427fb6b 100644 --- a/assembly/pom.xml +++ b/assembly/pom.xml @@ -22,7 +22,8 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 + ../pom.xml pom 4.0.0 diff --git a/conf/application-dss.yml b/conf/application-dss.yml index b0473838d8..46ee82c7a1 100644 --- a/conf/application-dss.yml +++ b/conf/application-dss.yml @@ -1,13 +1,11 @@ - eureka: client: + registry-fetch-interval-seconds: 8 serviceUrl: defaultZone: http://127.0.0.1:20303/eureka/ - #instance: - #prefer-ip-address: true - #instance-id: ${spring.cloud.client.ip-address}:${server.port} - #metadata-map: - #test: wedatasphere + instance: + lease-renewal-interval-in-seconds: 4 + lease-expiration-duration-in-seconds: 12 management: endpoints: @@ -17,7 +15,13 @@ management: logging: config: classpath:log4j2.xml -#mybatis: -# configuration: -# log-impl: org.apache.ibatis.logging.stdout.StdOutImpl +ribbon: + ReadTimeout: 300000 + ConnectTimeout: 300000 +feign: + client: + config: + default: + connectTimeout: 90000 + readTimeout: 90000 diff --git a/conf/atlas-application.properties b/conf/atlas-application.properties deleted file mode 100644 index e69de29bb2..0000000000 diff --git a/conf/dss-apiservice-server.properties b/conf/dss-apiservice-server.properties deleted file mode 100644 index 5162f42bcd..0000000000 --- a/conf/dss-apiservice-server.properties +++ /dev/null @@ -1,42 +0,0 @@ -# -# Copyright 2019 WeBank -# 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. -# -# - -# Spring configurations -spring.server.port=9206 -spring.spring.application.name=dss-apiservice-server - -wds.linkis.server.mybatis.mapperLocations=classpath*:com/webank/wedatasphere/dss/apiservice/core/dao/mapper/*.xml -wds.linkis.server.mybatis.typeAliasesPackage=com.webank.wedatasphere.dss.apiservice.core.bo,com.webank.wedatasphere.dss.apiservice.core.vo -wds.linkis.server.mybatis.BasePackage=com.webank.wedatasphere.dss.apiservice.core.dao - -wds.linkis.server.restful.scan.packages=com.webank.wedatasphere.dss.apiservice.core.restful - -#sit -wds.linkis.server.version=v1 -wds.linkis.server.url= - -#test -wds.linkis.test.mode=false -wds.linkis.test.user= - - -#dsm -wds.linkis.server.dsm.admin.users= - - -#用于执行的datasource配置 -wds.linkis.datasource.hikari.maximumPoolSize=100 -wds.linkis.datasource.hikari.minimumIdle=10 diff --git a/conf/dss-apps-server.properties b/conf/dss-apps-server.properties new file mode 100644 index 0000000000..6aac2f738e --- /dev/null +++ b/conf/dss-apps-server.properties @@ -0,0 +1,29 @@ +#### apiservice ##### +# Spring configurations +spring.server.port=9206 +spring.spring.application.name=dss-apps-server + +wds.linkis.server.mybatis.mapperLocations=classpath*:com/webank/wedatasphere/dss/apiservice/core/dao/mapper/*.xml,classpath*:com/webank/wedatasphere/dss/scriptis/dao/mapper/*.xml,classpath*:com/webank/wedatasphere/dss/guide/server/dao/impl/*.xml +wds.linkis.server.mybatis.typeAliasesPackage=com.webank.wedatasphere.dss.apiservice.core.bo,com.webank.wedatasphere.dss.apiservice.core.vo,com.webank.wedatasphere.dss.scriptis.vo,com.webank.wedatasphere.dss.guide.server.entity +wds.linkis.server.mybatis.BasePackage=com.webank.wedatasphere.dss.apiservice.core.dao,com.webank.wedatasphere.dss.scriptis.dao,com.webank.wedatasphere.dss.guide.server.dao +wds.linkis.server.restful.scan.packages=com.webank.wedatasphere.dss.apiservice.core.restful,com.webank.wedatasphere.dss.scriptis.restful,com.webank.wedatasphere.dss.guide.server.restful + +wds.linkis.server.version=v1 +wds.linkis.datasource.hikari.maximumPoolSize=100 +wds.linkis.datasource.hikari.minimumIdle=10 + +#### scriptis###### +wds.dss.scriptis.global.limits.exportResEnable=false +wds.dss.scriptis.global.limits.exportTableEnable=false +wds.dss.scriptis.global.limits.downloadResEnable=false +wds.dss.scriptis.global.limits.resCopyEnable=false +wds.dss.scriptis.global.limits.proxyEnable=false + +#####user guide##### +spring.jackson.date-format=yyyy-MM-dd HH:mm:ss +spring.jackson.time-zone=GMT+8 +## guide_images_path +guide.content.images.path=/opt/dss/dss-guide-server/guide_images/ +guide.chapter.images.path=/opt/dss/dss-guide-server/guide_images/ + + diff --git a/conf/dss-data-api-server.properties b/conf/dss-data-api-server.properties deleted file mode 100644 index fe386097ed..0000000000 --- a/conf/dss-data-api-server.properties +++ /dev/null @@ -1,39 +0,0 @@ -# -# /* -# * Copyright 2019 WeBank -# * -# * 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. -# */ -# - -# Spring configurations -spring.server.port=9208 -spring.spring.application.name=dss-data-api-server - -wds.linkis.log.clear=true - -wds.linkis.server.version=v1 - -##restful -wds.linkis.server.restful.scan.packages=com.webank.wedatasphere.dss.data.api.server.restful - -##mybatis -wds.linkis.server.mybatis.mapperLocations=classpath*:com/webank/wedatasphere/dss/data/api/server/dao/impl/*.xml - -wds.linkis.server.mybatis.typeAliasesPackage=com.webank.wedatasphere.dss.data.api.server.entity - -wds.linkis.server.mybatis.BasePackage=com.webank.wedatasphere.dss.data.api.server.dao - -#wds.linkis.gateway.ip=127.0.0.1 -#wds.linkis.gateway.port=9001 -#wds.linkis.gateway.url=http://127.0.0.1:9001/ diff --git a/conf/dss-data-governance-server.properties b/conf/dss-data-governance-server.properties deleted file mode 100644 index b3980c6ccc..0000000000 --- a/conf/dss-data-governance-server.properties +++ /dev/null @@ -1,53 +0,0 @@ -# -# /* -# * Copyright 2019 WeBank -# * -# * 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. -# */ -# - -# Spring configurations -spring.server.port=9209 -spring.spring.application.name=dss-data-governance-server - -wds.linkis.log.clear=true - -wds.linkis.server.version=v1 - -##restful -wds.linkis.server.restful.scan.packages=com.webank.wedatasphere.dss.data.asset.restful,com.webank.wedatasphere.dss.data.classification.restful - -##mybatis -wds.linkis.server.mybatis.mapperLocations=classpath*:com/webank/wedatasphere/dss/data/asset/dao/impl/*.xml -wds.linkis.server.mybatis.typeAliasesPackage=com.webank.wedatasphere.dss.data.asset.entity -wds.linkis.server.mybatis.BasePackage=com.webank.wedatasphere.dss.data.asset.dao,com.webank.wedatasphere.dss.data.warehouse.dao,com.webank.wedatasphere.dss.data.warehouse.mapper - -#wds.linkis.gateway.ip=127.0.0.1 -#wds.linkis.gateway.port=9001 -#wds.linkis.gateway.url=http://127.0.0.1:9001/ - - -# atlas config -atlas.rest.address=http://xxxxxxx:21000 -atlas.username=xxxxxxxx -atlas.password=yyyyyyyyy -atlas.client.readTimeoutMSecs=60000 -atlas.client.connectTimeoutMSecs=60000 - -atlas.cluster.name=primary - -# hive metadata config -metastore.datasource.driver=com.mysql.jdbc.Driver -metastore.datasource.url=jdbc:mysql://xxxxxx:yyyy/metastore?characterEncoding=UTF-8 -metastore.datasource.username=xxxxxx -metastore.datasource.password=yyyyyy \ No newline at end of file diff --git a/conf/dss-flow-execution-server.properties b/conf/dss-flow-execution-server.properties deleted file mode 100644 index 1d8eb8d285..0000000000 --- a/conf/dss-flow-execution-server.properties +++ /dev/null @@ -1,55 +0,0 @@ -# -# Copyright 2019 WeBank -# 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. -# -# - -# Spring configurations -spring.server.port=9006 - -spring.spring.application.name=dss-flow-entrance - -##mybatis -wds.linkis.server.mybatis.mapperLocations=classpath*:com/webank/wedatasphere/dss/flow/execution/entrance/dao/impl/*.xml,classpath*:org/apache/linkis/jobhistory/dao/impl/*.xml - -wds.linkis.server.mybatis.typeAliasesPackage= - -wds.linkis.server.mybatis.BasePackage=com.webank.wedatasphere.dss.flow.execution.entrance.dao,org.apache.linkis.jobhistory.dao - - -wds.linkis.server.restful.scan.packages=org.apache.linkis.entrance.restful,com.webank.wedatasphere.dss.flow.execution.entrance.restful - -#wds.linkis.server.component.exclude.classes=org.apache.linkis.DataWorkCloudApplication - -wds.linkis.engine.application.name=flowExecutionEngine -wds.linkis.enginemanager.application.name=flowExecution - -wds.linkis.query.application.name=linkis-ps-publicservice - -wds.linkis.console.config.application.name=linkis-ps-publicservice -wds.linkis.engine.creation.wait.time.max=20m -wds.linkis.server.version=v1 - -wds.linkis.server.socket.mode=true - -wds.linkis.client.flow.adminuser=ws -wds.linkis.client.flow.author.user.token=WS-AUTH - -wds.linkis.server.component.exclude.classes=org.apache.linkis.entranceclient.conf.ClientForEntranceSpringConfiguration,org.apache.linkis.entranceclient.conf.ClientSpringConfiguration - -wds.linkis.server.component.exclude.packages=org.apache.linkis.entrance.restful. -spring.spring.main.allow-bean-definition-overriding=true - -wds.linkis.entrance.config.log.path=file:///appcom/tmp/dss/ -wds.linkis.spark.engine.version=2.4.3 -wds.linkis.hive.engine.version=2.3.3 diff --git a/conf/dss-framework-orchestrator-server.properties b/conf/dss-framework-orchestrator-server.properties deleted file mode 100644 index a2b589a16b..0000000000 --- a/conf/dss-framework-orchestrator-server.properties +++ /dev/null @@ -1,41 +0,0 @@ -# -# Copyright 2019 WeBank -# 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. -# -# - -# Spring configurations -spring.server.port=9003 -spring.spring.application.name=dss-framework-orchestrator-server-dev - -wds.linkis.test.mode=false - -wds.linkis.test.user=hadoop - -wds.linkis.log.clear=true - -wds.linkis.server.version=v1 - -##restful -wds.linkis.server.restful.scan.packages=com.webank.wedatasphere.dss.orchestrator.server.restful - -##mybatis -wds.linkis.server.mybatis.mapperLocations=classpath*:com/webank/wedatasphere/dss/framework/appconn/dao/impl/*.xml,classpath*:com/webank/wedatasphere/dss/orchestrator/core/dao/impl/*.xml,classpath*:com/webank/wedatasphere/dss/server/dao/impl/*.xml,classpath*:com/webank/wedatasphere/dss/application/dao/impl/*.xml,classpath*:com/webank/wedatasphere/dss/workspace/mapper/impl/*.xml,classpath*:com/webank/wedatasphere/dss/workspace/common/dao/impl/*.xml,classpath*:com/webank/wedatasphere/dss/orchestrator/db/dao/impl/*.xml - -wds.linkis.server.mybatis.typeAliasesPackage=com.webank.wedatasphere.dss.server.entity,com.webank.wedatasphere.dss.application.entity,com.webank.wedatasphere.dss.framework.appconn.entity - -wds.linkis.server.mybatis.BasePackage=com.webank.wedatasphere.dss.framework.appconn.dao,com.webank.wedatasphere.dss.orchestrator.core.dao,com.webank.wedatasphere.dss.server.dao,com.webank.wedatasphere.dss.application.dao,com.webank.wedatasphere.dss.workspace.mapper,com.webank.wedatasphere.dss.workspace.common.dao,com.webank.wedatasphere.dss.workspace.common.dao,com.webank.wedatasphere.dss.orchestrator.db.dao - - -##export file dir -wds.dss.server.export.url=/appcom/tmp/dss diff --git a/conf/dss-framework-project-server.properties b/conf/dss-framework-project-server.properties deleted file mode 100644 index 6b8086de83..0000000000 --- a/conf/dss-framework-project-server.properties +++ /dev/null @@ -1,37 +0,0 @@ -# -# Copyright 2019 WeBank -# 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. -# -# - -# Spring configurations -spring.server.port=9202 -spring.spring.application.name=dss-framework-project-server - -wds.linkis.log.clear=true - -wds.linkis.server.version=v1 - -##restful -wds.linkis.server.restful.scan.packages=com.webank.wedatasphere.dss.framework.workspace.restful,com.webank.wedatasphere.dss.framework.project.restful,com.webank.wedatasphere.dss.framework.release.restful,com.webank.wedatasphere.dss.framework.appconn.restful,com.webank.wedatasphere.dss.framework.admin.restful - -##mybatis -wds.linkis.server.mybatis.mapperLocations=classpath*:com/webank/wedatasphere/dss/framework/workspace/dao/impl/*.xml,classpath*:com/webank/wedatasphere/dss/application/dao/impl/*.xml,classpath*:com/webank/wedatasphere/dss/framework/project/dao/impl/*Mapper.xml,classpath*:com/webank/wedatasphere/dss/framework/appconn/dao/impl/*.xml,classpath*:com/webank/wedatasphere/dss/framework/release/dao/impl/*.xml,classpath*:com/webank/wedatasphere/dss/framework/admin/xml/impl/*.xml - -wds.linkis.server.mybatis.typeAliasesPackage=com.webank.wedatasphere.dss.application.entity,com.webank.wedatasphere.dss.common.entity,com.webank.wedatasphere.dss.framework.workspace.bean,com.webank.wedatasphere.dss.framework.project.entity,com.webank.wedatasphere.dss.framework.appconn.entity,com.webank.wedatasphere.dss.framework.release.entity,com.webank.wedatasphere.dss.framework.admin.pojo.entity - -wds.linkis.server.mybatis.BasePackage=com.webank.wedatasphere.dss.framework.workspace.dao,com.webank.wedatasphere.dss.application.dao,com.webank.wedatasphere.dss.framework.project.dao,com.webank.wedatasphere.dss.framework.appconn.dao,com.webank.wedatasphere.dss.framework.release.dao,com.webank.wedatasphere.dss.framework.admin.xml - -wds.dss.appconn.checker.development.ignore.list=workflow,sendemail -wds.dss.appconn.checker.project.ignore.list=visualis - diff --git a/conf/dss-guide-server.properties b/conf/dss-guide-server.properties deleted file mode 100644 index 3a2f488d7f..0000000000 --- a/conf/dss-guide-server.properties +++ /dev/null @@ -1,55 +0,0 @@ -# -# /* -# * Copyright 2019 WeBank -# * -# * 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. -# */ -# - -# Spring configurations -spring.server.port=9210 -spring.spring.application.name=dss-guide-server - -spring.jackson.date-format=yyyy-MM-dd HH:mm:ss -spring.jackson.time-zone=GMT+8 - -wds.linkis.server.version=v1 - -wds.linkis.log.clear=true - - - -##restful -wds.linkis.server.restful.scan.packages=com.webank.wedatasphere.dss.guide.server.restful - -##mybatis -wds.linkis.server.mybatis.mapperLocations=classpath*:com/webank/wedatasphere/dss/guide/server/dao/impl/*.xml -wds.linkis.server.mybatis.typeAliasesPackage=com.webank.wedatasphere.dss.guide.server.entity -wds.linkis.server.mybatis.BasePackage=com.webank.wedatasphere.dss.guide.server.dao - - -## guide_images_path -guide.content.images.path=/opt/dss/dss-guide-server/guide_images/ -guide.chapter.images.path=/opt/dss/dss-guide-server/guide_images/ - -#gitbook -#The machine where the file exists -target.ip.address=127.0.0.1 -#The file path of the machine where the file is stored -host.gitbook.path=/appcom/Install/ApacheInstall/gitbook_books -#The path to copy the file to the current machine -target.gitbook.path=/appcom/Install/ApacheInstall -#SUMMARY.md ignore absolve -summary.ignore.model=km -#sync model include: gitbook or database -guide.sync.model=gitbook diff --git a/conf/dss-scriptis-server.properties b/conf/dss-scriptis-server.properties deleted file mode 100644 index cda41694b9..0000000000 --- a/conf/dss-scriptis-server.properties +++ /dev/null @@ -1,36 +0,0 @@ -# -# Copyright 2019 WeBank -# 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. -# -# - -# Spring configurations -spring.server.port=9009 -spring.spring.application.name=dss-scriptis-server - -wds.linkis.server.mybatis.mapperLocations=classpath*:com/webank/wedatasphere/dss/scriptis/dao/mapper/*.xml -wds.linkis.server.mybatis.typeAliasesPackage=com.webank.wedatasphere.dss.scriptis.vo -wds.linkis.server.mybatis.BasePackage=com.webank.wedatasphere.dss.scriptis.dao - -wds.linkis.server.restful.scan.packages=com.webank.wedatasphere.dss.scriptis.restful - -#sit -wds.linkis.server.version=v1 -wds.linkis.server.url= - -#test -wds.linkis.test.mode=false -wds.linkis.test.user= - -wds.dss.scriptis.global.limits.exportResEnable=false -wds.dss.scriptis.global.limits.exportTableEnable=false \ No newline at end of file diff --git a/conf/dss-server.properties b/conf/dss-server.properties new file mode 100644 index 0000000000..1c5264195a --- /dev/null +++ b/conf/dss-server.properties @@ -0,0 +1,66 @@ +# Spring configurations +wds.dss.appconn.framework.ismanager=true +spring.server.port=9503 +spring.spring.application.name=dss-server-dev + +wds.linkis.log.clear=true + +wds.linkis.server.version=v1 + +##restful +wds.linkis.server.restful.scan.packages=com.webank.wedatasphere.dss.framework.workspace.restful,com.webank.wedatasphere.dss.framework.project.restful,com.webank.wedatasphere.dss.framework.release.restful,com.webank.wedatasphere.dss.framework.appconn.restful,com.webank.wedatasphere.dss.framework.admin.restful,com.webank.wedatasphere.dss.orchestrator.server.restful,com.webank.wedatasphere.dss.flow.execution.entrance.restful,com.webank.wedatasphere.dss.workflow.restful + +##mybatis +wds.linkis.server.mybatis.mapperLocations=classpath*:com/webank/wedatasphere/dss/framework/workspace/dao/impl/*.xml,classpath*:com/webank/wedatasphere/dss/framework/project/dao/impl/*Mapper.xml,classpath*:com/webank/wedatasphere/dss/framework/release/dao/impl/*.xml,classpath*:com/webank/wedatasphere/dss/framework/admin/xml/impl/*.xml,classpath*:com/webank/wedatasphere/dss/framework/appconn/dao/impl/*.xml,classpath*:com/webank/wedatasphere/dss/orchestrator/core/dao/impl/*.xml,classpath*:com/webank/wedatasphere/dss/server/dao/impl/*.xml,classpath*:com/webank/wedatasphere/dss/application/dao/impl/*.xml,classpath*:com/webank/wedatasphere/dss/workspace/mapper/impl/*.xml,classpath*:com/webank/wedatasphere/dss/workspace/common/dao/impl/*.xml,classpath*:com/webank/wedatasphere/dss/orchestrator/db/dao/impl/*.xml,classpath*:com/webank/wedatasphere/dss/flow/execution/entrance/dao/impl/*.xml,classpath*:org/apache/linkis/jobhistory/dao/impl/*.xml,classpath*:com/webank/wedatasphere/dss/workflow/dao/impl/*.xml,classpath*:com/webank/wedatasphere/dss/scriptis/dao/mapper/*.xml,classpath*:com/webank/wedatasphere/dss/common/dao/impl/*.xml + +wds.linkis.server.mybatis.typeAliasesPackage=com.webank.wedatasphere.dss.application.entity,com.webank.wedatasphere.dss.common.entity,com.webank.wedatasphere.dss.framework.workspace.bean,com.webank.wedatasphere.dss.framework.project.entity,com.webank.wedatasphere.dss.framework.appconn.entity,com.webank.wedatasphere.dss.framework.release.entity,com.webank.wedatasphere.dss.framework.admin.pojo.entity,com.webank.wedatasphere.dss.server.entity,com.webank.wedatasphere.dss.application.entity,com.webank.wedatasphere.dss.framework.appconn.entity,com.webank.wedatasphere.dss.workflow.entity,com.webank.wedatasphere.dss.framework.appconn.entity,com.webank.wedatasphere.dss.scriptis.pojo.entity,com.webank.wedatasphere.dss.common.entity + +wds.linkis.server.mybatis.BasePackage=com.webank.wedatasphere.dss.framework.workspace.dao,com.webank.wedatasphere.dss.application.dao,com.webank.wedatasphere.dss.framework.project.dao,com.webank.wedatasphere.dss.framework.appconn.dao,com.webank.wedatasphere.dss.framework.release.dao,com.webank.wedatasphere.dss.orchestrator.db.dao,com.webank.wedatasphere.dss.framework.admin.xml,com.webank.wedatasphere.dss.common.server.dao,com.webank.wedatasphere.dss.orchestrator.core.dao,com.webank.wedatasphere.dss.server.dao,com.webank.wedatasphere.dss.application.dao,com.webank.wedatasphere.dss.workspace.mapper,com.webank.wedatasphere.dss.workspace.common.dao,com.webank.wedatasphere.dss.workspace.common.dao,com.webank.wedatasphere.dss.orchestrator.db.dao,com.webank.wedatasphere.dss.flow.execution.entrance.dao,org.apache.linkis.jobhistory.dao,com.webank.wedatasphere.dss.workflow.dao,com.webank.wedatasphere.dss.scriptis.dao,com.webank.wedatasphere.dss.common.dao + +####project +wds.dss.appconn.checker.development.ignore.list=orchestrator-framework,workflow,sendemail +wds.dss.appconn.checker.project.ignore.list=visualis + +####exclude appconn +wds.dss.appconn.disabled=exchangis + +##import file dir +wds.dss.server.scheduling.clear.cs.cron=0/5 * * * * ? + +wds.dss.publish.max.remain.version=10 + +####workflow +wds.dss.appconn.scheduler.project.store.dir=file:///appcom/tmp/wds/scheduler +wds.dss.appconn.scheduler.azkaban.login.passwd=userpwd +##import file dir +wds.dss.file.upload.dir=/appcom/tmp/uploads +wds.dss.server.flow.edit.lock.timeout=180000 + +wds.dss.server.export.env=DEV +wds.dss.server.import.env=DEV + +###flow-execution +wds.linkis.engine.application.name=flowExecutionEngine +wds.linkis.enginemanager.application.name=flowExecution + +wds.linkis.query.application.name=linkis-ps-publicservice +wds.linkis.jobhistory.application.name=linkis-ps-publicservice +wds.linkis.console.config.application.name=linkis-ps-publicservice +wds.linkis.engine.creation.wait.time.max=20m + +#hadoop config dir +wds.linkis.entrance.config.log.path=hdfs:///appcom/logs/linkis +wds.linkis.resultSet.store.path=hdfs:///tmp/linkis + +wds.linkis.server.socket.mode=true + +wds.linkis.client.flow.adminuser=ws +wds.linkis.client.flow.author.user.token=WS-AUTH + +wds.linkis.server.component.exclude.classes=org.apache.linkis.entranceclient.conf.ClientForEntranceSpringConfiguration,org.apache.linkis.entranceclient.conf.ClientSpringConfiguration + +wds.linkis.server.component.exclude.packages=org.apache.linkis.entrance.restful. + +spring.spring.main.allow-bean-definition-overriding=true + +wds.linkis.flow.job.creator.v1=nodeexecution \ No newline at end of file diff --git a/conf/dss-workflow-server.properties b/conf/dss-workflow-server.properties deleted file mode 100644 index df4bd6dea5..0000000000 --- a/conf/dss-workflow-server.properties +++ /dev/null @@ -1,44 +0,0 @@ -# -# Copyright 2019 WeBank -# 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. -# -# - -# Spring configurations -spring.server.port=9207 -spring.spring.application.name=dss-workflow-server-dev - -wds.linkis.test.mode=false - -wds.linkis.test.user= - -wds.linkis.log.clear=true - -wds.linkis.server.version=v1 - -##restful -wds.linkis.server.restful.scan.packages=com.webank.wedatasphere.dss.workflow.restful - -##mybatis -wds.linkis.server.mybatis.mapperLocations=classpath*:com/webank/wedatasphere/dss/workflow/dao/impl/*.xml,classpath*:com/webank/wedatasphere/dss/framework/appconn/dao/impl/*.xml - -wds.linkis.server.mybatis.typeAliasesPackage=com.webank.wedatasphere.dss.workflow.entity,com.webank.wedatasphere.dss.framework.appconn.entity - -wds.linkis.server.mybatis.BasePackage=com.webank.wedatasphere.dss.workflow.dao,com.webank.wedatasphere.dss.framework.appconn.dao - -##import file dir -wds.dss.file.upload.dir=/appcom/tmp/uploads - -wds.dss.server.export.env=DEV -wds.dss.server.import.env=DEV - diff --git a/conf/dss.properties b/conf/dss.properties index 9aa8c8e576..169b91ff8d 100644 --- a/conf/dss.properties +++ b/conf/dss.properties @@ -24,14 +24,14 @@ wds.linkis.server.mybatis.datasource.url= wds.linkis.server.mybatis.datasource.username= ***REMOVED*** -wds.dss.esb.appid= -wds.dss.esb.token= +wds.dss.check.server.active.period=30 wds.dss.appconn.scheduler.job.label=dev - wds.linkis.reflect.scan.package=org.apache.linkis,com.webank.wedatasphere.dss spring.spring.mvc.servlet.path=/api/rest_j/v1 spring.spring.servlet.multipart.max-file-size=200MB spring.spring.servlet.multipart.max-request-size=200MB wds.dss.project.strict.mode=true +wds.linkis.rpc.spring.params.enable=true +wds.linkis.gateway.conf.publicservice.list=query,application,filesystem,udf,variable,microservice diff --git a/conf/log4j2.xml b/conf/log4j2.xml index f3d6e66bcb..1d65cd6fc8 100644 --- a/conf/log4j2.xml +++ b/conf/log4j2.xml @@ -27,8 +27,18 @@ + + + + + + + + + diff --git a/db/dss_ddl.sql b/db/dss_ddl.sql index 938961d097..b32a9c958d 100644 --- a/db/dss_ddl.sql +++ b/db/dss_ddl.sql @@ -10,6 +10,7 @@ CREATE TABLE `dss_appconn` ( `class_name` varchar(255) DEFAULT NULL COMMENT '需要关联的某一个AppConn标识', `appconn_class_path` varchar(255) DEFAULT NULL COMMENT '需要关联的某一个AppConn标识', `resource` varchar(255) DEFAULT NULL COMMENT 'bml的资源ID', + `is_micro_app` tinyint(1) NOT NULL DEFAULT '0' COMMENT '是否微应用嵌入', PRIMARY KEY (`id`), UNIQUE KEY `idx_appconn_name` (`appconn_name`) ) ENGINE=InnoDB AUTO_INCREMENT=8 DEFAULT CHARSET=utf8mb4 COMMENT='dss appconn表'; @@ -20,7 +21,7 @@ CREATE TABLE `dss_appconn_instance` ( `appconn_id` int(20) NOT NULL COMMENT 'appconn的主键', `label` varchar(128) NOT NULL COMMENT '实例的标签', `url` varchar(128) DEFAULT NULL COMMENT '访问第三方的url', - `enhance_json` varchar(1024) DEFAULT NULL COMMENT 'json格式的配置', + `enhance_json` varchar(2048) DEFAULT NULL COMMENT 'json格式的配置', `homepage_uri` varchar(255) DEFAULT NULL COMMENT '主页uri,非URL', PRIMARY KEY (`id`) ) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8mb4 COMMENT='dss instance的实例表'; @@ -161,6 +162,7 @@ CREATE TABLE `dss_workspace` ( `last_update_time` datetime DEFAULT NULL, `last_update_user` varchar(30) DEFAULT NULL COMMENT '最新修改用户', `workspace_type` varchar(20) DEFAULT NULL comment '工作空间类型', + `admin_permission` tinyint(1) DEFAULT 1 NOT NULL COMMENT '工作空间管理员是否有权限查看该空间下所有项目,1可以,0不可以', PRIMARY KEY (`id`), UNIQUE KEY `name` (`name`) ) ENGINE=InnoDB AUTO_INCREMENT=224 DEFAULT CHARSET=utf8; @@ -327,11 +329,12 @@ CREATE TABLE `dss_workflow` ( `rank` int(10) DEFAULT NULL, `project_id` bigint(20) DEFAULT NULL, `has_saved` tinyint(1) DEFAULT NULL, - `uses` varchar(255) DEFAULT NULL, + `uses` varchar(500) DEFAULT NULL, `bml_version` varchar(255) DEFAULT NULL, `resource_id` varchar(255) DEFAULT NULL, `linked_appconn_names` varchar(255) DEFAULT NULL, `dss_labels` varchar(255) DEFAULT NULL, + `metrics` varchar(1024) NULL, PRIMARY KEY (`id`) USING BTREE ) ENGINE=InnoDB AUTO_INCREMENT=455 DEFAULT CHARSET=utf8mb4 ROW_FORMAT=COMPACT; @@ -370,6 +373,7 @@ CREATE TABLE `dss_workflow_task` ( `log_path` varchar(200) DEFAULT NULL COMMENT 'File path of the log files', `result_location` varchar(200) DEFAULT NULL COMMENT 'File path of the result', `status` varchar(50) DEFAULT NULL COMMENT 'Script execution status, must be one of the following: Inited, WaitForRetry, Scheduled, Running, Succeed, Failed, Cancelled, Timeout', + `instance_name` varchar(128) DEFAULT NULL COMMENT 'Execute task instance', `created_time` datetime DEFAULT NULL COMMENT 'Creation time', `updated_time` datetime DEFAULT NULL COMMENT 'Update time', `run_type` varchar(50) DEFAULT NULL COMMENT 'Further refinement of execution_application_time, e.g, specifying whether to run pySpark or SparkR', @@ -574,6 +578,8 @@ CREATE TABLE `dss_workspace_user_role` ( `create_time` datetime DEFAULT NULL, `created_by` varchar(255) DEFAULT NULL, `user_id` bigint(20) DEFAULT NULL, + `update_user` varchar(32) DEFAULT NULL, + `update_time` datetime DEFAULT NULL, PRIMARY KEY (`id`) ) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8 comment '空间用户角色关系表'; @@ -590,3 +596,245 @@ CREATE TABLE `dss_proxy_user` ( PRIMARY KEY (`id`) ) ENGINE=InnoDB AUTO_INCREMENT=214 DEFAULT CHARSET=utf8; +DROP TABLE IF EXISTS `dss_orchestrator_copy_info`; +CREATE TABLE `dss_orchestrator_copy_info` ( + `id` VARCHAR(128) NOT NULL COMMENT '主键', + `username` VARCHAR(128) DEFAULT NULL COMMENT '用户名', + `type` VARCHAR(128) DEFAULT NULL COMMENT '编排类别', + `source_orchestrator_id` INT(20) DEFAULT NULL COMMENT '源编排ID', + `source_orchestrator_name` VARCHAR(255) DEFAULT NULL COMMENT '源编排名', + `target_orchestrator_name` VARCHAR(255) DEFAULT NULL COMMENT '目标编排名', + `source_project_name` VARCHAR(255) DEFAULT NULL COMMENT '源工程名', + `target_project_name` VARCHAR(255) DEFAULT NULL COMMENT '目标工程名', + `workspace_id` INT(20) DEFAULT NULL COMMENT '工作空间ID', + `workflow_node_suffix` VARCHAR(255) DEFAULT NULL COMMENT '目标工作流节点后缀', + `microserver_name` VARCHAR(128) COMMENT '微服务名', + `exception_info` VARCHAR(128) COMMENT '异常信息', + `status` int(1) DEFAULT 0 COMMENT '复制任务最终状态', + `instance_name` varchar(128) DEFAULT NULL COMMENT '执行任务的实例', + `is_copying` int(1) DEFAULT 0 COMMENT '编排是否在被复制', + `success_node` TEXT COMMENT '复制成功节点', + `start_time` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '复制开始时间', + `end_time` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '复制结束时间', + PRIMARY KEY (`id`), + INDEX index_soi(source_orchestrator_id) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 ROW_FORMAT=COMPACT COMMENT='编排复制信息表'; + +DROP TABLE IF EXISTS `dss_project_operate_record`; +CREATE TABLE `dss_project_operate_record` +( + `id` bigint(20) NOT NULL AUTO_INCREMENT, + `record_id` varchar(64) NOT NULL, + `workspace_id` bigint(20) NOT NULL COMMENT '空间id', + `project_id` bigint(20) NOT NULL COMMENT '项目id', + `operate_type` int(11) NOT NULL COMMENT '操作类型', + `status` int(11) NOT NULL COMMENT '操作状态', + `instance_name` VARCHAR(128) DEFAULT NULL COMMENT '执行任务的实例', + `content` longtext DEFAULT NULL COMMENT '操作内容详情', + `result_resource_uri` text DEFAULT NULL COMMENT '操作结果资源的uri,是一个json', + `creator` varchar(100) NOT NULL, + `create_time` datetime NOT NULL, + PRIMARY KEY (`id`), + KEY `idx_record_id` (`record_id`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='项目操作记录表'; + +DROP TABLE IF EXISTS `dss_release_task`; +CREATE TABLE `dss_release_task` ( + `id` bigint(20) NOT NULL AUTO_INCREMENT, + `project_id` bigint(20) NOT NULL, + `orchestrator_version_id` bigint(20) NOT NULL, + `orchestrator_id` bigint(20) NOT NULL, + `release_user` varchar(128) NOT NULL, + `create_time` datetime DEFAULT NULL, + `update_time` datetime DEFAULT NULL, + `status` varchar(64) DEFAULT 'init', + `instance_name` varchar(128) DEFAULT NULL COMMENT '执行任务的实例', + `error_msg` varchar(500) DEFAULT NULL COMMENT '发布错误信息', + `comment` varchar(500) DEFAULT NULL COMMENT '发布描述', + `log_msg` varchar(255) DEFAULT NULL COMMENT '日志信息或日志路径', + `bak` varchar(255) DEFAULT NULL COMMENT '备用字段', + PRIMARY KEY (`id`) USING BTREE +) ENGINE=InnoDB AUTO_INCREMENT=605 DEFAULT CHARSET=utf8mb4 ROW_FORMAT=COMPACT; + +-- 首页公告表 +DROP TABLE IF EXISTS `dss_notice`; +CREATE TABLE `dss_notice` +( + `id` int(11) NOT NULL AUTO_INCREMENT COMMENT '主键ID', + `content` text DEFAULT NULL COMMENT '公告内容', + `start_time` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '生效时间', + `end_time` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '失效时间', + PRIMARY KEY (`id`) +) ENGINE = InnoDB + DEFAULT CHARSET = utf8mb4 COLLATE=utf8mb4_bin COMMENT ='首页公告内容'; + +DROP TABLE IF EXISTS `dss_orchestrator_job_info`; +CREATE TABLE `dss_orchestrator_job_info` ( + `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT '主键', + `job_id` varchar(64) DEFAULT NULL COMMENT 'job ID', + `conversion_job_json` varchar(1024) DEFAULT NULL COMMENT 'job信息', + `create_time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间', + `update_time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '修改时间', + `instance_name` varchar(128) DEFAULT NULL COMMENT '执行任务的实例', + `status` varchar(128) DEFAULT NULL COMMENT '转换任务状态', + `error_msg` varchar(2048) DEFAULT NULL COMMENT '转换任务异常信息', + PRIMARY KEY (`id`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='dss_orchestrator_job_info表'; + +DROP TABLE IF EXISTS `dss_project_copy_task`; +CREATE TABLE IF NOT EXISTS `dss_project_copy_task` ( + `id` BIGINT(20) NOT NULL AUTO_INCREMENT COMMENT '主键', + `workspace_id` BIGINT(20) COMMENT '空间ID', + `source_project_id` BIGINT(20) COMMENT '(源)复制工程ID', + `source_project_name` VARCHAR(200) COMMENT '(源)复制工程名称', + `copy_project_id` BIGINT(20) COMMENT '复制工程ID', + `copy_project_name` VARCHAR(200) COMMENT '复制工程名称', + `surplus_count` INT(3) COMMENT '剩余复制数量', + `sum_count` INT(3) COMMENT '总数', + `status` INT(1) COMMENT '状态 0:初始化,1:复制中,2:复制成功', + `instance_name` VARCHAR(128) DEFAULT NULL COMMENT '执行任务的实例', + `create_by` VARCHAR(200) COMMENT '创建人', + `create_time` datetime COMMENT '创建时间', + `update_time` datetime COMMENT '上个复制时间', + `error_msg` text COMMENT '失败原因', + `error_orc` VARCHAR(2048) DEFAULT '' COMMENT '拷贝异常编排', + PRIMARY KEY (`id`) +) ENGINE=INNODB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8 COLLATE=utf8_bin COMMENT='复制工程任务表'; + + +-- webank内部表,待开源相关功能使用 +DROP TABLE IF EXISTS `dss_streamis_proxy_user`; +CREATE TABLE `dss_streamis_proxy_user` +( + `id` int(11) NOT NULL AUTO_INCREMENT, + `user_name` varchar(64) DEFAULT NULL COMMENT '实名用户名', + `proxy_user_name` varchar(64) DEFAULT NULL COMMENT '代理用户名', + `create_by` varchar(64) DEFAULT NULL COMMENT '创建者', + `create_time` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间', + PRIMARY KEY (`id`) +) ENGINE = InnoDB DEFAULT CHARSET = utf8mb4 COMMENT='流式应用代理用户表'; + +-- webank内部表,待开源相关功能使用 +DROP TABLE IF EXISTS `dss_workspace_associate_departments`; +CREATE TABLE `dss_workspace_associate_departments` ( + `id` bigint(20) NOT NULL AUTO_INCREMENT, + `workspace_id` bigint(20) DEFAULT NULL, + `departments` text DEFAULT NULL COMMENT '关联的部门-科室列表,逗号分割,若部门后不接科室则代表关联整个部门', + `role_ids` varchar(128) DEFAULT NULL COMMENT '角色id列表,逗号分割', + `create_time` datetime DEFAULT NULL, + `update_time` datetime DEFAULT NULL, + `create_by` varchar(128) DEFAULT NULL, + `update_by` varchar(128) DEFAULT NULL, + PRIMARY KEY (`id`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='空间自动加入用户绑定的部门科室信息'; + +-- 版本发布时的releaseNote信息。webank内部表,待开源相关功能使用 +DROP TABLE IF EXISTS `dss_release_note_content`; +CREATE TABLE `dss_release_note_content` +( + `id` int(11) NOT NULL AUTO_INCREMENT COMMENT '主键ID', + `name` varchar(200) DEFAULT NULL COMMENT '名称', + `title` varchar(200) DEFAULT NULL COMMENT '标题', + `url` varchar(300) DEFAULT NULL COMMENT 'url', + `url_type` int(1) DEFAULT '1' COMMENT 'url类型: 0-内部系统,1-外部系统;默认是外部', + `release_type` int(1) DEFAULT '0' COMMENT '发布形式: 0-作为dss整体发布,1-单独发布scriptis;默认是dss', + `create_time` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间', + PRIMARY KEY (`id`) +) ENGINE = InnoDB DEFAULT CHARSET = utf8mb4 COMMENT ='releaseNote表'; + +-- 用户访问行为统计表,初期只有登录行为统计。webank内部表,待开源相关功能使用 +DROP TABLE IF EXISTS `dss_user_access_audit`; +CREATE TABLE `dss_user_access_audit` +( + `id` int(11) NOT NULL AUTO_INCREMENT COMMENT '主键ID', + `user_name` varchar(64) DEFAULT NULL COMMENT '用户名', + `login_count` BIGINT DEFAULT 0 COMMENT '登录次数', + `first_login` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '第一次登录时间', + `last_login` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '上一次登录时间', + PRIMARY KEY (`id`), + UNIQUE KEY `idx_user_name` (`user_name`) +) ENGINE = InnoDB DEFAULT CHARSET = utf8mb4 COMMENT ='用户访问行为次数统计'; + +-- webank内部表,待开源相关功能使用 +DROP TABLE IF EXISTS `dss_ec_release_strategy`; +CREATE TABLE `dss_ec_release_strategy` +( + `id` int(11) NOT NULL AUTO_INCREMENT COMMENT '主键', + `strategy_id` varchar(64) NOT NULL UNIQUE COMMENT '规则id', + `workspace_id` bigint(20) NOT NULL COMMENT '工作空间id', + `name` varchar(64) NOT NULL COMMENT '规则名', + `description` varchar(128) DEFAULT NULL COMMENT '规则描述', + `queue` varchar(128) NOT NULL UNIQUE COMMENT '关联队列', + `trigger_condition_conf` varchar(1024) NOT NULL COMMENT '触发条件(json)', + `terminate_condition_conf` varchar(1024) NOT NULL COMMENT '终止条件(json)', + `ims_conf` varchar(2048) NOT NULL COMMENT '告警设置(json)', + `status` int(1) DEFAULT 0 COMMENT '规则状态:0禁用 1开启', + `creator` varchar(64) NOT NULL COMMENT '创建人', + `create_time` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间', + `modifier` varchar(64) NOT NULL COMMENT '修改人', + `modify_time` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '修改时间', + `execute_instance` varchar(128) DEFAULT NULL COMMENT '最近处理该规则的服务实例', + `execute_time` datetime DEFAULT NULL COMMENT '最近处理该规则的时间起点', + PRIMARY KEY (`id`), + KEY `idx_strategy_id` (`strategy_id`), + KEY `idx_workspace_id` (`workspace_id`) +) ENGINE = InnoDB + DEFAULT CHARSET = utf8mb4 COLLATE=utf8mb4_bin COMMENT ='EC自动释放规则配置'; + +-- 工作空间关联的队列表。webank内部表,待开源相关功能使用 +DROP TABLE IF EXISTS `dss_queue_in_workspace`; +CREATE TABLE `dss_queue_in_workspace` +( + `id` int(11) NOT NULL AUTO_INCREMENT COMMENT '主键', + `workspace_id` bigint(20) NOT NULL COMMENT '工作空间id', + `queue` varchar(128) NOT NULL UNIQUE COMMENT '队列名', + `apply_user` varchar(64) NOT NULL COMMENT '申请人', + `approve_id` varchar(64) NOT NULL COMMENT '申请单号', + `create_time` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间', + PRIMARY KEY (`id`) +) ENGINE = InnoDB + DEFAULT CHARSET = utf8mb4 COLLATE=utf8mb4_bin COMMENT ='工作空间关联的队列'; + +-- EC释放通知发送记录表。webank内部表,待开源相关功能使用 +DROP TABLE IF EXISTS `dss_ec_release_ims_record`; +CREATE TABLE `dss_ec_release_ims_record` +( + `id` bigint(11) NOT NULL AUTO_INCREMENT COMMENT '主键', + `record_id` varchar(64) NOT NULL COMMENT '发送记录id', + `strategy_id` varchar(64) NOT NULL COMMENT '释放规则id', + `workspace_id` bigint(20) NOT NULL COMMENT '工作空间id', + `content` varchar(1024) NOT NULL COMMENT '发送内容', + `status` int(1) DEFAULT 0 COMMENT '通知状态:0未发送 1已发送 2发送失败', + `execute_instance` varchar(128) NOT NULL COMMENT '负责发送的服务实例', + `create_time` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间', + `modify_time` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '修改时间', + PRIMARY KEY (`id`), + KEY `idx_record_id` (`record_id`) +) ENGINE = InnoDB + DEFAULT CHARSET = utf8mb4 COLLATE=utf8mb4_bin COMMENT ='EC释放通知发送记录'; + +-- 请求释放EC历史。webank内部表,待开源相关功能使用 +DROP TABLE IF EXISTS `dss_ec_kill_history`; +CREATE TABLE `dss_ec_kill_history` +( + `id` bigint(11) NOT NULL AUTO_INCREMENT COMMENT '主键', + `strategy_id` varchar(64) NOT NULL COMMENT '释放规则id', + `workspace_id` bigint(20) NOT NULL COMMENT '工作空间id', + `instance` varchar(128) NOT NULL COMMENT '释放的EC实例名', + `engine_type` varchar(64) NOT NULL COMMENT 'EC类型', + `queue` varchar(128) NOT NULL COMMENT '队列名', + `driver_core` int(11) DEFAULT 0 COMMENT '本地释放核数', + `driver_memory` bigint(11) DEFAULT 0 COMMENT '本地释放内存,单位Byte', + `yarn_core` int(11) DEFAULT 0 COMMENT 'yarn释放核数', + `yarn_memory` bigint(11) DEFAULT 0 COMMENT 'yarn释放内存,单位Byte', + `unlock_duration` bigint(11) DEFAULT 0 COMMENT 'EC空闲时长,单位秒', + `owner` varchar(64) NOT NULL COMMENT 'EC创建者', + `killer` varchar(64) NOT NULL COMMENT 'EC释放触发者', + `ec_start_time` varchar(64) NOT NULL COMMENT 'EC创建的时间', + `kill_time` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '请求killEC的时间', + `create_time` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间', + `execute_instance` varchar(128) DEFAULT NULL COMMENT '负责发送的服务实例', + PRIMARY KEY (`id`), + KEY `idx_workspace_id` (`workspace_id`) +) ENGINE = InnoDB + DEFAULT CHARSET = utf8mb4 COLLATE=utf8mb4_bin COMMENT ='请求释放EC历史'; \ No newline at end of file diff --git a/db/dss_dml.sql b/db/dss_dml.sql index 7ae770c638..3246ab3509 100644 --- a/db/dss_dml.sql +++ b/db/dss_dml.sql @@ -163,6 +163,8 @@ insert into `dss_workflow_node_ui`(`id`,`key`,`description`,`description_en`,`l insert into `dss_workflow_node_ui`(`id`,`key`,`description`,`description_en`,`lable_name`,`lable_name_en`,`ui_type`,`required`,`value`,`default_value`,`is_hidden`,`condition`,`is_advanced`,`order`,`node_menu_type`,`is_base_info`,`position`) values (41,'executeUser','请填写执行用户','Please enter execute user','执行用户','executeUser','Input',1,NULL,NULL,0,NULL,0,1,1,0,'runtime'); insert into `dss_workflow_node_ui`(`id`,`key`,`description`,`description_en`,`lable_name`,`lable_name_en`,`ui_type`,`required`,`value`,`default_value`,`is_hidden`,`condition`,`is_advanced`,`order`,`node_menu_type`,`is_base_info`,`position`) values (42,'Filter','请填写过滤条件','Please enter filter','过滤条件','Filter','Input',1,NULL,NULL,0,NULL,0,1,1,0,'runtime'); INSERT INTO `dss_workflow_node_ui`(`id`,`key`,`description`,`description_en`,`lable_name`,`lable_name_en`,`ui_type`,`required`,`value`,`default_value`,`is_hidden`,`condition`,`is_advanced`,`order`,`node_menu_type`,`is_base_info`,`position`) values (45,'ReuseEngine','请选择是否复用引擎','Please choose to reuse engin or not','是否复用引擎','reuse-engine-or-not','Select',1,'[\"true\",\"false\"]','true',0,NULL,0,1,1,0,'startup'); +INSERT INTO `dss_workflow_node_ui`(`id`,`key`,`description`,`description_en`,`lable_name`,`lable_name_en`,`ui_type`,`required`,`value`,`default_value`,`is_hidden`,`condition`,`is_advanced`,`order`,`node_menu_type`,`is_base_info`,`position`) VALUES (46, 'spark.conf', 'spark自定义参数配置输入,例如spark.sql.shuffle.partitions=10。多个参数使用分号分隔。', 'input spark params config, eg: spark.sql.shuffle.partitions=10. Use semi-colon to split multi-params', 'spark.conf', 'spark.conf', 'Text', 0, NULL, "", 0, NULL, 0, 1, 1, 0, 'startup'); + DELETE FROM dss_workflow_node_to_ui; select @workflow_node_sql:=id from dss_workflow_node where name='sql'; @@ -206,6 +208,14 @@ select @node_ui_job_desc:=id from dss_workflow_node_ui where `key`='job.desc'; select @node_ui_upStreams:=id from dss_workflow_node_ui where `key`='upStreams'; select @node_ui_executeUser:=id from dss_workflow_node_ui where `key`='executeUser'; select @node_ui_ReuseEngine:=id from dss_workflow_node_ui where `key`='ReuseEngine'; +select @sparkConfUiId:=id from dss_workflow_node_ui where `key`="spark.conf"; +select @sqlNodeId:=id from dss_workflow_node where node_type="linkis.spark.sql"; +select @pysparkNodeId:=id from dss_workflow_node where node_type="linkis.spark.py"; +select @scalaNodeId:=id from dss_workflow_node where node_type="linkis.spark.scala"; + +insert into `dss_workflow_node_to_ui`(`workflow_node_id`,`ui_id`) values(@sqlNodeId, @sparkConfUiId); +insert into `dss_workflow_node_to_ui`(`workflow_node_id`,`ui_id`) values(@pysparkNodeId, @sparkConfUiId); +insert into `dss_workflow_node_to_ui`(`workflow_node_id`,`ui_id`) values(@scalaNodeId, @sparkConfUiId); insert into `dss_workflow_node_to_ui`(`workflow_node_id`,`ui_id`) values (@workflow_node_sql,@node_ui_title); insert into `dss_workflow_node_to_ui`(`workflow_node_id`,`ui_id`) values (@workflow_node_sql,@node_ui_desc); @@ -309,7 +319,7 @@ insert into `dss_workflow_node_ui_validate` (`id`, `validate_type`, `validate_ra insert into `dss_workflow_node_ui_validate` (`id`, `validate_type`, `validate_range`, `error_msg`, `error_msg_en`, `trigger`) values('58','Regex','(.+)@(.+)@(.+)','此格式错误,例如:ProjectName@WFName@jobName','Invalid format,example:ProjectName@WFName@jobName','blur'); INSERT INTO `dss_workflow_node_ui_validate` (`id`, `validate_type`, `validate_range`, `error_msg`, `error_msg_en`, `trigger`) values('59','OFT','["true","false"]','请填写是否复用引擎,false:不复用,true:复用','Please fill in whether or not to reuse engine, true: reuse, false: not reuse','blur'); insert into `dss_workflow_node_ui_validate` (`id`, `validate_type`, `validate_range`, `error_msg`, `error_msg_en`, `trigger`) values('60', 'Regex', '^[0-9.]*g{0,1}$', 'Spark内存设置如2g', 'Drive memory size, default value: 2', 'blur'); -insert into `dss_workflow_node_ui_validate` (`id`, `validate_type`, `validate_range`, `error_msg`, `error_msg_en`, `trigger`) values('61','Regex','^(.|\s){1,500}$','长度在1到5000个字符','The length is between 1 and 5000 characters','blur'); +insert into `dss_workflow_node_ui_validate` (`id`, `validate_type`, `validate_range`, `error_msg`, `error_msg_en`, `trigger`) values('61','Regex','^(.|\s){1,5000}$','长度在1到5000个字符','The length is between 1 and 5000 characters','blur'); insert into `dss_workflow_node_ui_validate` (`id`, `validate_type`, `validate_range`, `error_msg`, `error_msg_en`, `trigger`) values('62','Regex','^.{1,150}$','长度在1到150个字符','The length is between 1 and 150 characters','blur'); DELETE FROM dss_workflow_node_ui_to_validate; @@ -389,7 +399,34 @@ insert into `dss_workflow_node_ui_to_validate`(`ui_id`,`validate_id`) values (@ insert into `dss_workflow_node_ui_to_validate`(`ui_id`,`validate_id`) values (@node_ui_spark_driver_memory,60); insert into `dss_workflow_node_ui_to_validate`(`ui_id`,`validate_id`) values (@node_ui_spark_executor_memory,60); insert into `dss_workflow_node_ui_to_validate`(`ui_id`,`validate_id`) values (@node_ui_job_desc,61); - +insert into dss_workflow_node_ui_to_validate(`ui_id`,`validate_id`) values(@sparkConfUiId, 41); +-- 去除节点描述的中文限制 +delete nutv from dss_workflow_node_ui_to_validate nutv , dss_workflow_node_ui nu , dss_workflow_node_ui_validate nuv + where nu.id = nutv.ui_id + AND nutv.validate_id = nuv.id and nu.lable_name ='节点描述' and nuv.error_msg='此值不能输入中文'; +-- 调整执行器内存大小限制为1-28 +update + dss_workflow_node_ui_to_validate nutv , dss_workflow_node_ui nu , dss_workflow_node_ui_validate nuv +set + nuv.validate_type='Regex', + nuv.validate_range='^([1-9]|1[0-9]|2[0-8])(g|G){0,1}$', + nuv.error_msg='设置范围为[1,28],设置超出限制', + nuv.error_msg_en='must be between 1 and 28', + nuv.trigger='blur' +where nu.id = nutv.ui_id + AND nutv.validate_id = nuv.id AND nu.key='spark.executor.memory' ; + +-- fix 驱动器内存大小设置不能带g +update + dss_workflow_node_ui_to_validate nutv , dss_workflow_node_ui nu , dss_workflow_node_ui_validate nuv +set + nuv.validate_type='Regex', + nuv.validate_range='^([1-9]|1[0-5])(g|G){0,1}$', + nuv.error_msg='设置范围为[1,15],设置超出限制', + nuv.error_msg_en='must be between 1 and 15', + nuv.trigger='blur' +where nu.id = nutv.ui_id + AND nutv.validate_id = nuv.id AND nu.key='spark.driver.memory' ; DELETE FROM dss_workspace_appconn_role; INSERT INTO `dss_workspace_appconn_role` (`workspace_id`, `appconn_id`, `role_id`, `priv`, `update_time`, `updateby`) VALUES('-1',@scriptis_appconn_id,'1','1',now(),'system'); @@ -412,5 +449,4 @@ INSERT INTO `dss_workspace_appconn_role` (`workspace_id`, `appconn_id`, `role_id INSERT INTO `dss_workspace_appconn_role` (`workspace_id`, `appconn_id`, `role_id`, `priv`, `update_time`, `updateby`) VALUES('224',@workflow_appconn_id,'1','1',now(),'system'); INSERT INTO `dss_workspace_appconn_role` (`workspace_id`, `appconn_id`, `role_id`, `priv`, `update_time`, `updateby`) VALUES('224',@apiservice_appconn_id,'1','1',now(),'system'); - INSERT INTO `dss_workspace_admin_dept` (`id`, `parent_id`, `ancestors`, `dept_name`, `order_num`, `leader`, `phone`, `email`, `status`, `del_flag`, `create_by`, `create_time`, `update_by`, `update_time`) VALUES('100','0','0','基础科技','0','leader01','1888888888','123@qq.com','0','0','admin',now(),'admin',now()); diff --git a/db/version_update/from_v101_to_v110.sql b/db/version_update/from_v101_to_v110.sql index 83fa849762..5797c91265 100644 --- a/db/version_update/from_v101_to_v110.sql +++ b/db/version_update/from_v101_to_v110.sql @@ -20,10 +20,13 @@ INSERT INTO `dss_workflow_node_ui_validate` (`validate_type`, `validate_range`, CREATE TABLE `dss_orchestrator_job_info` ( `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT '主键', - `job_id` bigint(20) DEFAULT NULL COMMENT 'job ID', - `conversion_job_json` text, + `job_id` varchar(1024) DEFAULT NULL COMMENT 'job ID', + `conversion_job_json` varchar(1024) DEFAULT NULL COMMENT 'job信息', `create_time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间', - `updated_time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '修改时间', + `update_time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '修改时间', + `instance_name` varchar(128) DEFAULT NULL COMMENT '执行任务的实例', + `status` varchar(128) DEFAULT NULL COMMENT '转换任务状态', + `error_msg` varchar(2048) DEFAULT NULL COMMENT '转换任务异常信息', PRIMARY KEY (`id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='dss_orchestrator_job_info表'; diff --git a/db/version_update/from_v111_to_v112.sql b/db/version_update/from_v111_to_v112.sql new file mode 100644 index 0000000000..e50dd70608 --- /dev/null +++ b/db/version_update/from_v111_to_v112.sql @@ -0,0 +1,282 @@ +CREATE TABLE IF NOT EXISTS `dss_orchestrator_copy_info` ( + `id` VARCHAR(128) NOT NULL COMMENT '主键', + `username` VARCHAR(128) DEFAULT NULL COMMENT '用户名', + `type` VARCHAR(128) DEFAULT NULL COMMENT '编排类别', + `source_orchestrator_id` INT(20) DEFAULT NULL COMMENT '源编排ID', + `source_orchestrator_name` VARCHAR(255) DEFAULT NULL COMMENT '源编排名', + `target_orchestrator_name` VARCHAR(255) DEFAULT NULL COMMENT '目标编排名', + `source_project_name` VARCHAR(255) DEFAULT NULL COMMENT '源工程名', + `target_project_name` VARCHAR(255) DEFAULT NULL COMMENT '目标工程名', + `workspace_id` INT(20) DEFAULT NULL COMMENT '工作空间ID', + `workflow_node_suffix` VARCHAR(255) DEFAULT NULL COMMENT '目标工作流节点后缀', + `microserver_name` VARCHAR(128) COMMENT '微服务名', + `exception_info` VARCHAR(128) COMMENT '异常信息', + `status` int(1) DEFAULT 0 COMMENT '复制任务最终状态', + `instance_name` varchar(128) DEFAULT NULL COMMENT '执行任务的实例', + `is_copying` int(1) DEFAULT 0 COMMENT '编排是否在被复制', + `success_node` TEXT COMMENT '复制成功节点', + `start_time` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '复制开始时间', + `end_time` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '复制结束时间', + PRIMARY KEY (`id`), + INDEX index_soi(source_orchestrator_id) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 ROW_FORMAT=COMPACT COMMENT='编排复制信息表'; + +CREATE TABLE IF NOT EXISTS `dss_project_operate_record` +( + `id` bigint(20) NOT NULL AUTO_INCREMENT, + `record_id` varchar(64) NOT NULL, + `workspace_id` bigint(20) NOT NULL COMMENT '空间id', + `project_id` bigint(20) NOT NULL COMMENT '项目id', + `operate_type` int(11) NOT NULL COMMENT '操作类型', + `status` int(11) NOT NULL COMMENT '操作状态', + `instance_name` VARCHAR(128) DEFAULT NULL COMMENT '执行任务的实例', + `content` longtext DEFAULT NULL COMMENT '操作内容详情', + `result_resource_uri` text DEFAULT NULL COMMENT '操作结果资源的uri,是一个json', + `creator` varchar(100) NOT NULL, + `create_time` datetime NOT NULL, + PRIMARY KEY (`id`), + KEY `idx_record_id` (`record_id`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='项目操作记录表'; + +CREATE TABLE IF NOT EXISTS `dss_release_task` ( + `id` bigint(20) NOT NULL AUTO_INCREMENT, + `project_id` bigint(20) NOT NULL, + `orchestrator_version_id` bigint(20) NOT NULL, + `orchestrator_id` bigint(20) NOT NULL, + `release_user` varchar(128) NOT NULL, + `create_time` datetime DEFAULT NULL, + `update_time` datetime DEFAULT NULL, + `status` varchar(64) DEFAULT 'init', + `instance_name` varchar(128) DEFAULT NULL COMMENT '执行任务的实例', + `error_msg` varchar(500) DEFAULT NULL COMMENT '发布错误信息', + `comment` varchar(500) DEFAULT NULL COMMENT '发布描述', + `log_msg` varchar(255) DEFAULT NULL COMMENT '日志信息或日志路径', + `bak` varchar(255) DEFAULT NULL COMMENT '备用字段', + PRIMARY KEY (`id`) USING BTREE +) ENGINE=InnoDB AUTO_INCREMENT=605 DEFAULT CHARSET=utf8mb4 ROW_FORMAT=COMPACT; + +CREATE TABLE IF NOT EXISTS `dss_orchestrator_job_info` ( + `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT '主键', + `job_id` varchar(64) DEFAULT NULL COMMENT 'job ID', + `conversion_job_json` varchar(1024) DEFAULT NULL COMMENT 'job信息', + `create_time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间', + `update_time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '修改时间', + `instance_name` varchar(128) DEFAULT NULL COMMENT '执行任务的实例', + `status` varchar(128) DEFAULT NULL COMMENT '转换任务状态', + `error_msg` varchar(2048) DEFAULT NULL COMMENT '转换任务异常信息', + PRIMARY KEY (`id`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='dss_orchestrator_job_info表'; + +CREATE TABLE IF NOT EXISTS `dss_project_copy_task` ( + `id` BIGINT(20) NOT NULL AUTO_INCREMENT COMMENT '主键', + `workspace_id` BIGINT(20) COMMENT '空间ID', + `source_project_id` BIGINT(20) COMMENT '(源)复制工程ID', + `source_project_name` VARCHAR(200) COMMENT '(源)复制工程名称', + `copy_project_id` BIGINT(20) COMMENT '复制工程ID', + `copy_project_name` VARCHAR(200) COMMENT '复制工程名称', + `surplus_count` INT(3) COMMENT '剩余复制数量', + `sum_count` INT(3) COMMENT '总数', + `status` INT(1) COMMENT '状态 0:初始化,1:复制中,2:复制成功', + `instance_name` VARCHAR(128) DEFAULT NULL COMMENT '执行任务的实例', + `create_by` VARCHAR(200) COMMENT '创建人', + `create_time` datetime COMMENT '创建时间', + `update_time` datetime COMMENT '上个复制时间', + `error_msg` text COMMENT '失败原因', + `error_orc` VARCHAR(2048) DEFAULT '' COMMENT '拷贝异常编排', + PRIMARY KEY (`id`) +) ENGINE=INNODB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8 COLLATE=utf8_bin COMMENT='复制工程任务表'; + +-- 首页公告表 +CREATE TABLE IF NOT EXISTS `dss_notice` +( + `id` int(11) NOT NULL AUTO_INCREMENT COMMENT '主键ID', + `content` text DEFAULT NULL COMMENT '公告内容', + `start_time` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '生效时间', + `end_time` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '失效时间', + PRIMARY KEY (`id`) +) ENGINE = InnoDB + DEFAULT CHARSET = utf8mb4 COLLATE=utf8mb4_bin COMMENT ='首页公告内容'; + +INSERT INTO dss_workflow_node_ui +( `key`, description, description_en, lable_name, lable_name_en, ui_type, required, value, default_value, is_hidden, `condition`, is_advanced, `order`, node_menu_type, is_base_info, `position`) +VALUES('spark.conf', 'spark自定义参数配置输入,例如spark.sql.shuffle.partitions=10。多个参数使用分号分隔。', 'input spark params config, eg: spark.sql.shuffle.partitions=10. Use semi-colon to split multi-params', 'spark.conf', 'spark.conf', 'Text', 0, NULL, "", 0, NULL, 0, 1, 1, 0, 'startup'); +select @sparkConfUiId:=id from dss_workflow_node_ui where `key`="spark.conf"; +select @sqlNodeId:=id from dss_workflow_node where node_type="linkis.spark.sql"; +select @pysparkNodeId:=id from dss_workflow_node where node_type="linkis.spark.py"; +select @scalaNodeId:=id from dss_workflow_node where node_type="linkis.spark.scala"; +insert into dss_workflow_node_to_ui(`workflow_node_id`,`ui_id`) values(@sqlNodeId, @sparkConfUiId); +insert into dss_workflow_node_to_ui(`workflow_node_id`,`ui_id`) values(@pysparkNodeId, @sparkConfUiId); +insert into dss_workflow_node_to_ui(`workflow_node_id`,`ui_id`) values(@scalaNodeId, @sparkConfUiId); +select @len500valId:=id from dss_workflow_node_ui_validate where error_msg="长度在1到500个字符"; +insert into dss_workflow_node_ui_to_validate(`ui_id`,`validate_id`) values(@sparkConfUiId, @len500valId); + +-- 去除节点描述的中文限制 +delete nutv from dss_workflow_node_ui_to_validate nutv , dss_workflow_node_ui nu , dss_workflow_node_ui_validate nuv + where nu.id = nutv.ui_id + AND nutv.validate_id = nuv.id and nu.lable_name ='节点描述' and nuv.error_msg='此值不能输入中文'; + +-- 调整执行器内存大小限制为1-28 +update + dss_workflow_node_ui_to_validate nutv , dss_workflow_node_ui nu , dss_workflow_node_ui_validate nuv +set + nuv.validate_type='Regex', + nuv.validate_range='^([1-9]|1[0-9]|2[0-8])(g|G){0,1}$', + nuv.error_msg='设置范围为[1,28],设置超出限制', + nuv.error_msg_en='must be between 1 and 28', + nuv.trigger='blur' +where nu.id = nutv.ui_id + AND nutv.validate_id = nuv.id AND nu.key='spark.executor.memory' ; +-- fix 驱动器内存大小设置不能带g +update + dss_workflow_node_ui_to_validate nutv , dss_workflow_node_ui nu , dss_workflow_node_ui_validate nuv +set + nuv.validate_type='Regex', + nuv.validate_range='^([1-9]|1[0-5])(g|G){0,1}$', + nuv.error_msg='设置范围为[1,15],设置超出限制', + nuv.error_msg_en='must be between 1 and 15', + nuv.trigger='blur' +where nu.id = nutv.ui_id + AND nutv.validate_id = nuv.id AND nu.key='spark.driver.memory' ; + +ALTER TABLE dss_workflow ADD metrics varchar(1024) NULL; + +ALTER TABLE dss_workspace ADD COLUMN (`admin_permission` tinyint(1) DEFAULT 1 NOT NULL COMMENT '工作空间管理员是否有权限查看该空间下所有项目,1可以,0不可以'); +-- appconn表新增微应用标记字段 +ALTER TABLE `dss_appconn` ADD `is_micro_app` tinyint(1) NOT NULL DEFAULT '0' COMMENT '是否微应用嵌入'; +-- enhance_json字段大小从1024增到到2048 +ALTER TABLE dss_appconn_instance change `enhance_json` `enhance_json` varchar(2048) DEFAULT NULL COMMENT 'json格式的配置'; + +-- 表dss_workflow_task添加instance_name字段 +ALTER TABLE `dss_workflow_task` ADD `instance_name` varchar(128) DEFAULT NULL COMMENT '执行任务的实例' AFTER `status`; +-- 表dss_workspace_user_role添加update_user和update_time字段 +ALTER TABLE `dss_workspace_user_role` add `update_user` varchar(32) DEFAULT NULL COMMENT '更新人'; +ALTER TABLE `dss_workspace_user_role` add `update_time` datetime DEFAULT NULL COMMENT '更新时间'; + +-- webank内部表,待开源相关功能使用 +CREATE TABLE IF NOT EXISTS `dss_streamis_proxy_user` +( + `id` int(11) NOT NULL AUTO_INCREMENT, + `user_name` varchar(64) DEFAULT NULL COMMENT '实名用户名', + `proxy_user_name` varchar(64) DEFAULT NULL COMMENT '代理用户名', + `create_by` varchar(64) DEFAULT NULL COMMENT '创建者', + `create_time` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间', + PRIMARY KEY (`id`) +) ENGINE = InnoDB DEFAULT CHARSET = utf8mb4 COMMENT='流式应用代理用户表'; + +-- webank内部表,待开源相关功能使用 +CREATE TABLE IF NOT EXISTS `dss_workspace_associate_departments` ( + `id` bigint(20) NOT NULL AUTO_INCREMENT, + `workspace_id` bigint(20) DEFAULT NULL, + `departments` text DEFAULT NULL COMMENT '关联的部门-科室列表,逗号分割,若部门后不接科室则代表关联整个部门', + `role_ids` varchar(128) DEFAULT NULL COMMENT '角色id列表,逗号分割', + `create_time` datetime DEFAULT NULL, + `update_time` datetime DEFAULT NULL, + `create_by` varchar(128) DEFAULT NULL, + `update_by` varchar(128) DEFAULT NULL, + PRIMARY KEY (`id`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='空间自动加入用户绑定的部门科室信息'; + +-- 版本发布时的releaseNote信息。webank内部表,待开源相关功能使用 +CREATE TABLE IF NOT EXISTS `dss_release_note_content` +( + `id` int(11) NOT NULL AUTO_INCREMENT COMMENT '主键ID', + `name` varchar(200) DEFAULT NULL COMMENT '名称', + `title` varchar(200) DEFAULT NULL COMMENT '标题', + `url` varchar(300) DEFAULT NULL COMMENT 'url', + `url_type` int(1) DEFAULT '1' COMMENT 'url类型: 0-内部系统,1-外部系统;默认是外部', + `release_type` int(1) DEFAULT '0' COMMENT '发布形式: 0-作为dss整体发布,1-单独发布scriptis;默认是dss', + `create_time` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间', + PRIMARY KEY (`id`) +) ENGINE = InnoDB DEFAULT CHARSET = utf8mb4 COMMENT ='releaseNote表'; + +-- 用户访问行为统计表,初期只有登录行为统计。webank内部表,待开源相关功能使用 +CREATE TABLE IF NOT EXISTS `dss_user_access_audit` +( + `id` int(11) NOT NULL AUTO_INCREMENT COMMENT '主键ID', + `user_name` varchar(64) DEFAULT NULL COMMENT '用户名', + `login_count` BIGINT DEFAULT 0 COMMENT '登录次数', + `first_login` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '第一次登录时间', + `last_login` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '上一次登录时间', + PRIMARY KEY (`id`), + UNIQUE KEY `idx_user_name` (`user_name`) +) ENGINE = InnoDB DEFAULT CHARSET = utf8mb4 COMMENT ='用户访问行为次数统计'; + +-- webank内部表,待开源相关功能使用 +CREATE TABLE IF NOT EXISTS `dss_ec_release_strategy` +( + `id` int(11) NOT NULL AUTO_INCREMENT COMMENT '主键', + `strategy_id` varchar(64) NOT NULL UNIQUE COMMENT '规则id', + `workspace_id` bigint(20) NOT NULL COMMENT '工作空间id', + `name` varchar(64) NOT NULL COMMENT '规则名', + `description` varchar(128) DEFAULT NULL COMMENT '规则描述', + `queue` varchar(128) NOT NULL UNIQUE COMMENT '关联队列', + `trigger_condition_conf` varchar(1024) NOT NULL COMMENT '触发条件(json)', + `terminate_condition_conf` varchar(1024) NOT NULL COMMENT '终止条件(json)', + `ims_conf` varchar(2048) NOT NULL COMMENT '告警设置(json)', + `status` int(1) DEFAULT 0 COMMENT '规则状态:0禁用 1开启', + `creator` varchar(64) NOT NULL COMMENT '创建人', + `create_time` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间', + `modifier` varchar(64) NOT NULL COMMENT '修改人', + `modify_time` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '修改时间', + `execute_instance` varchar(128) DEFAULT NULL COMMENT '最近处理该规则的服务实例', + `execute_time` datetime DEFAULT NULL COMMENT '最近处理该规则的时间起点', + PRIMARY KEY (`id`), + KEY `idx_strategy_id` (`strategy_id`), + KEY `idx_workspace_id` (`workspace_id`) +) ENGINE = InnoDB + DEFAULT CHARSET = utf8mb4 COLLATE=utf8mb4_bin COMMENT ='EC自动释放规则配置'; + +-- 工作空间关联的队列表。webank内部表,待开源相关功能使用 +CREATE TABLE IF NOT EXISTS `dss_queue_in_workspace` +( + `id` int(11) NOT NULL AUTO_INCREMENT COMMENT '主键', + `workspace_id` bigint(20) NOT NULL COMMENT '工作空间id', + `queue` varchar(128) NOT NULL UNIQUE COMMENT '队列名', + `apply_user` varchar(64) NOT NULL COMMENT '申请人', + `approve_id` varchar(64) NOT NULL COMMENT '申请单号', + `create_time` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间', + PRIMARY KEY (`id`) +) ENGINE = InnoDB + DEFAULT CHARSET = utf8mb4 COLLATE=utf8mb4_bin COMMENT ='工作空间关联的队列'; + +-- EC释放通知发送记录表。webank内部表,待开源相关功能使用 +CREATE TABLE IF NOT EXISTS `dss_ec_release_ims_record` +( + `id` bigint(11) NOT NULL AUTO_INCREMENT COMMENT '主键', + `record_id` varchar(64) NOT NULL COMMENT '发送记录id', + `strategy_id` varchar(64) NOT NULL COMMENT '释放规则id', + `workspace_id` bigint(20) NOT NULL COMMENT '工作空间id', + `content` varchar(1024) NOT NULL COMMENT '发送内容', + `status` int(1) DEFAULT 0 COMMENT '通知状态:0未发送 1已发送 2发送失败', + `execute_instance` varchar(128) NOT NULL COMMENT '负责发送的服务实例', + `create_time` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间', + `modify_time` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '修改时间', + PRIMARY KEY (`id`), + KEY `idx_record_id` (`record_id`) +) ENGINE = InnoDB + DEFAULT CHARSET = utf8mb4 COLLATE=utf8mb4_bin COMMENT ='EC释放通知发送记录'; + +-- 请求释放EC历史。webank内部表,待开源相关功能使用 +CREATE TABLE IF NOT EXISTS `dss_ec_kill_history` +( + `id` bigint(11) NOT NULL AUTO_INCREMENT COMMENT '主键', + `strategy_id` varchar(64) NOT NULL COMMENT '释放规则id', + `workspace_id` bigint(20) NOT NULL COMMENT '工作空间id', + `instance` varchar(128) NOT NULL COMMENT '释放的EC实例名', + `engine_type` varchar(64) NOT NULL COMMENT 'EC类型', + `queue` varchar(128) NOT NULL COMMENT '队列名', + `driver_core` int(11) DEFAULT 0 COMMENT '本地释放核数', + `driver_memory` bigint(11) DEFAULT 0 COMMENT '本地释放内存,单位Byte', + `yarn_core` int(11) DEFAULT 0 COMMENT 'yarn释放核数', + `yarn_memory` bigint(11) DEFAULT 0 COMMENT 'yarn释放内存,单位Byte', + `unlock_duration` bigint(11) DEFAULT 0 COMMENT 'EC空闲时长,单位秒', + `owner` varchar(64) NOT NULL COMMENT 'EC创建者', + `killer` varchar(64) NOT NULL COMMENT 'EC释放触发者', + `ec_start_time` varchar(64) NOT NULL COMMENT 'EC创建的时间', + `kill_time` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '请求killEC的时间', + `create_time` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间', + `execute_instance` varchar(128) DEFAULT NULL COMMENT '负责发送的服务实例', + PRIMARY KEY (`id`), + KEY `idx_workspace_id` (`workspace_id`) +) ENGINE = InnoDB + DEFAULT CHARSET = utf8mb4 COLLATE=utf8mb4_bin COMMENT ='请求释放EC历史'; diff --git a/dss-appconn/appconns/dss-datachecker-appconn/pom.xml b/dss-appconn/appconns/dss-datachecker-appconn/pom.xml index 817fcb72a5..b6f87230b1 100644 --- a/dss-appconn/appconns/dss-datachecker-appconn/pom.xml +++ b/dss-appconn/appconns/dss-datachecker-appconn/pom.xml @@ -21,7 +21,7 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 ../../../pom.xml 4.0.0 @@ -44,6 +44,26 @@ json4s-jackson_2.11 org.json4s + + com.webank.wedatasphere.dss + dss-origin-sso-integration-standard + + + + + + com.webank.wedatasphere.dss + dss-origin-sso-integration-standard + 1.1.2 + + + org.apache.linkis + linkis-httpclient + + + org.apache.linkis + linkis-gateway-httpclient-support + @@ -60,16 +80,16 @@ ${dss.version} - - log4j - log4j - 1.2.17 - - com.squareup.okhttp3 okhttp 4.2.2 + + + org.jetbrains + annotations + + diff --git a/dss-appconn/appconns/dss-datachecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/datachecker/DataChecker.java b/dss-appconn/appconns/dss-datachecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/datachecker/DataChecker.java index c1805528bf..9a99bc4700 100644 --- a/dss-appconn/appconns/dss-datachecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/datachecker/DataChecker.java +++ b/dss-appconn/appconns/dss-datachecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/datachecker/DataChecker.java @@ -28,12 +28,26 @@ public class DataChecker { public final static String SOURCE_TYPE = "source.type"; public final static String DATA_OBJECT = "check.object"; + /** + * 检查对象在节点中的序号 + */ + public final static String DATA_OBJECT_NUM = "check.sn.object.num"; public final static String WAIT_TIME = "max.check.hours"; public final static String QUERY_FREQUENCY = "query.frequency"; public final static String TIME_SCAPE = "time.scape"; public final static String MASK_URL = "bdp.mask.url"; public final static String MASK_APP_ID = "bdp.mask.app.id"; public final static String MASK_APP_TOKEN = "bdp.mask.app.token"; + public final static String CONTEXTID_USER = "contextId.user"; + public final static String CONTEXTID_PROJECT_NAME = "contextId.projectName"; + public final static String CONTEXTID_FLOW_NAME = "contextId.flowName"; + public final static String NAME_NAME = "nodeName"; + + public final static String QUALITIS_CHECK = "qualitis.check"; + public final static String QUALITIS_SWITCH = "job.eventchecker.qualitis.switch"; + public final static String QUALITIS_CHECK_DEFAULT = "qualitis.check.default"; + + private Properties p; private static final Logger logger = LoggerFactory.getLogger(DataChecker.class);; @@ -67,23 +81,23 @@ public void run() { begineCheck(dataCheckerAction); }catch (Exception ex){ dataCheckerAction.setState(RefExecutionState.Failed); - throw new RuntimeException("get DataChecker result failed", ex); + throw ex; } } - public void begineCheck(RefExecutionAction action){ + public void begineCheck(DataCheckerExecutionAction action){ boolean success=false; try { success= wbDao.validateTableStatusFunction(p, logger,action); }catch (Exception ex){ dataCheckerAction.setState(RefExecutionState.Failed); logger.error("datacheck error",ex); - throw new RuntimeException("get DataChecker result failed", ex); + throw ex; } if(success) { dataCheckerAction.setState(RefExecutionState.Success); - }else { + } else if (dataCheckerAction.getState() != RefExecutionState.Failed) { dataCheckerAction.setState(RefExecutionState.Running); } } diff --git a/dss-appconn/appconns/dss-datachecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/datachecker/common/CheckDataObject.java b/dss-appconn/appconns/dss-datachecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/datachecker/common/CheckDataObject.java new file mode 100644 index 0000000000..ab4a37559a --- /dev/null +++ b/dss-appconn/appconns/dss-datachecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/datachecker/common/CheckDataObject.java @@ -0,0 +1,85 @@ +package com.webank.wedatasphere.dss.appconn.datachecker.common; + +import org.apache.commons.lang3.StringUtils; + +/** + * 检查的数据对象 + * Author: xlinliu + * Date: 2023/5/11 + */ +public class CheckDataObject { + /** + * 检查对象的类型 + */ + private Type type; + private String dbName; + private String tableName; + /** + * 分区明,只有当type等于PARTITION才有值 + */ + private String partitionName; + + public CheckDataObject(String dbName, String tableName, String partitionName) { + this.dbName = dbName; + this.tableName = tableName; + this.partitionName = partitionName; + type= StringUtils.isNotEmpty(partitionName)?Type.PARTITION:Type.TABLE; + } + + public CheckDataObject(String dbName, String tableName) { + this.dbName = dbName; + this.tableName = tableName; + type = Type.TABLE; + } + + public Type getType() { + return type; + } + + public void setType(Type type) { + this.type = type; + } + + public String getDbName() { + return dbName; + } + + public void setDbName(String dbName) { + this.dbName = dbName; + } + + public String getTableName() { + return tableName; + } + + public void setTableName(String tableName) { + this.tableName = tableName; + } + + public String getPartitionName() { + return partitionName; + } + + public void setPartitionName(String partitionName) { + this.partitionName = partitionName; + } + + public enum Type{ + /** + * 分区对象 + */ + PARTITION, + /** + * 表对象 + */ + TABLE, + + } + + @Override + public String toString() { + final StringBuffer sb = new StringBuffer(); + sb.append(dbName).append(tableName).append(partitionName); + return sb.toString(); + } +} diff --git a/dss-appconn/appconns/dss-datachecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/datachecker/connector/DataCheckerDao.java b/dss-appconn/appconns/dss-datachecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/datachecker/connector/DataCheckerDao.java index 8cd9a35b7c..1240be7307 100644 --- a/dss-appconn/appconns/dss-datachecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/datachecker/connector/DataCheckerDao.java +++ b/dss-appconn/appconns/dss-datachecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/datachecker/connector/DataCheckerDao.java @@ -19,26 +19,26 @@ import com.alibaba.druid.pool.DruidDataSource; import com.webank.wedatasphere.dss.appconn.datachecker.DataChecker; +import com.webank.wedatasphere.dss.appconn.datachecker.DataCheckerExecutionAction; +import com.webank.wedatasphere.dss.appconn.datachecker.common.CheckDataObject; import com.webank.wedatasphere.dss.appconn.datachecker.common.MaskCheckNotExistException; import com.webank.wedatasphere.dss.appconn.datachecker.utils.HttpUtils; +import com.webank.wedatasphere.dss.appconn.datachecker.utils.QualitisUtil; import com.webank.wedatasphere.dss.standard.app.development.listener.common.RefExecutionAction; +import com.webank.wedatasphere.dss.standard.app.development.listener.common.RefExecutionState; import okhttp3.FormBody; import okhttp3.RequestBody; import okhttp3.Response; import okhttp3.ResponseBody; +import org.apache.commons.codec.digest.DigestUtils; import org.apache.commons.lang.StringUtils; import org.slf4j.Logger; import javax.sql.DataSource; import java.io.IOException; -import java.sql.Connection; -import java.sql.PreparedStatement; -import java.sql.ResultSet; -import java.sql.SQLException; -import java.util.HashMap; -import java.util.List; -import java.util.Map; -import java.util.Properties; +import java.math.BigDecimal; +import java.sql.*; +import java.util.*; import java.util.function.Predicate; import java.util.function.Supplier; import java.util.regex.Matcher; @@ -59,12 +59,22 @@ public class DataCheckerDao { private static final String SQL_SOURCE_TYPE_BDP_WITH_TIME_CONDITION = "SELECT * FROM desktop_bdapimport WHERE bdap_db_name = ? AND bdap_table_name = ? AND target_partition_name = ? " + "AND (UNIX_TIMESTAMP() - UNIX_TIMESTAMP(STR_TO_DATE(modify_time, '%Y-%m-%d %H:%i:%s'))) <= ? AND status = '1';"; + + private static final String SQL_DOPS_CHECK_TABLE = + "SELECT * FROM dops_clean_task_list WHERE db_name = ? AND tb_name = ? AND part_name is null AND task_state NOT IN (10,13) order by order_id desc limit 1"; + private static final String SQL_DOPS_CHECK_PARTITION = + "SELECT * FROM dops_clean_task_list WHERE db_name = ? AND tb_name = ? AND part_name = ? AND task_state NOT IN (10,13) order by order_id desc limit 1"; + + private static final String SQL_DOPS_CHECK_ALL_PARTITION = + "SELECT * FROM dops_clean_task_list WHERE db_name = ? AND tb_name = ? AND part_name is not null AND task_state != 13 order by order_id desc limit 1"; private static final String HIVE_SOURCE_TYPE = "hivedb"; private static final String MASK_SOURCE_TYPE = "maskdb"; private static DataSource jobDS; private static DataSource bdpDS; - private static DataCheckerDao instance; + + private static DataSource dopsDS; + private static volatile DataCheckerDao instance; public static DataCheckerDao getInstance() { if (instance == null) { @@ -77,18 +87,26 @@ public static DataCheckerDao getInstance() { return instance; } - public boolean validateTableStatusFunction(Properties props, Logger log, RefExecutionAction action) { + public boolean validateTableStatusFunction(Properties props, Logger log, DataCheckerExecutionAction action) { if (jobDS == null) { jobDS = DataDruidFactory.getJobInstance(props, log); if (jobDS == null) { - log.error("Error getting Druid DataSource instance"); + log.error("Error getting bdp Druid DataSource instance"); return false; } } if (bdpDS == null) { bdpDS = DataDruidFactory.getBDPInstance(props, log); if (bdpDS == null) { - log.warn("Error getting Druid DataSource instance"); + log.warn("Error getting job Druid DataSource instance"); + return false; + } + } + boolean systemCheck = Boolean.valueOf(props.getProperty(DataChecker.QUALITIS_SWITCH)); + if (systemCheck && dopsDS == null) { + dopsDS = DataDruidFactory.getDopsInstance(props, log);//通过alibaba的druid数据库连接池获取JOB数据库连接 + if (dopsDS == null) { + log.error("Error getting Druid DataSource instance"); return false; } } @@ -112,14 +130,15 @@ public boolean validateTableStatusFunction(Properties props, Logger log, RefExec dataObjectList.forEach(checkObject -> { log.info(checkObject.keySet().toString()); }); - + QualitisUtil qualitisUtil = new QualitisUtil(props); try (Connection jobConn = jobDS.getConnection(); - Connection bdpConn = bdpDS.getConnection()) { + Connection bdpConn = bdpDS.getConnection(); + Connection dopsConn = dopsDS != null ? dopsDS.getConnection() : null) { List allCheckRes = dataObjectList - .stream() + .parallelStream() .map(proObjectMap -> { log.info("Begin to Check dataObject:" + proObjectMap.entrySet().toString()); - boolean checkRes = getDataCheckResult(proObjectMap, jobConn, bdpConn, props, log); + boolean checkRes = getDataCheckResult(proObjectMap, jobConn, bdpConn, dopsConn, props, log,action,qualitisUtil); if (null != action.getExecutionRequestRefContext()) { if (checkRes) { action.getExecutionRequestRefContext().appendLog("Database table partition info : " + proObjectMap.get(DataChecker.DATA_OBJECT) + " has arrived"); @@ -133,9 +152,9 @@ public boolean validateTableStatusFunction(Properties props, Logger log, RefExec }).collect(Collectors.toList()); boolean flag = allCheckRes.stream().allMatch(res -> res.equals(true)); if (flag) { - log.info("=============================Data Check End=========================================="); + log.info("=============================Data Check End,check result:true=========================================="); if (null != action.getExecutionRequestRefContext()) { - action.getExecutionRequestRefContext().appendLog("=============================Data Check End=========================================="); + action.getExecutionRequestRefContext().appendLog("=============================Data Check End,check result:true=========================================="); } return true; } @@ -144,14 +163,36 @@ public boolean validateTableStatusFunction(Properties props, Logger log, RefExec throw new RuntimeException("get DataChecker result failed", e); } - log.info("=============================Data Check End=========================================="); + log.info("=============================Data Check End,check result:false=========================================="); if (null != action.getExecutionRequestRefContext()) { - action.getExecutionRequestRefContext().appendLog("=============================Data Check End=========================================="); + action.getExecutionRequestRefContext().appendLog("=============================Data Check End,,check result:false=========================================="); } return false; } - private boolean getDataCheckResult(Map proObjectMap, Connection jobConn, Connection bdpConn, Properties props, Logger log) { + + + + private boolean getDataCheckResult(Map proObjectMap, + Connection jobConn, + Connection bdpConn, + Connection dopsConn, + Properties props, + Logger log, + DataCheckerExecutionAction action, + QualitisUtil qualitisUtil ) { + String dataObjectStr = proObjectMap.get(DataChecker.DATA_OBJECT) == null ? "" : proObjectMap.get(DataChecker.DATA_OBJECT); + if (StringUtils.isNotBlank(dataObjectStr)) { + dataObjectStr = dataObjectStr.replace(" ", "").trim(); + } + String objectNum = proObjectMap.get(DataChecker.DATA_OBJECT_NUM); + CheckDataObject dataObject; + try { + dataObject = parseDataObject(dataObjectStr); + } catch (SQLException e) { + log.error("parse dataObject failed", e); + return false; + } Predicate> hasDataSource = p -> { if (StringUtils.isEmpty(proObjectMap.get(DataChecker.SOURCE_TYPE))) { return false; @@ -163,36 +204,52 @@ private boolean getDataCheckResult(Map proObjectMap, Connection Supplier sourceType = () -> proObjectMap.get(DataChecker.SOURCE_TYPE).toLowerCase(); Predicate> isJobDataSource = p -> sourceType.get().equals("hivedb") || sourceType.get().equals("job"); Predicate> isBdpDataSource = p -> sourceType.get().equals("maskdb") || sourceType.get().equals("bdp"); - Predicate> isOdsDB = p -> { - String dataObject = proObjectMap.get(DataChecker.DATA_OBJECT) - .replace(" ", "").trim(); - String dbName = dataObject.split("\\.")[0]; - return dbName.contains("_ods"); - }; + Predicate> isOdsDB = p -> dataObject.getDbName().contains("_ods"); Predicate> isNotOdsDB = isOdsDB.negate(); Predicate> isCheckMetadata = (hasDataSource.and(isJobDataSource)).or(hasNotDataSource.and(isNotOdsDB)); Predicate> isCheckMask = (hasDataSource.and(isBdpDataSource)).or(hasNotDataSource.and(isOdsDB)); + boolean normalCheck; if (isCheckMetadata.test(proObjectMap)) { + if (null != action.getExecutionRequestRefContext()){ + action.getExecutionRequestRefContext().appendLog(dataObjectStr+" start to check hive meta"); + } + log.info("start to check hive meta"); proObjectMap.put(DataChecker.SOURCE_TYPE, HIVE_SOURCE_TYPE); - return getJobTotalCount(proObjectMap, jobConn, log) > 0; + normalCheck= getJobTotalCount(dataObject, jobConn, log) > 0; + if (null != action.getExecutionRequestRefContext()){ + action.getExecutionRequestRefContext().appendLog(dataObjectStr+" check hive meta end,check result:"+normalCheck); + } + log.info("check hive meta end,check result:"+normalCheck); + } else { if (isCheckMask.test(proObjectMap)) { + if (null != action.getExecutionRequestRefContext()){ + action.getExecutionRequestRefContext().appendLog(dataObjectStr+" start to check maskis"); + } + log.info("start to check maskis"); proObjectMap.put(DataChecker.SOURCE_TYPE, MASK_SOURCE_TYPE); - return (getBdpTotalCount(proObjectMap, bdpConn, log, props) > 0 || "success".equals(fetchMaskCode(proObjectMap, log, props).get("maskStatus"))); + normalCheck= (getBdpTotalCount(dataObject, bdpConn, log, props) > 0 || "success".equals(fetchMaskCode(dataObject, log, props).get("maskStatus"))); + if (null != action.getExecutionRequestRefContext()){ + action.getExecutionRequestRefContext().appendLog(dataObjectStr+" check maskis end,check result:"+normalCheck); + } + log.info("check maskis end,check result:"+normalCheck); + }else { + normalCheck = false; } + } + if(!normalCheck){ return false; } - } - - private void sleep(long sleepTime) { - try { - Thread.sleep(sleepTime); - } catch (InterruptedException e) { - e.printStackTrace(); + boolean qualitisCheck = checkQualitisData(objectNum, dataObject, log, action, props, dopsConn, qualitisUtil, dataObjectStr ); + if(!qualitisCheck){ + //如果是qualitis校验失败,则直接终止任务 + throw new RuntimeException(dataObjectStr+ " does not pass qualitis check(qualitis校验未通过)"); } + return qualitisCheck; } + private void removeBlankSpace(Properties props) { try { props.entrySet().forEach(entry -> { @@ -223,6 +280,7 @@ private Map key2Map(Object key, Properties p) { proMap.put(DataChecker.SOURCE_TYPE, String.valueOf(p.get(stKey))); } proMap.put(DataChecker.DATA_OBJECT, String.valueOf(p.get(doKey))); + proMap.put(DataChecker.DATA_OBJECT_NUM, keyNum); } else { String stKey = DataChecker.SOURCE_TYPE; String doKey = DataChecker.DATA_OBJECT; @@ -230,55 +288,35 @@ private Map key2Map(Object key, Properties p) { proMap.put(DataChecker.SOURCE_TYPE, String.valueOf(p.get(stKey))); } proMap.put(DataChecker.DATA_OBJECT, String.valueOf(p.get(doKey))); + proMap.put(DataChecker.DATA_OBJECT_NUM, "0"); } } return proMap; } - private PreparedStatement getJobStatement(Connection conn, String dataObject) throws SQLException { - String dataScape = dataObject.contains("{") ? "Partition" : "Table"; - String[] dataObjectArray = dataObject.split("\\."); - String dbName = dataObject.split("\\.")[0]; - String tableName = dataObject.split("\\.")[1]; - if (dataScape.equals("Partition")) { - Pattern pattern = Pattern.compile("\\{([^\\}]+)\\}"); - Matcher matcher = pattern.matcher(dataObject); - String partitionName = null; - if (matcher.find()) { - partitionName = matcher.group(1); - } - partitionName = partitionName.replace("\'", "").replace("\"", ""); - tableName = tableName.split("\\{")[0]; + /** + * 构造查询hive元数据库的查询 + */ + private PreparedStatement getJobStatement(Connection conn, CheckDataObject dataObject) throws SQLException { + if (CheckDataObject.Type.PARTITION==dataObject.getType()) { PreparedStatement pstmt = conn.prepareCall(SQL_SOURCE_TYPE_JOB_PARTITION); - pstmt.setString(1, dbName); - pstmt.setString(2, tableName); - pstmt.setString(3, partitionName); + pstmt.setString(1, dataObject.getDbName()); + pstmt.setString(2, dataObject.getTableName()); + pstmt.setString(3, dataObject.getPartitionName()); return pstmt; - } else if (dataObjectArray.length == 2) { + } else { PreparedStatement pstmt = conn.prepareCall(SQL_SOURCE_TYPE_JOB_TABLE); - pstmt.setString(1, dbName); - pstmt.setString(2, tableName); + pstmt.setString(1, dataObject.getDbName()); + pstmt.setString(2, dataObject.getTableName()); return pstmt; - } else { - throw new SQLException("Error for DataObject format!"); } } - private PreparedStatement getBdpStatement(Connection conn, String dataObject, String timeScape) throws SQLException { - String dataScape = dataObject.contains("{") ? "Partition" : "Table"; - String dbName = dataObject.split("\\.")[0]; - String tableName = dataObject.split("\\.")[1]; - String partitionName = ""; - Pattern pattern = Pattern.compile("\\{([^\\}]+)\\}"); - if (dataScape.equals("Partition")) { - Matcher matcher = pattern.matcher(dataObject); - if (matcher.find()) { - partitionName = matcher.group(1); - } - partitionName = partitionName.replace("\'", "").replace("\"", ""); - tableName = tableName.split("\\{")[0]; - } + /** + * 构造查询maskis的查询 + */ + private PreparedStatement getBdpStatement(Connection conn, CheckDataObject dataObject, String timeScape) throws SQLException { PreparedStatement pstmt = null; if (timeScape.equals("NULL")) { pstmt = conn.prepareCall(SQL_SOURCE_TYPE_BDP); @@ -286,66 +324,256 @@ private PreparedStatement getBdpStatement(Connection conn, String dataObject, St pstmt = conn.prepareCall(SQL_SOURCE_TYPE_BDP_WITH_TIME_CONDITION); pstmt.setInt(4, Integer.valueOf(timeScape) * 3600); } - pstmt.setString(1, dbName); - pstmt.setString(2, tableName); - pstmt.setString(3, partitionName); + if (dataObject.getPartitionName() == null) { + dataObject.setPartitionName(""); + } + pstmt.setString(1, dataObject.getDbName()); + pstmt.setString(2, dataObject.getTableName()); + pstmt.setString(3, dataObject.getPartitionName()); return pstmt; } - private long getJobTotalCount(Map proObjectMap, Connection conn, Logger log) { - String dataObject = proObjectMap.get(DataChecker.DATA_OBJECT); - if (dataObject != null) { - dataObject = dataObject.replace(" ", "").trim(); + /** + * 构造查询dops库的查询 + */ + private PreparedStatement getDopsStatement(Connection conn, CheckDataObject dataObject) throws SQLException { + if (CheckDataObject.Type.PARTITION==dataObject.getType()) { + PreparedStatement pstmt = conn.prepareCall(SQL_DOPS_CHECK_PARTITION); + pstmt.setString(1, dataObject.getDbName()); + pstmt.setString(2, dataObject.getTableName()); + pstmt.setString(3, dataObject.getPartitionName()); + return pstmt; + } else { + PreparedStatement pstmt = conn.prepareCall(SQL_DOPS_CHECK_TABLE); + pstmt.setString(1, dataObject.getDbName()); + pstmt.setString(2, dataObject.getTableName()); + return pstmt; + } + } + + /** + * 构造查询dops库的查询,分区表全表校验场景 + */ + private PreparedStatement getDopsStatementCheckAllPartition(Connection conn, CheckDataObject dataObject) throws SQLException { + PreparedStatement pstmt = conn.prepareCall(SQL_DOPS_CHECK_ALL_PARTITION); + pstmt.setString(1, dataObject.getDbName()); + pstmt.setString(2, dataObject.getTableName()); + return pstmt; + + } + + + /** + * 反序列化检查对象 + * @param dataObjectStr 字符串形式的对象 + * @return 发序列化后的对象 + */ + private CheckDataObject parseDataObject(String dataObjectStr)throws SQLException{ + CheckDataObject dataObject; + if(!dataObjectStr.contains(".")){ + throw new SQLException("Error for DataObject format!"+dataObjectStr); } + String dbName = dataObjectStr.split("\\.")[0]; + String tableName = dataObjectStr.split("\\.")[1]; + if (dataObjectStr.contains("{")) { + String partitionName = ""; + Pattern pattern = Pattern.compile("\\{([^\\}]+)\\}"); + Matcher matcher = pattern.matcher(dataObjectStr); + if (matcher.find()) { + partitionName = matcher.group(1); + } + partitionName = partitionName.replace("\'", "").replace("\"", ""); + tableName = tableName.split("\\{")[0]; + dataObject = new CheckDataObject(dbName, tableName, partitionName); + }else{ + dataObject=new CheckDataObject(dbName,tableName); + } + return dataObject; + } + /** + * 查hive 元数据库 + */ + private long getJobTotalCount(CheckDataObject dataObject, Connection conn, Logger log) { log.info("-------------------------------------- search hive/spark/mr data "); - log.info("-------------------------------------- : " + dataObject); + log.info("-------------------------------------- dataObject: " + dataObject); try (PreparedStatement pstmt = getJobStatement(conn, dataObject)) { ResultSet rs = pstmt.executeQuery(); - return rs.last() ? rs.getRow() : 0; + long ret = rs.last() ? rs.getRow() : 0; + log.info("-------------------------------------- hive/spark/mr data result:"+ret); + return ret; } catch (SQLException e) { log.error("fetch data from Hive MetaStore error", e); return 0; } } - private long getBdpTotalCount(Map proObjectMap, Connection conn, Logger log, Properties props) { - String dataObject = proObjectMap.get(DataChecker.DATA_OBJECT); - if (dataObject != null) { - dataObject = dataObject.replace(" ", "").trim(); - } + /** + * 查mask db + */ + private long getBdpTotalCount(CheckDataObject dataObject, Connection conn, Logger log, Properties props) { String timeScape = props.getOrDefault(DataChecker.TIME_SCAPE, "NULL").toString(); log.info("-------------------------------------- search bdp data "); - log.info("-------------------------------------- : " + dataObject); + log.info("-------------------------------------- dataObject: " + dataObject.toString()); try (PreparedStatement pstmt = getBdpStatement(conn, dataObject, timeScape)) { ResultSet rs = pstmt.executeQuery(); - return rs.last() ? rs.getRow() : 0; + long ret=rs.last() ? rs.getRow() : 0; + log.info("-------------------------------------- bdp data result:"+ret); + return ret; } catch (SQLException e) { - log.error("fetch data from Hive MetaStore error", e); + log.error("fetch data from bdp error", e); return 0; } } - private Map fetchMaskCode(Map proObjectMap, Logger log, Properties props) { - log.info("=============================调用BDP MASK接口查询数据状态=========================================="); - Map resultMap = new HashMap(); - String maskUrl = props.getProperty(DataChecker.MASK_URL); - String dataObject = proObjectMap.get(DataChecker.DATA_OBJECT); - if (dataObject != null) { - dataObject = dataObject.replace(" ", "").trim(); + /** + * - 返回0表示未找到任何记录 ; + * - 返回1表示非分区表的全表校验场景找到了记录; + * - 返回2表示分区表的分区校验场景找到了记录; + * - 返回3表示分区表的全表校验场景找到了记录; + * - 返回4表示查询出错了 + */ + private int checkDops(CheckDataObject dataObject, Connection conn, Logger log){ + log.info("-------------------------------------- search dops data "); + log.info("-------------------------------------- dataObject: " + dataObject.toString()); + try (PreparedStatement pstmt = getDopsStatement(conn, dataObject)) { + ResultSet rs = pstmt.executeQuery(); + long count = rs.last() ? rs.getRow() : 0; + log.info("-------------------------------------- dops data check table or partition,count:"+count); + if(count>0){ + return CheckDataObject.Type.PARTITION == dataObject.getType() ? 2 : 1; + }else if(CheckDataObject.Type.PARTITION == dataObject.getType()){ + //分区校验没找到记录,直接返回0。 + return 0; + } + } catch (SQLException e) { + log.error("fetch data from dops error while check table or partition", e); + //如果查询出错,还是认为dops处理过这个表/分区 + return 4; } - String dataScape = dataObject.contains("{") ? "Partition" : "Table"; - String dbName = dataObject.split("\\.")[0]; - String tableName = dataObject.split("\\.")[1]; - String partitionName = ""; - Pattern pattern = Pattern.compile("\\{([^\\}]+)\\}"); - if (dataScape.equals("Partition")) { - Matcher matcher = pattern.matcher(dataObject); - if (matcher.find()) { - partitionName = matcher.group(1); + + try(PreparedStatement pstmt = getDopsStatementCheckAllPartition(conn, dataObject)){ + ResultSet rs = pstmt.executeQuery(); + long count = rs.last() ? rs.getRow() : 0; + log.info("-------------------------------------- dops data check all partition result count:"+count); + if(count>0){ + return 3; } - partitionName = partitionName.replace("\'", "").replace("\"", ""); - tableName = tableName.split("\\{")[0]; + }catch (SQLException e) { + log.error("fetch data from dops error while check all partition", e); + //如果查询出错,还是认为dops处理过这个表/分区 + return 4; + } + return 0; + } + + /** + * 从qualitis去check数据 + */ + private boolean checkQualitisData(String objectNum,CheckDataObject dataObject, Logger log, + DataCheckerExecutionAction action,Properties props,Connection conn, + QualitisUtil qualitisUtil,String dataObjectStr ) { + boolean systemCheck = Boolean.valueOf(props.getProperty(DataChecker.QUALITIS_SWITCH)); + String userCheckDefault=props.getProperty(DataChecker.QUALITIS_CHECK_DEFAULT); + String userCheckStr = StringUtils.isBlank(props.getProperty(DataChecker.QUALITIS_CHECK)) ? + userCheckDefault : + props.getProperty(DataChecker.QUALITIS_CHECK); + boolean userCheck = Boolean.valueOf(userCheckStr); + log.info("systemCheck:{},userCheckDefault:{},userCheck:{}",systemCheck,userCheckDefault,userCheck); + if (systemCheck && userCheck ) { + if (null != action.getExecutionRequestRefContext()){ + action.getExecutionRequestRefContext().appendLog(dataObjectStr+" start to check dops"); + } + log.info("start to check dops"); + int dopsState=checkDops(dataObject,conn,log); + if (null != action.getExecutionRequestRefContext()){ + action.getExecutionRequestRefContext().appendLog(dataObjectStr+" check dops end,check result:"+dopsState); + } + log.info("check dops end,check result:"+dopsState); + if(dopsState==0){ + //没找到记录,直接通过校验 + return true; + } else if (dopsState == 3 || dopsState == 4) { + //找记录失败、或者是找到了分区表的全表校验记录,直接校验不通过。 + return false; + } + // 其他情况,继续走qualitis校验 + if (null != action.getExecutionRequestRefContext()){ + action.getExecutionRequestRefContext().appendLog(dataObjectStr+" Data Check Qualitis Start"); + } + log.info( + "=============================Data Check Qualitis Start=========================================="); + try { + String projectName = props.getProperty(DataChecker.CONTEXTID_PROJECT_NAME); + String user = props.getProperty(DataChecker.CONTEXTID_USER); + String flowName = props.getProperty(DataChecker.CONTEXTID_FLOW_NAME); + String nodeName=props.getProperty(DataChecker.NAME_NAME); + + String ruleName = getMD5Str(projectName + flowName + nodeName + objectNum); + String applicationId = qualitisUtil + .createAndSubmitRule(dataObject,projectName,ruleName,user); + if (StringUtils.isEmpty(applicationId)) { + throw new SQLException("applicationId is empty"); + } + long startTime = System.currentTimeMillis(); + while (System.currentTimeMillis() - startTime < Integer + .parseInt(props.getProperty("qualitis.getStatus.all.timeout"))) { + int status = new BigDecimal(qualitisUtil.getTaskStatus(applicationId)).intValue(); + switch (status) { + case 1: + case 3: + case 10: + case 12: + try { + Thread + .sleep(Double.valueOf(props.getProperty("qualitis.getStatus.interval")).longValue()); + } catch (InterruptedException e) { + log.error("get datachecker result from qualitis InterruptedException", e); + } + break; + case 4: + if (null != action.getExecutionRequestRefContext()){ + action.getExecutionRequestRefContext().appendLog(dataObjectStr+" check qualitis end,check result:true"); + } + log.info("check qualitis end,check result:true"); + return true; + default: + if (null != action.getExecutionRequestRefContext()){ + action.getExecutionRequestRefContext().appendLog(dataObjectStr+" check qualitis end,check result:false"); + } + log.info("check qualitis end,check result:false"); + return false; + } + + } + if (null != action.getExecutionRequestRefContext()){ + action.getExecutionRequestRefContext().appendLog(dataObjectStr+" Data Check Qualitis time out,check result set to false"); + } + log.info( + "=============================Data Check Qualitis time out,check result set to false=========================================="); + return false; + } catch (Exception e) { + if (null != action.getExecutionRequestRefContext()){ + action.getExecutionRequestRefContext().appendLog(dataObjectStr+" check qualitis failed,check result set to false.cause by:"+e.getMessage()); + } + log.error("get datachecker result from qualitis failed", e); + return false; + } + + } else { + return true; } + } + public static String getMD5Str(String str){ + return DigestUtils.md5Hex(str); + } + + private Map fetchMaskCode(CheckDataObject dataObject, Logger log, Properties props) { + log.info("=============================调用BDP MASK接口查询数据状态=========================================="); + Map resultMap = new HashMap(); + String maskUrl = props.getProperty(DataChecker.MASK_URL); + String dbName = dataObject.getDbName(); + String tableName = dataObject.getTableName(); + String partitionName = dataObject.getPartitionName() == null ? "" : dataObject.getPartitionName(); try { RequestBody requestBody = new FormBody.Builder() .add("targetDb", dbName) diff --git a/dss-appconn/appconns/dss-datachecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/datachecker/connector/DataDruidFactory.java b/dss-appconn/appconns/dss-datachecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/datachecker/connector/DataDruidFactory.java index 86a2a40c42..e91ef5b320 100644 --- a/dss-appconn/appconns/dss-datachecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/datachecker/connector/DataDruidFactory.java +++ b/dss-appconn/appconns/dss-datachecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/datachecker/connector/DataDruidFactory.java @@ -24,24 +24,26 @@ import java.util.Properties; public class DataDruidFactory { - private static DruidDataSource jobInstance; - private static DruidDataSource bdpInstance; - private static DruidDataSource msgInstance; - - public static DruidDataSource getJobInstance(Properties props, Logger log) { - if (jobInstance == null ) { - synchronized (DataDruidFactory.class) { - if(jobInstance == null) { - try { - jobInstance = createDataSource(props, log, "Job"); - } catch (Exception e) { - throw new RuntimeException("Error creating Druid DataSource", e); - } - } - } - } - return jobInstance; - } + private static volatile DruidDataSource jobInstance; + private static volatile DruidDataSource bdpInstance; + + private static volatile DruidDataSource dopsInstance; + private static volatile DruidDataSource msgInstance; + + public static DruidDataSource getJobInstance(Properties props, Logger log) { + if (jobInstance == null ) { + synchronized (DataDruidFactory.class) { + if(jobInstance == null) { + try { + jobInstance = createDataSource(props, log, "Job"); + } catch (Exception e) { + throw new RuntimeException("Error creating job Druid DataSource", e); + } + } + } + } + return jobInstance; + } public static DruidDataSource getBDPInstance(Properties props, Logger log) { if (bdpInstance == null ) { synchronized (DataDruidFactory.class) { @@ -49,109 +51,140 @@ public static DruidDataSource getBDPInstance(Properties props, Logger log) { try { bdpInstance = createDataSource(props, log, "BDP"); } catch (Exception e) { - throw new RuntimeException("Error creating Druid DataSource", e); + throw new RuntimeException("Error creating BDP Druid DataSource", e); } } } } return bdpInstance; } - - public static DruidDataSource getMsgInstance(Properties props, Logger log) { - if (msgInstance == null ) { - synchronized (DataDruidFactory.class) { - if(msgInstance == null) { - try { - msgInstance = createDataSource(props, log, "Msg"); - } catch (Exception e) { - throw new RuntimeException("Error creating Druid DataSource", e); - } - } - } - } - return msgInstance; - } - - private static DruidDataSource createDataSource(Properties props, Logger log, String type) { - String name = null; - String url = null; - String username = null; - String password = null; - String loginType = null; - if (type.equals("Job")) { - name = props.getProperty("job.datachecker.jdo.option.name"); - url = props.getProperty("job.datachecker.jdo.option.url"); - username = props.getProperty("job.datachecker.jdo.option.username"); - password = props.getProperty("job.datachecker.jdo.option.password"); - loginType = props.getProperty("job.datachecker.jdo.option.login.type"); - log.info("job url is:"+url+"and name is:"+username); - try { - if("base64".equals(loginType)) { - password = new String(Base64.getDecoder().decode(props.getProperty("job.datachecker.jdo.option.password").getBytes()), "UTF-8"); - }else { - password = props.getProperty("job.datachecker.jdo.option.password"); - } - } catch (Exception e){ - log.error("password decore failed" + e); - } - }else if (type.equals("BDP")) { + + public static DruidDataSource getDopsInstance(Properties props, Logger log) { + if (dopsInstance == null) { + synchronized (DataDruidFactory.class) { + if (dopsInstance == null) { + try { + dopsInstance = createDataSource(props, log, "Dops"); + } catch (Exception e) { + throw new RuntimeException("Error creating DOPS Druid DataSource", e); + } + } + } + } + return dopsInstance; + } + + public static DruidDataSource getMsgInstance(Properties props, Logger log) { + if (msgInstance == null ) { + synchronized (DataDruidFactory.class) { + if(msgInstance == null) { + try { + msgInstance = createDataSource(props, log, "Msg"); + } catch (Exception e) { + throw new RuntimeException("Error creating Druid DataSource", e); + } + } + } + } + return msgInstance; + } + + private static DruidDataSource createDataSource(Properties props, Logger log, String type) { + String name = null; + String url = null; + String username = null; + String password = null; + String loginType = null; + if (type.equals("Job")) { + name = props.getProperty("job.datachecker.jdo.option.name"); + url = props.getProperty("job.datachecker.jdo.option.url"); + username = props.getProperty("job.datachecker.jdo.option.username"); + password = props.getProperty("job.datachecker.jdo.option.password"); + loginType = props.getProperty("job.datachecker.jdo.option.login.type"); + log.info("job url is:" + url + "and name is:" + username); + try { + if ("base64".equals(loginType)) { + password = new String(Base64.getDecoder().decode(props.getProperty("job.datachecker.jdo.option.password").getBytes()), "UTF-8"); + } else { + password = props.getProperty("job.datachecker.jdo.option.password"); + } + } catch (Exception e) { + log.error("password decore failed" + e); + } + } else if (type.equals("BDP")) { name = props.getProperty("bdp.datachecker.jdo.option.name"); url = props.getProperty("bdp.datachecker.jdo.option.url"); username = props.getProperty("bdp.datachecker.jdo.option.username"); password = props.getProperty("bdp.datachecker.jdo.option.password"); - loginType = props.getProperty("bdp.datachecker.jdo.option.login.type"); - log.info("bdp url is:"+url+"and name is:"+username); + loginType = props.getProperty("bdp.datachecker.jdo.option.login.type"); + log.info("bdp url is:" + url + "and name is:" + username); try { - if("base64".equals(loginType)) { - password = new String(Base64.getDecoder().decode(props.getProperty("bdp.datachecker.jdo.option.password").getBytes()), "UTF-8"); - }else { - password = props.getProperty("bdp.datachecker.jdo.option.password"); - } - } catch (Exception e){ + if ("base64".equals(loginType)) { + password = new String(Base64.getDecoder().decode(props.getProperty("bdp.datachecker.jdo.option.password").getBytes()), "UTF-8"); + } else { + password = props.getProperty("bdp.datachecker.jdo.option.password"); + } + } catch (Exception e) { log.error("password decore failed" + e); } + }else if ("Dops".equals(type)) { + name = props.getProperty("dops.datachecker.jdo.option.name"); + url = props.getProperty("dops.datachecker.jdo.option.url"); + username = props.getProperty("dops.datachecker.jdo.option.username"); + password=props.getProperty("dops.datachecker.jdo.option.password"); + loginType = props.getProperty("dops.datachecker.jdo.option.login.type"); + log.info("dops url is:" + url + "and name is:" + username); + try { + if ("base64".equals(loginType)) { + password = new String(Base64.getDecoder().decode(props.getProperty("dops.datachecker.jdo.option.password").getBytes()), "UTF-8"); + } else { + password = props.getProperty("dops.datachecker.jdo.option.password"); + } + } catch (Exception e) { + log.error("password decore failed", e); + } + } + int initialSize = Integer.valueOf(props.getProperty("datachecker.jdo.option.initial.size", "1")); + int maxActive = Integer.valueOf(props.getProperty("datachecker.jdo.option.max.active", "100")); + int minIdle = Integer.valueOf(props.getProperty("datachecker.jdo.option.min.idle", "1")); + long maxWait = Long.valueOf(props.getProperty("datachecker.jdo.option.max.wait", "60000")); + String validationQuery = props.getProperty("datachecker.jdo.option.validation.quert", "SELECT 'x'"); + long timeBetweenEvictionRunsMillis = Long.valueOf(props.getProperty("datachecker.jdo.option.time.between.eviction.runs.millis", "6000")); + long minEvictableIdleTimeMillis = Long.valueOf(props.getProperty("datachecker.jdo.option.evictable.idle,time.millis", "300000")); + boolean testOnBorrow = Boolean.valueOf(props.getProperty("datachecker.jdo.option.test.on.borrow", "true")); + int maxOpenPreparedStatements = Integer.valueOf(props.getProperty("datachecker.jdo.option.max.open.prepared.statements", "-1")); + + + if (timeBetweenEvictionRunsMillis > minEvictableIdleTimeMillis) { + timeBetweenEvictionRunsMillis = minEvictableIdleTimeMillis; } - int initialSize = Integer.valueOf(props.getProperty("datachecker.jdo.option.initial.size", "1")); - int maxActive = Integer.valueOf(props.getProperty("datachecker.jdo.option.max.active", "100")); - int minIdle = Integer.valueOf(props.getProperty("datachecker.jdo.option.min.idle", "1")); - long maxWait = Long.valueOf(props.getProperty("datachecker.jdo.option.max.wait", "60000")); - String validationQuery = props.getProperty("datachecker.jdo.option.validation.quert", "SELECT 'x'"); - long timeBetweenEvictionRunsMillis = Long.valueOf(props.getProperty("datachecker.jdo.option.time.between.eviction.runs.millis", "6000")); - long minEvictableIdleTimeMillis = Long.valueOf(props.getProperty("datachecker.jdo.option.evictable.idle,time.millis", "300000")); - boolean testOnBorrow = Boolean.valueOf(props.getProperty("datachecker.jdo.option.test.on.borrow", "true")); - int maxOpenPreparedStatements = Integer.valueOf(props.getProperty("datachecker.jdo.option.max.open.prepared.statements", "-1")); - - - if (timeBetweenEvictionRunsMillis > minEvictableIdleTimeMillis) { - timeBetweenEvictionRunsMillis = minEvictableIdleTimeMillis; - } - - DruidDataSource ds = new DruidDataSource(); - - if (StringUtils.isNotBlank(name)) { - ds.setName(name); - } - - ds.setUrl(url); - ds.setDriverClassName("com.mysql.jdbc.Driver"); - ds.setUsername(username); - ds.setPassword(password); - ds.setInitialSize(initialSize); - ds.setMinIdle(minIdle); - ds.setMaxActive(maxActive); - ds.setMaxWait(maxWait); - ds.setTestOnBorrow(testOnBorrow); - ds.setValidationQuery(validationQuery); - ds.setTimeBetweenEvictionRunsMillis(timeBetweenEvictionRunsMillis); - ds.setMinEvictableIdleTimeMillis(minEvictableIdleTimeMillis); - if (maxOpenPreparedStatements > 0) { - ds.setPoolPreparedStatements(true); - ds.setMaxPoolPreparedStatementPerConnectionSize( - maxOpenPreparedStatements); - } else { - ds.setPoolPreparedStatements(false); - } - log.info("Druid data source initialed!"); - return ds; - } + + DruidDataSource ds = new DruidDataSource(); + + if (StringUtils.isNotBlank(name)) { + ds.setName(name); + } + + ds.setUrl(url); + ds.setDriverClassName("com.mysql.jdbc.Driver"); + ds.setUsername(username); + ds.setPassword(password); + ds.setInitialSize(initialSize); + ds.setMinIdle(minIdle); + ds.setMaxActive(maxActive); + ds.setMaxWait(maxWait); + ds.setTestOnBorrow(testOnBorrow); + ds.setValidationQuery(validationQuery); + ds.setTimeBetweenEvictionRunsMillis(timeBetweenEvictionRunsMillis); + ds.setMinEvictableIdleTimeMillis(minEvictableIdleTimeMillis); + if (maxOpenPreparedStatements > 0) { + ds.setPoolPreparedStatements(true); + ds.setMaxPoolPreparedStatementPerConnectionSize( + maxOpenPreparedStatements); + } else { + ds.setPoolPreparedStatements(false); + } + log.info("Druid data source initialed!"); + return ds; + } } diff --git a/dss-appconn/appconns/dss-datachecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/datachecker/utils/HttpUtils.java b/dss-appconn/appconns/dss-datachecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/datachecker/utils/HttpUtils.java index 85a9284f18..9cf1eb1cef 100644 --- a/dss-appconn/appconns/dss-datachecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/datachecker/utils/HttpUtils.java +++ b/dss-appconn/appconns/dss-datachecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/datachecker/utils/HttpUtils.java @@ -24,7 +24,8 @@ import okhttp3.*; import org.apache.commons.codec.digest.DigestUtils; import org.apache.commons.lang3.RandomStringUtils; -import org.apache.log4j.Logger; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; import java.io.IOException; import java.util.HashMap; @@ -34,7 +35,7 @@ public class HttpUtils { - private static final Logger logger = Logger.getLogger(HttpUtils.class); + private static final Logger logger = LoggerFactory.getLogger(HttpUtils.class); public static Response httpClientHandleBase(String actionUrl, RequestBody requestBody, Map urlMap) throws IOException { String maskUrl = actionUrl + "appid=" + urlMap.get("appid") + "&&nonce=" + urlMap.get("nonce") diff --git a/dss-appconn/appconns/dss-datachecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/datachecker/utils/QualitisUtil.java b/dss-appconn/appconns/dss-datachecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/datachecker/utils/QualitisUtil.java new file mode 100644 index 0000000000..3cc8013562 --- /dev/null +++ b/dss-appconn/appconns/dss-datachecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/datachecker/utils/QualitisUtil.java @@ -0,0 +1,313 @@ +package com.webank.wedatasphere.dss.appconn.datachecker.utils; + + +import com.webank.wedatasphere.dss.appconn.datachecker.DataChecker; +import com.webank.wedatasphere.dss.appconn.datachecker.common.CheckDataObject; +import com.webank.wedatasphere.dss.common.utils.DSSCommonUtils; +import okhttp3.*; +import org.apache.commons.codec.digest.DigestUtils; +import org.apache.commons.lang.RandomStringUtils; +import org.apache.commons.lang.StringUtils; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.IOException; +import java.net.URISyntaxException; +import java.security.NoSuchAlgorithmException; +import java.util.HashMap; +import java.util.List; +import java.util.Map; +import java.util.Properties; +import java.util.concurrent.TimeUnit; + +/** + * Tools for qualitis + * + * @author lebronwang + * @date 2022/08/04 + **/ +public class QualitisUtil { + + private static Logger logger = LoggerFactory.getLogger(QualitisUtil.class); + + String SUBMIT_TASK_PATH = ""; + private final static String GET_TASK_STATUS_PATH = "qualitis/outer/api/v1/application/{applicationId}/status/"; + private final static String createSubmitRulePath = "qualitis/outer/api/v1/bdp_client/create_and_submit"; + String baseUrl = ""; + String appId = ""; + String appToken = ""; + long getStatusTimeout; + private Properties properties; + + + + public QualitisUtil(Properties properties) { + this.properties = properties; + this.baseUrl = this.properties.getProperty("qualitis.baseUrl"); + this.appId = this.properties.getProperty("qualitis.appId"); + this.appToken = this.properties.getProperty("qualitis.appToken"); + this.getStatusTimeout =Double + .valueOf(this.properties.getProperty("qualitis.getStatus.timeout", "60000")).longValue(); + } + + /** + * 提交 Qualitis 任务 + * + * @param groupId 规则组 ID + * @param createUser + * @param executionUser + * @return + * @throws IOException + */ + public String submitTask(long groupId, String createUser, String executionUser) + throws Exception { + + logger.info("Submitting Qualitis task(groupId: " + groupId + ") ... "); + String applicationId = ""; + String url = ""; + try { + url = buildUrI(baseUrl, SUBMIT_TASK_PATH, appId, appToken, + RandomStringUtils.randomNumeric(5), String.valueOf(System.currentTimeMillis())); + } catch (NoSuchAlgorithmException | URISyntaxException e) { + logger.error("Build Qualitis URL error: " + e.getMessage()); + } + + Map param = new HashMap<>(); + param.put("group_id", groupId); + param.put("create_user", createUser); + param.put("execution_user", executionUser); + + String json = DSSCommonUtils.COMMON_GSON.toJson(param); + logger.info("Request Json: " + json); + MediaType applicationJson = MediaType.parse("application/json;charset=utf-8"); + RequestBody requestBody = RequestBody.create(json, applicationJson); + + OkHttpClient okHttpClient = new OkHttpClient.Builder() + .connectTimeout(10, TimeUnit.SECONDS) + .writeTimeout(20, TimeUnit.SECONDS) + .readTimeout(20, TimeUnit.SECONDS) + .build(); + + Request request = new Request.Builder() + .url(url) + .post(requestBody) + .build(); + + Call call = okHttpClient.newCall(request); + Response response = call.execute(); + String resultJson = response.body().string(); + logger.info("Response Json: " + resultJson); + if (StringUtils.isNotEmpty(resultJson)) { + Map resultMap = DSSCommonUtils.COMMON_GSON.fromJson(resultJson,Map.class); + logger.info(String.valueOf(resultMap)); + String code = (String) resultMap.get("code"); + if ("200".equals(code)) { + applicationId = (String) ((Map) resultMap.get("data")).get( + "application_id"); + logger.info(resultMap.get("message").toString()); + } else { + throw new RuntimeException(resultMap.get("message").toString()); + } + } + return applicationId; + } + + public String createAndSubmitRule(CheckDataObject dataObject,String projectName,String ruleName,String user) throws IOException { + logger.info(""); + String applicationId = ""; + String url = ""; + try { + url = buildUrI(baseUrl, createSubmitRulePath, appId, appToken, + RandomStringUtils.randomNumeric(5), String.valueOf(System.currentTimeMillis())); + } catch (NoSuchAlgorithmException | URISyntaxException e) { + logger.error("Build Qualitis URL error: " + e.getMessage()); + } + + // JSON 请求参数 + Map param = new HashMap<>(); + param.put("create_user", user); + param.put("execution_user",user); + // 集群名 + String clusterName = properties.getProperty("cluster.name"); + // 子系统名 WTSS-BDPWFM/WTSS-BDAPWFM + StringBuilder sb = new StringBuilder("expectFileAmountCount(\"").append(clusterName) + .append(".").append(dataObject.getDbName()) + .append(".").append(dataObject.getTableName()) + .append(dataObject.getType()== CheckDataObject.Type.PARTITION? ":" + dataObject.getPartitionName() : "") + .append("\", null, false).addRuleMetricWithCheck(\"") + .append(String.format(properties.getProperty("qualitis.rule.metric"), + ruleName)) + .append("\", false, false, false).fixValueNotEqual(0)"); + logger.info("template_function:{}",sb); + param.put("template_function", sb.toString()); + param.put("project_name", projectName); + param.put("rule_name", ruleName); + + String json = DSSCommonUtils.COMMON_GSON.toJson(param); + logger.info("start to call qualitis,url:{} ,Request Json:{} ", url, json ); + MediaType applicationJson = MediaType.parse("application/json;charset=utf-8"); + RequestBody requestBody = RequestBody.create(json, applicationJson); + + OkHttpClient okHttpClient = new OkHttpClient.Builder() + .callTimeout(Double.valueOf(properties.getProperty("qualitis.submitTask.timeout")).longValue(), TimeUnit.MILLISECONDS) + .connectTimeout(10, TimeUnit.SECONDS) + .writeTimeout(20, TimeUnit.SECONDS) + .readTimeout(20, TimeUnit.SECONDS) + .build(); + + Request request = new Request.Builder() + .url(url) + .post(requestBody) + .build(); + + Call call = okHttpClient.newCall(request); + Response response = call.execute(); + String resultJson = response.body().string(); + logger.info("call qualitis end,Response Json:{} " , resultJson); + if (StringUtils.isNotEmpty(resultJson)) { + Map resultMap = DSSCommonUtils.COMMON_GSON.fromJson(resultJson,Map.class); + String code = (String) resultMap.get("code"); + if ("200".equals(code)) { + Map applicationDetail = (Map) ((Map) (resultMap.get( + "data"))).get("application_detail"); + Map project_application = ((List>) (applicationDetail.get( + "project_applications"))).get(0); + applicationId = (String) project_application.get("application_id"); + logger.info(resultMap.get("message").toString()); + } else { + throw new RuntimeException(resultMap.get("message").toString()); + } + } + return applicationId; + } + + /** + * 获取 Qualitis 任务状态 + * + * @param applicationId + * @return + */ + public String getTaskStatus(String applicationId){ + + logger.info("Getting Qualitis task status(ApplicationId: " + applicationId + ") ... "); + if (StringUtils.isEmpty(applicationId)) { + logger.error("Empty Qualitis application ID"); + } + String url = ""; + try { + url = buildUrI(baseUrl, GET_TASK_STATUS_PATH.replace("{applicationId}", applicationId) + , appId, appToken, RandomStringUtils.randomNumeric(5), + String.valueOf(System.currentTimeMillis())); + } catch (NoSuchAlgorithmException | URISyntaxException e) { + logger.error("Build Qualitis URL error: " + e.getMessage()); + } + + OkHttpClient okHttpClient = new OkHttpClient.Builder() + .callTimeout(this.getStatusTimeout, TimeUnit.MILLISECONDS) + .connectTimeout(10, TimeUnit.SECONDS) + .writeTimeout(20, TimeUnit.SECONDS) + .readTimeout(20, TimeUnit.SECONDS) + .build(); + + Request request = new Request.Builder() + .url(url) + .get() + .build(); + + Call call = okHttpClient.newCall(request); + Response response = null; + String resultJson = ""; + String status = ""; + try { + response = call.execute(); + resultJson = response.body().string(); + logger.info("Response Json: " + resultJson); + if (StringUtils.isNotEmpty(resultJson)) { + Map resultMap = DSSCommonUtils.COMMON_GSON.fromJson(resultJson,Map.class); + String code = (String) resultMap.get("code"); + if ("200".equals(code)) { + status = ((Map) (resultMap.get("data"))).get( + "application_status") + ""; + } + } + } catch (IOException e) { + logger.error("Request to Qualitis failed: " + e.getMessage()); + } + + return status; + } + + public static String buildUrI(String baseUrl, String path, String appId, String appToken, + String nonce, String timestamp) throws NoSuchAlgorithmException, URISyntaxException { + String signature = getSignature(appId, appToken, nonce, timestamp); + StringBuffer uriBuffer = new StringBuffer(baseUrl); + uriBuffer.append(path).append("?") + .append("app_id=").append(appId).append("&") + .append("nonce=").append(nonce).append("&") + .append("timestamp=").append(timestamp).append("&") + .append("signature=").append(signature); + + return uriBuffer.toString(); + } + + public static String getSignature(String appId, String appToken, String nonce, String timestamp) + throws NoSuchAlgorithmException { + return Sha256Utils.getSHA256L32(Sha256Utils.getSHA256L32(appId + nonce + timestamp) + appToken); + } + + public static void main(String[] args) throws Exception { + + /*Map ruleGroup = new HashMap<>(); + ruleGroup.put(39211L, 0); + ruleGroup.put(37456L, 0); + + + String applicationId = submitTask(ruleList, "allenzhou", "allenzhou"); + System.out.println(applicationId); + System.out.println( + "---------------------------------------------------------------------------"); + String taskStatus = getTaskStatus(applicationId); + System.out.println(taskStatus);*/ + + String appId = "linkis_id"; + String appToken = "a33693de51"; + String nonce = RandomStringUtils.randomNumeric(5); + String timestamp = String.valueOf(System.currentTimeMillis()); + String signature = getSignature(appId, appToken, nonce, timestamp); + + System.out.println("nonce = " + nonce); + System.out.println("timeStamp = " + timestamp); + System.out.println("signature = " + signature); + + Properties properties = new Properties(); + properties.setProperty("qualitis.baseUrl", "http://***REMOVED***:7080/"); + properties.setProperty("qualitis.appId", appId); + properties.setProperty("qualitis.appToken", appToken); + properties.setProperty("qualitis.createSubmitRule.path", + "qualitis/outer/api/v1/bdp_client/create_and_submit"); + properties.setProperty("qualitis.getTaskStatus.path", + "qualitis/outer/api/v1/application/{applicationId}/status/"); + + properties.setProperty("user.to.proxy", "hadoop"); + properties.setProperty("user.to.proxy", "hadoop"); + properties.setProperty("cluster.name", "HDP-GZPC-BDAP-UAT"); + properties.setProperty("qualitis.submitTask.timeout","180000"); + properties.setProperty("subsystem.name", "WTSS-BDPWFM"); + properties.setProperty("azkaban.flow.flowid", "test"); + properties.setProperty("azkaban.flow.projectname", "test"); + properties.setProperty("qualitis.rule.metric","WTSS-BDPWFM_general-metric_%s_Daily"); + + String dBName = "dqm_test"; + String tableName = "test_dqm_left"; + String partitionName = "ds=2023-05-08"; + + QualitisUtil qualitisUtil = new QualitisUtil(properties); + String applicationId = qualitisUtil.createAndSubmitRule(new CheckDataObject(dBName, tableName, partitionName), "test", "4134314","hadoop"); + String applicationId1 = qualitisUtil.createAndSubmitRule(new CheckDataObject(dBName, tableName, partitionName), "test", "4134314","hadoop"); + System.out.println(applicationId); + System.out.println(applicationId1); + + String taskStatus = qualitisUtil.getTaskStatus(applicationId); + System.out.println(taskStatus); + } +} diff --git a/dss-appconn/appconns/dss-datachecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/datachecker/utils/Sha256Utils.java b/dss-appconn/appconns/dss-datachecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/datachecker/utils/Sha256Utils.java new file mode 100644 index 0000000000..242fc1a5ea --- /dev/null +++ b/dss-appconn/appconns/dss-datachecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/datachecker/utils/Sha256Utils.java @@ -0,0 +1,30 @@ +package com.webank.wedatasphere.dss.appconn.datachecker.utils; + +import org.apache.commons.lang.StringUtils; + +import java.math.BigInteger; +import java.security.MessageDigest; +import java.security.NoSuchAlgorithmException; + +/** + * @author lebronwang + * @date 2022/08/04 + **/ +public class Sha256Utils { + + public static String getSHA256L32(String str) throws NoSuchAlgorithmException { + try { + // Generate a instance of sha-256 + MessageDigest hash = MessageDigest.getInstance("SHA-256"); + // calculate + hash.update(str.getBytes()); + String result = new BigInteger(1, hash.digest()).toString(16); + + return StringUtils.leftPad(result, 32, '0'); + } catch (NoSuchAlgorithmException e) { + throw new NoSuchAlgorithmException("Error generating sha256 of string", e); + } + } + + +} diff --git a/dss-appconn/appconns/dss-datachecker-appconn/src/main/resources/appconn.properties b/dss-appconn/appconns/dss-datachecker-appconn/src/main/resources/appconn.properties index ac52d9b551..deed2cfe61 100644 --- a/dss-appconn/appconns/dss-datachecker-appconn/src/main/resources/appconn.properties +++ b/dss-appconn/appconns/dss-datachecker-appconn/src/main/resources/appconn.properties @@ -26,8 +26,35 @@ bdp.datachecker.jdo.option.username= bdp.datachecker.jdo.option.password= bdp.datachecker.jdo.option.login.type=base64 +dops.datachecker.jdo.option.name=dops +dops.datachecker.jdo.option.url= +dops.datachecker.jdo.option.username= +dops.datachecker.jdo.option.password= +dops.datachecker.jdo.option.login.type=base64 + bdp.mask.url= bdp.mask.app.id= bdp.mask.app.token= + + +job.eventchecker.qualitis.switch=true +# Qualitis ?? URL +qualitis.baseUrl=http://***REMOVED***:8090/ +# Qualitis ???? +qualitis.appId=linkis_id +qualitis.appToken=a33693de51 +# ?? Qualitis ?????????????? +qualitis.getStatus.interval=60000 +# Qualitis ???????????????? +qualitis.submitTask.timeout=180000 +# Qualitis ?????????? +qualitis.getStatus.timeout=60000 +# ??? +cluster.name=HDP-GZPC-BDAP-UAT +# Qualitis ???? +qualitis.rule.metric=DSS-IDE_general-metric_%s_Daily +# Qualitis ?????????? +qualitis.getStatus.all.timeout=60000 + diff --git a/dss-appconn/appconns/dss-datachecker-appconn/src/main/scala/com/webank/wedatasphere/dss/appconn/datachecker/DataCheckerRefExecutionOperation.scala b/dss-appconn/appconns/dss-datachecker-appconn/src/main/scala/com/webank/wedatasphere/dss/appconn/datachecker/DataCheckerRefExecutionOperation.scala index f9511f39c2..655a89afa6 100644 --- a/dss-appconn/appconns/dss-datachecker-appconn/src/main/scala/com/webank/wedatasphere/dss/appconn/datachecker/DataCheckerRefExecutionOperation.scala +++ b/dss-appconn/appconns/dss-datachecker-appconn/src/main/scala/com/webank/wedatasphere/dss/appconn/datachecker/DataCheckerRefExecutionOperation.scala @@ -16,7 +16,8 @@ package com.webank.wedatasphere.dss.appconn.datachecker -import com.webank.wedatasphere.dss.common.utils.VariableUtils + +import com.webank.wedatasphere.dss.common.utils.DSSCommonUtils import java.util import java.util.{Properties, UUID} @@ -25,7 +26,7 @@ import com.webank.wedatasphere.dss.standard.app.development.listener.core.{Killa import com.webank.wedatasphere.dss.standard.app.development.listener.ref.ExecutionResponseRef.ExecutionResponseRefBuilder import com.webank.wedatasphere.dss.standard.app.development.listener.ref.{AsyncExecutionResponseRef, ExecutionResponseRef, RefExecutionRequestRef} import org.apache.linkis.common.log.LogUtils -import org.apache.linkis.common.utils.Utils +import org.apache.linkis.common.utils.{Utils, VariableUtils} import scala.collection.mutable @@ -42,11 +43,22 @@ class DataCheckerRefExecutionOperation val nodeAction = new DataCheckerExecutionAction() nodeAction.setId(UUID.randomUUID().toString) import scala.collection.JavaConversions.mapAsScalaMap + // appconn里的配置 val InstanceConfig = this.service.getAppInstance.getConfig + // 节点上的配置 val runTimeParams = requestRef.getExecutionRequestRefContext.getRuntimeMap + // 自定义变量 val variableParams: mutable.Map[String, Object]= requestRef.getRefJobContent.get("variable"). asInstanceOf[java.util.Map[String,Object]] val inputParams = runTimeParams ++ variableParams val properties = new Properties() + // 解析contextId数据 + val contextID = DSSCommonUtils.COMMON_GSON.fromJson(runTimeParams.get("contextID").toString, classOf[util.Map[String, String]]) + // 去掉左右两侧花括号 以及字符的引号 + val value = contextID.get("value").drop(1).dropRight(1).replaceAll("\"", "") + val info = value.split(",").map(_.split(":")).map(arr => (arr(0), arr(1))).toMap + properties.put(DataChecker.CONTEXTID_USER, info("user")) + properties.put(DataChecker.CONTEXTID_PROJECT_NAME, info("project")) + properties.put(DataChecker.CONTEXTID_FLOW_NAME, info("flow")) InstanceConfig.foreach { case (key: String, value: Object) => //避免密码被打印 @@ -92,6 +104,7 @@ class DataCheckerRefExecutionOperation } } } + logger.info("datachecker properties :{}", properties) Utils.tryCatch({ val dc = new DataChecker(properties, nodeAction) dc.run() diff --git a/dss-appconn/appconns/dss-dolphinscheduler-appconn/pom.xml b/dss-appconn/appconns/dss-dolphinscheduler-appconn/pom.xml index fc00a8473b..411b9fd682 100644 --- a/dss-appconn/appconns/dss-dolphinscheduler-appconn/pom.xml +++ b/dss-appconn/appconns/dss-dolphinscheduler-appconn/pom.xml @@ -6,7 +6,7 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 ../../../pom.xml 4.0.0 @@ -81,7 +81,7 @@ com.google.guava guava - 30.0-jre + 28.2-android diff --git a/dss-appconn/appconns/dss-dolphinscheduler-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/dolphinscheduler/conversion/DolphinSchedulerWorkflowToRelConverter.java b/dss-appconn/appconns/dss-dolphinscheduler-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/dolphinscheduler/conversion/DolphinSchedulerWorkflowToRelConverter.java index 8b0e112027..4ccdd0634c 100644 --- a/dss-appconn/appconns/dss-dolphinscheduler-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/dolphinscheduler/conversion/DolphinSchedulerWorkflowToRelConverter.java +++ b/dss-appconn/appconns/dss-dolphinscheduler-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/dolphinscheduler/conversion/DolphinSchedulerWorkflowToRelConverter.java @@ -24,7 +24,7 @@ public ConvertedRel convertToRel(PreConversionRel rel) { .entrySet().stream().filter(entry -> entry.getValue() > 1).map(Map.Entry::getKey) .collect(Collectors.joining(", ")); if (StringUtils.isNotEmpty(repeatNodes)) { - throw new DSSRuntimeException(80001, "重复的节点名称:" + repeatNodes); + throw new DSSRuntimeException(80001, "重复的节点名称。项目中不同工作流(或子工作流)里存在重名节点,请修改节点名避免重名。重名节点:" + repeatNodes); } DolphinSchedulerConvertedRel dolphinSchedulerConvertedRel = new DolphinSchedulerConvertedRel(rel); return dolphinSchedulerConvertedRel; diff --git a/dss-appconn/appconns/dss-dolphinscheduler-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/dolphinscheduler/sso/AbstractDolphinSchedulerTokenManager.java b/dss-appconn/appconns/dss-dolphinscheduler-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/dolphinscheduler/sso/AbstractDolphinSchedulerTokenManager.java index b670fbcc1b..62e0a0bdb7 100644 --- a/dss-appconn/appconns/dss-dolphinscheduler-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/dolphinscheduler/sso/AbstractDolphinSchedulerTokenManager.java +++ b/dss-appconn/appconns/dss-dolphinscheduler-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/dolphinscheduler/sso/AbstractDolphinSchedulerTokenManager.java @@ -39,6 +39,10 @@ public abstract class AbstractDolphinSchedulerTokenManager implements DolphinSch protected String baseUrl; protected SSORequestOperation ssoRequestOperation; + private static final Object lock_1 = new Object(); + + private static final Object lock_2 = new Object(); + @Override public String getBaseUrl() { return baseUrl; @@ -66,7 +70,7 @@ public int getUserId(String userName) { if(userTokens.containsKey(userName)) { return userTokens.get(userName).getUserId(); } - synchronized (userName.intern()) { + synchronized (lock_1) { if(userTokens.containsKey(userName)) { return userTokens.get(userName).getUserId(); } @@ -107,7 +111,7 @@ public final String getToken(String userName) { } DolphinSchedulerAccessToken userToken = null; Integer userId; - synchronized (userName.intern()) { + synchronized (lock_2) { userId = fetchUserId(userName); if(userId == null) { // 用户不存在,创建该用户 diff --git a/dss-appconn/appconns/dss-eventchecker-appconn/pom.xml b/dss-appconn/appconns/dss-eventchecker-appconn/pom.xml index 0bcd00b2f6..ef59cd8672 100644 --- a/dss-appconn/appconns/dss-eventchecker-appconn/pom.xml +++ b/dss-appconn/appconns/dss-eventchecker-appconn/pom.xml @@ -21,7 +21,7 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 ../../../pom.xml 4.0.0 @@ -34,6 +34,20 @@ com.webank.wedatasphere.dss dss-appconn-core ${dss.version} + + + org.apache.linkis + linkis-common + + + org.apache.linkis + linkis-gateway-httpclient-support + + + org.apache.linkis + linkis-httpclient + + @@ -60,16 +74,11 @@ 1.0.28 - - log4j - log4j - 1.2.17 - - org.apache.linkis linkis-cs-client ${linkis.version} + provided linkis-common diff --git a/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/adapter/EventCheckAdapter.java b/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/adapter/EventCheckAdapter.java index f7004bb38e..f562ac4e74 100644 --- a/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/adapter/EventCheckAdapter.java +++ b/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/adapter/EventCheckAdapter.java @@ -16,7 +16,7 @@ package com.webank.wedatasphere.dss.appconn.eventchecker.adapter; -import org.apache.log4j.Logger; +import org.slf4j.Logger; import java.util.Properties; diff --git a/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/connector/EventDruidFactory.java b/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/connector/EventDruidFactory.java index ca8825fdce..9715e43c7a 100644 --- a/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/connector/EventDruidFactory.java +++ b/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/connector/EventDruidFactory.java @@ -19,7 +19,7 @@ import com.alibaba.druid.pool.DruidDataSource; import org.apache.commons.lang3.StringUtils; -import org.apache.log4j.Logger; +import org.slf4j.Logger; import java.util.Base64; import java.util.Properties; diff --git a/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/entity/EventChecker.java b/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/entity/EventChecker.java index d5158d3d1b..91477baabe 100644 --- a/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/entity/EventChecker.java +++ b/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/entity/EventChecker.java @@ -22,7 +22,9 @@ import com.webank.wedatasphere.dss.appconn.eventchecker.execution.EventCheckerExecutionAction; import com.webank.wedatasphere.dss.standard.app.development.listener.common.RefExecutionState; import org.apache.commons.lang3.StringUtils; -import org.apache.log4j.Logger; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + import java.util.HashMap; import java.util.Map; import java.util.Properties; @@ -55,7 +57,7 @@ public class EventChecker implements Runnable{ private static Pattern pattern = Pattern.compile("[a-zA-Z_0-9@\\-]+"); - private static final Logger logger = Logger.getRootLogger(); + private static final org.slf4j.Logger logger = LoggerFactory.getLogger(Logger.ROOT_LOGGER_NAME); public EventChecker(Properties p, EventCheckerExecutionAction action) { this.p = p; diff --git a/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/service/AbstractEventCheck.java b/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/service/AbstractEventCheck.java index e8af842b34..f5c5c488b3 100644 --- a/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/service/AbstractEventCheck.java +++ b/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/service/AbstractEventCheck.java @@ -21,7 +21,7 @@ import com.webank.wedatasphere.dss.appconn.eventchecker.adapter.EventCheckAdapter; import com.webank.wedatasphere.dss.appconn.eventchecker.entity.EventChecker; -import org.apache.log4j.Logger; +import org.slf4j.Logger; import java.net.InetAddress; import java.net.NetworkInterface; diff --git a/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/service/AbstractEventCheckReceiver.java b/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/service/AbstractEventCheckReceiver.java index dd48920fd6..da650d4211 100644 --- a/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/service/AbstractEventCheckReceiver.java +++ b/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/service/AbstractEventCheckReceiver.java @@ -20,7 +20,7 @@ import org.apache.commons.lang3.StringUtils; import org.apache.commons.lang3.time.DateFormatUtils; -import org.apache.log4j.Logger; +import org.slf4j.Logger; import java.sql.Connection; import java.sql.PreparedStatement; diff --git a/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/service/DefaultEventcheckReceiver.java b/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/service/DefaultEventcheckReceiver.java index 9fa17274f6..62ec9f36b7 100644 --- a/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/service/DefaultEventcheckReceiver.java +++ b/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/service/DefaultEventcheckReceiver.java @@ -19,11 +19,12 @@ import org.apache.commons.lang3.time.DateFormatUtils; -import org.apache.log4j.Logger; +import org.slf4j.Logger; import java.text.DateFormat; import java.text.ParseException; import java.text.SimpleDateFormat; +import java.util.Arrays; import java.util.Date; import java.util.Properties; @@ -59,7 +60,7 @@ public boolean reciveMsg(int jobId, Properties props, Logger log) { result = updateMsgOffset(jobId,props,log,consumedMsgInfo,lastMsgId); } }else{ - log.error("executeType error {} " + executeType.toString()); + log.error("executeType error {} " + Arrays.toString(executeType)); return result; } }catch (Exception e){ @@ -113,7 +114,7 @@ private void waitForTime(Logger log,Long waitTime){ String waitForTime = wait_for_time; String formatWaitForTime = DateFormatUtils.format(new Date(),"yyyy-MM-dd " + waitForTime + ":00"); DateFormat fmt =new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); - Date targetWaitTime = null; + Date targetWaitTime = new Date(); try { targetWaitTime = fmt.parse(formatWaitForTime); } catch (ParseException e) { diff --git a/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/service/EventCheckSender.java b/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/service/EventCheckSender.java index 8e21ab72df..c0c3928745 100644 --- a/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/service/EventCheckSender.java +++ b/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/service/EventCheckSender.java @@ -17,7 +17,7 @@ package com.webank.wedatasphere.dss.appconn.eventchecker.service; import org.apache.commons.lang3.time.DateFormatUtils; -import org.apache.log4j.Logger; +import org.slf4j.Logger; import java.sql.Connection; import java.sql.PreparedStatement; diff --git a/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/service/EventCheckerService.java b/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/service/EventCheckerService.java index 2ca27bec30..4583b6ea89 100644 --- a/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/service/EventCheckerService.java +++ b/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/service/EventCheckerService.java @@ -16,12 +16,12 @@ package com.webank.wedatasphere.dss.appconn.eventchecker.service; -import org.apache.log4j.Logger; +import org.slf4j.Logger; import java.util.Properties; public class EventCheckerService { - private static EventCheckerService instance; + private static volatile EventCheckerService instance; public static EventCheckerService getInstance() { if (instance == null) { @@ -35,9 +35,9 @@ public static EventCheckerService getInstance() { } public boolean sendMsg(int jobId, Properties props, Logger log) { - if(props!=null){ - return new EventCheckSender(props).sendMsg(jobId,props,log); - }else{ + if (props != null) { + return new EventCheckSender(props).sendMsg(jobId, props, log); + } else { log.error("create EventCheckSender failed {}"); return false; } @@ -51,9 +51,9 @@ public boolean sendMsg(int jobId, Properties props, Logger log) { * when the set target is not exceeded. */ public boolean reciveMsg(int jobId, Properties props, Logger log) { - if(props!=null){ - return new DefaultEventcheckReceiver(props).reciveMsg(jobId,props,log); - }else{ + if (props != null) { + return new DefaultEventcheckReceiver(props).reciveMsg(jobId, props, log); + } else { log.error("create EventCheckSender failed {}"); return false; } diff --git a/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/utils/Props.java b/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/utils/Props.java index e4e8cc9b8a..e4f133e56d 100644 --- a/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/utils/Props.java +++ b/dss-appconn/appconns/dss-eventchecker-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/eventchecker/utils/Props.java @@ -18,7 +18,7 @@ import com.webank.wedatasphere.dss.appconn.eventchecker.exception.UndefinedPropertyException; -import org.apache.log4j.Logger; +import org.slf4j.Logger; import java.io.BufferedInputStream; import java.io.BufferedOutputStream; diff --git a/dss-appconn/appconns/dss-schedulis-appconn/pom.xml b/dss-appconn/appconns/dss-schedulis-appconn/pom.xml index a7af986cba..23c18b6f97 100644 --- a/dss-appconn/appconns/dss-schedulis-appconn/pom.xml +++ b/dss-appconn/appconns/dss-schedulis-appconn/pom.xml @@ -21,7 +21,7 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 ../../../pom.xml 4.0.0 @@ -35,6 +35,12 @@ com.webank.wedatasphere.dss dss-scheduler-appconn ${dss.version} + + + org.apache.linkis + linkis-common + + @@ -51,6 +57,14 @@ json4s-jackson_2.11 org.json4s + + org.apache.linkis + linkis-gateway-httpclient-support + + + org.apache.linkis + linkis-httpclient + diff --git a/dss-appconn/appconns/dss-schedulis-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/schedulis/SchedulisAppConn.java b/dss-appconn/appconns/dss-schedulis-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/schedulis/SchedulisAppConn.java index 1fba3e4801..ce01b58efa 100644 --- a/dss-appconn/appconns/dss-schedulis-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/schedulis/SchedulisAppConn.java +++ b/dss-appconn/appconns/dss-schedulis-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/schedulis/SchedulisAppConn.java @@ -25,6 +25,7 @@ public class SchedulisAppConn extends AbstractSchedulerAppConn { public static final String SCHEDULIS_APPCONN_NAME = "Schedulis"; + @Override public ConversionIntegrationStandard getOrCreateConversionStandard() { return super.getOrCreateConversionStandard(); } diff --git a/dss-appconn/appconns/dss-schedulis-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/schedulis/conversion/AzkabanWorkflowToRelSynchronizer.java b/dss-appconn/appconns/dss-schedulis-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/schedulis/conversion/AzkabanWorkflowToRelSynchronizer.java index 1ae53f6604..eba775090a 100644 --- a/dss-appconn/appconns/dss-schedulis-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/schedulis/conversion/AzkabanWorkflowToRelSynchronizer.java +++ b/dss-appconn/appconns/dss-schedulis-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/schedulis/conversion/AzkabanWorkflowToRelSynchronizer.java @@ -20,15 +20,20 @@ import com.webank.wedatasphere.dss.appconn.schedulis.entity.AzkabanConvertedRel; import com.webank.wedatasphere.dss.appconn.schedulis.utils.SchedulisHttpUtils; import com.webank.wedatasphere.dss.common.exception.DSSRuntimeException; +import com.webank.wedatasphere.dss.common.utils.DSSCommonUtils; import com.webank.wedatasphere.dss.common.utils.ZipHelper; import com.webank.wedatasphere.dss.orchestrator.converter.standard.operation.DSSToRelConversionOperation; +import com.webank.wedatasphere.dss.orchestrator.converter.standard.ref.ProjectToRelConversionRequestRef; import com.webank.wedatasphere.dss.standard.app.sso.Workspace; import com.webank.wedatasphere.dss.standard.app.sso.origin.request.action.DSSUploadAction; -import com.webank.wedatasphere.dss.standard.app.sso.request.SSORequestService; -import com.webank.wedatasphere.dss.standard.common.desc.AppInstance; +import com.webank.wedatasphere.dss.standard.app.structure.project.ProjectSearchOperation; +import com.webank.wedatasphere.dss.standard.app.structure.project.ref.ProjectResponseRef; +import com.webank.wedatasphere.dss.standard.app.structure.project.ref.RefProjectContentRequestRef; +import com.webank.wedatasphere.dss.standard.common.exception.operation.ExternalOperationFailedException; import com.webank.wedatasphere.dss.workflow.conversion.entity.ConvertedRel; import com.webank.wedatasphere.dss.workflow.conversion.operation.WorkflowToRelSynchronizer; import org.apache.commons.io.IOUtils; +import org.apache.commons.lang.StringUtils; import org.apache.commons.lang.exception.ExceptionUtils; import org.apache.linkis.httpclient.request.BinaryBody; import org.slf4j.Logger; @@ -38,7 +43,11 @@ import java.io.FileInputStream; import java.io.InputStream; import java.util.ArrayList; +import java.util.HashMap; import java.util.List; +import java.util.Map; +import java.util.regex.Matcher; +import java.util.regex.Pattern; public class AzkabanWorkflowToRelSynchronizer implements WorkflowToRelSynchronizer { @@ -46,10 +55,13 @@ public class AzkabanWorkflowToRelSynchronizer implements WorkflowToRelSynchroniz private String projectUrl; private DSSToRelConversionOperation dssToRelConversionOperation; + //匹配wtss返回的错误信息 + private static final Pattern ERROR_PATTERN = Pattern.compile("(?<=Error uploading project properties)[\\s\\S]+.job"); + private static final int SCHEDULIS_MAX_SIZE = 250; public void init() { String baseUrl = dssToRelConversionOperation.getConversionService().getAppInstance().getBaseUrl(); - this.projectUrl = baseUrl.endsWith("/") ? baseUrl + "manager": baseUrl + "/manager"; + this.projectUrl = baseUrl.endsWith("/") ? baseUrl + "manager" : baseUrl + "/manager"; } @Override @@ -62,36 +74,75 @@ public void setDSSToRelConversionOperation(DSSToRelConversionOperation dssToRelC public void syncToRel(ConvertedRel convertedRel) { String tmpSavePath; AzkabanConvertedRel azkabanConvertedRel = (AzkabanConvertedRel) convertedRel; + ProjectToRelConversionRequestRef projectToRelConversionRequestRef = azkabanConvertedRel.getDSSToRelConversionRequestRef(); try { + String projectName = projectToRelConversionRequestRef.getDSSProject().getName(); + //前置检查,若项目在schedulis不存在,直接返回 + SchedulisAppConn schedulisAppConn = (SchedulisAppConn) dssToRelConversionOperation.getConversionService().getAppStandard().getAppConn(); + ProjectSearchOperation projectSearchOperation = schedulisAppConn.getOrCreateStructureStandard() + .getProjectService(dssToRelConversionOperation.getConversionService().getAppInstance()).getProjectSearchOperation(); + ProjectResponseRef responseRef = projectSearchOperation.searchProject(new RefProjectContentRequestRef.RefProjectContentRequestRefImpl() + .setProjectName(projectName).setWorkspace(projectToRelConversionRequestRef.getWorkspace())); + if (responseRef.isFailed()) { + //接口调用返回其他错误,如网络错误 + throw new ExternalOperationFailedException(90012, responseRef.getErrorMsg()); + } + if (responseRef.isSucceed() && responseRef.getRefProjectId() == null) { + //项目在schedulis不存在 + throw new DSSRuntimeException(90012, "the project: " + projectName + " is not exists in schedulis.(工作流对应项目在schedulis已被删除,请在schedulis中重新创建同名项目)"); + } + //项目存在,则继续执行如下步骤 String projectPath = azkabanConvertedRel.getStorePath(); tmpSavePath = ZipHelper.zip(projectPath); //upload zip to Azkaban - uploadProject(azkabanConvertedRel.getDSSToRelConversionRequestRef().getWorkspace(), tmpSavePath, - azkabanConvertedRel.getDSSToRelConversionRequestRef().getDSSProject().getName(), azkabanConvertedRel.getDSSToRelConversionRequestRef().getUserName()); + uploadProject(projectToRelConversionRequestRef.getWorkspace(), tmpSavePath, + projectToRelConversionRequestRef.getDSSProject().getName(), + projectToRelConversionRequestRef.getUserName(), + projectToRelConversionRequestRef.getApprovalId()); } catch (Exception e) { throw new DSSRuntimeException(90012, ExceptionUtils.getRootCauseMessage(e), e); } } - private void uploadProject(Workspace workspace, String tmpSavePath, String projectName, String releaseUser) throws Exception { + private void uploadProject(Workspace workspace, String tmpSavePath, String projectName, String releaseUser, String approvalId) throws Exception { File file = new File(tmpSavePath); InputStream inputStream = new FileInputStream(file); try { - BinaryBody binaryBody = BinaryBody.apply("file",inputStream,file.getName(),"application/zip"); - List binaryBodyList =new ArrayList<>(); + BinaryBody binaryBody = BinaryBody.apply("file", inputStream, file.getName(), "application/zip"); + List binaryBodyList = new ArrayList<>(); binaryBodyList.add(binaryBody); DSSUploadAction uploadAction = new DSSUploadAction(binaryBodyList); uploadAction.getFormParams().put("ajax", "upload"); + uploadAction.getParameters().put("ajax", "upload"); + uploadAction.getFormParams().put("project", projectName); uploadAction.getParameters().put("project", projectName); - uploadAction.getParameters().put("ajax", "upload"); + + if (StringUtils.isNotBlank(approvalId)) { + uploadAction.getFormParams().put("itsmId", approvalId); + uploadAction.getParameters().put("itsmId", approvalId); + } uploadAction.setUrl(projectUrl); - SchedulisHttpUtils.getHttpResult(projectUrl, uploadAction, - dssToRelConversionOperation.getConversionService().getSSORequestService() - .createSSORequestOperation(SchedulisAppConn.SCHEDULIS_APPCONN_NAME), workspace); + String body = + SchedulisHttpUtils.getHttpResult(projectUrl, uploadAction, + dssToRelConversionOperation.getConversionService().getSSORequestService() + .createSSORequestOperation(SchedulisAppConn.SCHEDULIS_APPCONN_NAME), workspace); + if (body != null && DSSCommonUtils.COMMON_GSON.fromJson(body, Map.class).get("error") != null) { + throw new ExternalOperationFailedException(50063, "upload project to schedulis failed." + body); + } + } catch (Exception e) { + throw new DSSRuntimeException(90012, dealSchedulisErrorMsg(ExceptionUtils.getRootCauseMessage(e))); } finally { IOUtils.closeQuietly(inputStream); } } + + private String dealSchedulisErrorMsg(String errorMsg) { + Matcher matcher = ERROR_PATTERN.matcher(errorMsg); + if (matcher.find() && matcher.group().length() >= SCHEDULIS_MAX_SIZE) { + errorMsg = "wokflow name " + matcher.group().split("/")[1] + " is to long, please abide the rules of schedulis: projectName + workflowName*3 + 12 <= 250 "; + } + return errorMsg; + } } diff --git a/dss-appconn/appconns/dss-schedulis-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/schedulis/conversion/ProjectInfoWorkflowToRelConverter.java b/dss-appconn/appconns/dss-schedulis-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/schedulis/conversion/ProjectInfoWorkflowToRelConverter.java index 87df14a570..dcadb03150 100644 --- a/dss-appconn/appconns/dss-schedulis-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/schedulis/conversion/ProjectInfoWorkflowToRelConverter.java +++ b/dss-appconn/appconns/dss-schedulis-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/schedulis/conversion/ProjectInfoWorkflowToRelConverter.java @@ -51,7 +51,7 @@ public ConvertedRel convertToRel(PreConversionRel rel) { String repeatNodes = nodeNames.stream().collect(Collectors.groupingBy(Function.identity(), Collectors.counting())) .entrySet().stream().filter(entry -> entry.getValue() > 1).map(Map.Entry::getKey).collect(Collectors.joining(", ")); if (StringUtils.isNotEmpty(repeatNodes)) { - throw new DSSRuntimeException(80001, "重复的节点名称:" + repeatNodes); + throw new DSSRuntimeException(80001, "重复的节点名称。项目中不同工作流(或子工作流)里存在重名节点,请修改节点名避免重名。重名节点:" + repeatNodes); } AzkabanConvertedRel azkabanConvertedRel = new AzkabanConvertedRel(projectPreConversionRel); //1. Assign a value to the storepath of azkabanschedulerproject. @@ -93,7 +93,12 @@ private void removeProjectStoreDirAndZip(AzkabanConvertedRel rel) { File zipFile = new File(projectZip); if (zipFile.exists()) { LOGGER.info("exist project zip{} before publish ,now remove it", projectZip); - zipFile.delete(); + boolean flag = zipFile.delete(); + if(flag){ + LOGGER.info("zip file delete success!"); + }else { + LOGGER.info("zip file delete failed!"); + } } } catch (Exception e) { LOGGER.error("delete project dir or zip failed,reaseon:", e); diff --git a/dss-appconn/appconns/dss-schedulis-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/schedulis/operation/SchedulisProjectCreationOperation.java b/dss-appconn/appconns/dss-schedulis-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/schedulis/operation/SchedulisProjectCreationOperation.java index c01ca93d6f..dfe4206404 100644 --- a/dss-appconn/appconns/dss-schedulis-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/schedulis/operation/SchedulisProjectCreationOperation.java +++ b/dss-appconn/appconns/dss-schedulis-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/schedulis/operation/SchedulisProjectCreationOperation.java @@ -42,6 +42,8 @@ public class SchedulisProjectCreationOperation private String managerUrl; + public static final int WTSS_MAX_PROJECT_NAME_SIZE = 64; + @Override protected String getAppConnName() { return SchedulisAppConn.SCHEDULIS_APPCONN_NAME; @@ -57,11 +59,14 @@ public void init() { @Override public ProjectResponseRef createProject(DSSProjectContentRequestRef.DSSProjectContentRequestRefImpl requestRef) throws ExternalOperationFailedException { logger.info("begin to create project in schedulis, project name is {}.", requestRef.getDSSProject().getName()); + if (requestRef.getDSSProject().getName().length() > WTSS_MAX_PROJECT_NAME_SIZE) { + throw new ExternalOperationFailedException(60021, "project name is too long, it must be less then " + WTSS_MAX_PROJECT_NAME_SIZE + " in schedulis! "); + } if (CollectionUtils.isNotEmpty(requestRef.getDSSProjectPrivilege().getReleaseUsers())) { // 先校验运维用户是否存在于 Schedulis,如果不存在,则不能成功创建工程。 requestRef.getDSSProjectPrivilege().getReleaseUsers().forEach(releaseUser -> { if (!AzkabanUserService.containsUser(releaseUser, getBaseUrl(), ssoRequestOperation, requestRef.getWorkspace())) { - throw new ExternalOperationFailedException(100323, "当前设置的发布用户: " + releaseUser + ", 在 Schedulis 系统中不存在,请联系 Schedulis 管理员创建该用户!"); + throw new ExternalOperationFailedException(100323, "当前设置的发布用户: " + releaseUser + ", 在 Schedulis 系统中不存在,请根据'查看解决方案'中给出的指引在ITSM上提单!"); } }); } @@ -71,7 +76,7 @@ public ProjectResponseRef createProject(DSSProjectContentRequestRef.DSSProjectCo params.put("description", requestRef.getDSSProject().getDescription()); try { String entStr = SchedulisHttpUtils.getHttpPostResult(projectUrl, params, ssoRequestOperation, requestRef.getWorkspace()); - logger.error("新建工程 {}, Schedulis 返回的信息是 {}.", requestRef.getName(), entStr); + logger.info("新建工程 {}, Schedulis 返回的信息是 {}.", requestRef.getDSSProject().getName(), entStr); String message = AzkabanUtils.handleAzkabanEntity(entStr); if (!"success".equals(message)) { throw new ExternalOperationFailedException(90008, "Schedulis 新建工程失败, 原因: " + message); diff --git a/dss-appconn/appconns/dss-schedulis-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/schedulis/operation/SchedulisProjectUpdateOperation.java b/dss-appconn/appconns/dss-schedulis-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/schedulis/operation/SchedulisProjectUpdateOperation.java index e2195cdf8a..99bb26c686 100644 --- a/dss-appconn/appconns/dss-schedulis-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/schedulis/operation/SchedulisProjectUpdateOperation.java +++ b/dss-appconn/appconns/dss-schedulis-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/schedulis/operation/SchedulisProjectUpdateOperation.java @@ -47,7 +47,7 @@ public ResponseRef updateProject(ProjectUpdateRequestRef.ProjectUpdateRequestRef // 先校验运维用户是否存在于 Schedulis,如果不存在,则不能成功创建工程。 projectRef.getDSSProjectPrivilege().getReleaseUsers().forEach(releaseUser -> { if (!AzkabanUserService.containsUser(releaseUser, getBaseUrl(), ssoRequestOperation, projectRef.getWorkspace())) { - throw new ExternalOperationFailedException(100323, "当前设置的发布用户: " + releaseUser + ", 在 Schedulis 系统中不存在,请联系 Schedulis 管理员创建该用户!"); + throw new ExternalOperationFailedException(100323, "当前设置的发布用户: " + releaseUser + ", 在 Schedulis 系统中不存在,请根据'查看解决方案'中给出的指引在ITSM上提单!"); } }); } diff --git a/dss-appconn/appconns/dss-schedulis-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/schedulis/service/AzkabanUserService.java b/dss-appconn/appconns/dss-schedulis-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/schedulis/service/AzkabanUserService.java index da9f85a1af..34973da195 100644 --- a/dss-appconn/appconns/dss-schedulis-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/schedulis/service/AzkabanUserService.java +++ b/dss-appconn/appconns/dss-schedulis-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/schedulis/service/AzkabanUserService.java @@ -26,10 +26,7 @@ import org.slf4j.Logger; import org.slf4j.LoggerFactory; -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; -import java.util.Map; +import java.util.*; import java.util.concurrent.ConcurrentHashMap; import java.util.function.Supplier; import java.util.stream.Collectors; @@ -64,9 +61,17 @@ private static void requestUserId(String baseUrl, SSORequestOperation ssoRequest } } List entityList = schedulisUserMap.get(baseUrl); - List newEntityList = ((List) map.get("systemUserList")).stream().map(e -> - DSSCommonUtils.COMMON_GSON.fromJson(e.toString(), AzkabanUserEntity.class) - ).collect(Collectors.toList()); + List newEntityList = ((List) map.get("systemUserList")).stream().map(e -> { + AzkabanUserEntity userEntity; + try { + userEntity = DSSCommonUtils.COMMON_GSON.fromJson(e.toString(), AzkabanUserEntity.class); + } catch (Exception ex) { + LOGGER.warn("AzkabanUserEntity: {} parsed from json failed!", e.toString()); + userEntity = null; + } + return userEntity; + } + ).filter(Objects::nonNull).collect(Collectors.toList()); synchronized (entityList) { entityList.clear(); entityList.addAll(newEntityList); @@ -79,7 +84,7 @@ private static void requestUserId(String baseUrl, SSORequestOperation ssoRequest } public static boolean containsUser(String releaseUser, String baseUrl, - SSORequestOperation ssoRequestOperation, Workspace workspace) { + SSORequestOperation ssoRequestOperation, Workspace workspace) { Supplier supplier = () -> schedulisUserMap.containsKey(baseUrl) && schedulisUserMap.get(baseUrl).stream().anyMatch(entity -> entity.getUsername().equals(releaseUser)); if (!supplier.get()) { @@ -89,8 +94,8 @@ public static boolean containsUser(String releaseUser, String baseUrl, } public static String getUserId(String user, String baseUrl, - SSORequestOperation ssoRequestOperation, Workspace workspace) { - if(containsUser(user, baseUrl, ssoRequestOperation, workspace)) { + SSORequestOperation ssoRequestOperation, Workspace workspace) { + if (containsUser(user, baseUrl, ssoRequestOperation, workspace)) { return schedulisUserMap.get(baseUrl).stream().filter(entity -> entity.getUsername().equals(user)).findAny().get().getId(); } else { throw new ExternalOperationFailedException(10823, "Not exists user in Schedulis " + user); diff --git a/dss-appconn/appconns/dss-schedulis-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/schedulis/utils/SchedulisHttpUtils.java b/dss-appconn/appconns/dss-schedulis-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/schedulis/utils/SchedulisHttpUtils.java index 12cc31e7e2..37b76c17a1 100644 --- a/dss-appconn/appconns/dss-schedulis-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/schedulis/utils/SchedulisHttpUtils.java +++ b/dss-appconn/appconns/dss-schedulis-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/schedulis/utils/SchedulisHttpUtils.java @@ -16,6 +16,7 @@ package com.webank.wedatasphere.dss.appconn.schedulis.utils; +import com.webank.wedatasphere.dss.common.utils.DSSCommonUtils; import com.webank.wedatasphere.dss.standard.app.sso.Workspace; import com.webank.wedatasphere.dss.standard.app.sso.builder.SSOUrlBuilderOperation; import com.webank.wedatasphere.dss.standard.app.sso.origin.request.action.DSSGetAction; @@ -53,10 +54,11 @@ public static String getHttpResult(String url, action.setUrl(ssoUrlBuilderOperation.getBuiltUrl()); HttpResult previewResult = ssoRequestOperation.requestWithSSO(ssoUrlBuilderOperation, action); if (previewResult.getStatusCode() == 200 || previewResult.getStatusCode() == 0) { + logger.info("request Schedulis success, responseBody is {}", DSSCommonUtils.COMMON_GSON.toJson(previewResult)); return previewResult.getResponseBody(); } else { logger.error("request Schedulis failed, responseBody is {}.", previewResult.getResponseBody()); - throw new ExternalOperationFailedException(50063, "request Schedulis failed."); + throw new ExternalOperationFailedException(50063, "request Schedulis failed." + previewResult.getResponseBody()); } } diff --git a/dss-appconn/appconns/dss-schedulis-appconn/src/main/resources/uninstall.sql b/dss-appconn/appconns/dss-schedulis-appconn/src/main/resources/uninstall.sql new file mode 100644 index 0000000000..724cc848ad --- /dev/null +++ b/dss-appconn/appconns/dss-schedulis-appconn/src/main/resources/uninstall.sql @@ -0,0 +1,6 @@ +select @appconnId:=id from `dss_appconn` where `appconn_name` = 'schedulis'; +delete from `dss_appconn_instance` where `appconn_id` = @appconnId; + +delete from dss_appconn where appconn_name = "schedulis"; + +delete from dss_workspace_menu_appconn where title_en = "Schedulis"; diff --git a/dss-appconn/appconns/dss-scriptis-appconn/pom.xml b/dss-appconn/appconns/dss-scriptis-appconn/pom.xml index fef31c8f2f..a7e4a0d2d4 100644 --- a/dss-appconn/appconns/dss-scriptis-appconn/pom.xml +++ b/dss-appconn/appconns/dss-scriptis-appconn/pom.xml @@ -21,7 +21,7 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 ../../../pom.xml 4.0.0 diff --git a/dss-appconn/appconns/dss-sendemail-appconn/pom.xml b/dss-appconn/appconns/dss-sendemail-appconn/pom.xml index 780c124bd2..538913077b 100644 --- a/dss-appconn/appconns/dss-sendemail-appconn/pom.xml +++ b/dss-appconn/appconns/dss-sendemail-appconn/pom.xml @@ -21,8 +21,8 @@ dss com.webank.wedatasphere.dss - 1.1.1 - ../../pom.xml + 1.1.2 + ../../../pom.xml 4.0.0 diff --git a/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/pom.xml b/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/pom.xml index 394da4a185..4c80160b45 100644 --- a/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/pom.xml +++ b/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/pom.xml @@ -21,8 +21,8 @@ dss com.webank.wedatasphere.dss - 1.1.1 - ../pom.xml + 1.1.2 + ../../../../pom.xml 4.0.0 @@ -33,6 +33,20 @@ dss-appconn-core ${dss.version} compile + + + org.apache.linkis + linkis-httpclient + + + org.apache.linkis + linkis-common + + + org.apache.linkis + linkis-gateway-httpclient-support + + @@ -40,6 +54,56 @@ dss-common ${dss.version} compile + + + org.apache.linkis + linkis-mybatis + + + org.apache.linkis + linkis-cs-common + + + org.apache.linkis + linkis-gateway-authentication + + + org.apache.linkis + linkis-gateway-core + + + org.apache.linkis + linkis-gateway-server-support + + + org.apache.linkis + linkis-gateway-httpclient-support + + + org.apache.linkis + linkis-httpclient + + + org.apache.linkis + linkis-instance-label-server + + + org.apache.linkis + linkis-rpc + + + org.apache.linkis + linkis-common + + + org.apache.linkis + linkis-protocol + + + org.apache.linkis + linkis-label-common + + @@ -67,7 +131,12 @@ org.springframework spring-context-support - 5.2.5.RELEASE + ${spring-framework.version} + + + org.springframework + spring-aop + ${spring-framework.version} javax.mail @@ -79,11 +148,12 @@ org.apache.linkis linkis-cs-client ${linkis.version} + provided org.apache.httpcomponents httpclient - 4.5.13 + 4.5.4 compile diff --git a/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/java/com/webank/wedatasphere/dss/appconn/sendemail/emailcontent/HtmlItem.java b/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/java/com/webank/wedatasphere/dss/appconn/sendemail/emailcontent/HtmlItem.java new file mode 100644 index 0000000000..ad8a6680b1 --- /dev/null +++ b/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/java/com/webank/wedatasphere/dss/appconn/sendemail/emailcontent/HtmlItem.java @@ -0,0 +1,59 @@ +package com.webank.wedatasphere.dss.appconn.sendemail.emailcontent; + +public class HtmlItem { + + // 文件名 + private String fileName; + + // 文件类型 + private String fileType; + + // 内容id + private String contentId; + + // 内容类型 + private String contentType; + + // 内容 + private String content; + + public String getFileName() { + return fileName; + } + + public void setFileName(String fileName) { + this.fileName = fileName; + } + + public String getFileType() { + return fileType; + } + + public void setFileType(String fileType) { + this.fileType = fileType; + } + + public String getContentId() { + return contentId; + } + + public void setContentId(String contentId) { + this.contentId = contentId; + } + + public String getContentType() { + return contentType; + } + + public void setContentType(String contentType) { + this.contentType = contentType; + } + + public String getContent() { + return content; + } + + public void setContent(String content) { + this.content = content; + } +} diff --git a/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/java/com/webank/wedatasphere/dss/appconn/sendemail/hook/HttpClientUtil.java b/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/java/com/webank/wedatasphere/dss/appconn/sendemail/hook/HttpClientUtil.java index f2bf89f9de..1d0783d558 100644 --- a/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/java/com/webank/wedatasphere/dss/appconn/sendemail/hook/HttpClientUtil.java +++ b/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/java/com/webank/wedatasphere/dss/appconn/sendemail/hook/HttpClientUtil.java @@ -36,14 +36,10 @@ import org.apache.http.util.EntityUtils; import org.slf4j.Logger; import org.slf4j.LoggerFactory; - -import javax.net.ssl.TrustManager; -import javax.net.ssl.X509TrustManager; import java.io.IOException; import java.io.UnsupportedEncodingException; import java.net.SocketTimeoutException; import java.net.URLEncoder; -import java.security.cert.CertificateException; import java.text.SimpleDateFormat; import java.util.*; @@ -54,21 +50,6 @@ public final class HttpClientUtil { private static PoolingHttpClientConnectionManager connManager = null; private static CloseableHttpClient httpclient = null; - private static TrustManager trustAllManager = new X509TrustManager() { - @Override - public void checkClientTrusted(java.security.cert.X509Certificate[] arg0, String arg1) - throws CertificateException { - } - @Override - public void checkServerTrusted(java.security.cert.X509Certificate[] arg0, String arg1) - throws CertificateException { - } - @Override - public java.security.cert.X509Certificate[] getAcceptedIssuers() { - return null; - } - - }; static { httpclient = HttpClients.createDefault(); diff --git a/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/SendEmailRefExecutionOperation.scala b/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/SendEmailRefExecutionOperation.scala index d4cae247a9..8ff5b59d84 100644 --- a/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/SendEmailRefExecutionOperation.scala +++ b/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/SendEmailRefExecutionOperation.scala @@ -53,16 +53,14 @@ class SendEmailRefExecutionOperation val email = Utils.tryCatch { sendEmailAppConnHooks.foreach(_.preGenerate(requestRef)) val email = emailGenerator.generateEmail(requestRef) - emailContentParsers.foreach{ - p => Utils.tryQuietly(p.parse(email)) - } + emailContentParsers.foreach(_.parse(email)) emailContentGenerators.foreach{ g => Utils.tryQuietly(g.generate(email)) } sendEmailAppConnHooks.foreach(_.preSend(requestRef, email)) email }{ t => - return putErrorMsg("解析邮件内容失败!", t) + return putErrorMsg(t.getMessage, t) } Utils.tryCatch { emailSender.send(email) diff --git a/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/conf/SendEmailAppConnConfiguration.scala b/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/conf/SendEmailAppConnConfiguration.scala index cdd2f37287..2dc58e06c1 100644 --- a/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/conf/SendEmailAppConnConfiguration.scala +++ b/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/conf/SendEmailAppConnConfiguration.scala @@ -25,9 +25,13 @@ object SendEmailAppConnConfiguration { val EMAIL_HOOK_CLASSES = CommonVars("wds.dss.appconn.email.hook.classes", "com.webank.wedatasphere.dss.appconn.sendemail.hook.SendEmailItsmCheckHook," + + "com.webank.wedatasphere.dss.appconn.sendemail.hook.SendEmailCheckHook," + "com.webank.wedatasphere.dss.appconn.sendemail.hook.SendEmailVisualisContentLimitHook," + - "com.webank.wedatasphere.dss.appconn.sendemail.hook.SendEmailTableauCheckHook") + "com.webank.wedatasphere.dss.appconn.sendemail.hook.SendEmailTableauCheckHook," + + "com.webank.wedatasphere.dss.appconn.sendemail.hook.SendEmailMetaBaseCheckHook") + val EMAIL_IMAGE_MAXSIZE = CommonVars("wds.dss.appconn.email.image.maxsize", 5000*30000) + val CHECK_EMAIL_IMAGE_SWITCH = CommonVars("wds.dss.appconn.email.image.check", true) val EMAIL_IMAGE_HEIGHT = CommonVars("wds.dss.appconn.email.image.height", 500) val EMAIL_IMAGE_WIDTH = CommonVars("wds.dss.appconn.email.image.width", 1920) val DEFAULT_EMAIL_FROM = CommonVars("wds.dss.appconn.email.from.default", "") @@ -45,6 +49,8 @@ object SendEmailAppConnConfiguration { val EMAIL_SMTP_SSL_ENABLED = CommonVars("wds.dss.appconn.email.smtp.ssl.enable", "true") val EMAIL_SMTP_TIMEOUT: CommonVars[Integer] = CommonVars("wds.dss.appconn.email.smtp.timeout", 25000) - val EMAIL_ENTITY_CLASSES = CommonVars("wds.dss.appconn.email.hook.entity.classes","com.webank.wedatasphere.dss.appconn.sendemail.hook.entity.newvisualis.NewVisualisEmailInfo," + - "com.webank.wedatasphere.dss.appconn.sendemail.hook.entity.visualis.VisualisEmailInfo") + val EMAIL_ENTITY_CLASSES = CommonVars("wds.dss.appconn.email.hook.entity.classes", + "com.webank.wedatasphere.dss.appconn.sendemail.hook.entity.newvisualis.NewVisualisEmailInfo," + + "com.webank.wedatasphere.dss.appconn.sendemail.hook.entity.visualis.VisualisEmailInfo," + + "com.webank.wedatasphere.dss.appconn.sendemail.hook.entity.visualis.MetaBaseEmailInfo") } diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/aop/DSSUWorkspaceIdCheckerAspect.java b/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/email/domain/CsvAttachment.scala similarity index 64% rename from dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/aop/DSSUWorkspaceIdCheckerAspect.java rename to dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/email/domain/CsvAttachment.scala index 3f09b9f162..56857c52d3 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/aop/DSSUWorkspaceIdCheckerAspect.java +++ b/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/email/domain/CsvAttachment.scala @@ -14,16 +14,18 @@ * */ -package com.webank.wedatasphere.dss.framework.workspace.aop; +package com.webank.wedatasphere.dss.appconn.sendemail.email.domain -import org.aspectj.lang.annotation.Aspect; -import org.springframework.stereotype.Component; +import java.io.File +class CsvAttachment(name: String, b64: String) extends Attachment { -@Aspect -@Component -public class DSSUWorkspaceIdCheckerAspect { + override def getName: String = name + override def getBase64Str: String = b64 + override def getFile: File = null -} + override def getMediaType: String = "text/csv" + +} \ No newline at end of file diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/aop/DSSPrivCheckerAspect.java b/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/email/domain/PdfAttachment.scala similarity index 62% rename from dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/aop/DSSPrivCheckerAspect.java rename to dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/email/domain/PdfAttachment.scala index ff8e4b8188..1bea5f5e66 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/aop/DSSPrivCheckerAspect.java +++ b/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/email/domain/PdfAttachment.scala @@ -14,15 +14,18 @@ * */ -package com.webank.wedatasphere.dss.framework.workspace.aop; +package com.webank.wedatasphere.dss.appconn.sendemail.email.domain -import org.aspectj.lang.annotation.Aspect; -import org.springframework.stereotype.Component; +import java.io.File +class PdfAttachment(name: String, b64: String) extends Attachment { -@Aspect -@Component -public class DSSPrivCheckerAspect { + override def getName: String = name + override def getBase64Str: String = b64 -} + override def getFile: File = null //TODO write b64 to file + + override def getMediaType: String = "application/pdf" + +} \ No newline at end of file diff --git a/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/email/generate/AbstractEmailGenerator.scala b/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/email/generate/AbstractEmailGenerator.scala index 556303371e..a39ac1b07e 100644 --- a/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/email/generate/AbstractEmailGenerator.scala +++ b/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/email/generate/AbstractEmailGenerator.scala @@ -18,10 +18,9 @@ package com.webank.wedatasphere.dss.appconn.sendemail.email.generate import com.webank.wedatasphere.dss.appconn.sendemail.email.domain.AbstractEmail import com.webank.wedatasphere.dss.appconn.sendemail.email.{Email, EmailGenerator} -import com.webank.wedatasphere.dss.common.utils.VariableUtils import com.webank.wedatasphere.dss.standard.app.development.listener.core.ExecutionRequestRefContext import com.webank.wedatasphere.dss.standard.app.development.listener.ref.RefExecutionRequestRef -import org.apache.linkis.common.utils.Logging +import org.apache.linkis.common.utils.{Logging, VariableUtils} trait AbstractEmailGenerator extends EmailGenerator with Logging{ diff --git a/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/email/generate/MultiContentEmailGenerator.scala b/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/email/generate/MultiContentEmailGenerator.scala index 7043d54d55..cd3b82fdbb 100644 --- a/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/email/generate/MultiContentEmailGenerator.scala +++ b/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/email/generate/MultiContentEmailGenerator.scala @@ -19,13 +19,19 @@ package com.webank.wedatasphere.dss.appconn.sendemail.email.generate import com.google.gson.internal.LinkedTreeMap import com.webank.wedatasphere.dss.appconn.sendemail.cs.EmailCSHelper import com.webank.wedatasphere.dss.appconn.sendemail.email.domain.{AbstractEmail, MultiContentEmail} -import com.webank.wedatasphere.dss.appconn.sendemail.emailcontent.domain.PictureEmailContent +import com.webank.wedatasphere.dss.appconn.sendemail.emailcontent.domain.{HtmlEmailContent, PictureEmailContent} import com.webank.wedatasphere.dss.appconn.sendemail.exception.EmailSendFailedException import com.webank.wedatasphere.dss.common.utils.DSSCommonUtils import com.webank.wedatasphere.dss.standard.app.development.listener.ref.RefExecutionRequestRef import org.apache.commons.lang3.StringUtils +import org.apache.linkis.common.log.LogUtils +import org.apache.linkis.common.utils.Utils +import org.apache.linkis.server.JSONUtils +import org.apache.linkis.storage.LineMetaData import org.apache.linkis.storage.resultset.ResultSetFactory +import java.util + class MultiContentEmailGenerator extends AbstractEmailGenerator { override protected def createEmail(): AbstractEmail = new MultiContentEmail @@ -39,12 +45,26 @@ class MultiContentEmailGenerator extends AbstractEmailGenerator { val resultSetFactory = ResultSetFactory.getInstance EmailCSHelper.getJobIds(refContext).foreach { jobId => refContext.fetchLinkisJobResultSetPaths(jobId).foreach { fsPath => + var fileType = ""; + val reader = refContext.getResultSetReader(fsPath) + val meta = reader.getMetaData.cloneMeta() + Utils.tryFinally(meta match { + case metadata: LineMetaData => + val data = JSONUtils.gson.fromJson(metadata.getMetaData, classOf[util.Map[String, String]]) + // 如果是pdf附件存字段pdf到multiContentEmail + if (data.get("type") != null && data.get("format") != null) { + fileType = data.get("format").toString + } + case _ => + })(Utils.tryQuietly(reader.close())) val resultSet = resultSetFactory.getResultSetByPath(fsPath) val emailContent = resultSet.resultSetType() match { - case ResultSetFactory.PICTURE_TYPE => new PictureEmailContent(fsPath) - case ResultSetFactory.HTML_TYPE => throw new EmailSendFailedException(80003 ,"html result set is not allowed")//new HtmlEmailContent(fsPath) - case ResultSetFactory.TABLE_TYPE => throw new EmailSendFailedException(80003 ,"table result set is not allowed")//new TableEmailContent(fsPath) - case ResultSetFactory.TEXT_TYPE => throw new EmailSendFailedException(80003 ,"text result set is not allowed")//new FileEmailContent(fsPath) + case ResultSetFactory.PICTURE_TYPE => new PictureEmailContent(fsPath, fileType) + case ResultSetFactory.HTML_TYPE => + multiContentEmail.setEmailType("html") + new HtmlEmailContent (fsPath, fileType) + case ResultSetFactory.TABLE_TYPE => throw new EmailSendFailedException(80003 ,"table result set is not allowed") + case ResultSetFactory.TEXT_TYPE => throw new EmailSendFailedException(80003 ,"text result set is not allowed") } multiContentEmail.addEmailContent(emailContent) } @@ -53,13 +73,9 @@ class MultiContentEmailGenerator extends AbstractEmailGenerator { multiContentEmail.setEmailType(emailType) } } - case "file" => throw new EmailSendFailedException(80003 ,"file content is not allowed") //addContentEmail(c => new FileEmailContent(new FsPath(c))) - case "text" => throw new EmailSendFailedException(80003 ,"text content is not allowed")//addContentEmail(new TextEmailContent(_)) - case "link" => throw new EmailSendFailedException(80003 ,"link content is not allowed")//addContentEmail(new UrlEmailContent(_)) + case "file" => throw new EmailSendFailedException(80003 ,"file content is not allowed") + case "text" => throw new EmailSendFailedException(80003 ,"text content is not allowed") + case "link" => throw new EmailSendFailedException(80003 ,"link content is not allowed") } } - - - - } \ No newline at end of file diff --git a/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/emailcontent/domain/FsPathStoreEmailContent.scala b/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/emailcontent/domain/FsPathStoreEmailContent.scala index ebbe876fcf..a90d10b561 100644 --- a/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/emailcontent/domain/FsPathStoreEmailContent.scala +++ b/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/emailcontent/domain/FsPathStoreEmailContent.scala @@ -20,7 +20,11 @@ import org.apache.linkis.common.io.FsPath trait FsPathStoreEmailContent { private var fsPath: FsPath = _ + private var fileType: String = _ def getFsPath: FsPath = fsPath def setFsPath(fsPath: FsPath): Unit = this.fsPath = fsPath + + def getFileType: String = fileType + def setFileType(fileType: String): Unit = this.fileType = fileType } \ No newline at end of file diff --git a/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/emailcontent/domain/package.scala b/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/emailcontent/domain/package.scala index 016398556f..c85c7c8870 100644 --- a/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/emailcontent/domain/package.scala +++ b/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/emailcontent/domain/package.scala @@ -31,13 +31,20 @@ class PictureEmailContent extends ArrayEmailContent with FsPathStoreEmailContent setFsPath(filePath) } + def this(filePath: FsPath, fileType: String) = { + this() + setFsPath(filePath) + setFileType(fileType) + } + } class HtmlEmailContent extends StringEmailContent with FsPathStoreEmailContent { - def this(filePath: FsPath) = { + def this(filePath: FsPath, fileType: String) = { this() setFsPath(filePath) + setFileType(fileType) } } diff --git a/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/emailcontent/generator/MultiEmailContentGenerator.scala b/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/emailcontent/generator/MultiEmailContentGenerator.scala index af40d314cd..2d1ea7fd38 100644 --- a/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/emailcontent/generator/MultiEmailContentGenerator.scala +++ b/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/emailcontent/generator/MultiEmailContentGenerator.scala @@ -28,7 +28,11 @@ class MultiEmailContentGenerator extends AbstractEmailContentGenerator with Logg case multiContentEmail: MultiContentEmail => formatSubjectOfOldVersion(email) formatSubject(multiContentEmail) - formatContent(multiContentEmail) + if (multiContentEmail.getEmailType.equals("html")) { + setHtmlContent(multiContentEmail) + } else { + formatContent(multiContentEmail) + } } protected def formatContent(email: MultiContentEmail): Unit = { @@ -36,7 +40,9 @@ class MultiEmailContentGenerator extends AbstractEmailContentGenerator with Logg sb.append("") email.getEmailContents.foreach { case emailContent: ArrayEmailContent => - emailContent.getContent.foreach(content => sb.append("")) + if (emailContent.getContent != null) { + emailContent.getContent.foreach(content => sb.append("")) + } case emailContent: StringEmailContent => sb.append("") } @@ -45,4 +51,13 @@ class MultiEmailContentGenerator extends AbstractEmailContentGenerator with Logg email.setContent(sb.toString) } + protected def setHtmlContent(email: MultiContentEmail): Unit = { + email.getEmailContents.foreach { + case emailContent: StringEmailContent => + if (emailContent.getContent != null) { + email.setContent(emailContent.getContent) + } + } + } + } \ No newline at end of file diff --git a/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/emailcontent/parser/AbstractEmailContentParser.scala b/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/emailcontent/parser/AbstractEmailContentParser.scala index 2bbcb6c217..92bfaec8f6 100644 --- a/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/emailcontent/parser/AbstractEmailContentParser.scala +++ b/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/emailcontent/parser/AbstractEmailContentParser.scala @@ -26,7 +26,7 @@ import org.apache.linkis.common.io.resultset.ResultSetReader import org.apache.linkis.common.io.{MetaData, Record} import org.apache.linkis.common.utils.Utils import org.apache.linkis.storage.LineRecord -import org.apache.linkis.storage.resultset.ResultSetReader +import org.apache.linkis.storage.resultset.ResultSetReaderFactory import org.apache.commons.io.IOUtils abstract class AbstractEmailContentParser[T] extends EmailContentParser { @@ -41,8 +41,8 @@ abstract class AbstractEmailContentParser[T] extends EmailContentParser { case _ => } - protected def getResultSetReader(fsPathStore: FsPathStoreEmailContent): ResultSetReader[_ <: MetaData, _ <: Record] = { - val reader = ResultSetReader.getResultSetReader(fsPathStore.getFsPath.getSchemaPath) + protected def getResultSetReader(fsPathStore: FsPathStoreEmailContent): ResultSetReader[_, _ ] = { + val reader = ResultSetReaderFactory.getResultSetReader(fsPathStore.getFsPath.getSchemaPath) reader.getMetaData reader } diff --git a/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/emailcontent/parser/HtmlEmailContentParser.scala b/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/emailcontent/parser/HtmlEmailContentParser.scala index cab083f70c..d2b5520d88 100644 --- a/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/emailcontent/parser/HtmlEmailContentParser.scala +++ b/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/emailcontent/parser/HtmlEmailContentParser.scala @@ -16,12 +16,40 @@ package com.webank.wedatasphere.dss.appconn.sendemail.emailcontent.parser -import com.webank.wedatasphere.dss.appconn.sendemail.email.domain.MultiContentEmail +import com.webank.wedatasphere.dss.appconn.sendemail.email.domain.{CsvAttachment, MultiContentEmail, PngAttachment} +import com.webank.wedatasphere.dss.appconn.sendemail.emailcontent.HtmlItem import com.webank.wedatasphere.dss.appconn.sendemail.emailcontent.domain.HtmlEmailContent +import org.apache.linkis.server.JSONUtils + +import java.nio.charset.StandardCharsets +import java.util.Base64 +import scala.sys.error object HtmlEmailContentParser extends AbstractEmailContentParser[HtmlEmailContent] { override protected def parseEmailContent(emailContent: HtmlEmailContent, multiContentEmail: MultiContentEmail): Unit = { - getFirstLineRecord(emailContent).foreach(emailContent.setContent) + getFirstLineRecord(emailContent).foreach(htmlStr => + emailContent.getFileType match { + case "html" => + val htmlItems: Array[HtmlItem] = JSONUtils.gson.fromJson(htmlStr, classOf[Array[HtmlItem]]) + htmlItems.foreach { + case htmlItem: HtmlItem => + if (htmlItem.getFileType.equals("attachment") && htmlItem.getContentType.equals("csv")) { + val csvName = htmlItem.getFileName + val csvContent = htmlItem.getContent + val csvContentBytes = csvContent.getBytes + val csvBase64BaseContent = new String(Base64.getEncoder.encode(csvContentBytes), StandardCharsets.UTF_8) + multiContentEmail.addAttachment(new CsvAttachment(csvName, csvBase64BaseContent)) + } else if (htmlItem.getContentType.equals("image") && htmlItem.getFileType.equals("inline")) { + multiContentEmail.addAttachment(new PngAttachment(htmlItem.getContentId, htmlItem.getContent)) + } else if (htmlItem.getContentType.equals("html")) { + emailContent.setContent(htmlItem.getContent) + } else { + error("unknow content type: " + emailContent.getFileType) + } + } + case _ => + } + ) } } diff --git a/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/emailcontent/parser/PictureEmailContentParser.scala b/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/emailcontent/parser/PictureEmailContentParser.scala index 5162486081..6a39b16a14 100644 --- a/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/emailcontent/parser/PictureEmailContentParser.scala +++ b/dss-appconn/appconns/dss-sendemail-appconn/sendemail-appconn-core/src/main/scala/com/webank/wedatasphere/dss/appconn/sendemail/emailcontent/parser/PictureEmailContentParser.scala @@ -17,36 +17,82 @@ package com.webank.wedatasphere.dss.appconn.sendemail.emailcontent.parser import java.awt.image.BufferedImage -import java.io.{ByteArrayInputStream, ByteArrayOutputStream} -import java.util.{Base64, UUID} +import java.io.{ByteArrayInputStream, ByteArrayOutputStream, InputStream} +import java.util +import java.util.{Base64, Iterator, UUID} -import com.webank.wedatasphere.dss.appconn.sendemail.email.domain.{AbstractEmail, MultiContentEmail, PngAttachment} +import com.webank.wedatasphere.dss.appconn.sendemail.email.domain.{AbstractEmail, MultiContentEmail, PdfAttachment, PngAttachment} import com.webank.wedatasphere.dss.appconn.sendemail.emailcontent.domain.PictureEmailContent import org.apache.linkis.common.conf.Configuration -import javax.imageio.ImageIO +import javax.imageio.{ImageIO, ImageReader} import org.apache.commons.codec.binary.Base64OutputStream import com.webank.wedatasphere.dss.appconn.sendemail.conf.SendEmailAppConnConfiguration._ +import com.webank.wedatasphere.dss.appconn.sendemail.exception.EmailSendFailedException +import javax.imageio.stream.ImageInputStream +import org.apache.commons.io.IOUtils +import org.apache.linkis.common.utils.Utils object PictureEmailContentParser extends AbstractEmailContentParser[PictureEmailContent] { override protected def parseEmailContent(emailContent: PictureEmailContent, multiContentEmail: MultiContentEmail): Unit = { getFirstLineRecord(emailContent).foreach { imageStr => - val decoder = Base64.getDecoder - val byteArr = decoder.decode(imageStr) - val inputStream = new ByteArrayInputStream(byteArr) - val image = ImageIO.read(inputStream) - val contents = generateImage(image, multiContentEmail) - emailContent.setContent(contents) + emailContent.getFileType match { + case "checkData" => + //对于邮件校验数据不进行处理 + case "pdf" => + val pdfUUID: String = UUID.randomUUID.toString + val pdfName = pdfUUID + ".pdf" + val decoder = Base64.getDecoder + val byteArr = decoder.decode(imageStr) + multiContentEmail.addAttachment(new PdfAttachment(pdfName, Base64.getEncoder.encodeToString(byteArr))) + case _ => + var inputStream: ByteArrayInputStream = null + Utils.tryFinally({ + val decoder = Base64.getDecoder + val byteArr = decoder.decode(imageStr) + if (CHECK_EMAIL_IMAGE_SWITCH.getValue) { + checkImageSize(byteArr) + } + inputStream = new ByteArrayInputStream(byteArr) + val image = ImageIO.read(inputStream) + val contents = generateImage(image, multiContentEmail) + emailContent.setContent(contents) + })(IOUtils.closeQuietly(inputStream)) + + } } } + protected def checkImageSize(byteArr: Array[Byte]): Unit = { + var reader: ImageReader = null + val inputStream: InputStream = new ByteArrayInputStream(byteArr) + var imageInputStream: ImageInputStream = null + Utils.tryFinally({ + imageInputStream = ImageIO.createImageInputStream(inputStream) + val imageReaders: util.Iterator[ImageReader] = ImageIO.getImageReaders(imageInputStream) + if (!imageReaders.hasNext) throw new EmailSendFailedException(80002,"Unsupported image format!") + reader = imageReaders.next + reader.setInput(imageInputStream) + val height = reader.getHeight(0) + val width = reader.getWidth(0) + if ((height * width) > EMAIL_IMAGE_MAXSIZE.getValue) { + throw new EmailSendFailedException(80002, "too large picture size :" + (height * width) + ", expect max picture size is :" + EMAIL_IMAGE_MAXSIZE.getValue) + } + })({ + if (reader != null) reader.dispose() + IOUtils.closeQuietly(imageInputStream) + IOUtils.closeQuietly(inputStream) + }) + + } + protected def generateImage(bufferedImage: BufferedImage, email: AbstractEmail): Array[String] = { val imageUUID: String = UUID.randomUUID.toString val width: Int = bufferedImage.getWidth val height: Int = bufferedImage.getHeight // 只支持修改visualis图片大小,后续如果有新增其他类型的邮件需要修改图片大小,需要在if中加上该邮件类型 - val imagesCuts = if (email.getEmailType.contains("visualis") && height > EMAIL_IMAGE_HEIGHT.getValue) { + val imagesCuts = if (height > EMAIL_IMAGE_HEIGHT.getValue) { val numOfCut = Math.ceil(height.toDouble / EMAIL_IMAGE_HEIGHT.getValue).toInt val realHeight = height / numOfCut (0 until numOfCut).map(i => bufferedImage.getSubimage(0, i * realHeight, width, realHeight)).toArray @@ -62,8 +108,7 @@ object PictureEmailContentParser extends AbstractEmailContentParser[PictureEmail var iHeight = image.getHeight var iWidth = image.getWidth - - if (email.getEmailType.contains("visualis") && iWidth > EMAIL_IMAGE_WIDTH.getValue) { + if (iWidth > EMAIL_IMAGE_WIDTH.getValue) { iHeight = ((EMAIL_IMAGE_WIDTH.getValue.toDouble / iWidth.toDouble) * iHeight.toDouble).toInt iWidth = EMAIL_IMAGE_WIDTH.getValue } diff --git a/dss-appconn/appconns/dss-sso-appconn/pom.xml b/dss-appconn/appconns/dss-sso-appconn/pom.xml index e17d4ae5f4..492f72f572 100644 --- a/dss-appconn/appconns/dss-sso-appconn/pom.xml +++ b/dss-appconn/appconns/dss-sso-appconn/pom.xml @@ -21,7 +21,7 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 ../../../pom.xml 4.0.0 diff --git a/dss-appconn/appconns/dss-workflow-appconn/pom.xml b/dss-appconn/appconns/dss-workflow-appconn/pom.xml index 937ba44ee8..58d501e912 100644 --- a/dss-appconn/appconns/dss-workflow-appconn/pom.xml +++ b/dss-appconn/appconns/dss-workflow-appconn/pom.xml @@ -21,7 +21,7 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 ../../../pom.xml 4.0.0 diff --git a/dss-appconn/appconns/dss-workflow-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/workflow/opertion/WorkflowRefCopyOperation.java b/dss-appconn/appconns/dss-workflow-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/workflow/opertion/WorkflowRefCopyOperation.java index 28d8a97812..2a6740f373 100644 --- a/dss-appconn/appconns/dss-workflow-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/workflow/opertion/WorkflowRefCopyOperation.java +++ b/dss-appconn/appconns/dss-workflow-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/workflow/opertion/WorkflowRefCopyOperation.java @@ -16,6 +16,7 @@ package com.webank.wedatasphere.dss.appconn.workflow.opertion; +import com.webank.wedatasphere.dss.common.utils.RpcAskUtils; import com.webank.wedatasphere.dss.orchestrator.common.ref.OrchestratorRefConstant; import com.webank.wedatasphere.dss.sender.service.DSSSenderServiceFactory; import com.webank.wedatasphere.dss.standard.app.development.operation.AbstractDevelopmentOperation; @@ -28,6 +29,7 @@ import java.util.HashMap; import java.util.Map; +import java.util.Optional; public class WorkflowRefCopyOperation @@ -45,10 +47,15 @@ public RefJobContentResponseRef copyRef(ThirdlyRequestRef.CopyWitContextRequestR //插入version String version = workflowCopyRequestRef.getNewVersion(); String description = (String) workflowCopyRequestRef.getRefJobContent().get(OrchestratorRefConstant.ORCHESTRATION_DESCRIPTION); + Long targetProjectId = workflowCopyRequestRef.getRefProjectId(); + Optional nodeSuffix = Optional.ofNullable(workflowCopyRequestRef.getRefJobContent().get(OrchestratorRefConstant.ORCHESTRATION_NODE_SUFFIX)); + Optional newFlowName = Optional.ofNullable(workflowCopyRequestRef.getRefJobContent().get(OrchestratorRefConstant.ORCHESTRATION_NAME)); RequestCopyWorkflow requestCopyWorkflow = new RequestCopyWorkflow(userName, workflowCopyRequestRef.getWorkspace(), appId, contextIdStr, - projectName, version, description, workflowCopyRequestRef.getDSSLabels()); - ResponseCopyWorkflow responseCopyWorkflow = (ResponseCopyWorkflow) sender.ask(requestCopyWorkflow); + projectName, version, description, workflowCopyRequestRef.getDSSLabels(), + targetProjectId, (String) nodeSuffix.orElse(null), (String) newFlowName.orElse(null)); + ResponseCopyWorkflow responseCopyWorkflow = RpcAskUtils.processAskException(sender.ask(requestCopyWorkflow), + ResponseCopyWorkflow.class, RequestCopyWorkflow.class); Map refJobContent = new HashMap<>(2); refJobContent.put(OrchestratorRefConstant.ORCHESTRATION_ID_KEY, responseCopyWorkflow.getDssFlow().getId()); refJobContent.put(OrchestratorRefConstant.ORCHESTRATION_CONTENT_KEY, responseCopyWorkflow.getDssFlow().getFlowJson()); diff --git a/dss-appconn/appconns/dss-workflow-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/workflow/opertion/WorkflowRefCreationOperation.java b/dss-appconn/appconns/dss-workflow-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/workflow/opertion/WorkflowRefCreationOperation.java index 9fcbc14073..da9fb14775 100644 --- a/dss-appconn/appconns/dss-workflow-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/workflow/opertion/WorkflowRefCreationOperation.java +++ b/dss-appconn/appconns/dss-workflow-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/workflow/opertion/WorkflowRefCreationOperation.java @@ -18,6 +18,7 @@ import com.webank.wedatasphere.dss.common.label.DSSLabel; import com.webank.wedatasphere.dss.common.utils.MapUtils; +import com.webank.wedatasphere.dss.common.utils.RpcAskUtils; import com.webank.wedatasphere.dss.orchestrator.common.entity.DSSOrchestratorInfo; import com.webank.wedatasphere.dss.orchestrator.common.ref.OrchestratorRefConstant; import com.webank.wedatasphere.dss.sender.service.DSSSenderServiceFactory; @@ -35,7 +36,7 @@ import java.util.Map; public class WorkflowRefCreationOperation - extends AbstractDevelopmentOperation + extends AbstractDevelopmentOperation implements RefCreationOperation { private Sender sender = DSSSenderServiceFactory.getOrCreateServiceInstance().getWorkflowSender(); @@ -58,10 +59,10 @@ public RefJobContentResponseRef createRef(ThirdlyRequestRef.DSSJobContentWithCon dssOrchestratorInfo.getUses() : "uses"; RequestCreateWorkflow requestCreateWorkflow = new RequestCreateWorkflow(userName, dssOrchestratorInfo.getProjectId(), workflowName, contextId, description, parentFlowId, uses, linkedAppConnNames, dssLabels, orcVersion, schedulerAppConnName); - - ResponseCreateWorkflow responseCreateWorkflow = (ResponseCreateWorkflow) sender.ask(requestCreateWorkflow); + ResponseCreateWorkflow responseCreateWorkflow = RpcAskUtils.processAskException(sender.ask(requestCreateWorkflow), + ResponseCreateWorkflow.class, RequestCreateWorkflow.class); Map refJobContent = MapUtils.newCommonMap(OrchestratorRefConstant.ORCHESTRATION_ID_KEY, responseCreateWorkflow.getDssFlow().getId(), - OrchestratorRefConstant.ORCHESTRATION_CONTENT_KEY, responseCreateWorkflow.getDssFlow().getFlowJson()); + OrchestratorRefConstant.ORCHESTRATION_CONTENT_KEY, responseCreateWorkflow.getDssFlow().getFlowJson()); return RefJobContentResponseRef.newBuilder().setRefJobContent(refJobContent).success(); } } diff --git a/dss-appconn/appconns/dss-workflow-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/workflow/opertion/WorkflowRefDeletionOperation.java b/dss-appconn/appconns/dss-workflow-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/workflow/opertion/WorkflowRefDeletionOperation.java index fbf6fd3652..a2b870d476 100644 --- a/dss-appconn/appconns/dss-workflow-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/workflow/opertion/WorkflowRefDeletionOperation.java +++ b/dss-appconn/appconns/dss-workflow-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/workflow/opertion/WorkflowRefDeletionOperation.java @@ -19,6 +19,7 @@ import com.webank.wedatasphere.dss.common.label.DSSLabel; import com.webank.wedatasphere.dss.common.protocol.JobStatus; import com.webank.wedatasphere.dss.common.protocol.RequestDeleteWorkflow; +import com.webank.wedatasphere.dss.common.utils.RpcAskUtils; import com.webank.wedatasphere.dss.orchestrator.common.ref.OrchestratorRefConstant; import com.webank.wedatasphere.dss.sender.service.DSSSenderServiceFactory; import com.webank.wedatasphere.dss.standard.app.development.operation.AbstractDevelopmentOperation; @@ -33,7 +34,7 @@ public class WorkflowRefDeletionOperation - extends AbstractDevelopmentOperation + extends AbstractDevelopmentOperation implements RefDeletionOperation { @Override @@ -43,7 +44,8 @@ public ResponseRef deleteRef(OnlyDevelopmentRequestRef.RefJobContentRequestRefIm RequestDeleteWorkflow requestDeleteWorkflow = new RequestDeleteWorkflow(userName, flowId); List dssLabels = requestRef.getDSSLabels(); Sender tempSend = DSSSenderServiceFactory.getOrCreateServiceInstance().getWorkflowSender(dssLabels); - ResponseDeleteWorkflow responseDeleteWorkflow = (ResponseDeleteWorkflow) tempSend.ask(requestDeleteWorkflow); + ResponseDeleteWorkflow responseDeleteWorkflow = RpcAskUtils.processAskException(tempSend.ask(requestDeleteWorkflow), + ResponseDeleteWorkflow.class, RequestDeleteWorkflow.class); if(responseDeleteWorkflow.getJobStatus() == JobStatus.Success) { return ResponseRef.newInternalBuilder().success(); } else { diff --git a/dss-appconn/appconns/dss-workflow-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/workflow/opertion/WorkflowRefExportOperation.java b/dss-appconn/appconns/dss-workflow-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/workflow/opertion/WorkflowRefExportOperation.java index 9b1c5c12c4..038db59133 100644 --- a/dss-appconn/appconns/dss-workflow-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/workflow/opertion/WorkflowRefExportOperation.java +++ b/dss-appconn/appconns/dss-workflow-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/workflow/opertion/WorkflowRefExportOperation.java @@ -18,6 +18,7 @@ import com.webank.wedatasphere.dss.common.protocol.RequestExportWorkflow; import com.webank.wedatasphere.dss.common.protocol.ResponseExportWorkflow; +import com.webank.wedatasphere.dss.common.utils.RpcAskUtils; import com.webank.wedatasphere.dss.orchestrator.common.ref.OrchestratorRefConstant; import com.webank.wedatasphere.dss.sender.service.DSSSenderServiceFactory; import com.webank.wedatasphere.dss.standard.app.development.operation.AbstractDevelopmentOperation; @@ -49,7 +50,8 @@ public ExportResponseRef exportRef(ThirdlyRequestRef.RefJobContentRequestRefImpl toJson(requestRef.getWorkspace()), requestRef.getDSSLabels()); Sender sender = DSSSenderServiceFactory.getOrCreateServiceInstance().getWorkflowSender(requestRef.getDSSLabels()); - ResponseExportWorkflow responseExportWorkflow = (ResponseExportWorkflow) sender.ask(requestExportWorkflow); + ResponseExportWorkflow responseExportWorkflow = RpcAskUtils.processAskException(sender.ask(requestExportWorkflow), + ResponseExportWorkflow.class, RequestExportWorkflow.class); Map resourceMap = new HashMap<>(2); resourceMap.put(ImportRequestRef.RESOURCE_ID_KEY, responseExportWorkflow.resourceId()); resourceMap.put(ImportRequestRef.RESOURCE_VERSION_KEY, responseExportWorkflow.version()); diff --git a/dss-appconn/appconns/dss-workflow-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/workflow/opertion/WorkflowRefImportOperation.java b/dss-appconn/appconns/dss-workflow-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/workflow/opertion/WorkflowRefImportOperation.java index 19c2172bfa..650dad7235 100644 --- a/dss-appconn/appconns/dss-workflow-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/workflow/opertion/WorkflowRefImportOperation.java +++ b/dss-appconn/appconns/dss-workflow-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/workflow/opertion/WorkflowRefImportOperation.java @@ -18,6 +18,7 @@ import com.webank.wedatasphere.dss.common.protocol.JobStatus; import com.webank.wedatasphere.dss.common.utils.MapUtils; +import com.webank.wedatasphere.dss.common.utils.RpcAskUtils; import com.webank.wedatasphere.dss.orchestrator.common.ref.OrchestratorRefConstant; import com.webank.wedatasphere.dss.sender.service.DSSSenderServiceFactory; import com.webank.wedatasphere.dss.standard.app.development.operation.AbstractDevelopmentOperation; @@ -34,7 +35,7 @@ import java.util.Map; public class WorkflowRefImportOperation - extends AbstractDevelopmentOperation + extends AbstractDevelopmentOperation implements RefImportOperation { @Override @@ -48,7 +49,8 @@ public RefJobContentResponseRef importRef(ThirdlyRequestRef.ImportWitContextRequ requestRef.getContextId(), requestRef.getDSSLabels()); Sender sender = DSSSenderServiceFactory.getOrCreateServiceInstance().getWorkflowSender(requestRef.getDSSLabels()); - ResponseImportWorkflow responseImportWorkflow = (ResponseImportWorkflow) sender.ask(requestImportWorkflow); + ResponseImportWorkflow responseImportWorkflow = RpcAskUtils.processAskException(sender.ask(requestImportWorkflow), + ResponseImportWorkflow.class, RequestImportWorkflow.class); if(responseImportWorkflow.getStatus() == JobStatus.Success) { if(MapUtils.isEmpty(responseImportWorkflow.getWorkflows())) { return RefJobContentResponseRef.newBuilder() diff --git a/dss-appconn/appconns/dss-workflow-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/workflow/opertion/WorkflowRefUpdateOperation.java b/dss-appconn/appconns/dss-workflow-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/workflow/opertion/WorkflowRefUpdateOperation.java index b461d9497d..aca380566f 100644 --- a/dss-appconn/appconns/dss-workflow-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/workflow/opertion/WorkflowRefUpdateOperation.java +++ b/dss-appconn/appconns/dss-workflow-appconn/src/main/java/com/webank/wedatasphere/dss/appconn/workflow/opertion/WorkflowRefUpdateOperation.java @@ -18,6 +18,7 @@ import com.webank.wedatasphere.dss.common.protocol.JobStatus; import com.webank.wedatasphere.dss.common.protocol.RequestUpdateWorkflow; +import com.webank.wedatasphere.dss.common.utils.RpcAskUtils; import com.webank.wedatasphere.dss.orchestrator.common.ref.OrchestratorRefConstant; import com.webank.wedatasphere.dss.sender.service.DSSSenderServiceFactory; import com.webank.wedatasphere.dss.standard.app.development.operation.AbstractDevelopmentOperation; @@ -43,7 +44,8 @@ public ResponseRef updateRef(ThirdlyRequestRef.UpdateRequestRefImpl requestRef) String description = (String) requestRef.getDSSJobContent().get(OrchestratorRefConstant.ORCHESTRATION_DESCRIPTION); String uses = (String) requestRef.getDSSJobContent().get(OrchestratorRefConstant.ORCHESTRATION_USES); RequestUpdateWorkflow requestUpdateWorkflow = new RequestUpdateWorkflow(userName, flowId, flowName, description, uses); - ResponseUpdateWorkflow responseUpdateWorkflow = (ResponseUpdateWorkflow) sender.ask(requestUpdateWorkflow); + ResponseUpdateWorkflow responseUpdateWorkflow = RpcAskUtils.processAskException(sender.ask(requestUpdateWorkflow), + ResponseUpdateWorkflow.class, RequestUpdateWorkflow.class); if(responseUpdateWorkflow.getJobStatus() == JobStatus.Success) { return ResponseRef.newInternalBuilder().success(); } else { diff --git a/dss-appconn/dss-appconn-core/pom.xml b/dss-appconn/dss-appconn-core/pom.xml index 72ff338825..2e00952837 100644 --- a/dss-appconn/dss-appconn-core/pom.xml +++ b/dss-appconn/dss-appconn-core/pom.xml @@ -21,14 +21,15 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 + ../../pom.xml 4.0.0 dss-appconn-core - 2.11.0 + 2.14.1 3.2.2 diff --git a/dss-appconn/dss-appconn-loader/pom.xml b/dss-appconn/dss-appconn-loader/pom.xml index 968740a92f..db5afc17c1 100644 --- a/dss-appconn/dss-appconn-loader/pom.xml +++ b/dss-appconn/dss-appconn-loader/pom.xml @@ -21,7 +21,8 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 + ../../pom.xml 4.0.0 diff --git a/dss-appconn/dss-appconn-loader/src/main/java/com/webank/wedatasphere/dss/appconn/loader/loader/AppConnLoaderFactory.java b/dss-appconn/dss-appconn-loader/src/main/java/com/webank/wedatasphere/dss/appconn/loader/loader/AppConnLoaderFactory.java index c84f592c33..44ee8ed90d 100644 --- a/dss-appconn/dss-appconn-loader/src/main/java/com/webank/wedatasphere/dss/appconn/loader/loader/AppConnLoaderFactory.java +++ b/dss-appconn/dss-appconn-loader/src/main/java/com/webank/wedatasphere/dss/appconn/loader/loader/AppConnLoaderFactory.java @@ -28,7 +28,7 @@ public class AppConnLoaderFactory { private static final Logger logger = LoggerFactory.getLogger(AppConnLoaderFactory.class); private static Class clazz = CommonAppConnLoader.class; - private static AppConnLoader appConnLoader = null; + private static volatile AppConnLoader appConnLoader = null; @SuppressWarnings("unchecked") public static AppConnLoader getAppConnLoader(){ diff --git a/dss-appconn/dss-appconn-loader/src/main/java/com/webank/wedatasphere/dss/appconn/loader/loader/CommonAppConnLoader.java b/dss-appconn/dss-appconn-loader/src/main/java/com/webank/wedatasphere/dss/appconn/loader/loader/CommonAppConnLoader.java index ca04a6c24f..47fdaf9d28 100644 --- a/dss-appconn/dss-appconn-loader/src/main/java/com/webank/wedatasphere/dss/appconn/loader/loader/CommonAppConnLoader.java +++ b/dss-appconn/dss-appconn-loader/src/main/java/com/webank/wedatasphere/dss/appconn/loader/loader/CommonAppConnLoader.java @@ -54,7 +54,7 @@ public AppConn getAppConn(String appConnName, String spi, String homePath) throw } LOGGER.info("The libPath url of AppConn {} is {}.", appConnName, libPathUrl); List jars = AppConnUtils.getJarsUrlsOfPath(libPathUrl); - ClassLoader classLoader = AppStandardClassUtils.getClassLoader(appConnName, () -> new AppConnClassLoader(jars.toArray(new URL[1]), currentClassLoader)); + ClassLoader classLoader = AppStandardClassUtils.refreshClassloader(appConnName, () -> new AppConnClassLoader(jars.toArray(new URL[1]), currentClassLoader)); Thread.currentThread().setContextClassLoader(classLoader); String fullClassName; if (StringUtils.isEmpty(spi)) { diff --git a/dss-appconn/dss-appconn-loader/src/main/java/com/webank/wedatasphere/dss/appconn/loader/utils/AppConnUtils.java b/dss-appconn/dss-appconn-loader/src/main/java/com/webank/wedatasphere/dss/appconn/loader/utils/AppConnUtils.java index bfab51f1ff..64e161b05f 100644 --- a/dss-appconn/dss-appconn-loader/src/main/java/com/webank/wedatasphere/dss/appconn/loader/utils/AppConnUtils.java +++ b/dss-appconn/dss-appconn-loader/src/main/java/com/webank/wedatasphere/dss/appconn/loader/utils/AppConnUtils.java @@ -99,16 +99,17 @@ public static List getJarsUrlsOfPath(String path) throws MalformedURLExcept */ private static List getClassNameFrom(String jarName) throws IOException { List fileList = new ArrayList<>(); - JarFile jarFile = new JarFile(new File(jarName)); - Enumeration en = jarFile.entries(); - while (en.hasMoreElements()) { - String name1 = en.nextElement().getName(); - if (!name1.endsWith(".class")) { - continue; + try (JarFile jarFile = new JarFile(new File(jarName))) { + Enumeration en = jarFile.entries(); + while (en.hasMoreElements()) { + String name1 = en.nextElement().getName(); + if (!name1.endsWith(".class")) { + continue; + } + String name2 = name1.substring(0, name1.lastIndexOf(".class")); + String name3 = name2.replaceAll("/", "."); + fileList.add(name3); } - String name2 = name1.substring(0, name1.lastIndexOf(".class")); - String name3 = name2.replaceAll("/", "."); - fileList.add(name3); } return fileList; } diff --git a/dss-appconn/dss-appconn-manager/dss-appconn-manager-client/pom.xml b/dss-appconn/dss-appconn-manager/dss-appconn-manager-client/pom.xml index 26df350a48..c8ca5215bc 100644 --- a/dss-appconn/dss-appconn-manager/dss-appconn-manager-client/pom.xml +++ b/dss-appconn/dss-appconn-manager/dss-appconn-manager-client/pom.xml @@ -21,7 +21,8 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 + ../../../pom.xml 4.0.0 diff --git a/dss-appconn/dss-appconn-manager/dss-appconn-manager-client/src/main/java/com/webank/wedatasphere/dss/appconn/manager/AppConnManagerClientImpl.java b/dss-appconn/dss-appconn-manager/dss-appconn-manager-client/src/main/java/com/webank/wedatasphere/dss/appconn/manager/AppConnManagerClientImpl.java new file mode 100644 index 0000000000..9854d18d8a --- /dev/null +++ b/dss-appconn/dss-appconn-manager/dss-appconn-manager-client/src/main/java/com/webank/wedatasphere/dss/appconn/manager/AppConnManagerClientImpl.java @@ -0,0 +1,35 @@ +/* + * Copyright 2019 WeBank + * 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. + * + */ + +package com.webank.wedatasphere.dss.appconn.manager; + +import com.webank.wedatasphere.dss.appconn.manager.impl.AppConnManagerImpl; +import com.webank.wedatasphere.dss.appconn.manager.service.AppConnInfoService; +import com.webank.wedatasphere.dss.appconn.manager.service.AppConnInfoServiceImpl; +import com.webank.wedatasphere.dss.appconn.manager.service.AppConnResourceService; +import com.webank.wedatasphere.dss.appconn.manager.service.AppConnResourceServiceImpl; + +public class AppConnManagerClientImpl extends AppConnManagerImpl { + @Override + protected AppConnInfoService createAppConnInfoService() { + return new AppConnInfoServiceImpl(); + } + + @Override + protected AppConnResourceService createAppConnResourceService() { + return new AppConnResourceServiceImpl(); + } +} diff --git a/dss-appconn/dss-appconn-manager/dss-appconn-manager-client/src/main/java/com/webank/wedatasphere/dss/appconn/manager/conf/AppConnManagerClientConfiguration.java b/dss-appconn/dss-appconn-manager/dss-appconn-manager-client/src/main/java/com/webank/wedatasphere/dss/appconn/manager/conf/AppConnManagerClientConfiguration.java index c95e3ed53f..c8f1a4aba0 100644 --- a/dss-appconn/dss-appconn-manager/dss-appconn-manager-client/src/main/java/com/webank/wedatasphere/dss/appconn/manager/conf/AppConnManagerClientConfiguration.java +++ b/dss-appconn/dss-appconn-manager/dss-appconn-manager-client/src/main/java/com/webank/wedatasphere/dss/appconn/manager/conf/AppConnManagerClientConfiguration.java @@ -17,12 +17,14 @@ package com.webank.wedatasphere.dss.appconn.manager.conf; import org.apache.linkis.common.conf.CommonVars; +import org.apache.linkis.common.conf.TimeType; public class AppConnManagerClientConfiguration { public final static CommonVars DSS_APPCONN_CLIENT_TOKEN = CommonVars.apply("wds.dss.appconn.client.user.token","WS-AUTH"); public final static CommonVars LINKIS_ADMIN_USER = CommonVars.apply("wds.dss.appconn.client.user","ws"); + public final static CommonVars APPCONN_WAIT_MAX_TIME = CommonVars.apply("wds.dss.appconn.client.load.wait.max.time",new TimeType("3m")); } diff --git a/dss-appconn/dss-appconn-manager/dss-appconn-manager-client/src/main/java/com/webank/wedatasphere/dss/appconn/manager/service/AppConnResourceServiceImpl.java b/dss-appconn/dss-appconn-manager/dss-appconn-manager-client/src/main/java/com/webank/wedatasphere/dss/appconn/manager/service/AppConnResourceServiceImpl.java index a29a313478..2f16d6fdb8 100644 --- a/dss-appconn/dss-appconn-manager/dss-appconn-manager-client/src/main/java/com/webank/wedatasphere/dss/appconn/manager/service/AppConnResourceServiceImpl.java +++ b/dss-appconn/dss-appconn-manager/dss-appconn-manager-client/src/main/java/com/webank/wedatasphere/dss/appconn/manager/service/AppConnResourceServiceImpl.java @@ -23,7 +23,8 @@ import com.webank.wedatasphere.dss.common.entity.Resource; import com.webank.wedatasphere.dss.common.exception.DSSErrorException; import com.webank.wedatasphere.dss.common.utils.ZipHelper; -import org.apache.commons.lang.StringUtils; +import org.apache.commons.io.FileUtils; +import org.apache.commons.lang3.StringUtils; import org.apache.linkis.bml.client.BmlClient; import org.apache.linkis.bml.client.BmlClientFactory; import org.apache.linkis.common.utils.Utils; @@ -36,13 +37,19 @@ import java.nio.file.Path; import java.nio.file.Paths; import java.util.Arrays; +import java.util.HashMap; import java.util.List; +import java.util.Map; +import java.util.function.Supplier; import java.util.stream.Collectors; import java.util.stream.IntStream; +import static com.webank.wedatasphere.dss.appconn.manager.conf.AppConnManagerClientConfiguration.APPCONN_WAIT_MAX_TIME; + public class AppConnResourceServiceImpl implements AppConnResourceService { private static final Logger LOGGER = LoggerFactory.getLogger(AppConnResourceServiceImpl.class); + private final Map appConnIsLoaded = new HashMap<>(); private BmlClient bmlClient = BmlClientFactory.createBmlClient(); @@ -57,69 +64,105 @@ public String getAppConnHome(AppConnInfo appConnInfo) { } File appConnPath = new File(appConnHomePath, appConnName); Resource resource = appConnInfo.getAppConnResource(); - if(!appConnPath.exists() && !appConnPath.mkdir()) { - throw new AppConnHomeNotExistsWarnException(20350, "Cannot create dir " + appConnPath.getPath() + " for AppConn " + appConnName); - } - if(AppConnIndexFileUtils.isLatestIndex(appConnPath, resource)) { + Supplier isLatest = () -> appConnPath.exists() && appConnPath.isDirectory() + && AppConnIndexFileUtils.isLatestIndex(appConnPath, resource); + if(isLatest.get()) { + if(!appConnIsLoaded.containsKey(appConnName)) { + synchronized (appConnIsLoaded) { + if(!appConnIsLoaded.containsKey(appConnName)) { + appConnIsLoaded.put(appConnName, true); + LOGGER.warn("AppConn {} is newest, no necessary to reload it, just use it.", appConnName); + } + } + } return appConnPath.getPath(); } - if (StringUtils.isNotBlank(appConnInfo.getReference())) { - return appConnPath.getPath(); + File zipFilePath = new File(appConnHomePath, appConnName + "_" + resource.getVersion() + ".zip"); + if(zipFilePath.exists()) { + long startTime = System.currentTimeMillis(); + LOGGER.warn("I found the {} is exists, maybe another service is loading the AppConn {}, I will try to wait for {} at max.", + zipFilePath, appConnName, APPCONN_WAIT_MAX_TIME.getValue().toString()); + while(!isLatest.get()) { + Utils.sleepQuietly(3000); + if(System.currentTimeMillis() - startTime >= APPCONN_WAIT_MAX_TIME.getValue().toLong()) { + break; + } + } + if(isLatest.get()) { + LOGGER.warn("AppConn {} is loaded by another service, now just use it.", appConnName); + synchronized (appConnIsLoaded) { + appConnIsLoaded.put(appConnName, true); + } + return appConnPath.getPath(); + } else { + LOGGER.warn("Since waited for {}, the AppConn {} has not been loaded by others, now I will try to load it by myself.", APPCONN_WAIT_MAX_TIME.getValue().toString(), appConnName); + deleteFile(zipFilePath, "Delete the zip file " + zipFilePath.getName() + " of AppConn " + appConnName + " failed"); + } } - LOGGER.info("Try to download latest resource {} in version {} from BML for AppConn {}.", resource.getResourceId(), - resource.getVersion(), appConnName); + try { + Files.createFile(zipFilePath.toPath()); + } catch (IOException e) { + throw new AppConnHomeNotExistsWarnException(20350, "Cannot create zip file " + zipFilePath.getPath() + " for AppConn " + + appConnName, e); + } + LOGGER.warn("Try to load AppConn {}......", appConnName); + LOGGER.info("First, download latest resource {} in version {} from BML for AppConn {}, and write it into file {}.", resource.getResourceId(), + resource.getVersion(), appConnName, zipFilePath); // At first, Download AppConn files from bml. - String zipFilePath = new File(appConnHomePath, appConnName + "_" + resource.getVersion() + ".zip").getPath(); bmlClient.downloadResource(Utils.getJvmUser(), resource.getResourceId(), resource.getVersion(), - "file://" + zipFilePath, true); + "file://" + zipFilePath.getPath(), true); // Then, try to unzip it. - if(!deleteAppConnDir(appConnPath)) { - throw new AppConnHomeNotExistsWarnException(20350, "Cannot delete dir " + appConnPath.getPath() + " for AppConn " + appConnName); + if(appConnPath.exists()) { + try { + FileUtils.deleteDirectory(appConnPath); + } catch (IOException e) { + throw new AppConnHomeNotExistsWarnException(20350, "Cannot delete dir " + appConnPath.getPath() + " for AppConn " + appConnName, e); + } } + LOGGER.info("Then, unzip the latest resource file {}.", zipFilePath); try { - ZipHelper.unzip(zipFilePath); + ZipHelper.unzip(zipFilePath.getPath()); } catch (DSSErrorException e) { - throw new AppConnHomeNotExistsWarnException(20350, "Unzip " + zipFilePath + " failed, AppConn " + appConnName, e); + throw new AppConnHomeNotExistsWarnException(20350, "Unzip " + zipFilePath + " failed, AppConn is " + appConnName, e); } File oldIndexFile = AppConnIndexFileUtils.getIndexFile(appConnPath); - - // update index file. - if (oldIndexFile != null && !oldIndexFile.delete()) { - throw new AppConnHomeNotExistsWarnException(20350, "Delete index file " + oldIndexFile.getName() + " failed, please ensure the permission is all right."); + // delete old index file. + if (oldIndexFile != null) { + LOGGER.info("Thirdly, delete the old index file {} for AppConn {}.", oldIndexFile, appConnName); + deleteFile(oldIndexFile, "Delete the index file " + oldIndexFile.getName() + " of AppConn " + appConnName + " failed"); } - -// TODO ZipUtils.fileToUnzip(zipFilePath, appConnHomePath.getPath()); - // Only reserve latest 2 version files. - File[] historyZipFiles = appConnHomePath.listFiles((p, fileName) -> fileName.startsWith(appConnName) && fileName.endsWith(".zip")); - if(historyZipFiles.length > 2) { - List files = Arrays.stream(historyZipFiles).sorted().collect(Collectors.toList()); - // ignore delete failed. - IntStream.range(0, files.size() - 2).forEach(index -> files.get(index).delete()); - } - // Finally, write index file. + // create new index file. Path indexFile = Paths.get(appConnPath.getPath(), AppConnIndexFileUtils.getIndexFileName(resource)); + LOGGER.info("Finally, create the latest index file {} for AppConn {}.", indexFile.toFile(), appConnName); try { Files.createFile(indexFile); } catch (IOException e) { throw new AppConnHomeNotExistsWarnException(20350, "Cannot create index file " + indexFile.toFile().getPath() + " for AppConn " + appConnName, e); } + synchronized (appConnIsLoaded) { + appConnIsLoaded.put(appConnName, true); + } + // Only reserve latest 2 version files. + File[] historyZipFiles = appConnHomePath.listFiles((p, fileName) -> fileName.startsWith(appConnName) && fileName.endsWith(".zip")); + if(historyZipFiles != null && historyZipFiles.length > 2) { + List files = Arrays.stream(historyZipFiles).sorted().collect(Collectors.toList()); + // ignore the failed deletion. + IntStream.range(0, files.size() - 2).forEach(index -> deleteFile(files.get(index), null)); + } + LOGGER.warn("AppConn {} is loaded.", appConnName); return appConnPath.getPath(); } - public boolean deleteAppConnDir(File f){ - if(f.isDirectory()){ - File[] files = f.listFiles(); - for (File key : files) { - if(key.isFile()){ - key.delete(); - }else{ - deleteAppConnDir(key); - } + private void deleteFile(File file, String errorMsg) { + try { + FileUtils.forceDelete(file); + } catch (IOException e) { + if(StringUtils.isNotEmpty(errorMsg)) { + throw new AppConnHomeNotExistsWarnException(20350, errorMsg + ". Please ensure the permission is all right.", e); } } - return f.delete(); } } diff --git a/dss-appconn/dss-appconn-manager/dss-appconn-manager-core/pom.xml b/dss-appconn/dss-appconn-manager/dss-appconn-manager-core/pom.xml index dfc2bbbe55..45b21e5872 100644 --- a/dss-appconn/dss-appconn-manager/dss-appconn-manager-core/pom.xml +++ b/dss-appconn/dss-appconn-manager/dss-appconn-manager-core/pom.xml @@ -21,7 +21,8 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 + ../../../pom.xml 4.0.0 @@ -34,6 +35,7 @@ dss-appconn-loader ${dss.version} + diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/aop/DSSWorkspaceUMAspect.java b/dss-appconn/dss-appconn-manager/dss-appconn-manager-core/src/main/java/com/webank/wedatasphere/dss/appconn/manager/conf/AppConnManagerCoreConf.java similarity index 61% rename from dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/aop/DSSWorkspaceUMAspect.java rename to dss-appconn/dss-appconn-manager/dss-appconn-manager-core/src/main/java/com/webank/wedatasphere/dss/appconn/manager/conf/AppConnManagerCoreConf.java index e42e2b6a95..54e1e90ad9 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/aop/DSSWorkspaceUMAspect.java +++ b/dss-appconn/dss-appconn-manager/dss-appconn-manager-core/src/main/java/com/webank/wedatasphere/dss/appconn/manager/conf/AppConnManagerCoreConf.java @@ -14,13 +14,11 @@ * */ -package com.webank.wedatasphere.dss.framework.workspace.aop; +package com.webank.wedatasphere.dss.appconn.manager.conf; -import org.aspectj.lang.annotation.Aspect; -import org.springframework.stereotype.Component; +import org.apache.linkis.common.conf.CommonVars; - -@Aspect -@Component -public class DSSWorkspaceUMAspect { +public class AppConnManagerCoreConf { + // 默认值为false,只有dev的其中一个服务需要开启该配置作为manager节点 + public static final CommonVars IS_APPCONN_MANAGER = CommonVars.apply("wds.dss.appconn.framework.ismanager", false); } diff --git a/dss-appconn/dss-appconn-manager/dss-appconn-manager-core/src/main/java/com/webank/wedatasphere/dss/appconn/manager/impl/AbstractAppConnManager.java b/dss-appconn/dss-appconn-manager/dss-appconn-manager-core/src/main/java/com/webank/wedatasphere/dss/appconn/manager/impl/AbstractAppConnManager.java index 9340464c0b..2581e47c0d 100644 --- a/dss-appconn/dss-appconn-manager/dss-appconn-manager-core/src/main/java/com/webank/wedatasphere/dss/appconn/manager/impl/AbstractAppConnManager.java +++ b/dss-appconn/dss-appconn-manager/dss-appconn-manager-core/src/main/java/com/webank/wedatasphere/dss/appconn/manager/impl/AbstractAppConnManager.java @@ -17,9 +17,12 @@ package com.webank.wedatasphere.dss.appconn.manager.impl; import com.webank.wedatasphere.dss.appconn.core.AppConn; +import com.webank.wedatasphere.dss.appconn.core.exception.AppConnErrorException; +import com.webank.wedatasphere.dss.appconn.core.exception.AppConnWarnException; import com.webank.wedatasphere.dss.appconn.loader.loader.AppConnLoader; import com.webank.wedatasphere.dss.appconn.loader.loader.AppConnLoaderFactory; import com.webank.wedatasphere.dss.appconn.manager.AppConnManager; +import com.webank.wedatasphere.dss.appconn.manager.conf.AppConnManagerCoreConf; import com.webank.wedatasphere.dss.appconn.manager.entity.AppConnInfo; import com.webank.wedatasphere.dss.appconn.manager.entity.AppInstanceInfo; import com.webank.wedatasphere.dss.appconn.manager.service.AppConnInfoService; @@ -68,7 +71,11 @@ public static AppConnManager getAppConnManager() { } synchronized (AbstractAppConnManager.class) { if (appConnManager == null) { - appConnManager = ClassUtils.getInstanceOrDefault(AppConnManager.class, new AppConnManagerImpl()); + //appconn-manager-core包无法引入manager-client包,会有maven循环依赖,这里通过反射获取client的实现类 + //ismanager=false时,获取client端的AppConnManager实现类,ismanager=true时,获取appconn-framework端的AppConnManager实现类。 + appConnManager = !AppConnManagerCoreConf.IS_APPCONN_MANAGER.getValue() ? ClassUtils.getInstanceOrWarn(AppConnManagerImpl.class) : + //通过包名过滤 + ClassUtils.getInstanceOrDefault(AppConnManager.class, c -> c.getPackage().getName().contains("com.webank.wedatasphere.dss.framework.appconn"), new AppConnManagerImpl()); LOGGER.info("The instance of AppConnManager is {}.", appConnManager.getClass().getName()); appConnManager.init(); } @@ -217,6 +224,9 @@ private void lazyLoadAppConns() { @Override public List listAppConns() { lazyLoadAppConns(); + if(appConnList==null){ + throw new AppConnWarnException(25344,"appconn list has not been loaded,please try again later."); + } return appConnList; } @@ -275,4 +285,21 @@ public void reloadAppConn(AppConnInfo appConnInfo) { LOGGER.info("Reloaded AppConn {}.", appConnInfo.getAppConnName()); } + void deleteAppConn(AppConnInfo appConnInfo) { + lazyLoadAppConns(); + if (this.appConns.containsKey(appConnInfo.getAppConnName())) { + synchronized (this.appConns) { + if (this.appConns.containsKey(appConnInfo.getAppConnName())) { + this.appConns.remove(appConnInfo.getAppConnName()); + appConnList = Collections.unmodifiableList(new ArrayList<>(appConns.values())); + LOGGER.info("Deleted AppConn {}.", appConnInfo.getAppConnName()); + } + } + } + } + + public AppConnRefreshThread getAppConnRefreshThread() { + return appConnRefreshThread; + } + } diff --git a/dss-appconn/dss-appconn-manager/dss-appconn-manager-core/src/main/java/com/webank/wedatasphere/dss/appconn/manager/impl/AppConnManagerImpl.java b/dss-appconn/dss-appconn-manager/dss-appconn-manager-core/src/main/java/com/webank/wedatasphere/dss/appconn/manager/impl/AppConnManagerImpl.java index c06025d8e7..c16ea970a9 100644 --- a/dss-appconn/dss-appconn-manager/dss-appconn-manager-core/src/main/java/com/webank/wedatasphere/dss/appconn/manager/impl/AppConnManagerImpl.java +++ b/dss-appconn/dss-appconn-manager/dss-appconn-manager-core/src/main/java/com/webank/wedatasphere/dss/appconn/manager/impl/AppConnManagerImpl.java @@ -28,6 +28,7 @@ public class AppConnManagerImpl extends AbstractAppConnManager { @Override protected AppConnInfoService createAppConnInfoService() { try { + //由于maven不能循环依赖,这里不能返回client端的实现类 return ClassUtils.getInstance(AppConnInfoService.class); } catch (DSSErrorException e) { throw new DSSRuntimeException(25000, "Cannot find a useful AppConnInfoService.", e); diff --git a/dss-appconn/dss-appconn-manager/dss-appconn-manager-core/src/main/java/com/webank/wedatasphere/dss/appconn/manager/impl/AppConnRefreshThread.java b/dss-appconn/dss-appconn-manager/dss-appconn-manager-core/src/main/java/com/webank/wedatasphere/dss/appconn/manager/impl/AppConnRefreshThread.java index 406bda7e8c..2344821ce4 100644 --- a/dss-appconn/dss-appconn-manager/dss-appconn-manager-core/src/main/java/com/webank/wedatasphere/dss/appconn/manager/impl/AppConnRefreshThread.java +++ b/dss-appconn/dss-appconn-manager/dss-appconn-manager-core/src/main/java/com/webank/wedatasphere/dss/appconn/manager/impl/AppConnRefreshThread.java @@ -1,11 +1,17 @@ package com.webank.wedatasphere.dss.appconn.manager.impl; import com.webank.wedatasphere.dss.appconn.manager.entity.AppConnInfo; +import com.webank.wedatasphere.dss.appconn.manager.entity.AppInstanceInfo; +import com.webank.wedatasphere.dss.appconn.manager.service.AppConnRefreshListener; +import com.webank.wedatasphere.dss.common.utils.DSSCommonUtils; +import com.webank.wedatasphere.dss.common.utils.MapUtils; +import com.webank.wedatasphere.dss.standard.common.desc.AppInstance; +import org.apache.commons.collections4.CollectionUtils; +import org.apache.commons.lang3.StringUtils; import org.slf4j.Logger; import org.slf4j.LoggerFactory; -import java.util.List; -import java.util.Optional; +import java.util.*; /** * @author enjoyyin @@ -18,6 +24,7 @@ public class AppConnRefreshThread implements Runnable { private AbstractAppConnManager appConnManager; private volatile List appConnInfos; + private List refreshListeners = new ArrayList<>(); public AppConnRefreshThread(AbstractAppConnManager appConnManager, List appConnInfos) { @@ -47,17 +54,27 @@ public void run() { appConnInfos.forEach(appConnInfo -> { Optional oldOne = this.appConnInfos.stream().filter(old -> old.getAppConnName().equals(appConnInfo.getAppConnName())).findAny(); if(!oldOne.isPresent() || isChanged(oldOne.get(), appConnInfo)) { - LOGGER.warn("AppConn info {} has updated, now try to refresh it.", appConnInfo.getAppConnName()); - try { - appConnManager.reloadAppConn(appConnInfo); - } catch (Exception e) { - // If update failed, it seems like some error happened in this AppConn. - // this AppConn will not be refreshed any more, unless the admin changes the AppConn plugin files to optimize it. - LOGGER.warn("Reload AppConn {} failed, ignore it", appConnInfo.getAppConnName(), e); + LOGGER.warn("The appConnInfo of AppConn {} has changed, now try to refresh it.", appConnInfo.getAppConnName()); + LOGGER.warn("The new appConnInfo of AppConn {} is {}.", appConnInfo.getAppConnName(), appConnInfo); + reloadAppConn(appConnInfo); + } else { + List appInstanceInfos = appConnManager.appConnInfoService.getAppInstancesByAppConnInfo(appConnInfo); + List oldAppInstanceList = appConnManager.getAppConn(appConnInfo.getAppConnName()).getAppDesc().getAppInstances(); + if(isChanged(appInstanceInfos, oldAppInstanceList)) { + LOGGER.warn("The appInstanceInfo of AppConn {} has changed, now try to refresh it.", appConnInfo.getAppConnName()); + LOGGER.warn("The old appInstanceInfo of AppConn {} is {}.", appConnInfo.getAppConnName(), oldAppInstanceList); + LOGGER.warn("The new appInstanceInfo of AppConn {} is {}.", appConnInfo.getAppConnName(), appInstanceInfos); + reloadAppConn(appConnInfo); } } }); - // now, do not support to delete exists AppConn, since deletion operation is very dangerous. + // Now, try to delete not exists AppConn. + // Since deletion is very dangerous, it is not suggested to do this operation. + this.appConnInfos.stream().filter(appConnInfo -> appConnInfos.stream().noneMatch(newOne -> appConnInfo.getAppConnName().equals(newOne.getAppConnName()))) + .forEach(appConnInfo -> { + LOGGER.warn("The AppConn {} is not exists in DSS DB, it seems like that admin has deleted it, now try to delete it.", appConnInfo.getAppConnName()); + appConnManager.deleteAppConn(appConnInfo); + }); this.appConnInfos = appConnInfos; LOGGER.info("all AppConns have refreshed."); } @@ -71,4 +88,39 @@ private boolean isChanged(AppConnInfo one, AppConnInfo other) { return !one.getAppConnResource().equals(other.getAppConnResource()); } } + + private boolean isChanged(List one, List other) { + if (CollectionUtils.isEmpty(one) && CollectionUtils.isEmpty(other)) { + return false; + } else if (CollectionUtils.isEmpty(one) || CollectionUtils.isEmpty(other) || one.size() != other.size()) { + return true; + } else { + // 判断条件为:一种是找不到 ID 相同的,一种是 ID 相同但是属性有变化 + // 这里不判断标签,因为标签是不会改动的 + return one.stream().anyMatch(newOne -> other.stream().noneMatch(otherOne -> otherOne.getId().equals(newOne.getId())) + || other.stream().anyMatch(otherOne -> otherOne.getId().equals(newOne.getId()) + && (!StringUtils.equals(newOne.getUrl(), otherOne.getBaseUrl()) + || ((StringUtils.isNotEmpty(newOne.getHomepageUri()) || StringUtils.isNotEmpty(otherOne.getHomepageUri())) + && !StringUtils.equals(newOne.getHomepageUri(), otherOne.getHomepageUri())) + || ((StringUtils.isNotEmpty(newOne.getEnhanceJson()) || MapUtils.isNotEmpty(otherOne.getConfig())) && + //考虑到enhanceJson中key和value之间含有空格的情况,这里需要比较map而不是map的jsonString,否则由map转换为json时不会再包含空格 + !otherOne.getConfig().equals(DSSCommonUtils.COMMON_GSON.fromJson(newOne.getEnhanceJson(), Map.class))))) + ); + } + } + + private void reloadAppConn(AppConnInfo appConnInfo) { + try { + appConnManager.reloadAppConn(appConnInfo); + refreshListeners.forEach(listener -> listener.afterRefresh(appConnInfo.getAppConnName())); + } catch (Exception e) { + // If update failed, it seems like some error happened in this AppConn. + // this AppConn will not be refreshed any more, unless the admin changes the AppConn plugin files to optimize it. + LOGGER.warn("Reload AppConn {} failed, ignore it", appConnInfo.getAppConnName(), e); + } + } + + public void registerRefreshListener(AppConnRefreshListener listener) { + refreshListeners.add(listener); + } } diff --git a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/IOEnv.java b/dss-appconn/dss-appconn-manager/dss-appconn-manager-core/src/main/java/com/webank/wedatasphere/dss/appconn/manager/service/AppConnRefreshListener.java similarity index 80% rename from dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/IOEnv.java rename to dss-appconn/dss-appconn-manager/dss-appconn-manager-core/src/main/java/com/webank/wedatasphere/dss/appconn/manager/service/AppConnRefreshListener.java index a4f8e248ae..f1e41a96e8 100644 --- a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/IOEnv.java +++ b/dss-appconn/dss-appconn-manager/dss-appconn-manager-core/src/main/java/com/webank/wedatasphere/dss/appconn/manager/service/AppConnRefreshListener.java @@ -14,11 +14,9 @@ * */ -package com.webank.wedatasphere.dss.common.entity; +package com.webank.wedatasphere.dss.appconn.manager.service; -public enum IOEnv { - /** - * - */ - BDP_DEV,BDP_PRODUCTION,BDAP_DEV,BDAP_PRODUCTION +public interface AppConnRefreshListener { + + void afterRefresh(String appconnName); } diff --git a/dss-appconn/dss-appconn-manager/pom.xml b/dss-appconn/dss-appconn-manager/pom.xml index ec4db13c95..298a25c684 100644 --- a/dss-appconn/dss-appconn-manager/pom.xml +++ b/dss-appconn/dss-appconn-manager/pom.xml @@ -21,7 +21,8 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 + ../../pom.xml 4.0.0 diff --git a/dss-appconn/dss-scheduler-appconn/pom.xml b/dss-appconn/dss-scheduler-appconn/pom.xml index 531dc3ad96..f69417f55d 100644 --- a/dss-appconn/dss-scheduler-appconn/pom.xml +++ b/dss-appconn/dss-scheduler-appconn/pom.xml @@ -21,7 +21,8 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 + ../../pom.xml 4.0.0 diff --git a/dss-appconn/linkis-appconn-engineplugin/pom.xml b/dss-appconn/linkis-appconn-engineplugin/pom.xml index e7572b08b8..1447f0cda8 100644 --- a/dss-appconn/linkis-appconn-engineplugin/pom.xml +++ b/dss-appconn/linkis-appconn-engineplugin/pom.xml @@ -21,7 +21,8 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 + ../../pom.xml 4.0.0 @@ -63,14 +64,12 @@ org.hibernate.validator - provided org.apache.linkis linkis-storage ${linkis.version} - provided com.webank.wedatasphere.dss @@ -107,6 +106,28 @@ org.apache.linkis linkis-mybatis ${linkis.version} + + + org.springframework.boot + spring-boot-starter + + + org.springframework.boot + spring-boot-autoconfigure + + + org.springframework + spring-beans + + + org.springframework + spring-jdbc + + + com.zaxxer + HikariCP + + @@ -119,7 +140,6 @@ org.apache.linkis linkis-common ${linkis.version} - provided diff --git a/dss-appconn/linkis-appconn-engineplugin/src/main/assembly/distribution.xml b/dss-appconn/linkis-appconn-engineplugin/src/main/assembly/distribution.xml index 4ebadd9af1..0e7dc7c4fb 100644 --- a/dss-appconn/linkis-appconn-engineplugin/src/main/assembly/distribution.xml +++ b/dss-appconn/linkis-appconn-engineplugin/src/main/assembly/distribution.xml @@ -32,7 +32,7 @@ - /dist/v1/lib + /dist/1/lib true true false @@ -60,7 +60,7 @@ log4j2*.xml 0777 - /dist/v1/conf + /dist/1/conf unix diff --git a/dss-appconn/linkis-appconn-engineplugin/src/main/resources/log4j2.xml b/dss-appconn/linkis-appconn-engineplugin/src/main/resources/log4j2.xml index 062ae6e206..7d38cb41de 100644 --- a/dss-appconn/linkis-appconn-engineplugin/src/main/resources/log4j2.xml +++ b/dss-appconn/linkis-appconn-engineplugin/src/main/resources/log4j2.xml @@ -32,9 +32,21 @@ + + + + + + + + + + + diff --git a/dss-appconn/linkis-appconn-engineplugin/src/main/scala/org/apache/linkis/manager/engineplugin/appconn/AppConnEngineConnPlugin.scala b/dss-appconn/linkis-appconn-engineplugin/src/main/scala/org/apache/linkis/manager/engineplugin/appconn/AppConnEngineConnPlugin.scala index fea56f4e14..42c8f9a99a 100644 --- a/dss-appconn/linkis-appconn-engineplugin/src/main/scala/org/apache/linkis/manager/engineplugin/appconn/AppConnEngineConnPlugin.scala +++ b/dss-appconn/linkis-appconn-engineplugin/src/main/scala/org/apache/linkis/manager/engineplugin/appconn/AppConnEngineConnPlugin.scala @@ -45,7 +45,7 @@ class AppConnEngineConnPlugin extends EngineConnPlugin { private val defaultLabels: util.List[Label[_]] = new util.ArrayList[Label[_]]() - override def init(params: util.Map[String, Any]): Unit = { + override def init(params: util.Map[String, AnyRef]): Unit = { val engineTypeLabel = EngineTypeLabelCreator.createEngineTypeLabel(EngineType.APPCONN.toString) this.defaultLabels.add(engineTypeLabel) } diff --git a/dss-appconn/linkis-appconn-engineplugin/src/main/scala/org/apache/linkis/manager/engineplugin/appconn/executor/AbstractExecutionRequestRefContext.scala b/dss-appconn/linkis-appconn-engineplugin/src/main/scala/org/apache/linkis/manager/engineplugin/appconn/executor/AbstractExecutionRequestRefContext.scala index 3d91c60192..b631b447d5 100644 --- a/dss-appconn/linkis-appconn-engineplugin/src/main/scala/org/apache/linkis/manager/engineplugin/appconn/executor/AbstractExecutionRequestRefContext.scala +++ b/dss-appconn/linkis-appconn-engineplugin/src/main/scala/org/apache/linkis/manager/engineplugin/appconn/executor/AbstractExecutionRequestRefContext.scala @@ -17,7 +17,6 @@ package org.apache.linkis.manager.engineplugin.appconn.executor import java.util - import com.webank.wedatasphere.dss.standard.app.development.listener.core.ExecutionRequestRefContext import com.webank.wedatasphere.dss.standard.app.development.listener.exception.AppConnExecutionErrorException import org.apache.linkis.common.io.resultset.{ResultSet, ResultSetReader, ResultSetWriter} @@ -28,7 +27,7 @@ import org.apache.linkis.manager.engineplugin.appconn.conf.AppConnEngineConnConf import org.apache.linkis.rpc.Sender import org.apache.linkis.storage.FSFactory import org.apache.linkis.storage.fs.FileSystem -import org.apache.linkis.storage.resultset.{ResultSetFactory, ResultSetReader} +import org.apache.linkis.storage.resultset.{ResultSetFactory, ResultSetReaderFactory} abstract class AbstractExecutionRequestRefContext(engineExecutorContext: EngineExecutionContext, user: String, @@ -70,8 +69,10 @@ abstract class AbstractExecutionRequestRefContext(engineExecutorContext: EngineE resultSetAlias: String): ResultSetWriter[M, R] = engineExecutorContext.createResultSetWriter(resultSet, resultSetAlias).asInstanceOf[ResultSetWriter[M, R]] - override def getResultSetReader[M <: MetaData, R <: Record](fsPath: FsPath): ResultSetReader[M, R] = - ResultSetReader.getResultSetReader(fsPath.getSchemaPath).asInstanceOf[ResultSetReader[M, R]] + override def getResultSetReader[M <: MetaData, R <: Record](fsPath: FsPath): ResultSetReader[M, R] = { + + ResultSetReaderFactory.getResultSetReader(fsPath.getSchemaPath).asInstanceOf[ResultSetReader[M, R]] + } private def createResultSetWriter[M <: MetaData, R <: Record](resultSetType: String, resultSetAlias: String): ResultSetWriter[M, R] = engineExecutorContext.createResultSetWriter(resultSetType, resultSetAlias).asInstanceOf[ResultSetWriter[M, R]] diff --git a/dss-appconn/linkis-appconn-engineplugin/src/main/scala/org/apache/linkis/manager/engineplugin/appconn/executor/AppConnEngineConnExecutor.scala b/dss-appconn/linkis-appconn-engineplugin/src/main/scala/org/apache/linkis/manager/engineplugin/appconn/executor/AppConnEngineConnExecutor.scala index ebe3a0f5c7..52ea42ee9e 100644 --- a/dss-appconn/linkis-appconn-engineplugin/src/main/scala/org/apache/linkis/manager/engineplugin/appconn/executor/AppConnEngineConnExecutor.scala +++ b/dss-appconn/linkis-appconn-engineplugin/src/main/scala/org/apache/linkis/manager/engineplugin/appconn/executor/AppConnEngineConnExecutor.scala @@ -18,9 +18,9 @@ package org.apache.linkis.manager.engineplugin.appconn.executor import java.util import java.util.concurrent.{ConcurrentHashMap, TimeUnit} - import com.webank.wedatasphere.dss.appconn.core.AppConn import com.webank.wedatasphere.dss.appconn.core.ext.OnlyDevelopmentAppConn +import com.webank.wedatasphere.dss.appconn.loader.utils.AppConnUtils import com.webank.wedatasphere.dss.appconn.manager.AppConnManager import com.webank.wedatasphere.dss.common.label.{DSSLabel, EnvDSSLabel, LabelKeyConvertor} import com.webank.wedatasphere.dss.common.utils.DSSCommonUtils @@ -30,6 +30,7 @@ import com.webank.wedatasphere.dss.standard.app.development.listener.ref.{AsyncE import com.webank.wedatasphere.dss.standard.app.sso.Workspace import com.webank.wedatasphere.dss.standard.common.desc.AppInstance import com.webank.wedatasphere.dss.standard.common.entity.ref.ResponseRef +import org.apache.commons.io.FileUtils import org.apache.commons.lang.StringUtils import org.apache.linkis.common.utils.{OverloadUtils, Utils} import org.apache.linkis.engineconn.computation.executor.async.AsyncConcurrentComputationExecutor @@ -46,6 +47,7 @@ import org.apache.linkis.protocol.engine.JobProgressInfo import org.apache.linkis.scheduler.executer.{AsynReturnExecuteResponse, ErrorExecuteResponse, ExecuteResponse, SuccessExecuteResponse} import org.apache.linkis.server.BDPJettyServerHelper +import java.io.File import scala.collection.JavaConverters._ class AppConnEngineConnExecutor(override val outputPrintLimit: Int, val id: Int) @@ -69,7 +71,7 @@ class AppConnEngineConnExecutor(override val outputPrintLimit: Int, val id: Int) def setUser(user: String): Unit = this.user = user override def executeLine(engineExecutorContext: EngineExecutionContext, code: String): ExecuteResponse = { - info(s"The execution code is: $code, runtime properties is: ${BDPJettyServerHelper.gson.toJson(engineExecutorContext.getProperties)}.") + info(s"The execution code is: $code, jobId is: ${engineExecutorContext.getJobId}, runtime properties is: ${BDPJettyServerHelper.gson.toJson(engineExecutorContext.getProperties)}.") val source = engineExecutorContext.getProperties.get(GovernanceConstant.TASK_SOURCE_MAP_KEY) match { case map: util.Map[String, Object] => map @@ -85,7 +87,7 @@ class AppConnEngineConnExecutor(override val outputPrintLimit: Int, val id: Int) val appConnName = getAppConnName(getValue(engineExecutorContext.getProperties, NODE_TYPE)) val appConn = AppConnManager.getAppConnManager.getAppConn(appConnName) if (appConn == null) { - error(s"Cannot find AppConn $appConnName.") + error(s"Cannot find AppConn $appConnName. jobId is: ${engineExecutorContext.getJobId}") throw AppConnExecuteFailedException(510001, "Cannot Find appConnName: " + appConnName) } val labels = engineExecutorContext.getProperties.get("labels").toString @@ -101,6 +103,7 @@ class AppConnEngineConnExecutor(override val outputPrintLimit: Int, val id: Int) case map: util.Map[String, Object] => map case _ => new util.HashMap[String, Object]() } + info(s"try to execute appconn job for jobId: ${engineExecutorContext.getJobId}") val responseRef = Utils.tryCatch { AppConnExecutionUtils.tryToOperation(refExecutionService, getValue(engineExecutorContext.getProperties, CONTEXT_ID_KEY), getValue(source, PROJECT_NAME_STR), new ExecutionRequestRefContextImpl(engineExecutorContext, user, submitUser), @@ -236,6 +239,19 @@ class AppConnEngineConnExecutor(override val outputPrintLimit: Int, val id: Int) } super.killTask(taskID) } + + override def tryShutdown(): Boolean = { + val toClearPath = AppConnUtils.getAppConnHomePath + val file = new File(toClearPath) + if (!file.exists()) { + warn(s"the appconn path is not exists, path is: ${toClearPath}") + } + info(s"try to clear dss-appconns for this engine, path is: ${toClearPath}") + Utils.tryCatch(FileUtils.deleteDirectory(file))(t => { + error(s"you have no permission to delete this path: ${toClearPath}, error: $t") + }) + super.tryShutdown() + } } diff --git a/dss-appconn/linkis-appconn-engineplugin/src/main/scala/org/apache/linkis/manager/engineplugin/appconn/launch/AppConnProcessEngineConnLaunchBuilder.scala b/dss-appconn/linkis-appconn-engineplugin/src/main/scala/org/apache/linkis/manager/engineplugin/appconn/launch/AppConnProcessEngineConnLaunchBuilder.scala index b1e09691a6..43f27c6f57 100644 --- a/dss-appconn/linkis-appconn-engineplugin/src/main/scala/org/apache/linkis/manager/engineplugin/appconn/launch/AppConnProcessEngineConnLaunchBuilder.scala +++ b/dss-appconn/linkis-appconn-engineplugin/src/main/scala/org/apache/linkis/manager/engineplugin/appconn/launch/AppConnProcessEngineConnLaunchBuilder.scala @@ -36,4 +36,7 @@ class AppConnProcessEngineConnLaunchBuilder extends JavaProcessEngineConnLaunchB super.getBmlResources } + override def enablePublicModule: Boolean = true + + } diff --git a/dss-appconn/pom.xml b/dss-appconn/pom.xml index e93fbe43fb..db2bb78104 100644 --- a/dss-appconn/pom.xml +++ b/dss-appconn/pom.xml @@ -21,7 +21,7 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 ../pom.xml 4.0.0 diff --git a/dss-apps/dss-apiservice-server/pom.xml b/dss-apps/dss-apiservice-server/pom.xml index 23b7687761..ee30668d96 100644 --- a/dss-apps/dss-apiservice-server/pom.xml +++ b/dss-apps/dss-apiservice-server/pom.xml @@ -21,7 +21,7 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 ../../pom.xml 4.0.0 @@ -204,23 +204,13 @@ - - - - - - + hk2-api org.glassfish.hk2 2.4.0-b34 - - - - - org.modelmapper modelmapper @@ -246,7 +236,6 @@ test - com.h2database h2 @@ -285,67 +274,23 @@ 1.5.2 test - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - xstream com.thoughtworks.xstream 1.4.19 + + com.webank.wedatasphere.dss + dss-sso-integration-standard + ${dss.version} + provided + + + + com.sun.jersey + jersey-core + 1.19.1 + diff --git a/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/exception/BeanValidationExceptionMapper.java b/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/exception/BeanValidationExceptionMapper.java index 0c1bc10613..ff63a7b74a 100644 --- a/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/exception/BeanValidationExceptionMapper.java +++ b/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/exception/BeanValidationExceptionMapper.java @@ -16,24 +16,20 @@ package com.webank.wedatasphere.dss.apiservice.core.exception; -import com.webank.wedatasphere.dss.common.utils.MessageUtils; import org.apache.linkis.server.Message; import javax.validation.ConstraintViolationException; import javax.validation.ValidationException; -import javax.ws.rs.core.Response; -import javax.ws.rs.ext.ExceptionMapper; import javax.ws.rs.ext.Provider; @Provider -public class BeanValidationExceptionMapper implements ExceptionMapper { - @Override - public Response toResponse(ValidationException exception) { +public class BeanValidationExceptionMapper { + public Message toResponse(ValidationException exception) { StringBuilder stringBuilder = new StringBuilder(); ((ConstraintViolationException)exception) .getConstraintViolations().forEach(constraintViolation -> stringBuilder.append(constraintViolation.getMessage()).append(";")); Message message = Message.error("Bean validation error[实例校验出错], detail:" + stringBuilder.toString()); - return MessageUtils.messageToResponse(message); + return message; } } diff --git a/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/restful/ApiServiceCoreRestfulApi.java b/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/restful/ApiServiceCoreRestfulApi.java index 9fd4e21309..313f6a0612 100644 --- a/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/restful/ApiServiceCoreRestfulApi.java +++ b/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/restful/ApiServiceCoreRestfulApi.java @@ -26,6 +26,11 @@ import com.webank.wedatasphere.dss.apiservice.core.vo.ApiVersionVo; import com.webank.wedatasphere.dss.apiservice.core.vo.ApprovalVo; import com.webank.wedatasphere.dss.apiservice.core.vo.QueryParamVo; +import com.webank.wedatasphere.dss.common.auditlog.OperateTypeEnum; +import com.webank.wedatasphere.dss.common.auditlog.TargetTypeEnum; +import com.webank.wedatasphere.dss.common.utils.AuditLogUtils; +import com.webank.wedatasphere.dss.standard.app.sso.Workspace; +import com.webank.wedatasphere.dss.standard.sso.utils.SSOHelper; import org.apache.commons.collections.MapUtils; import org.apache.commons.lang.StringUtils; import org.apache.linkis.server.Message; @@ -61,44 +66,50 @@ public class ApiServiceCoreRestfulApi { @Autowired private Validator beanValidator; + @Autowired + private HttpServletRequest httpServletRequest; private static final Pattern WRITABLE_PATTERN = Pattern.compile("^\\s*(insert|update|delete|drop|alter|create).*", Pattern.CASE_INSENSITIVE | Pattern.DOTALL); + /** + * 新建一个ApiService. + * 走scriptis发起的请求用这个接口, + */ @RequestMapping(value = "/api",method = RequestMethod.POST) - public Message insert(@RequestBody ApiServiceVo apiService, HttpServletRequest req) { + public Message insert(@RequestBody ApiServiceVo apiServiceVo) { return ApiUtils.doAndResponse(() -> { - if (apiService.getWorkspaceId() == null){ - apiService.setWorkspaceId(180L); + if (apiServiceVo.getWorkspaceId() == null){ + apiServiceVo.setWorkspaceId(180L); } - if (StringUtils.isBlank(apiService.getAliasName())) { + if (StringUtils.isBlank(apiServiceVo.getAliasName())) { return Message.error("'api service alias name' is missing[缺少中文名]"); } - if (StringUtils.isBlank(apiService.getScriptPath())) { + if (StringUtils.isBlank(apiServiceVo.getScriptPath())) { return Message.error("'api service script path' is missing[缺少脚本路径]"); } - if (StringUtils.isBlank(apiService.getContent())) { + if (StringUtils.isBlank(apiServiceVo.getContent())) { return Message.error("'api service script content' is missing[缺少脚本内容]"); } - if (null == apiService.getWorkspaceId()) { + if (null == apiServiceVo.getWorkspaceId()) { return Message.error("'api service workspaceId ' is missing[缺少工作空间ID]"); } - if (apiService.getContent().contains(";")) { - if(!apiService.getContent().toLowerCase().startsWith("use ")) { + if (apiServiceVo.getContent().contains(";")) { + if(!apiServiceVo.getContent().toLowerCase().startsWith("use ")) { return Message.error("'api service script content exists semicolon[脚本内容包含分号]"); } } // check data change script - if (WRITABLE_PATTERN.matcher(apiService.getContent()).matches()) { + if (WRITABLE_PATTERN.matcher(apiServiceVo.getContent()).matches()) { return Message.error("'api service script content' only supports query[脚本内容只支持查询语句]"); } - Map metadata = apiService.getMetadata(); - if (apiService.getScriptPath().endsWith(".jdbc")) { + Map metadata = apiServiceVo.getMetadata(); + if (apiServiceVo.getScriptPath().endsWith(".jdbc")) { if (MapUtils.isEmpty(metadata)) { return Message.error("'api service metadata' is missing[请选择数据源]"); } @@ -114,13 +125,13 @@ public Message insert(@RequestBody ApiServiceVo apiService, HttpServletRequest r } } - String userName = SecurityFilter.getLoginUsername(req); - Set> result = beanValidator.validate(apiService, Default.class); + String userName = SecurityFilter.getLoginUsername(httpServletRequest); + Set> result = beanValidator.validate(apiServiceVo, Default.class); if (result.size() > 0) { throw new ConstraintViolationException(result); } - ApprovalVo approvalVo = apiService.getApprovalVo(); + ApprovalVo approvalVo = apiServiceVo.getApprovalVo(); // if (StringUtils.isBlank(approvalVo.getApprovalName())) { // return Message.error("'approvalName' is missing[缺少审批单名字]"); @@ -130,48 +141,57 @@ public Message insert(@RequestBody ApiServiceVo apiService, HttpServletRequest r return Message.error("'applyUser' is missing[缺少申请用户名字]"); } - apiService.setCreator(userName); - apiService.setModifier(userName); - this.apiService.save(apiService); - return Message.ok().data("insert_id", apiService.getId()).data("approval_no",approvalVo.getApprovalNo()); + apiServiceVo.setCreator(userName); + apiServiceVo.setModifier(userName); + Workspace workspace = SSOHelper.getWorkspace(httpServletRequest); + this.apiService.save(apiServiceVo); + AuditLogUtils.printLog(userName,workspace.getWorkspaceId(), workspace.getWorkspaceName(), TargetTypeEnum.APISERVICE, + apiServiceVo.getId(),apiServiceVo.getName(), OperateTypeEnum.CREATE,apiServiceVo); + return Message.ok().data("insert_id", apiServiceVo.getId()).data("approval_no",approvalVo.getApprovalNo()); }, "/apiservice/api", "Fail to insert service api[新增服务api失败]"); } + /** + * 也是创建apiservice,给数据采集用的。 + * 因为在bdp中,需要区分运维用户,业务用户,系统用户,所以入参是不一样的。 + * @param apiServiceVo + * @return + */ @RequestMapping(value = "/create",method = RequestMethod.POST) - public Message create(@RequestBody ApiServiceVo apiService, HttpServletRequest req) { + public Message create(@RequestBody ApiServiceVo apiServiceVo) { return ApiUtils.doAndResponse(() -> { - if (apiService.getWorkspaceId() == null){ - apiService.setWorkspaceId(180L); + if (apiServiceVo.getWorkspaceId() == null){ + apiServiceVo.setWorkspaceId(180L); } - if (StringUtils.isBlank(apiService.getAliasName())) { + if (StringUtils.isBlank(apiServiceVo.getAliasName())) { return Message.error("'api service alias name' is missing[缺少中文名]"); } - if (StringUtils.isBlank(apiService.getScriptPath())) { + if (StringUtils.isBlank(apiServiceVo.getScriptPath())) { return Message.error("'api service script path' is missing[缺少脚本路径]"); } - if (StringUtils.isBlank(apiService.getContent())) { + if (StringUtils.isBlank(apiServiceVo.getContent())) { return Message.error("'api service script content' is missing[缺少脚本内容]"); } - if (null == apiService.getWorkspaceId()) { + if (null == apiServiceVo.getWorkspaceId()) { return Message.error("'api service workspaceId ' is missing[缺少工作空间ID]"); } - if (apiService.getContent().contains(";")) { - if(!apiService.getContent().toLowerCase().startsWith("use ")) { + if (apiServiceVo.getContent().contains(";")) { + if(!apiServiceVo.getContent().toLowerCase().startsWith("use ")) { return Message.error("'api service script content exists semicolon[脚本内容包含分号]"); } } // check data change script - if (WRITABLE_PATTERN.matcher(apiService.getContent()).matches()) { + if (WRITABLE_PATTERN.matcher(apiServiceVo.getContent()).matches()) { return Message.error("'api service script content' only supports query[脚本内容只支持查询语句]"); } - Map metadata = apiService.getMetadata(); - if (apiService.getScriptPath().endsWith(".jdbc")) { + Map metadata = apiServiceVo.getMetadata(); + if (apiServiceVo.getScriptPath().endsWith(".jdbc")) { if (MapUtils.isEmpty(metadata)) { return Message.error("'api service metadata' is missing[请选择数据源]"); } @@ -187,13 +207,14 @@ public Message create(@RequestBody ApiServiceVo apiService, HttpServletRequest r } } - String userName = SecurityFilter.getLoginUsername(req); - Set> result = beanValidator.validate(apiService, Default.class); + String userName = SecurityFilter.getLoginUsername(httpServletRequest); + LOG.info("user {} begin to create api, params: {}", userName, apiServiceVo); + Set> result = beanValidator.validate(apiServiceVo, Default.class); if (result.size() > 0) { throw new ConstraintViolationException(result); } - ApprovalVo approvalVo = apiService.getApprovalVo(); + ApprovalVo approvalVo = apiServiceVo.getApprovalVo(); if (StringUtils.isBlank(approvalVo.getApprovalName())) { return Message.error("'approvalName' is missing[缺少审批单名字]"); @@ -203,44 +224,46 @@ public Message create(@RequestBody ApiServiceVo apiService, HttpServletRequest r return Message.error("'applyUser' is missing[缺少申请用户名字]"); } - apiService.setCreator(userName); - apiService.setModifier(userName); - this.apiService.saveByApp(apiService); - return Message.ok().data("insert_id", apiService.getId()).data("approval_no",approvalVo.getApprovalNo()); + apiServiceVo.setCreator(userName); + apiServiceVo.setModifier(userName); + ApiServiceVo saveResult = apiService.saveByApp(apiServiceVo); + Workspace workspace = SSOHelper.getWorkspace(httpServletRequest); + AuditLogUtils.printLog(userName, workspace.getWorkspaceId(), workspace.getWorkspaceName(),TargetTypeEnum.APISERVICE, + saveResult.getId(),saveResult.getName() ,OperateTypeEnum.CREATE,apiServiceVo); + return Message.ok().data("insert_id", apiServiceVo.getId()).data("approval_no",approvalVo.getApprovalNo()); }, "/apiservice/api", "Fail to insert service api[新增服务api失败]"); } @RequestMapping(value = "/api/{api_service_version_id}",method = RequestMethod.PUT) - public Message update(@RequestBody ApiServiceVo apiService, - @PathVariable("api_service_version_id") Long apiServiceVersionId, - HttpServletRequest req) { + public Message update(@RequestBody ApiServiceVo apiServiceVo, + @PathVariable("api_service_version_id") Long apiServiceVersionId) { return ApiUtils.doAndResponse(() -> { - if (StringUtils.isBlank(apiService.getScriptPath())) { + if (StringUtils.isBlank(apiServiceVo.getScriptPath())) { return Message.error("'api service script path' is missing[缺少脚本路径]"); } if(apiServiceVersionId !=0) { - if (StringUtils.isBlank(apiService.getPath())) { + if (StringUtils.isBlank(apiServiceVo.getPath())) { return Message.error("'api service api path' is missing[缺少api路径]"); } } - if (StringUtils.isBlank(apiService.getContent())) { + if (StringUtils.isBlank(apiServiceVo.getContent())) { return Message.error("'api service script content' is missing[缺少脚本内容]"); } - if (null == apiService.getWorkspaceId()) { + if (null == apiServiceVo.getWorkspaceId()) { return Message.error("'api service workspaceId ' is missing[缺少工作空间ID]"); } - if (null == apiService.getTargetServiceId()) { + if (null == apiServiceVo.getTargetServiceId()) { return Message.error("'api service update to target service id ' is missing[缺少更新目标服务ID]"); } - if (apiService.getContent().contains(";")) { + if (apiServiceVo.getContent().contains(";")) { return Message.error("'api service script content exists semicolon[脚本内容包含分号]"); } - ApprovalVo approvalVo = apiService.getApprovalVo(); + ApprovalVo approvalVo = apiServiceVo.getApprovalVo(); // if (StringUtils.isBlank(approvalVo.getApprovalName())) { // return Message.error("'approvalName' is missing[缺少审批单名字]"); @@ -254,12 +277,12 @@ public Message update(@RequestBody ApiServiceVo apiService, // } // check data change script - if (WRITABLE_PATTERN.matcher(apiService.getContent()).matches()) { + if (WRITABLE_PATTERN.matcher(apiServiceVo.getContent()).matches()) { return Message.error("'api service script content' only supports query[脚本内容只支持查询语句]"); } - Map metadata = apiService.getMetadata(); - if (apiService.getScriptPath().endsWith(".jdbc")) { + Map metadata = apiServiceVo.getMetadata(); + if (apiServiceVo.getScriptPath().endsWith(".jdbc")) { if (MapUtils.isEmpty(metadata)) { return Message.error("'api service metadata' is missing[请选择数据源]"); } @@ -275,16 +298,20 @@ public Message update(@RequestBody ApiServiceVo apiService, } } - String userName = SecurityFilter.getLoginUsername(req); + String userName = SecurityFilter.getLoginUsername(httpServletRequest); + LOG.info("user {} try to update service api, params: {}", userName, apiServiceVo); // Bean validation - Set> result = beanValidator.validate(apiService, Default.class); + Set> result = beanValidator.validate(apiServiceVo, Default.class); if (result.size() > 0) { throw new ConstraintViolationException(result); } - apiService.setLatestVersionId(apiServiceVersionId); - apiService.setModifier(userName); - apiService.setModifyTime(Calendar.getInstance().getTime()); - this.apiService.update(apiService); + apiServiceVo.setLatestVersionId(apiServiceVersionId); + apiServiceVo.setModifier(userName); + apiServiceVo.setModifyTime(Calendar.getInstance().getTime()); + ApiServiceVo updatedResult= apiService.update(apiServiceVo); + Workspace workspace = SSOHelper.getWorkspace(httpServletRequest); + AuditLogUtils.printLog(userName, workspace.getWorkspaceId(), workspace.getWorkspaceName(),TargetTypeEnum.APISERVICE, + updatedResult.getId(),updatedResult.getName() ,OperateTypeEnum.UPDATE,apiServiceVo); return Message.ok().data("update_id", apiServiceVersionId); }, "/apiservice/api/" + apiServiceVersionId, "Fail to update service api[更新服务api失败]"); } @@ -297,9 +324,8 @@ public Message query(@RequestParam(required = false, name = "name") String name, @RequestParam(required = false, name = "tag") String tag, @RequestParam(required = false, name = "status") Integer status, @RequestParam(required = false, name = "creator") String creator, - @RequestParam(required = false, name = "workspaceId") Integer workspaceId, - HttpServletRequest req) { - String userName = SecurityFilter.getLoginUsername(req); + @RequestParam(required = false, name = "workspaceId") Integer workspaceId) { + String userName = SecurityFilter.getLoginUsername(httpServletRequest); return ApiUtils.doAndResponse(() -> { if (null == workspaceId) { @@ -317,9 +343,8 @@ public Message query(@RequestParam(required = false, name = "name") String name, @RequestMapping(value = "/getUserServices",method = RequestMethod.GET) - public Message getUserServices(@RequestParam(required = false, name = "workspaceId") Integer workspaceId, - HttpServletRequest req){ - String userName = SecurityFilter.getLoginUsername(req); + public Message getUserServices(@RequestParam(required = false, name = "workspaceId") Integer workspaceId){ + String userName = SecurityFilter.getLoginUsername(httpServletRequest); return ApiUtils.doAndResponse(() -> { if(!this.apiService.checkUserWorkspace(userName,workspaceId) ){ return Message.error("'api service getUserServices workspaceId' is wrong[该用户不属于该工作空间Id]"); @@ -332,8 +357,8 @@ public Message getUserServices(@RequestParam(required = false, name = "workspace @RequestMapping(value = "/tags",method = RequestMethod.GET) - public Message query( HttpServletRequest req,@RequestParam(required = false, name = "workspaceId") Integer workspaceId) { - String userName = SecurityFilter.getLoginUsername(req); + public Message query(@RequestParam(required = false, name = "workspaceId") Integer workspaceId) { + String userName = SecurityFilter.getLoginUsername(httpServletRequest); return ApiUtils.doAndResponse(() -> { List tags= apiService.queryAllTags(userName,workspaceId); @@ -345,10 +370,9 @@ public Message query( HttpServletRequest req,@RequestParam(required = false, nam @RequestMapping(value = "/query",method = RequestMethod.GET) - public Message queryByScriptPath(@RequestParam(required = false, name = "scriptPath") String scriptPath, - HttpServletRequest req) { + public Message queryByScriptPath(@RequestParam(required = false, name = "scriptPath") String scriptPath) { return ApiUtils.doAndResponse(() -> { - String userName = SecurityFilter.getLoginUsername(req); + String userName = SecurityFilter.getLoginUsername(httpServletRequest); if (StringUtils.isBlank(scriptPath)) { return Message.error("'api service scriptPath' is missing[缺少脚本路径]"); } @@ -370,9 +394,8 @@ public Message queryByScriptPath(@RequestParam(required = false, name = "scriptP } @RequestMapping(value = "/queryById",method = RequestMethod.GET) - public Message queryById(@RequestParam(required = false, name = "id") Long id, - HttpServletRequest req) { - String userName = SecurityFilter.getLoginUsername(req); + public Message queryById(@RequestParam(required = false, name = "id") Long id) { + String userName = SecurityFilter.getLoginUsername(httpServletRequest); return ApiUtils.doAndResponse(() -> { if (id==null) { return Message.error("'api service id' is missing[缺少服务ID]"); @@ -414,68 +437,81 @@ public Message checkName(@RequestParam(required = false, name = "name") String n } @RequestMapping(value = "/apiDisable",method = RequestMethod.GET) - public Message apiDisable(@RequestParam(required = false, name = "id") Long id, - HttpServletRequest req) { + public Message apiDisable(@RequestParam(required = false, name = "id") Long id) { return ApiUtils.doAndResponse(() -> { - String userName = SecurityFilter.getLoginUsername(req); + String userName = SecurityFilter.getLoginUsername(httpServletRequest); if (null == id) { return Message.error("'api service api id' is missing[缺少api id]"); } - boolean resultFlag = apiService.disableApi(id,userName); + LOG.info("user {} begin to disable service api, id: {}", userName, id); + ApiServiceVo apiServiceVo = apiService.queryById(id,userName); + boolean resultFlag = apiService.disableApi(userName,apiServiceVo); + Workspace workspace = SSOHelper.getWorkspace(httpServletRequest); + AuditLogUtils.printLog(userName, workspace.getWorkspaceId(), workspace.getWorkspaceName(),TargetTypeEnum.APISERVICE, + apiServiceVo.getId(),apiServiceVo.getName() ,OperateTypeEnum.DISABLE,apiServiceVo); return Message.ok().data("result", resultFlag); }, "/apiservice/apiDisable", "Fail to disable api[禁用api失败]"); } @RequestMapping(value = "/apiEnable",method = RequestMethod.GET) - public Message apiEnable(@RequestParam(required = false, name = "id") Long id, - HttpServletRequest req) { + public Message apiEnable(@RequestParam(required = false, name = "id") Long id) { return ApiUtils.doAndResponse(() -> { - String userName = SecurityFilter.getLoginUsername(req); + String userName = SecurityFilter.getLoginUsername(httpServletRequest); if (null == id) { return Message.error("'api service api id' is missing[缺少api id]"); } - boolean resultFlag = apiService.enableApi(id,userName); + LOG.info("user {} begin to enable service api, id: {}", userName, id); + ApiServiceVo apiServiceVo = apiService.queryById(id,userName); + boolean resultFlag = apiService.enableApi(userName,apiServiceVo); + Workspace workspace = SSOHelper.getWorkspace(httpServletRequest); + AuditLogUtils.printLog(userName, workspace.getWorkspaceId(), workspace.getWorkspaceName(),TargetTypeEnum.APISERVICE, + apiServiceVo.getId(),apiServiceVo.getName() ,OperateTypeEnum.ENABLE,apiServiceVo); return Message.ok().data("result", resultFlag); }, "/apiservice/apiEnable", "Fail to enable api[启用api失败]"); } @RequestMapping(value = "/apiDelete",method = RequestMethod.GET) - public Message apiDelete(@RequestParam(required = false, name = "id") Long id, - HttpServletRequest req) { + public Message apiDelete(@RequestParam(required = false, name = "id") Long id) { //目前暂时不实际删除数据,只做不可见和不可用。 return ApiUtils.doAndResponse(() -> { - String userName = SecurityFilter.getLoginUsername(req); + String userName = SecurityFilter.getLoginUsername(httpServletRequest); if (null == id) { return Message.error("'api service api id' is missing[缺少api id]"); } - boolean resultFlag = apiService.deleteApi(id,userName); + LOG.info("user {} try to delete service api, id: {}", userName, id); + ApiServiceVo apiServiceVo = apiService.queryById(id,userName); + boolean resultFlag = apiService.deleteApi(userName,apiServiceVo); + Workspace workspace = SSOHelper.getWorkspace(httpServletRequest); + AuditLogUtils.printLog(userName, workspace.getWorkspaceId(), workspace.getWorkspaceName(),TargetTypeEnum.APISERVICE, + apiServiceVo.getId(),apiServiceVo.getName() ,OperateTypeEnum.DELETE,apiServiceVo); return Message.ok().data("result", resultFlag); }, "/apiservice/apiDelete", "Fail to delete api[删除api失败]"); } @RequestMapping(value = "/apiCommentUpdate",method = RequestMethod.POST) - public Message apiCommentUpdate(HttpServletRequest req, - @RequestBody ApiCommentUpdateRequest apiCommentUpdateRequest) { + public Message apiCommentUpdate(@RequestBody ApiCommentUpdateRequest apiCommentUpdateRequest) { Long id = apiCommentUpdateRequest.getId(); String comment = apiCommentUpdateRequest.getComment(); //目前暂时不实际删除数据,只做不可见和不可用。 return ApiUtils.doAndResponse(() -> { - String userName = SecurityFilter.getLoginUsername(req); + String userName = SecurityFilter.getLoginUsername(httpServletRequest); if (null == id) { return Message.error("'api service api id' is missing[缺少api id]"); } - boolean resultFlag = apiService.updateComment(id,comment,userName); + ApiServiceVo apiServiceVo = apiService.queryById(id,userName); + boolean resultFlag = apiService.updateComment(comment,userName,apiServiceVo); + Workspace workspace = SSOHelper.getWorkspace(httpServletRequest); + AuditLogUtils.printLog(userName, workspace.getWorkspaceId(), workspace.getWorkspaceName(),TargetTypeEnum.APISERVICE, + apiServiceVo.getId(),apiServiceVo.getName() ,OperateTypeEnum.UPDATE,apiCommentUpdateRequest); return Message.ok().data("result", resultFlag); - }, "/apiservice/apiDelete", "Fail to delete api[删除api失败]"); + }, "/apiservice/apiDelete", "Fail to update comment api[更新评论失败]"); } @RequestMapping(value = "/apiParamQuery",method = RequestMethod.GET) public Message apiParamQuery(@RequestParam(required = false, name = "scriptPath") String scriptPath, - @RequestParam(required = false, name = "versionId") Long versionId, - HttpServletRequest req) { + @RequestParam(required = false, name = "versionId") Long versionId) { return ApiUtils.doAndResponse(() -> { - String userName = SecurityFilter.getLoginUsername(req); if (StringUtils.isEmpty(scriptPath)) { return Message.error("'api service api scriptPath' is missing[缺少api scriptPath]"); } @@ -488,10 +524,9 @@ public Message apiParamQuery(@RequestParam(required = false, name = "scriptPath" } @RequestMapping(value = "/apiVersionQuery",method = RequestMethod.GET) - public Message apiVersionQuery(@RequestParam(required = false, name = "serviceId") Long serviceId, - HttpServletRequest req) { + public Message apiVersionQuery(@RequestParam(required = false, name = "serviceId") Long serviceId) { return ApiUtils.doAndResponse(() -> { - String userName = SecurityFilter.getLoginUsername(req); + String userName = SecurityFilter.getLoginUsername(httpServletRequest); if (null == serviceId) { return Message.error("'api service api serviceId' is missing[缺少api serviceId]"); } @@ -503,9 +538,8 @@ public Message apiVersionQuery(@RequestParam(required = false, name = "serviceId } @RequestMapping(value = "/apiContentQuery",method = RequestMethod.GET) - public Message apiContentQuery(@RequestParam(required = false, name = "versionId") Long versionId, - HttpServletRequest req) { - String userName = SecurityFilter.getLoginUsername(req); + public Message apiContentQuery(@RequestParam(required = false, name = "versionId") Long versionId) { + String userName = SecurityFilter.getLoginUsername(httpServletRequest); return ApiUtils.doAndResponse(() -> { if (null== versionId) { return Message.error("'api service api versionId' is missing[缺少api versionId]"); diff --git a/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/restful/ApiServiceExecuteRestfulApi.java b/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/restful/ApiServiceExecuteRestfulApi.java index 91e4278247..f6a66fc6ed 100644 --- a/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/restful/ApiServiceExecuteRestfulApi.java +++ b/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/restful/ApiServiceExecuteRestfulApi.java @@ -288,18 +288,18 @@ public Message getTaskByID(HttpServletRequest req, @PathVariable("id") Long task } private Message getResponse(String user,String path, QueryRequest queryRequest, String httpMethod) { - Response response = ApiUtils.doAndResponse(() -> { + Message response = ApiUtils.doAndResponse(() -> { validParam(queryRequest); String token = queryRequest.getParams().get(ApiServiceConfiguration.API_SERVICE_TOKEN_KEY.getValue()).toString(); - MessageVo messageVo = null; + Message messageVo = null; ApiServiceToken tokenDetail = null; boolean isParseRight = true; try { tokenDetail = JwtManager.parseToken(token); }catch (Exception e) { isParseRight = false; - messageVo = new MessageVo().setData("token解析错误,该token无效!"); + messageVo = Message.error("token解析错误,该token无效!"); } if(false == isParseRight) { return messageVo; @@ -310,20 +310,17 @@ private Message getResponse(String user,String path, QueryRequest queryRequest, queryRequest.getParams() == null ? new HashMap<>() : queryRequest.getParams(), queryRequest.getModuleName(), httpMethod,tokenDetail,user); if(null == query) { - messageVo = new MessageVo().setMessage("用户任务执行出错,用户参数错误!").setStatus(1); + messageVo = Message.error("用户输入了非法关键字!"); return messageVo; } - HashMap queryRes = new HashMap<>(); - queryRes.put("taskId",query.getTaskId()); - queryRes.put("execId",query.getExecId()); - messageVo = new MessageVo().setData(queryRes); + messageVo = Message.ok().data("taskId",query.getTaskId()).data("execId",query.getExecId()); }else { - messageVo = new MessageVo().setData("Token is not correct"); + messageVo = Message.error("Token is not correct"); } return messageVo; }); - return convertMessage(response); + return response; } //convert Response to Message diff --git a/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/service/ApiService.java b/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/service/ApiService.java index 8937efaeb3..a6571dc791 100644 --- a/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/service/ApiService.java +++ b/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/service/ApiService.java @@ -34,14 +34,14 @@ public interface ApiService { - void saveByApp(ApiServiceVo apiService) throws Exception; + ApiServiceVo saveByApp(ApiServiceVo apiService) throws Exception; /** * Update * * @param oneService oneService info */ - void update(ApiServiceVo oneService) throws Exception; + ApiServiceVo update(ApiServiceVo oneService) throws Exception; /** * query @@ -70,21 +70,20 @@ public interface ApiService { /** * enable api - * @param id api record id * @return */ - Boolean enableApi(Long id,String userName); + Boolean enableApi(String userName,ApiServiceVo apiServiceVo); /** * disable api - * @param id api record id + * @param apiServiceVo 需要禁用的apiservice * @return */ - Boolean disableApi(Long id,String userName); + Boolean disableApi(String userName,ApiServiceVo apiServiceVo); - Boolean deleteApi(Long id,String userName); + Boolean deleteApi(String userName,ApiServiceVo apiServiceVo); - Boolean updateComment(Long id,String comment,String userName); + Boolean updateComment(String comment, String userName,ApiServiceVo apiServiceVo); ApiServiceVo queryById(Long id,String userName); diff --git a/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/service/impl/ApiServiceImpl.java b/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/service/impl/ApiServiceImpl.java index 426de81a2e..f42bd1e142 100644 --- a/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/service/impl/ApiServiceImpl.java +++ b/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/service/impl/ApiServiceImpl.java @@ -165,7 +165,7 @@ public void save(ApiServiceVo apiService) throws Exception { @Override @Transactional(rollbackFor = Exception.class) - public void saveByApp(ApiServiceVo apiService) throws Exception { + public ApiServiceVo saveByApp(ApiServiceVo apiService) throws Exception { String user = apiService.getCreator(); String resourceId = null; try { @@ -212,6 +212,7 @@ public void saveByApp(ApiServiceVo apiService) throws Exception { //insert a token record for self genTokenForPublisher(apiService,apiVersionVo.getId()); + return apiService; } catch (Exception e) { LOG.error("one service insert error", e); if (StringUtils.isNotBlank(resourceId)) { @@ -230,7 +231,7 @@ public void saveByApp(ApiServiceVo apiService) throws Exception { @Override @Transactional(rollbackFor = Exception.class) - public void update(ApiServiceVo apiService) throws Exception { + public ApiServiceVo update(ApiServiceVo apiService) throws Exception { try { if(null!=apiService.getTargetServiceId()) { ApiVersionVo maxTargetApiVersionVo = getMaxVersion(apiService.getTargetServiceId()); @@ -288,6 +289,7 @@ public void update(ApiServiceVo apiService) throws Exception { //insert a token record for self genTokenForPublisher(apiService, apiServiceVersionVo.getId()); + return apiService; }else { throw new ApiServiceQueryException(800036,"Only can update the api service by owner! "); } @@ -414,14 +416,14 @@ public Integer queryCountByName(String name) { } @Override - public Boolean enableApi(Long id,String userName) { - ApiServiceVo apiServiceVo = apiServiceDao.queryById(id); + public Boolean enableApi(String userName,ApiServiceVo apiServiceVo) { + if(!checkUserWorkspace(userName,apiServiceVo.getWorkspaceId().intValue())){ LOG.error("api service check workspace error"); return false; } if(apiServiceVo.getCreator().equals(userName)) { - + long id=apiServiceVo.getId(); Integer updateCount = apiServiceDao.enableApi(id); List targetApiVersionList = apiServiceVersionDao.queryApiVersionByApiServiceId(id); ApiVersionVo maxTargetApiVersionVo = targetApiVersionList.stream().max(Comparator.comparing(ApiVersionVo::getVersion)).orElse(null); @@ -435,13 +437,13 @@ public Boolean enableApi(Long id,String userName) { } @Override - public Boolean disableApi(Long id,String userName) { - ApiServiceVo apiServiceVo = apiServiceDao.queryById(id); + public Boolean disableApi(String userName,ApiServiceVo apiServiceVo) { if(!checkUserWorkspace(userName,apiServiceVo.getWorkspaceId().intValue())){ LOG.error("api service check workspace error"); return false; } if(apiServiceVo.getCreator().equals(userName)) { + Long id=apiServiceVo.getId(); Integer updateCount = apiServiceDao.disableApi(id); apiServiceTokenManagerDao.disableTokenStatusByApiId(id); apiServiceVersionDao.updateAllApiVersionStatusByApiServiceId(id, 0); @@ -453,13 +455,13 @@ public Boolean disableApi(Long id,String userName) { @Override - public Boolean deleteApi(Long id,String userName) { - ApiServiceVo apiServiceVo = apiServiceDao.queryById(id); + public Boolean deleteApi(String userName,ApiServiceVo apiServiceVo) { if(!checkUserWorkspace(userName,apiServiceVo.getWorkspaceId().intValue())){ LOG.error("api service check workspace error"); return false; } if(apiServiceVo.getCreator().equals(userName)) { + Long id = apiServiceVo.getId(); Integer updateCount = apiServiceDao.deleteApi(id); apiServiceTokenManagerDao.disableTokenStatusByApiId(id); apiServiceVersionDao.updateAllApiVersionStatusByApiServiceId(id, 0); @@ -470,14 +472,13 @@ public Boolean deleteApi(Long id,String userName) { } @Override - public Boolean updateComment(Long id, String comment, String userName) { - ApiServiceVo apiServiceVo = apiServiceDao.queryById(id); + public Boolean updateComment(String comment, String userName,ApiServiceVo apiServiceVo) { if(!checkUserWorkspace(userName,apiServiceVo.getWorkspaceId().intValue())){ LOG.error("api service check workspace error"); return false; } if(apiServiceVo.getCreator().equals(userName)) { - Integer updateCount = apiServiceDao.updateApiServiceComment(id,comment); + Integer updateCount = apiServiceDao.updateApiServiceComment(apiServiceVo.getId(),comment); return updateCount > 0; }else { return false; @@ -490,7 +491,7 @@ private Map uploadBml(String userName, String scriptPath, Map variableList=null; if(metadata.entrySet().size() >0) { Variable[] v = VariableParser.getVariables(metadata); - variableList = Arrays.stream(v).filter(var -> !StringUtils.isEmpty(var.value())).collect(Collectors.toList()); + variableList = Arrays.stream(v).filter(var -> !StringUtils.isEmpty(var.getValue())).collect(Collectors.toList()); } if(variableList!=null) { @@ -519,7 +520,7 @@ private Map updateBml(String userName, String resourceId, String try { ScriptFsWriter writer = StorageScriptFsWriter.getScriptFsWriter(new FsPath(scriptPath), Consts.UTF_8.toString(), null); Variable[] v = VariableParser.getVariables(metadata); - List variableList = Arrays.stream(v).filter(var -> !StringUtils.isEmpty(var.value())).collect(Collectors.toList()); + List variableList = Arrays.stream(v).filter(var -> !StringUtils.isEmpty(var.getValue())).collect(Collectors.toList()); writer.addMetaData(new ScriptMetaData(variableList.toArray(new Variable[0]))); writer.addRecord(new ScriptRecord(scriptContent)); InputStream inputStream = writer.getInputStream(); diff --git a/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/service/impl/ApiServiceQueryServiceImpl.java b/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/service/impl/ApiServiceQueryServiceImpl.java index fbfb1f8033..b5054e140e 100644 --- a/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/service/impl/ApiServiceQueryServiceImpl.java +++ b/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/service/impl/ApiServiceQueryServiceImpl.java @@ -23,10 +23,8 @@ import com.webank.wedatasphere.dss.apiservice.core.bo.LinkisExecuteResult; import com.webank.wedatasphere.dss.apiservice.core.config.ApiServiceConfiguration; import com.webank.wedatasphere.dss.apiservice.core.constant.ParamType; -import com.webank.wedatasphere.dss.apiservice.core.constant.ParamTypeEnum; import com.webank.wedatasphere.dss.apiservice.core.constant.RequireEnum; import com.webank.wedatasphere.dss.apiservice.core.dao.*; -import com.webank.wedatasphere.dss.apiservice.core.exception.ApiExecuteException; import com.webank.wedatasphere.dss.apiservice.core.exception.ApiServiceQueryException; import com.webank.wedatasphere.dss.apiservice.core.execute.ApiServiceExecuteJob; import com.webank.wedatasphere.dss.apiservice.core.execute.DefaultApiServiceJob; @@ -37,22 +35,18 @@ import com.webank.wedatasphere.dss.apiservice.core.util.DateUtil; import com.webank.wedatasphere.dss.apiservice.core.util.SQLCheckUtil; import com.webank.wedatasphere.dss.apiservice.core.vo.*; -//import com.webank.wedatasphere.dss.oneservice.core.jdbc.JdbcUtil; import com.webank.wedatasphere.dss.apiservice.core.exception.ApiServiceRuntimeException; import com.webank.wedatasphere.dss.apiservice.core.service.ApiServiceQueryService; import com.webank.wedatasphere.dss.apiservice.core.util.AssertUtil; import com.webank.wedatasphere.dss.apiservice.core.util.ModelMapperUtil; -//import com.webank.wedatasphere.dss.oneservice.core.vo.*; import com.webank.wedatasphere.dss.apiservice.core.vo.ApiServiceVo; import org.apache.linkis.bml.client.BmlClient; import org.apache.linkis.bml.client.BmlClientFactory; import org.apache.linkis.bml.protocol.BmlDownloadResponse; import org.apache.linkis.common.io.FsPath; import org.apache.linkis.storage.source.FileSource; -import org.apache.linkis.storage.source.FileSource$; import org.apache.linkis.ujes.client.UJESClient; import org.apache.linkis.ujes.client.response.JobExecuteResult; -import org.apache.commons.collections4.MapUtils; import org.apache.commons.lang.StringUtils; import org.apache.commons.math3.util.Pair; import org.slf4j.Logger; @@ -141,11 +135,9 @@ public class ApiServiceQueryServiceImpl implements ApiServiceQueryService { @Autowired private ApiServiceTokenManagerDao apiServiceTokenManagerDao; - @Autowired private ApiService apiService; - @Autowired private ApiServiceAccessDao apiServiceAccessDao; @@ -169,9 +161,7 @@ public LinkisExecuteResult query(String path, ApiServiceToken tokenDetail, String loginUser) { // 根据path查询resourceId和version - // 得到metadata - // 执行查询 //path 必须唯一 ApiServiceVo apiServiceVo = apiServiceDao.queryByPath(path); @@ -241,10 +231,6 @@ public LinkisExecuteResult query(String path, } }); - -// AssertUtil.isTrue(MapUtils.isNotEmpty((Map) collect.getKey()), "数据源不能为空"); - - ApiServiceExecuteJob job = new DefaultApiServiceJob(); //sql代码封装成scala执行 job.setCode(ExecuteCodeHelper.packageCodeToExecute(executeCode, maxApiVersionVo.getMetadataInfo())); @@ -270,7 +256,6 @@ public LinkisExecuteResult query(String path, apiAccessVo.setAccessTime(DateUtil.getNow()); apiServiceAccessDao.addAccessRecord(apiAccessVo); - JobExecuteResult jobExecuteResult = LinkisJobSubmit.execute(job,ujesClient); //记录执行任务用户和代理用户关系,没有代理用户的统一设置为登录用户 @@ -280,7 +265,6 @@ public LinkisExecuteResult query(String path, apiServiceJob.setJobExecuteResult(jobExecuteResult); runJobs.put(jobExecuteResult.getTaskID(),apiServiceJob); - LinkisExecuteResult linkisExecuteResult = new LinkisExecuteResult(jobExecuteResult.getTaskID(), jobExecuteResult.getExecID()); return linkisExecuteResult; } catch (IOException e) { @@ -288,9 +272,6 @@ public LinkisExecuteResult query(String path, } } - - - @Override public ApiServiceVo queryByVersionId(String userName,Long versionId) throws ApiServiceQueryException { ApiVersionVo apiVersionVo = apiServiceVersionDao.queryApiVersionByVersionId(versionId); @@ -325,10 +306,8 @@ public List queryParamList(String scriptPath, Long versionId) { AssertUtil.notNull(targetApiVersionVo, "目标参数版本不存在,path=" + scriptPath+",version:"+versionId); - // todo~! List paramVoList = apiServiceParamDao.queryByVersionId(targetApiVersionVo.getId()); - List queryParamVoList = new ArrayList<>(); Map paramMap = paramVoList.stream() @@ -390,16 +369,16 @@ private Pair> queryBml(String userName, String resou InputStream inputStream = resource.inputStream(); - try (FileSource fileSource = FileSource$.MODULE$.create(new FsPath(scriptPath), inputStream)) { + try (FileSource fileSource = FileSource.create(new FsPath(scriptPath), inputStream)) { //todo 数组取了第一个 - collect = fileSource.collect()[0]; + Pair> sourcePair = fileSource.collect()[0]; + collect = new Pair<>(sourcePair.getKey(), new ArrayList<>(sourcePair.getValue())); bmlCache.put(key, collect); } } } } - return collect; } @@ -425,60 +404,6 @@ private Map queryConfigParam(long apiId, String version) { return collect; } - - - -// private Tuple3 getDatasourceInfo(final Map datasourceMap) { -// Tuple3 tuple3 = datasourceCache.getIfPresent(datasourceMap); -// -// if (tuple3 == null) { -// synchronized (this) { -// tuple3 = datasourceCache.getIfPresent(datasourceMap); -// if (tuple3 == null) { -// tuple3 = JdbcUtil.getDatasourceInfo(datasourceMap); -// datasourceCache.put(datasourceMap, tuple3); -// } -// } -// } -// -// return tuple3; -// } - -// private List> executeJob(String executeCode, -// Object datasourceMap, Map params) { -// -//// Tuple3 tuple3 = getDatasourceInfo((Map) datasourceMap); -//// final String jdbcUrl = tuple3._1().toString(); -//// final String username = tuple3._2().toString(); -//// final String password = tuple3._3().toString(); -// -//// NamedParameterJdbcTemplate namedParameterJdbcTemplate = datasourceService.getNamedParameterJdbcTemplate(jdbcUrl, username, password); -// -// String namedSql = genNamedSql(executeCode, params); -// -//// return namedParameterJdbcTemplate.query(namedSql, new MapSqlParameterSource(params), new ColumnAliasMapRowMapper()); -// -// } - - private static String genNamedSql(String executeCode, Map params) { - // 没有参数,无需生成namedSql - if (MapUtils.isEmpty(params)) { - return executeCode; - } - - for (String paramName : params.keySet()) { - for (String $name : new String[]{"'${" + paramName + "}'", "${" + paramName + "}", "\"${" + paramName + "}\""}) { - if (executeCode.contains($name)) { - executeCode = StringUtils.replace(executeCode, $name, ":" + paramName); - break; - } - } - } - - return executeCode; - } - - public static class ColumnAliasMapRowMapper implements RowMapper> { @Override public Map mapRow(ResultSet rs, int rowNum) throws SQLException { @@ -520,27 +445,7 @@ protected Object getColumnValue(ResultSet rs, int index) throws SQLException { @Override public ApiServiceJob getJobByTaskId(String taskId){ - ApiServiceJob apiServiceJob=runJobs.get(taskId); - return apiServiceJob; + return runJobs.get(taskId); } - - private static String getRunTypeFromScriptsPath(String scriptsPath) { - - String res = "sql"; - String fileFlag = scriptsPath.substring(scriptsPath.lastIndexOf(".") + 1); - switch (fileFlag) { - case "sh": - res = "shell"; - break; - case "py": - res= "pyspark"; - break; - default: - res = fileFlag; - break; - } - return res; - - } } diff --git a/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/util/ApiUtils.java b/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/util/ApiUtils.java index 972272be1c..cee61f8e1b 100644 --- a/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/util/ApiUtils.java +++ b/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/util/ApiUtils.java @@ -55,27 +55,27 @@ public static Message doAndResponse(TryOperation tryOperation, String method, St /** * @param tryOperation operate function */ - public static Response doAndResponse(Operation tryOperation) { - Object msg = null; + public static Message doAndResponse(Operation tryOperation) { + Message msg = null; try { msg = tryOperation.operateAndGetMessage(); - return Response.ok(msg).build(); + return msg; } catch (ConstraintViolationException e) { LOG.error("api error ", e); return new BeanValidationExceptionMapper().toResponse(e); } catch (WarnException e) { LOG.error("api error ", e); - return Response.ok(setMsg("系统异常")).build(); + return Message.error("系统异常"); } catch (AssertException e) { LOG.error("api error ", e); - return Response.ok(setMsg(e.getMessage())).build(); + return Message.error(e.getMessage()); }catch (ApiServiceRuntimeException e){ LOG.error("api error ", e); - return Response.ok(setMsg(e.getMessage())).build(); + return Message.error(e.getMessage()); } catch (Exception e) { LOG.error("api error ", e); - return Response.ok(setMsg(String.valueOf(e.getCause()))).build(); + return Message.error(String.valueOf(e.getCause())); } } @@ -106,6 +106,6 @@ public interface Operation { /** * Operate method */ - Object operateAndGetMessage() throws Exception; + Message operateAndGetMessage() throws Exception; } } diff --git a/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/util/HttpClientUtil.java b/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/util/HttpClientUtil.java index fc31316291..75c7808db0 100644 --- a/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/util/HttpClientUtil.java +++ b/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/util/HttpClientUtil.java @@ -35,14 +35,10 @@ import org.apache.http.util.EntityUtils; import org.slf4j.Logger; import org.slf4j.LoggerFactory; - -import javax.net.ssl.TrustManager; -import javax.net.ssl.X509TrustManager; import java.io.IOException; import java.io.UnsupportedEncodingException; import java.net.SocketTimeoutException; import java.net.URLEncoder; -import java.security.cert.CertificateException; import java.text.SimpleDateFormat; import java.util.*; import java.util.Map.Entry; @@ -53,21 +49,6 @@ public final class HttpClientUtil { private static PoolingHttpClientConnectionManager connManager = null; private static CloseableHttpClient httpclient = null; - private static TrustManager trustAllManager = new X509TrustManager() { - @Override - public void checkClientTrusted(java.security.cert.X509Certificate[] arg0, String arg1) - throws CertificateException { - } - @Override - public void checkServerTrusted(java.security.cert.X509Certificate[] arg0, String arg1) - throws CertificateException { - } - @Override - public java.security.cert.X509Certificate[] getAcceptedIssuers() { - return null; - } - - }; static { httpclient = HttpClients.createDefault(); diff --git a/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/vo/ApiServiceVo.java b/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/vo/ApiServiceVo.java index ff4858499a..3a1eb012af 100644 --- a/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/vo/ApiServiceVo.java +++ b/dss-apps/dss-apiservice-server/src/main/java/com/webank/wedatasphere/dss/apiservice/core/vo/ApiServiceVo.java @@ -276,4 +276,36 @@ public void setAliasName(String aliasName) { this.aliasName = aliasName; } + @Override + public String toString() { + return "ApiServiceVo{" + + "id=" + id + + ", name='" + name + '\'' + + ", aliasName='" + aliasName + '\'' + + ", path='" + path + '\'' + + ", protocol=" + protocol + + ", method='" + method + '\'' + + ", tag='" + tag + '\'' + + ", scope='" + scope + '\'' + + ", description='" + description + '\'' + + ", status=" + status + + ", type='" + type + '\'' + + ", runType='" + runType + '\'' + + ", createTime=" + createTime + + ", modifyTime=" + modifyTime + + ", creator='" + creator + '\'' + + ", modifier='" + modifier + '\'' + + ", scriptPath='" + scriptPath + '\'' + + ", approvalVo=" + approvalVo + + ", latestVersionId=" + latestVersionId + + ", userToken='" + userToken + '\'' + + ", params=" + params + + ", metadata=" + metadata + + ", content='" + content + '\'' + + ", versionVos=" + versionVos + + ", workspaceId=" + workspaceId + + ", targetServiceId=" + targetServiceId + + ", comment='" + comment + '\'' + + '}'; + } } \ No newline at end of file diff --git a/dss-apps/dss-apps-server/pom.xml b/dss-apps/dss-apps-server/pom.xml new file mode 100644 index 0000000000..4dffb74a3a --- /dev/null +++ b/dss-apps/dss-apps-server/pom.xml @@ -0,0 +1,143 @@ + + + + + + dss + com.webank.wedatasphere.dss + 1.1.2 + ../../pom.xml + + 4.0.0 + + dss-apps-server + + + + org.apache.linkis + linkis-module + ${linkis.version} + provided + + + org.springframework.cloud + spring-cloud-netflix + + + spring-cloud-starter-netflix-eureka-client + org.springframework.cloud + + + + + com.webank.wedatasphere.dss + dss-common + ${dss.version} + provided + + + + com.webank.wedatasphere.dss + dss-apiservice-server + ${dss.version} + + + com.webank.wedatasphere.dss + dss-scriptis-server + ${dss.version} + + + com.webank.wedatasphere.dss + dss-user-guide-server + ${dss.version} + + + + + + + org.apache.maven.plugins + maven-deploy-plugin + + + + net.alchim31.maven + scala-maven-plugin + + + org.apache.maven.plugins + maven-jar-plugin + + + org.apache.maven.plugins + maven-assembly-plugin + 2.3 + false + + + make-assembly + package + + single + + + + src/main/assembly/distribution.xml + + + + + + false + out + false + false + + src/main/assembly/distribution.xml + + + + + org.apache.maven.plugins + maven-compiler-plugin + + 8 + 8 + + + + + + src/main/java + + **/*.xml + + + + + + + + + + + + + + \ No newline at end of file diff --git a/dss-apps/dss-apps-server/src/main/assembly/distribution.xml b/dss-apps/dss-apps-server/src/main/assembly/distribution.xml new file mode 100644 index 0000000000..1e0339ec88 --- /dev/null +++ b/dss-apps/dss-apps-server/src/main/assembly/distribution.xml @@ -0,0 +1,43 @@ + + + + dss-apps-server + + dir + + true + dss-apps-server + + + + + + lib + true + true + false + true + true + + + + + + diff --git a/dss-apps/dss-apps-server/src/main/scala/com/webank/wedatasphere/dss/apps/DSSAppsServerApplication.scala b/dss-apps/dss-apps-server/src/main/scala/com/webank/wedatasphere/dss/apps/DSSAppsServerApplication.scala new file mode 100644 index 0000000000..18588139be --- /dev/null +++ b/dss-apps/dss-apps-server/src/main/scala/com/webank/wedatasphere/dss/apps/DSSAppsServerApplication.scala @@ -0,0 +1,55 @@ +/* +* +* * Copyright 2019 WeBank +* * +* * 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. +* +*/ + +package com.webank.wedatasphere.dss.apps + +import com.webank.wedatasphere.dss.common.utils.DSSMainHelper +import org.apache.linkis.DataWorkCloudApplication +import org.apache.linkis.common.conf.DSSConfiguration +import org.apache.linkis.common.utils.{Logging, Utils} + +import java.io.File + +object DSSAppsServerApplication extends Logging { + + val userName: String = System.getProperty("user.name") + val hostName: String = Utils.getComputerName + + val legacyServerConfigs: Array[String] = Array("dss-apiservice-server.properties", "dss-datapipe-server.properties", + "dss-guide-server.properties", "dss-scriptis-server.properties") + + def main(args: Array[String]): Unit = { + val serviceName = System.getProperty("serviceName") //ProjectConf.SERVICE_NAME.getValue + DSSMainHelper.formatPropertyFiles(serviceName) + // 若dss-apps-server配置不存在,则加载先前配置 + val serverConfFileURL = getClass.getClassLoader.getResource(serviceName + ".properties") + if (serverConfFileURL == null || !new File(serverConfFileURL.getPath).exists) { + logger.info(s"$serviceName.properties is not exists, now try to load legacy configs.") + DSSConfiguration.addLegacyConfiguration(legacyServerConfigs) + // server.port使用dss-scriptis-server.properties设置的值 + DSSConfiguration.setSpringApplicationName("dss-apps-server") + } + val allArgs = args ++ DSSMainHelper.getExtraSpringOptions + System.setProperty("hostName", hostName) + System.setProperty("userName", userName) + info(s"Ready to start $serviceName with args: ${allArgs.toList}.") + println(s"Test Ready to start $serviceName with args: ${allArgs.toList}.") + + DataWorkCloudApplication.main(allArgs) + } +} \ No newline at end of file diff --git a/dss-apps/dss-data-api/dss-api-sql-template/pom.xml b/dss-apps/dss-data-api/dss-api-sql-template/pom.xml index 854c15cb36..0df5a433f9 100644 --- a/dss-apps/dss-data-api/dss-api-sql-template/pom.xml +++ b/dss-apps/dss-data-api/dss-api-sql-template/pom.xml @@ -5,7 +5,7 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 ../../../pom.xml 4.0.0 @@ -40,7 +40,7 @@ junit junit - 4.13.1 + 4.12 test diff --git a/dss-apps/dss-data-api/dss-data-api-server/pom.xml b/dss-apps/dss-data-api/dss-data-api-server/pom.xml index e1d5829500..896e5b1d0e 100644 --- a/dss-apps/dss-data-api/dss-data-api-server/pom.xml +++ b/dss-apps/dss-data-api/dss-data-api-server/pom.xml @@ -5,7 +5,7 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 ../../../pom.xml 4.0.0 @@ -133,7 +133,7 @@ com.webank.wedatasphere.dss dss-sso-integration-standard - 1.1.1 + ${dss.version} provided diff --git a/dss-apps/dss-data-api/dss-data-api-server/src/main/java/com/webank/wedatasphere/dss/data/api/server/util/DateJsonDeserializer.java b/dss-apps/dss-data-api/dss-data-api-server/src/main/java/com/webank/wedatasphere/dss/data/api/server/util/DateJsonDeserializer.java index bb1d044bae..c8ca5d8583 100644 --- a/dss-apps/dss-data-api/dss-data-api-server/src/main/java/com/webank/wedatasphere/dss/data/api/server/util/DateJsonDeserializer.java +++ b/dss-apps/dss-data-api/dss-data-api-server/src/main/java/com/webank/wedatasphere/dss/data/api/server/util/DateJsonDeserializer.java @@ -11,7 +11,8 @@ public class DateJsonDeserializer extends JsonDeserializer { - public static final SimpleDateFormat format=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); + public static final ThreadLocal format + = ThreadLocal.withInitial(() -> new SimpleDateFormat("yyyy-MM-dd HH:mm:ss")); @Override public Date deserialize(JsonParser jsonParser, DeserializationContext deserializationContext) @@ -19,7 +20,7 @@ public Date deserialize(JsonParser jsonParser, DeserializationContext deserializ { try { - return format.parse(jsonParser.getText()); + return format.get().parse(jsonParser.getText()); } catch(Exception e) { diff --git a/dss-apps/dss-data-api/dss-data-api-server/src/main/java/com/webank/wedatasphere/dss/data/api/server/util/DateJsonSerializer.java b/dss-apps/dss-data-api/dss-data-api-server/src/main/java/com/webank/wedatasphere/dss/data/api/server/util/DateJsonSerializer.java index 1548167cde..b86cef3cac 100644 --- a/dss-apps/dss-data-api/dss-data-api-server/src/main/java/com/webank/wedatasphere/dss/data/api/server/util/DateJsonSerializer.java +++ b/dss-apps/dss-data-api/dss-data-api-server/src/main/java/com/webank/wedatasphere/dss/data/api/server/util/DateJsonSerializer.java @@ -11,12 +11,13 @@ public class DateJsonSerializer extends JsonSerializer { - public static final SimpleDateFormat format=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); + public static final ThreadLocal format + = ThreadLocal.withInitial (()->new SimpleDateFormat("yyyy-MM-dd HH:mm:ss")); @Override public void serialize(Date date, JsonGenerator jsonGenerator, SerializerProvider serializerProvider) throws IOException, JsonProcessingException { - jsonGenerator.writeString(format.format(date)); + jsonGenerator.writeString(format.get().format(date)); } } diff --git a/dss-apps/dss-data-api/dss-data-api-server/src/main/java/com/webank/wedatasphere/dss/data/api/server/util/JdbcUtil.java b/dss-apps/dss-data-api/dss-data-api-server/src/main/java/com/webank/wedatasphere/dss/data/api/server/util/JdbcUtil.java index 89aa9b679a..ff418d913b 100644 --- a/dss-apps/dss-data-api/dss-data-api-server/src/main/java/com/webank/wedatasphere/dss/data/api/server/util/JdbcUtil.java +++ b/dss-apps/dss-data-api/dss-data-api-server/src/main/java/com/webank/wedatasphere/dss/data/api/server/util/JdbcUtil.java @@ -13,9 +13,10 @@ public class JdbcUtil { public static ResultSet query(String sql, Connection connection) throws SQLException { - PreparedStatement preparedStatement = connection.prepareStatement(sql); - ResultSet resultSet = preparedStatement.executeQuery(); - return resultSet; + try(PreparedStatement preparedStatement = connection.prepareStatement(sql)){ + ResultSet resultSet = preparedStatement.executeQuery(); + return resultSet; + } } public static Connection getConnection(DataSource ds) throws SQLException, ClassNotFoundException { @@ -123,15 +124,15 @@ public static List getRDBMSColumnProperties(Connection conn, String typ sql = "show full columns from " + table; } log.info(sql); - - resultSet = conn.prepareStatement(sql).executeQuery(); - while (resultSet.next()) { - HashMap colProp = new HashMap<>(); - colProp.put("Comment",resultSet.getString("Comment")); - colProp.put("fieldType",resultSet.getString("Type")); - colProp.put("columnName",resultSet.getString("Field")); - - list.add(colProp); + try (PreparedStatement ps = conn.prepareStatement(sql)) { + resultSet = ps.executeQuery(); + while (resultSet.next()) { + HashMap colProp = new HashMap<>(); + colProp.put("Comment", resultSet.getString("Comment")); + colProp.put("fieldType", resultSet.getString("Type")); + colProp.put("columnName", resultSet.getString("Field")); + list.add(colProp); + } } return list; } catch (Exception e) { diff --git a/dss-apps/dss-data-api/pom.xml b/dss-apps/dss-data-api/pom.xml index 120c8e961b..f435dc9947 100644 --- a/dss-apps/dss-data-api/pom.xml +++ b/dss-apps/dss-data-api/pom.xml @@ -6,7 +6,7 @@ dss com.webank.wedatasphere.dss ../../pom.xml - 1.1.1 + 1.1.2 4.0.0 dss-data-api diff --git a/dss-apps/dss-data-governance/dss-data-asset-server/pom.xml b/dss-apps/dss-data-governance/dss-data-asset-server/pom.xml index 2934851463..56d85cb06a 100644 --- a/dss-apps/dss-data-governance/dss-data-asset-server/pom.xml +++ b/dss-apps/dss-data-governance/dss-data-asset-server/pom.xml @@ -5,7 +5,7 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 ../../../pom.xml 4.0.0 @@ -107,6 +107,28 @@ org.apache.linkis linkis-mybatis ${linkis.version} + + + org.springframework.boot + spring-boot-starter + + + org.springframework.boot + spring-boot-autoconfigure + + + org.springframework + spring-beans + + + org.springframework + spring-jdbc + + + com.zaxxer + HikariCP + + com.webank.wedatasphere.dss diff --git a/dss-apps/dss-data-governance/dss-data-asset-server/src/main/java/com/webank/wedatasphere/dss/data/asset/dao/impl/MetaInfoMapperImpl.java b/dss-apps/dss-data-governance/dss-data-asset-server/src/main/java/com/webank/wedatasphere/dss/data/asset/dao/impl/MetaInfoMapperImpl.java index 72989801d2..6463e6205b 100644 --- a/dss-apps/dss-data-governance/dss-data-asset-server/src/main/java/com/webank/wedatasphere/dss/data/asset/dao/impl/MetaInfoMapperImpl.java +++ b/dss-apps/dss-data-governance/dss-data-asset-server/src/main/java/com/webank/wedatasphere/dss/data/asset/dao/impl/MetaInfoMapperImpl.java @@ -1,174 +1,166 @@ -package com.webank.wedatasphere.dss.data.asset.dao.impl; - -import com.webank.wedatasphere.dss.data.asset.dao.MetaInfoMapper; -import com.webank.wedatasphere.dss.data.asset.entity.HivePartInfo; -import com.webank.wedatasphere.dss.data.asset.entity.HiveStorageInfo; -import com.webank.wedatasphere.dss.data.common.exception.DAOException; -import com.webank.wedatasphere.dss.data.common.utils.DataSourceUtil; -import com.webank.wedatasphere.dss.data.common.utils.DateUtil; - -import javax.sql.DataSource; -import java.sql.Connection; -import java.sql.PreparedStatement; -import java.sql.ResultSet; -import java.sql.SQLException; -import java.util.ArrayList; -import java.util.Collections; -import java.util.Comparator; -import java.util.List; - -public class MetaInfoMapperImpl implements MetaInfoMapper { - @Override - public Long getTableStorage() throws SQLException { - DataSource dataSource = DataSourceUtil.getDataSource(); - - Connection con =dataSource.getConnection(); - long num=0; - PreparedStatement ps=null; - ResultSet rs=null; - try { - String sql="select SUM(PARAM_VALUE) from TABLE_PARAMS WHERE PARAM_KEY='totalSize'"; - ps=con.prepareStatement(sql); - rs=ps.executeQuery(); - while (rs.next()){ - num =rs.getLong(1); - } - String sql2 ="select SUM(PARAM_VALUE) from PARTITION_PARAMS WHERE PARAM_KEY='totalSize'"; - ps=con.prepareStatement(sql2); - rs=ps.executeQuery(); - while (rs.next()){ - num= num + rs.getLong(1); - } - - } catch (DAOException | SQLException e){ - throw new DAOException(e.getMessage(),e); - } - finally { - con.close(); - } - - return num; - } - - @Override - public List getTop10Table() throws SQLException{ - DataSource dataSource = DataSourceUtil.getDataSource(); - Connection con =dataSource.getConnection(); - PreparedStatement ps=null; - ResultSet rs=null; - List hiveStorageInfos = new ArrayList<>(); - try { - String sql="SELECT DBS.NAME ,TBLS.TBL_NAME,CAST(TABLE_PARAMS.PARAM_VALUE AS UNSIGNED) AS totalSize from DBS, TBLS,TABLE_PARAMS where TBLS.TBL_ID=TABLE_PARAMS.TBL_ID AND TBLS.DB_ID=DBS.DB_ID AND TABLE_PARAMS.PARAM_KEY='totalSize' order by totalSize DESC limit 10"; - ps=con.prepareStatement(sql); - rs=ps.executeQuery(); - while (rs.next()){ - HiveStorageInfo tableinfo=new HiveStorageInfo(); - tableinfo.setTableName(rs.getString(1)+"."+rs.getString(2)); - tableinfo.setStorage(rs.getLong(3)); - hiveStorageInfos.add(tableinfo); - } - String sql2="SELECT DBS.NAME ,TBLS.TBL_NAME,SUM(CAST(PARTITION_PARAMS.PARAM_VALUE AS UNSIGNED)) AS totalSize from DBS,TBLS,PARTITIONS ,PARTITION_PARAMS where DBS.DB_ID=TBLS.DB_ID AND TBLS.TBL_ID=PARTITIONS.TBL_ID AND PARTITIONS.PART_ID =PARTITION_PARAMS.PART_ID AND PARTITION_PARAMS.PARAM_KEY='totalSize' group by TBLS.TBL_NAME order by totalSize desc limit 10"; - ps=con.prepareStatement(sql2); - rs=ps.executeQuery(); - while (rs.next()){ - HiveStorageInfo tableinfo=new HiveStorageInfo(); - tableinfo.setTableName(rs.getString(1)+"."+rs.getString(2)); - tableinfo.setStorage(rs.getLong(3)); - hiveStorageInfos.add(tableinfo); - } - /** - * 特别注意LONG类型相减超出INT范围 - * System.out.println((int) (4401131805L -1796673800L)) - * System.out.println(Long.parseLong("4401131805")-Long.parseLong("1796673800")) - */ - Collections.sort(hiveStorageInfos, new Comparator() { - @Override - public int compare(HiveStorageInfo o1, HiveStorageInfo o2) { - //return (int) (Long.valueOf(o2.getStorage())-Long.valueOf(o1.getStorage())) - if(o2.getStorage() > o1.getStorage()){ - return 1; - } - else if(o2.getStorage() < o1.getStorage()){ - return -1; - } - else{ - return 0; - } - } - }); - } catch (DAOException | SQLException e){ - throw new DAOException(e.getMessage(),e); - } - finally { - con.close(); - } - return hiveStorageInfos.subList(0,10); - } - - @Override - public int getTableInfo(String dbName, String tableName, Boolean isPartTable) throws SQLException { - DataSource dataSource = DataSourceUtil.getDataSource(); - Connection con =dataSource.getConnection(); - PreparedStatement ps=null; - ResultSet rs=null; - int res = 0; - try { - String sql=null; - if(isPartTable==false){ - sql="select TABLE_PARAMS.PARAM_VALUE as totalSize from DBS, TBLS,TABLE_PARAMS where TBLS.TBL_ID=TABLE_PARAMS.TBL_ID AND TBLS.DB_ID=DBS.DB_ID AND TABLE_PARAMS.PARAM_KEY='totalSize' AND DBS.NAME="+"'"+dbName+"' AND TBLS.TBL_NAME="+"'"+tableName+"'"; - } - else { - - sql="select SUM(PARTITION_PARAMS.PARAM_VALUE) as totalSize from DBS,TBLS,PARTITIONS ,PARTITION_PARAMS where DBS.DB_ID=TBLS.DB_ID AND TBLS.TBL_ID=PARTITIONS.TBL_ID AND PARTITIONS.PART_ID =PARTITION_PARAMS.PART_ID AND PARTITION_PARAMS.PARAM_KEY='totalSize' AND DBS.NAME="+"'"+dbName +"' AND TBLS.TBL_NAME="+"'"+tableName+"' group by TBLS.TBL_NAME"; - } - ps=con.prepareStatement(sql); - rs=ps.executeQuery(); - while (rs.next()){ - res=rs.getInt(1); - } - - } catch (DAOException | SQLException e){ - throw new DAOException(e.getMessage(),e); - } - finally { - con.close(); - } - return res; - } - - @Override - public List getPartInfo(String dbName, String tableName)throws SQLException { - DataSource dataSource = DataSourceUtil.getDataSource(); - Connection con =dataSource.getConnection(); - PreparedStatement ps=null; - ResultSet rs=null; - List hivePartInfos = new ArrayList<>(); - try { - String sql="select b.PART_NAME,b.CREATE_TIME,MAX(CASE c.PARAM_KEY WHEN 'transient_lastDdlTime' THEN c.PARAM_VALUE ELSE null END) transient_lastDdlTime ,MAX(CASE c.PARAM_KEY WHEN 'numRows' THEN c.PARAM_VALUE ELSE null END) numRows,MAX(CASE c.PARAM_KEY WHEN 'totalSize' THEN c.PARAM_VALUE ELSE null END) totalSize from TBLS a,PARTITIONS b,PARTITION_PARAMS c,DBS d where a.TBL_NAME="+"'"+tableName+"'"+"AND d.NAME="+"'"+dbName+"'" +"AND a.TBL_ID=b.TBL_ID AND a.DB_ID=d.DB_ID AND b.PART_ID=c.PART_ID GROUP BY c.PART_ID"; - ps=con.prepareStatement(sql); - rs=ps.executeQuery(); - while (rs.next()){ - HivePartInfo part =new HivePartInfo(); - part.setPartName(rs.getString(1)); - Long lastAccessTime = Long.valueOf(rs.getInt(3)); - if(lastAccessTime !=null && lastAccessTime !=0L) { - part.setLastAccessTime(DateUtil.unixToTimeStr(lastAccessTime * 1000)); - } - Long createTime = Long.valueOf(rs.getInt(2)); - if(createTime !=null && createTime !=0L) { - part.setCreateTime(DateUtil.unixToTimeStr(createTime * 1000)); - } - part.setReordCnt(rs.getInt(4)); - part.setStore(rs.getInt(5)); - hivePartInfos.add(part); - } - - } catch (DAOException | SQLException e){ - throw new DAOException(e.getMessage(),e); - } - finally { - con.close(); - } - return hivePartInfos; - } -} +//package com.webank.wedatasphere.dss.data.asset.dao.impl; +// +//import com.webank.wedatasphere.dss.data.asset.dao.MetaInfoMapper; +//import com.webank.wedatasphere.dss.data.asset.entity.HivePartInfo; +//import com.webank.wedatasphere.dss.data.asset.entity.HiveStorageInfo; +//import com.webank.wedatasphere.dss.data.common.exception.DAOException; +//import com.webank.wedatasphere.dss.data.common.utils.DataSourceUtil; +//import com.webank.wedatasphere.dss.data.common.utils.DateUtil; +// +//import javax.sql.DataSource; +//import java.sql.Connection; +//import java.sql.PreparedStatement; +//import java.sql.ResultSet; +//import java.sql.SQLException; +//import java.util.ArrayList; +//import java.util.Collections; +//import java.util.Comparator; +//import java.util.List; +// +//public class MetaInfoMapperImpl implements MetaInfoMapper { +// @Override +// public Long getTableStorage() throws SQLException { +// DataSource dataSource = DataSourceUtil.getDataSource(); +// +// Connection con = dataSource.getConnection(); +// long num = 0; +// PreparedStatement ps = null; +// ResultSet rs = null; +// try { +// String sql = "select SUM(PARAM_VALUE) from TABLE_PARAMS WHERE PARAM_KEY='totalSize'"; +// ps = con.prepareStatement(sql); +// rs = ps.executeQuery(); +// while (rs.next()) { +// num = rs.getLong(1); +// } +// ps.close(); +// String sql2 = "select SUM(PARAM_VALUE) from PARTITION_PARAMS WHERE PARAM_KEY='totalSize'"; +// ps = con.prepareStatement(sql2); +// rs = ps.executeQuery(); +// while (rs.next()) { +// num = num + rs.getLong(1); +// } +// ps.close(); +// } catch (DAOException | SQLException e) { +// throw new DAOException(e.getMessage(), e); +// } finally { +// con.close(); +// } +// +// return num; +// } +// +// @Override +// public List getTop10Table() throws SQLException { +// DataSource dataSource = DataSourceUtil.getDataSource(); +// Connection con = dataSource.getConnection(); +// PreparedStatement ps = null; +// ResultSet rs = null; +// List hiveStorageInfos = new ArrayList<>(); +// try { +// String sql = "SELECT DBS.NAME ,TBLS.TBL_NAME,CAST(TABLE_PARAMS.PARAM_VALUE AS UNSIGNED) AS totalSize from DBS, TBLS,TABLE_PARAMS where TBLS.TBL_ID=TABLE_PARAMS.TBL_ID AND TBLS.DB_ID=DBS.DB_ID AND TABLE_PARAMS.PARAM_KEY='totalSize' order by totalSize DESC limit 10"; +// ps = con.prepareStatement(sql); +// rs = ps.executeQuery(); +// while (rs.next()) { +// HiveStorageInfo tableinfo = new HiveStorageInfo(); +// tableinfo.setTableName(rs.getString(1) + "." + rs.getString(2)); +// tableinfo.setStorage(rs.getLong(3)); +// hiveStorageInfos.add(tableinfo); +// } +// ps.close(); +// String sql2 = "SELECT DBS.NAME ,TBLS.TBL_NAME,SUM(CAST(PARTITION_PARAMS.PARAM_VALUE AS UNSIGNED)) AS totalSize from DBS,TBLS,PARTITIONS ,PARTITION_PARAMS where DBS.DB_ID=TBLS.DB_ID AND TBLS.TBL_ID=PARTITIONS.TBL_ID AND PARTITIONS.PART_ID =PARTITION_PARAMS.PART_ID AND PARTITION_PARAMS.PARAM_KEY='totalSize' group by TBLS.TBL_NAME order by totalSize desc limit 10"; +// ps = con.prepareStatement(sql2); +// rs = ps.executeQuery(); +// while (rs.next()) { +// HiveStorageInfo tableinfo = new HiveStorageInfo(); +// tableinfo.setTableName(rs.getString(1) + "." + rs.getString(2)); +// tableinfo.setStorage(rs.getLong(3)); +// hiveStorageInfos.add(tableinfo); +// } +// ps.close(); +// /** +// * 特别注意LONG类型相减超出INT范围 +// * System.out.println((int) (4401131805L -1796673800L)) +// * System.out.println(Long.parseLong("4401131805")-Long.parseLong("1796673800")) +// */ +// Collections.sort(hiveStorageInfos, new Comparator() { +// @Override +// public int compare(HiveStorageInfo o1, HiveStorageInfo o2) { +// //return (int) (Long.valueOf(o2.getStorage())-Long.valueOf(o1.getStorage())) +// if (o2.getStorage() > o1.getStorage()) { +// return 1; +// } else if (o2.getStorage() < o1.getStorage()) { +// return -1; +// } else { +// return 0; +// } +// } +// }); +// } catch (DAOException | SQLException e) { +// throw new DAOException(e.getMessage(), e); +// } finally { +// con.close(); +// } +// return hiveStorageInfos.subList(0, 10); +// } +// +// @Override +// public int getTableInfo(String dbName, String tableName, Boolean isPartTable) throws SQLException { +// DataSource dataSource = DataSourceUtil.getDataSource(); +// Connection con = dataSource.getConnection(); +// int res = 0; +// try { +// String sql = null; +// if (isPartTable == false) { +// sql = "select TABLE_PARAMS.PARAM_VALUE as totalSize from DBS, TBLS,TABLE_PARAMS where TBLS.TBL_ID=TABLE_PARAMS.TBL_ID AND TBLS.DB_ID=DBS.DB_ID AND TABLE_PARAMS.PARAM_KEY='totalSize' AND DBS.NAME=" + "'" + dbName + "' AND TBLS.TBL_NAME=" + "'" + tableName + "'"; +// } else { +// +// sql = "select SUM(PARTITION_PARAMS.PARAM_VALUE) as totalSize from DBS,TBLS,PARTITIONS ,PARTITION_PARAMS where DBS.DB_ID=TBLS.DB_ID AND TBLS.TBL_ID=PARTITIONS.TBL_ID AND PARTITIONS.PART_ID =PARTITION_PARAMS.PART_ID AND PARTITION_PARAMS.PARAM_KEY='totalSize' AND DBS.NAME=" + "'" + dbName + "' AND TBLS.TBL_NAME=" + "'" + tableName + "' group by TBLS.TBL_NAME"; +// } +// try (PreparedStatement ps = con.prepareStatement(sql); ResultSet rs = ps.executeQuery()) { +// while (rs.next()) { +// res = rs.getInt(1); +// } +// } +// +// } catch (DAOException | SQLException e) { +// throw new DAOException(e.getMessage(), e); +// } finally { +// con.close(); +// } +// return res; +// } +// +// @Override +// public List getPartInfo(String dbName, String tableName) throws SQLException { +// DataSource dataSource = DataSourceUtil.getDataSource(); +// Connection con = dataSource.getConnection(); +// List hivePartInfos = new ArrayList<>(); +// try { +// String sql = "select b.PART_NAME,b.CREATE_TIME,MAX(CASE c.PARAM_KEY WHEN 'transient_lastDdlTime' THEN c.PARAM_VALUE ELSE null END) transient_lastDdlTime ,MAX(CASE c.PARAM_KEY WHEN 'numRows' THEN c.PARAM_VALUE ELSE null END) numRows,MAX(CASE c.PARAM_KEY WHEN 'totalSize' THEN c.PARAM_VALUE ELSE null END) totalSize from TBLS a,PARTITIONS b,PARTITION_PARAMS c,DBS d where a.TBL_NAME=" + "'" + tableName + "'" + "AND d.NAME=" + "'" + dbName + "'" + "AND a.TBL_ID=b.TBL_ID AND a.DB_ID=d.DB_ID AND b.PART_ID=c.PART_ID GROUP BY c.PART_ID"; +// try (PreparedStatement ps = con.prepareStatement(sql); ResultSet rs = ps.executeQuery()) { +// while (rs.next()) { +// HivePartInfo part = new HivePartInfo(); +// part.setPartName(rs.getString(1)); +// Long lastAccessTime = Long.valueOf(rs.getInt(3)); +// if (lastAccessTime != null && lastAccessTime != 0L) { +// part.setLastAccessTime(DateUtil.unixToTimeStr(lastAccessTime * 1000)); +// } +// Long createTime = Long.valueOf(rs.getInt(2)); +// if (createTime != null && createTime != 0L) { +// part.setCreateTime(DateUtil.unixToTimeStr(createTime * 1000)); +// } +// part.setReordCnt(rs.getInt(4)); +// part.setStore(rs.getInt(5)); +// hivePartInfos.add(part); +// } +// } +// +// } catch (DAOException | SQLException e) { +// throw new DAOException(e.getMessage(), e); +// } finally { +// con.close(); +// } +// return hivePartInfos; +// } +//} diff --git a/dss-apps/dss-data-governance/dss-data-classification-server/pom.xml b/dss-apps/dss-data-governance/dss-data-classification-server/pom.xml index 1543322457..b5e285e96e 100644 --- a/dss-apps/dss-data-governance/dss-data-classification-server/pom.xml +++ b/dss-apps/dss-data-governance/dss-data-classification-server/pom.xml @@ -5,7 +5,7 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 ../../../pom.xml 4.0.0 @@ -107,6 +107,28 @@ org.apache.linkis linkis-mybatis ${linkis.version} + + + org.springframework.boot + spring-boot-starter + + + org.springframework.boot + spring-boot-autoconfigure + + + org.springframework + spring-beans + + + org.springframework + spring-jdbc + + + com.zaxxer + HikariCP + + com.webank.wedatasphere.dss diff --git a/dss-apps/dss-data-governance/dss-data-governance-common/pom.xml b/dss-apps/dss-data-governance/dss-data-governance-common/pom.xml index bdd9f0c2f0..edd3f278ed 100644 --- a/dss-apps/dss-data-governance/dss-data-governance-common/pom.xml +++ b/dss-apps/dss-data-governance/dss-data-governance-common/pom.xml @@ -5,7 +5,7 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 ../../../pom.xml 4.0.0 diff --git a/dss-apps/dss-data-governance/dss-data-governance-common/src/main/java/com/webank/wedatasphere/dss/data/common/utils/DataSourceUtil.java b/dss-apps/dss-data-governance/dss-data-governance-common/src/main/java/com/webank/wedatasphere/dss/data/common/utils/DataSourceUtil.java index 5794f74f93..42d2a48a30 100644 --- a/dss-apps/dss-data-governance/dss-data-governance-common/src/main/java/com/webank/wedatasphere/dss/data/common/utils/DataSourceUtil.java +++ b/dss-apps/dss-data-governance/dss-data-governance-common/src/main/java/com/webank/wedatasphere/dss/data/common/utils/DataSourceUtil.java @@ -4,7 +4,7 @@ import com.webank.wedatasphere.dss.data.common.conf.AtlasConf; public class DataSourceUtil { - private static DruidDataSource druidDataSource =null; + private static volatile DruidDataSource druidDataSource =null; private DataSourceUtil(){} diff --git a/dss-apps/dss-data-governance/dss-data-governance-common/src/main/java/com/webank/wedatasphere/dss/data/common/utils/HttpClientUtil.java b/dss-apps/dss-data-governance/dss-data-governance-common/src/main/java/com/webank/wedatasphere/dss/data/common/utils/HttpClientUtil.java index e14fbe073f..3c32792785 100644 --- a/dss-apps/dss-data-governance/dss-data-governance-common/src/main/java/com/webank/wedatasphere/dss/data/common/utils/HttpClientUtil.java +++ b/dss-apps/dss-data-governance/dss-data-governance-common/src/main/java/com/webank/wedatasphere/dss/data/common/utils/HttpClientUtil.java @@ -19,14 +19,10 @@ import org.apache.http.util.EntityUtils; import org.slf4j.Logger; import org.slf4j.LoggerFactory; - -import javax.net.ssl.TrustManager; -import javax.net.ssl.X509TrustManager; import java.io.IOException; import java.io.UnsupportedEncodingException; import java.net.SocketTimeoutException; import java.net.URLEncoder; -import java.security.cert.CertificateException; import java.text.SimpleDateFormat; import java.util.ArrayList; import java.util.Date; @@ -42,25 +38,6 @@ public final class HttpClientUtil { private static PoolingHttpClientConnectionManager connManager = null; private static CloseableHttpClient httpclient = null; - /** - * 重写验证方法,取消检测ssl - */ - private static TrustManager trustAllManager = new X509TrustManager() { - @Override - public void checkClientTrusted(java.security.cert.X509Certificate[] arg0, String arg1) - throws CertificateException { - } - @Override - public void checkServerTrusted(java.security.cert.X509Certificate[] arg0, String arg1) - throws CertificateException { - } - @Override - public java.security.cert.X509Certificate[] getAcceptedIssuers() { - return null; - } - - }; - static { httpclient = HttpClients.createDefault(); } diff --git a/dss-apps/dss-data-governance/dss-data-governance-server/pom.xml b/dss-apps/dss-data-governance/dss-data-governance-server/pom.xml index 6451c5c60e..845abe2743 100644 --- a/dss-apps/dss-data-governance/dss-data-governance-server/pom.xml +++ b/dss-apps/dss-data-governance/dss-data-governance-server/pom.xml @@ -5,7 +5,7 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 ../../../pom.xml 4.0.0 diff --git a/dss-apps/dss-data-governance/pom.xml b/dss-apps/dss-data-governance/pom.xml index e63bff2d20..0d80bb2836 100644 --- a/dss-apps/dss-data-governance/pom.xml +++ b/dss-apps/dss-data-governance/pom.xml @@ -5,7 +5,7 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 ../../pom.xml 4.0.0 diff --git a/dss-apps/dss-scriptis-server/pom.xml b/dss-apps/dss-scriptis-server/pom.xml index f1b411b1c0..40e09c7991 100644 --- a/dss-apps/dss-scriptis-server/pom.xml +++ b/dss-apps/dss-scriptis-server/pom.xml @@ -22,7 +22,7 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 ../../pom.xml @@ -34,6 +34,11 @@ + + com.webank.wedatasphere.dss + dss-framework-proxy-user-service + ${dss.version} + com.webank.wedatasphere.dss dss-common @@ -212,7 +217,7 @@ junit junit - 4.13.1 + 4.12 test @@ -274,6 +279,12 @@ + + src/main/java + + **/*.xml + + ${basedir}/src/main/resources diff --git a/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/config/DSSScriptisConfiguration.java b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/config/DSSScriptisConfiguration.java index 86bae03e15..4aa6bb1646 100644 --- a/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/config/DSSScriptisConfiguration.java +++ b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/config/DSSScriptisConfiguration.java @@ -29,4 +29,5 @@ public class DSSScriptisConfiguration { public final static String GLOBAL_LIMITS_PREFIX = "wds.dss.scriptis.global.limits."; public final static String GLOBAL_LIMIT_PREFIX = "wds.dss.scriptis.global.limit."; + } diff --git a/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/config/ScriptisBeanConfiguration.java b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/config/ScriptisBeanConfiguration.java new file mode 100644 index 0000000000..801fc9262b --- /dev/null +++ b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/config/ScriptisBeanConfiguration.java @@ -0,0 +1,17 @@ +package com.webank.wedatasphere.dss.scriptis.config; + +import com.webank.wedatasphere.dss.scriptis.service.ScriptisAuthService; +import com.webank.wedatasphere.dss.scriptis.service.impl.ScriptisAuthServiceImpl; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; + +@Configuration +public class ScriptisBeanConfiguration { + + @Bean + @ConditionalOnMissingBean + public ScriptisAuthService getScriptisAuthService(){ + return new ScriptisAuthServiceImpl(); + } +} diff --git a/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/dao/DssAuditMapper.java b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/dao/DssAuditMapper.java new file mode 100644 index 0000000000..ea3cc493d3 --- /dev/null +++ b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/dao/DssAuditMapper.java @@ -0,0 +1,13 @@ +package com.webank.wedatasphere.dss.scriptis.dao; + +import com.baomidou.mybatisplus.core.mapper.BaseMapper; +import com.webank.wedatasphere.dss.scriptis.pojo.entity.DssScriptDownloadAudit; +import org.apache.ibatis.annotations.Param; + +import java.util.List; + +public interface DssAuditMapper extends BaseMapper { + + List getDownloadAuditList(@Param("creator") String creator, @Param("startTime") String startIme, @Param("endTime") String endTime); + +} diff --git a/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/dao/ScriptisProxyUserMapper.java b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/dao/ScriptisProxyUserMapper.java new file mode 100644 index 0000000000..aed2e9e94a --- /dev/null +++ b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/dao/ScriptisProxyUserMapper.java @@ -0,0 +1,16 @@ +package com.webank.wedatasphere.dss.scriptis.dao; + +import com.baomidou.mybatisplus.core.mapper.BaseMapper; +import com.webank.wedatasphere.dss.scriptis.pojo.entity.ScriptisProxyUser; +import org.apache.ibatis.annotations.Mapper; + +import java.util.List; + +@Mapper +public interface ScriptisProxyUserMapper extends BaseMapper { + + List selectProxyUserList(String userName); + + int insertUser(ScriptisProxyUser user); + +} diff --git a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/xml/impl/DssAuditMapper.xml b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/dao/mapper/DssAuditMapper.xml similarity index 77% rename from dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/xml/impl/DssAuditMapper.xml rename to dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/dao/mapper/DssAuditMapper.xml index 3030bc3f2f..79e1c61c24 100644 --- a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/xml/impl/DssAuditMapper.xml +++ b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/dao/mapper/DssAuditMapper.xml @@ -1,9 +1,9 @@ - + - - - + insert into dss_proxy_user( id, username, diff --git a/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/execute/LinkisJobSubmit.java b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/execute/LinkisJobSubmit.java index 54f4400522..ed3bfbc30f 100644 --- a/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/execute/LinkisJobSubmit.java +++ b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/execute/LinkisJobSubmit.java @@ -35,7 +35,7 @@ public class LinkisJobSubmit { - private static UJESClient ujesClient; + private static volatile UJESClient ujesClient; public static UJESClient getClient() { if(ujesClient == null) { diff --git a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/pojo/entity/DssScriptDownloadAudit.java b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/pojo/entity/DssScriptDownloadAudit.java similarity index 80% rename from dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/pojo/entity/DssScriptDownloadAudit.java rename to dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/pojo/entity/DssScriptDownloadAudit.java index 91e086ed0b..0293218ef0 100644 --- a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/pojo/entity/DssScriptDownloadAudit.java +++ b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/pojo/entity/DssScriptDownloadAudit.java @@ -1,4 +1,4 @@ -package com.webank.wedatasphere.dss.framework.admin.pojo.entity; +package com.webank.wedatasphere.dss.scriptis.pojo.entity; import com.baomidou.mybatisplus.annotation.IdType; import com.baomidou.mybatisplus.annotation.TableField; @@ -7,18 +7,9 @@ import com.fasterxml.jackson.annotation.JsonFormat; import org.springframework.format.annotation.DateTimeFormat; -import javax.validation.constraints.NotBlank; import javax.validation.constraints.NotEmpty; -import javax.validation.constraints.NotNull; -import javax.xml.bind.annotation.XmlRootElement; import java.util.Date; -/** - * @Auther: Han Tang - * @Date: 2022/1/10-01-10-15:48 - */ - - @TableName(value = "dss_workspace_download_audit") public class DssScriptDownloadAudit { @@ -83,4 +74,15 @@ public void setCreateTime(Date createTime) { this.createTime = createTime; } + @Override + public String toString() { + return "DssScriptDownloadAudit{" + + "id=" + id + + ", creator='" + creator + '\'' + + ", tenant='" + tenant + '\'' + + ", path='" + path + '\'' + + ", sql='" + sql + '\'' + + ", createTime=" + createTime + + '}'; + } } diff --git a/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/pojo/entity/ScriptisProxyUser.java b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/pojo/entity/ScriptisProxyUser.java new file mode 100644 index 0000000000..7c602289e2 --- /dev/null +++ b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/pojo/entity/ScriptisProxyUser.java @@ -0,0 +1,57 @@ +package com.webank.wedatasphere.dss.scriptis.pojo.entity; + +import com.baomidou.mybatisplus.annotation.IdType; +import com.baomidou.mybatisplus.annotation.TableId; +import com.baomidou.mybatisplus.annotation.TableName; +import com.webank.wedatasphere.dss.framework.proxy.pojo.entity.DssProxyUserImpl; + +import java.util.Date; + +/** + * @author enjoyyin + * @date 2022-09-06 + * @since 0.5.0 + */ +@TableName(value = "dss_scriptis_proxy_user") +public class ScriptisProxyUser extends DssProxyUserImpl { + + private static final long serialVersionUID = 1L; + + @TableId(type = IdType.AUTO) + private Long id; + private String createBy; + private Date createTime; + private String remark; + + public Long getId() { + return id; + } + + public void setId(Long id) { + this.id = id; + } + + public String getCreateBy() { + return createBy; + } + + public void setCreateBy(String createBy) { + this.createBy = createBy; + } + + public Date getCreateTime() { + return createTime; + } + + public void setCreateTime(Date createTime) { + this.createTime = createTime; + } + + public String getRemark() { + return remark; + } + + public void setRemark(String remark) { + this.remark = remark; + } +} diff --git a/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/restful/ScriptisAuthRestfulApi.java b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/restful/ScriptisAuthRestfulApi.java index 30efdf73b9..56eadc1147 100644 --- a/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/restful/ScriptisAuthRestfulApi.java +++ b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/restful/ScriptisAuthRestfulApi.java @@ -1,19 +1,23 @@ package com.webank.wedatasphere.dss.scriptis.restful; -import org.apache.commons.lang.StringUtils; +import com.webank.wedatasphere.dss.common.conf.DSSCommonConf; +import com.webank.wedatasphere.dss.common.utils.GlobalLimitsUtils; +import com.webank.wedatasphere.dss.scriptis.service.ScriptisAuthService; import org.apache.linkis.common.conf.BDPConfiguration; -import org.apache.linkis.protocol.util.ImmutablePair; import org.apache.linkis.server.Message; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; +import org.apache.linkis.server.conf.ServerConfiguration; +import org.apache.linkis.server.security.SecurityFilter; +import org.springframework.beans.factory.annotation.Autowired; import org.springframework.web.bind.annotation.PathVariable; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.RequestMethod; import org.springframework.web.bind.annotation.RestController; +import javax.annotation.PostConstruct; +import javax.servlet.http.HttpServletRequest; + import java.util.HashMap; import java.util.Map; -import java.util.function.Supplier; import static com.webank.wedatasphere.dss.scriptis.config.DSSScriptisConfiguration.GLOBAL_LIMITS_PREFIX; import static com.webank.wedatasphere.dss.scriptis.config.DSSScriptisConfiguration.GLOBAL_LIMIT_PREFIX; @@ -27,50 +31,26 @@ @RestController public class ScriptisAuthRestfulApi { - private final Logger logger = LoggerFactory.getLogger(ScriptisAuthRestfulApi.class); - private Map globalLimits; - private final Map> globalLimitContents = new HashMap<>(); + @Autowired + private ScriptisAuthService scriptisAuthService; + + @PostConstruct + public void init() { + BDPConfiguration.set(DSSCommonConf.ALL_GLOBAL_LIMITS_PREFIX.key(), GLOBAL_LIMITS_PREFIX); + BDPConfiguration.set(DSSCommonConf.GLOBAL_LIMIT_PREFIX.key(), GLOBAL_LIMIT_PREFIX); + } - @RequestMapping(value = "/globalLimits",method = RequestMethod.GET) - public Message globalLimits() { - if(globalLimits == null) { - synchronized (ScriptisAuthRestfulApi.class) { - if(globalLimits == null) { - globalLimits = getMap(GLOBAL_LIMITS_PREFIX); - logger.info("loaded Scriptis global limits is {}.", globalLimits); - } - } - } + @RequestMapping(value = "/globalLimits", method = RequestMethod.GET) + public Message globalLimits(HttpServletRequest req) { + String username = SecurityFilter.getLoginUsername(req); + Map globalLimits = scriptisAuthService.getGlobalLimits(username); return Message.ok().data("globalLimits", globalLimits); } @RequestMapping(value = "/globalLimits/{globalLimitName}",method = RequestMethod.GET) public Message globalLimit(@PathVariable("globalLimitName") String globalLimitName) { - Supplier function = () -> Message.ok().data("globalLimitName", globalLimitName).data("content", globalLimitContents.get(globalLimitName)); - if(globalLimitContents.containsKey(globalLimitName)) { - return function.get(); - } - synchronized (globalLimitContents) { - if(!globalLimitContents.containsKey(globalLimitName)) { - Map globalLimitContent = getMap(GLOBAL_LIMIT_PREFIX + globalLimitName + "."); - logger.info("loaded Scriptis global limit {} content are {}.", globalLimitName, globalLimitContent); - globalLimitContents.put(globalLimitName, globalLimitContent); - } - } - return function.get(); + return Message.ok().data("globalLimitName", globalLimitName) + .data("content", GlobalLimitsUtils.getGlobalLimitMap(globalLimitName)); } - private Map getMap(String prefix) { - return BDPConfiguration.properties().entrySet().stream() - .filter(entry -> entry.getKey().toString().startsWith(prefix) && entry.getValue() != null && StringUtils.isNotBlank(entry.getValue().toString())) - .map(entry -> { - String key = ((String) entry.getKey()).substring(prefix.length()); - if("true".equals(entry.getValue()) || "false".equals(entry.getValue())) { - return new ImmutablePair<>(key, Boolean.parseBoolean((String) entry.getValue())); - } else { - return new ImmutablePair<>(key, entry.getValue()); - } - }).collect(HashMap::new, (map, pair) -> map.put(pair.getKey(), pair.getValue()), - HashMap::putAll); - } } diff --git a/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/restful/ScriptisProxyUserController.java b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/restful/ScriptisProxyUserController.java new file mode 100644 index 0000000000..b568fb2dbc --- /dev/null +++ b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/restful/ScriptisProxyUserController.java @@ -0,0 +1,49 @@ +package com.webank.wedatasphere.dss.scriptis.restful; + +import com.webank.wedatasphere.dss.common.conf.DSSCommonConf; +import com.webank.wedatasphere.dss.framework.proxy.restful.DssProxyUserController; +import com.webank.wedatasphere.dss.scriptis.pojo.entity.ScriptisProxyUser; +import com.webank.wedatasphere.dss.scriptis.service.ScriptisProxyUserService; +import org.apache.commons.lang.exception.ExceptionUtils; +import org.apache.commons.lang3.ArrayUtils; +import org.apache.commons.lang3.StringUtils; +import org.apache.linkis.server.Message; +import org.apache.linkis.server.security.SecurityFilter; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.web.bind.annotation.RequestBody; +import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RequestMethod; +import org.springframework.web.bind.annotation.RestController; + +import javax.servlet.http.HttpServletRequest; + +@RequestMapping(path = "/dss/scriptis/proxy", produces = {"application/json"}) +@RestController +public class ScriptisProxyUserController extends DssProxyUserController { + + @Autowired + private ScriptisProxyUserService scriptisProxyUserService; + + @RequestMapping(path = "add", method = RequestMethod.POST) + public Message add(@RequestBody ScriptisProxyUser userRep, HttpServletRequest req) { + String username = SecurityFilter.getLoginUsername(req); + if(!ArrayUtils.contains(DSSCommonConf.SUPER_ADMIN_LIST, username)){ + return Message.error("Only super admin can add proxy users."); + } else if(StringUtils.isEmpty(userRep.getUserName())){ + return Message.error("userName is null."); + } else if(StringUtils.isEmpty(userRep.getProxyUserName())){ + return Message.error("proxyUser is null."); + } else if (dssProxyUserService.isExists(userRep.getUserName(), userRep.getProxyUserName(), null)) { + return Message.error("Failed to add proxy user,'userName:" + userRep.getUserName() + ", proxyName:"+userRep.getProxyUserName()+" already exists."); + } + LOGGER.info("admin {} try to add proxy user, params:{}.", username, userRep); + try { + scriptisProxyUserService.insertProxyUser(userRep); + } catch (Exception exception) { + LOGGER.error("Failed to add proxy user.", exception); + return Message.error(ExceptionUtils.getRootCauseMessage(exception)); + } + return Message.ok("Success to add proxy user."); + } + +} diff --git a/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/restful/ScriptisResultSetAuditController.java b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/restful/ScriptisResultSetAuditController.java new file mode 100644 index 0000000000..bea9e4f817 --- /dev/null +++ b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/restful/ScriptisResultSetAuditController.java @@ -0,0 +1,62 @@ +package com.webank.wedatasphere.dss.scriptis.restful; + +import com.github.pagehelper.PageHelper; +import com.github.pagehelper.PageInfo; +import com.webank.wedatasphere.dss.common.conf.DSSCommonConf; +import com.webank.wedatasphere.dss.scriptis.pojo.entity.DssScriptDownloadAudit; +import com.webank.wedatasphere.dss.scriptis.service.DssScriptDownloadService; +import org.apache.commons.lang3.ArrayUtils; +import org.apache.linkis.server.Message; +import org.apache.linkis.server.security.SecurityFilter; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.web.bind.annotation.*; + +import javax.servlet.http.HttpServletRequest; +import javax.validation.Valid; +import java.util.List; + + +@RestController +@RequestMapping(path = "/dss/scriptis/audit", produces = {"application/json"}) +public class ScriptisResultSetAuditController { + + private static final Logger LOGGER = LoggerFactory.getLogger(ScriptisResultSetAuditController.class); + @Autowired + private DssScriptDownloadService dssScriptDownloadService; + + @RequestMapping(path = "download/save", method = RequestMethod.POST) + public Message saveScriptDownload(@RequestBody @Valid DssScriptDownloadAudit dssScriptDownloadAudit, HttpServletRequest request) { + String userName = SecurityFilter.getLoginUsername(request); + dssScriptDownloadAudit.setCreator(userName); + String sql = dssScriptDownloadAudit.getSql(); + if (sql.length() > 2000) { + sql = sql.substring(0, 2000); + dssScriptDownloadAudit.setSql(sql); + } + LOGGER.info("user {} try to saveScriptDownload, params:{}", userName, dssScriptDownloadAudit); + dssScriptDownloadService.save(dssScriptDownloadAudit); + return Message.ok("保存成功!"); + } + + @RequestMapping(path = "download/query", method = RequestMethod.GET) + public Message getScriptDownload(HttpServletRequest req, + @RequestParam(required = false) String userName, @RequestParam(required = false) String startTime, + @RequestParam(required = false) String endTime, @RequestParam(required = false) Integer pn) { + if(pn == null || pn < 1) { + pn = 1; + } + String loginUser = SecurityFilter.getLoginUsername(req); + if(!ArrayUtils.contains(DSSCommonConf.SUPER_ADMIN_LIST, loginUser)) { + return Message.error("Only super admin can query the download history of user " + userName); + } + PageHelper.startPage(pn, 10, true); + LOGGER.info("user {} try to queryScriptDownload, query userName: {}, startTime:{}, endTime:{}, page:{}", loginUser, userName, startTime, endTime, pn); + List userPage = dssScriptDownloadService.getDownloadAuditList(userName, startTime, endTime); + PageInfo pageInfo = new PageInfo<>(userPage); + return Message.ok().data("data", pageInfo); + } +} + + diff --git a/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/service/DssScriptDownloadService.java b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/service/DssScriptDownloadService.java new file mode 100644 index 0000000000..9b1b635d28 --- /dev/null +++ b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/service/DssScriptDownloadService.java @@ -0,0 +1,13 @@ +package com.webank.wedatasphere.dss.scriptis.service; + +import com.baomidou.mybatisplus.extension.service.IService; +import com.webank.wedatasphere.dss.scriptis.pojo.entity.DssScriptDownloadAudit; + +import java.util.List; + + +public interface DssScriptDownloadService extends IService { + + List getDownloadAuditList(String userName, String startIme, String endTime); + +} diff --git a/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/service/ScriptisAuthService.java b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/service/ScriptisAuthService.java new file mode 100644 index 0000000000..677bfe14ec --- /dev/null +++ b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/service/ScriptisAuthService.java @@ -0,0 +1,8 @@ +package com.webank.wedatasphere.dss.scriptis.service; + +import java.util.Map; + +public interface ScriptisAuthService { + + Map getGlobalLimits(String username); +} diff --git a/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/service/ScriptisProxyUserService.java b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/service/ScriptisProxyUserService.java new file mode 100644 index 0000000000..96f08b2f3c --- /dev/null +++ b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/service/ScriptisProxyUserService.java @@ -0,0 +1,10 @@ +package com.webank.wedatasphere.dss.scriptis.service; + + +import com.webank.wedatasphere.dss.scriptis.pojo.entity.ScriptisProxyUser; + +public interface ScriptisProxyUserService { + + int insertProxyUser(ScriptisProxyUser dssProxyUser); + +} diff --git a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/service/impl/DssScriptDownloadServiceImpl.java b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/service/impl/DssScriptDownloadServiceImpl.java similarity index 62% rename from dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/service/impl/DssScriptDownloadServiceImpl.java rename to dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/service/impl/DssScriptDownloadServiceImpl.java index bddd3bd4b1..7b14a3d9fe 100644 --- a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/service/impl/DssScriptDownloadServiceImpl.java +++ b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/service/impl/DssScriptDownloadServiceImpl.java @@ -1,27 +1,23 @@ -package com.webank.wedatasphere.dss.framework.admin.service.impl; +package com.webank.wedatasphere.dss.scriptis.service.impl; import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; -import com.webank.wedatasphere.dss.framework.admin.pojo.entity.DssScriptDownloadAudit; -import com.webank.wedatasphere.dss.framework.admin.service.DssScriptDownloadService; -import com.webank.wedatasphere.dss.framework.admin.xml.DssAuditMapper; +import com.webank.wedatasphere.dss.scriptis.dao.DssAuditMapper; +import com.webank.wedatasphere.dss.scriptis.pojo.entity.DssScriptDownloadAudit; +import com.webank.wedatasphere.dss.scriptis.service.DssScriptDownloadService; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Service; import java.util.List; -/** - * @Auther: Han Tang - * @Date: 2022/1/11-01-11-15:29 - */ + @Service public class DssScriptDownloadServiceImpl extends ServiceImpl implements DssScriptDownloadService { @Autowired public DssAuditMapper dssAuditMapper; + @Override public List getDownloadAuditList(String userName, String startIme, String endTime) { return dssAuditMapper.getDownloadAuditList(userName, startIme, endTime); } - - } diff --git a/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/service/impl/ScriptisAuthServiceImpl.java b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/service/impl/ScriptisAuthServiceImpl.java new file mode 100644 index 0000000000..9e84239703 --- /dev/null +++ b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/service/impl/ScriptisAuthServiceImpl.java @@ -0,0 +1,14 @@ +package com.webank.wedatasphere.dss.scriptis.service.impl; + +import com.webank.wedatasphere.dss.common.utils.GlobalLimitsUtils; +import com.webank.wedatasphere.dss.scriptis.service.ScriptisAuthService; + +import java.util.Map; + +public class ScriptisAuthServiceImpl implements ScriptisAuthService { + + @Override + public Map getGlobalLimits(String username) { + return GlobalLimitsUtils.getAllGlobalLimits(); + } +} diff --git a/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/service/impl/ScriptisProxyUserServiceImpl.java b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/service/impl/ScriptisProxyUserServiceImpl.java new file mode 100644 index 0000000000..8ca8116ae4 --- /dev/null +++ b/dss-apps/dss-scriptis-server/src/main/java/com/webank/wedatasphere/dss/scriptis/service/impl/ScriptisProxyUserServiceImpl.java @@ -0,0 +1,31 @@ +package com.webank.wedatasphere.dss.scriptis.service.impl; + +import com.webank.wedatasphere.dss.common.entity.DSSWorkspace; +import com.webank.wedatasphere.dss.framework.proxy.pojo.entity.DssProxyUser; +import com.webank.wedatasphere.dss.framework.proxy.service.DssProxyUserService; +import com.webank.wedatasphere.dss.scriptis.dao.ScriptisProxyUserMapper; +import com.webank.wedatasphere.dss.scriptis.pojo.entity.ScriptisProxyUser; +import com.webank.wedatasphere.dss.scriptis.service.ScriptisProxyUserService; +import org.springframework.stereotype.Service; + +import javax.annotation.Resource; +import java.util.ArrayList; +import java.util.List; + +@Service +public class ScriptisProxyUserServiceImpl implements DssProxyUserService, ScriptisProxyUserService { + + @Resource + private ScriptisProxyUserMapper dssProxyUserMapper; + + @Override + public List selectProxyUserList(String userName, DSSWorkspace workspace) { + return new ArrayList<>(dssProxyUserMapper.selectProxyUserList(userName)); + } + + @Override + public int insertProxyUser(ScriptisProxyUser dssProxyUser) { + return dssProxyUserMapper.insertUser(dssProxyUser); + } + +} diff --git a/dss-apps/dss-user-guide/dss-user-guide-server/pom.xml b/dss-apps/dss-user-guide/dss-user-guide-server/pom.xml index 34b8e7d9c0..ebe9f90874 100644 --- a/dss-apps/dss-user-guide/dss-user-guide-server/pom.xml +++ b/dss-apps/dss-user-guide/dss-user-guide-server/pom.xml @@ -5,7 +5,8 @@ dss-user-guide com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 + ../pom.xml 4.0.0 diff --git a/dss-apps/dss-user-guide/dss-user-guide-server/src/main/java/com/webank/wedatasphere/dss/guide/server/conf/GuideConf.java b/dss-apps/dss-user-guide/dss-user-guide-server/src/main/java/com/webank/wedatasphere/dss/guide/server/conf/GuideConf.java index e3c0e111fe..1628beec63 100644 --- a/dss-apps/dss-user-guide/dss-user-guide-server/src/main/java/com/webank/wedatasphere/dss/guide/server/conf/GuideConf.java +++ b/dss-apps/dss-user-guide/dss-user-guide-server/src/main/java/com/webank/wedatasphere/dss/guide/server/conf/GuideConf.java @@ -12,7 +12,7 @@ public interface GuideConf { CommonVars HOST_IP_ADDRESS = CommonVars.apply("target.ip.address", "127.0.0.1"); - CommonVars TARGET_GITBOOK_PATH = CommonVars.apply("target.gitbook.path", "/appcom/Install/ApacheInstall"); + CommonVars TARGET_GITBOOK_PATH = CommonVars.apply("target.gitbook.path", "/appcom/Install/ApacheInstall/gitbook_books"); CommonVars SUMMARY_IGNORE_MODEL = CommonVars.apply("summary.ignore.model","km"); diff --git a/dss-apps/dss-user-guide/dss-user-guide-server/src/main/java/com/webank/wedatasphere/dss/guide/server/restful/KnowledgeGuideAdminRestful.java b/dss-apps/dss-user-guide/dss-user-guide-server/src/main/java/com/webank/wedatasphere/dss/guide/server/restful/KnowledgeGuideAdminRestful.java index dc956d4984..8d6bfe17f8 100644 --- a/dss-apps/dss-user-guide/dss-user-guide-server/src/main/java/com/webank/wedatasphere/dss/guide/server/restful/KnowledgeGuideAdminRestful.java +++ b/dss-apps/dss-user-guide/dss-user-guide-server/src/main/java/com/webank/wedatasphere/dss/guide/server/restful/KnowledgeGuideAdminRestful.java @@ -7,9 +7,7 @@ import com.webank.wedatasphere.dss.guide.server.service.GuideChapterService; import com.webank.wedatasphere.dss.guide.server.service.GuideGroupService; import com.webank.wedatasphere.dss.guide.server.util.FileUtils; -import com.webank.wedatasphere.dss.guide.server.util.ShellUtils; import lombok.AllArgsConstructor; -import org.apache.commons.lang3.StringUtils; import org.apache.linkis.common.utils.Utils; import org.apache.linkis.server.Message; import org.apache.linkis.server.security.SecurityFilter; @@ -203,32 +201,11 @@ public Message fileUpload(@RequestParam(required = true) MultipartFile file) { @PostConstruct public void syncKnowledge() { final String summaryPath = GuideConf.HOST_GITBOOK_PATH.getValue() + File.separator + SUMMARY; - final String savePath = GuideConf.TARGET_GITBOOK_PATH.getValue() + File.separator + "gitbook_books"; - final String scpCommand = "scp -r " - + " hadoop@" + GuideConf.HOST_IP_ADDRESS.getValue() + ":" - + GuideConf.HOST_GITBOOK_PATH.getValue() + " " - + GuideConf.TARGET_GITBOOK_PATH.getValue(); - String delMkdir = "rm -rf " + savePath; logger.info("开始执行定时任务..."); Utils.defaultScheduler().scheduleAtFixedRate(() -> { try { - if (StringUtils.equals(GuideConf.GUIDE_SYNC_MODEL.getValue(), MODEL_GITBOOK_SYNC)) { - String hostIp = ShellUtils.callShellQuery(SHELL_COMMAND_HOST_IP); - //如果不是当前节点,则需要拷贝文件 - if (!StringUtils.equals(hostIp, GuideConf.HOST_IP_ADDRESS.getValue())) { - //判断文件是否存在 - boolean flag = FileUtils.fileExist(savePath); - if (flag) { - //删除文件 - ShellUtils.callShellByExec(delMkdir); - } - //拷贝文件到相应节点 - ShellUtils.callShellByExec(scpCommand); - } - }else { - guideCatalogService.syncKnowledge(summaryPath, GuideConf.SUMMARY_IGNORE_MODEL.getValue()); - guideGroupService.asyncGuide(summaryPath, GuideConf.SUMMARY_IGNORE_MODEL.getValue()); - } + guideCatalogService.syncKnowledge(summaryPath, GuideConf.SUMMARY_IGNORE_MODEL.getValue()); + guideGroupService.asyncGuide(summaryPath, GuideConf.SUMMARY_IGNORE_MODEL.getValue()); } catch (Exception e) { logger.error("定时任务执行异常:" + e); throw new RuntimeException(e); diff --git a/dss-apps/dss-user-guide/dss-user-guide-server/src/main/java/com/webank/wedatasphere/dss/guide/server/util/MdAnalysis.java b/dss-apps/dss-user-guide/dss-user-guide-server/src/main/java/com/webank/wedatasphere/dss/guide/server/util/MdAnalysis.java index d60e0b148f..f2dad0d2f0 100644 --- a/dss-apps/dss-user-guide/dss-user-guide-server/src/main/java/com/webank/wedatasphere/dss/guide/server/util/MdAnalysis.java +++ b/dss-apps/dss-user-guide/dss-user-guide-server/src/main/java/com/webank/wedatasphere/dss/guide/server/util/MdAnalysis.java @@ -48,10 +48,11 @@ public static String readMd(String filePath) { return null; } FileReader fr = null; + BufferedReader br = null; StringBuffer sb = new StringBuffer(""); try { fr = new FileReader(filePath); - BufferedReader br = new BufferedReader(fr); + br = new BufferedReader(fr); String line = br.readLine(); while (line != null) { sb.append(line); @@ -62,9 +63,14 @@ public static String readMd(String filePath) { throw new RuntimeException(e); } finally { try { - fr.close(); + if (br != null) { + br.close(); + } + if (fr != null) { + fr.close(); + } } catch (IOException e) { - throw new RuntimeException(e); + logger.error(String.valueOf(e)); } } return sb.toString(); @@ -79,38 +85,39 @@ public static String readMd(String filePath) { */ public static List>> analysisMd(String filePath, String type, String ignoreModel) throws IOException { logger.info("开始解析summary.md文件=============》"); - BufferedReader br = new BufferedReader(new InputStreamReader(new FileInputStream(filePath), "UTF-8")); - List>> mapList = new ArrayList<>(); - String line = null; - boolean flag = false; - while ((line = br.readLine()) != null) { - Map> map = new HashMap<>(); - Map dataMap = new HashMap<>(); - if (StringUtils.equals(type, "guide")) { - if (StringUtils.equals(line.trim(), type)) { - logger.info("开始解析学习引导模块"); - flag = true; + try(BufferedReader br = new BufferedReader(new InputStreamReader(new FileInputStream(filePath), "UTF-8"))){ + List>> mapList = new ArrayList<>(); + String line = null; + boolean flag = false; + while ((line = br.readLine()) != null) { + Map> map = new HashMap<>(); + Map dataMap = new HashMap<>(); + if (StringUtils.equals(type, "guide")) { + if (StringUtils.equals(line.trim(), type)) { + logger.info("开始解析学习引导模块"); + flag = true; + } + if (StringUtils.equals(line.trim(), "knowledge") || ignoreModel.contains(line.trim())) { + flag = false; + } } - if (StringUtils.equals(line.trim(), "knowledge") || ignoreModel.contains(line.trim())) { - flag = false; - } - } - if (StringUtils.equals(type, "knowledge")) { - if (StringUtils.equals(line.trim(), type)) { - logger.info("开始解析知识库模块"); - flag = true; + if (StringUtils.equals(type, "knowledge")) { + if (StringUtils.equals(line.trim(), type)) { + logger.info("开始解析知识库模块"); + flag = true; + } + if (StringUtils.equals(line.trim(), "guide") || ignoreModel.contains(line.trim())) { + flag = false; + } } - if (StringUtils.equals(line.trim(), "guide") || ignoreModel.contains(line.trim())) { - flag = false; + if (flag) { + absolveKnowledge(line, dataMap, map, mapList); } } - if (flag) { - absolveKnowledge(line, dataMap, map, mapList); - } + Y = 0; + Z = 0; + return mapList; } - Y = 0; - Z = 0; - return mapList; } private static void absolveKnowledge(String line, Map dataMap, Map> map, List>> mapList) { diff --git a/dss-apps/dss-user-guide/dss-user-guide-server/src/main/java/com/webank/wedatasphere/dss/guide/server/util/ShellUtils.java b/dss-apps/dss-user-guide/dss-user-guide-server/src/main/java/com/webank/wedatasphere/dss/guide/server/util/ShellUtils.java deleted file mode 100644 index bb497155c7..0000000000 --- a/dss-apps/dss-user-guide/dss-user-guide-server/src/main/java/com/webank/wedatasphere/dss/guide/server/util/ShellUtils.java +++ /dev/null @@ -1,59 +0,0 @@ -package com.webank.wedatasphere.dss.guide.server.util; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import java.io.BufferedReader; -import java.io.InputStreamReader; - -public class ShellUtils { - - private static final Logger logger = LoggerFactory.getLogger(ShellUtils.class); - - /** - * 执行shell命令 - * - * @param shellString - */ - public static void callShellByExec(String shellString) { - try { - logger.error("shellString:" + shellString); - Process process = Runtime.getRuntime().exec(shellString); - int exitValue = process.waitFor(); - if (0 != exitValue) { - logger.error("call shell failed. error code is :" + exitValue); - } - } catch (Throwable e) { - logger.error("call shell failed. " + e); - } - } - - /** - * 执行shell命令 - * - * @param shellString - */ - public static String callShellQuery(String shellString) { - logger.error("shellString:" + shellString); - BufferedReader reader = null; - String result = ""; - try { - Process process = Runtime.getRuntime().exec(shellString); - int exitValue = process.waitFor(); - if (0 != exitValue) { - logger.error("call shell failed. error code is :" + exitValue); - } - // 返回值 - reader = new BufferedReader(new InputStreamReader(process.getInputStream())); - String line = null; - while ((line = reader.readLine()) != null) { - result += line; - } - } catch (Throwable e) { - logger.error("call shell failed. " + e); - } - logger.info("call shell query result:" + result); - return result.trim(); - } - -} diff --git a/dss-apps/dss-user-guide/pom.xml b/dss-apps/dss-user-guide/pom.xml index 9f1094218f..3d3f10d8fd 100644 --- a/dss-apps/dss-user-guide/pom.xml +++ b/dss-apps/dss-user-guide/pom.xml @@ -5,7 +5,8 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 + ../../pom.xml 4.0.0 diff --git a/dss-apps/pom.xml b/dss-apps/pom.xml new file mode 100644 index 0000000000..eb6d5ee0c0 --- /dev/null +++ b/dss-apps/pom.xml @@ -0,0 +1,39 @@ + + + + + + dss + com.webank.wedatasphere.dss + 1.1.2 + ../pom.xml + + 4.0.0 + + dss-apps + pom + + + dss-apiservice-server + dss-scriptis-server + dss-user-guide + dss-apps-server + + + \ No newline at end of file diff --git a/dss-commons/dss-common/pom.xml b/dss-commons/dss-common/pom.xml index b48afb1ab7..d8fbfec56b 100644 --- a/dss-commons/dss-common/pom.xml +++ b/dss-commons/dss-common/pom.xml @@ -22,7 +22,8 @@ dss-commons com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 + ../pom.xml dss-common @@ -44,10 +45,6 @@ com.fasterxml.jackson.core jackson-databind - - org.codehaus.jackson - jackson-mapper-asl - org.scala-lang scala-library @@ -121,17 +118,43 @@ + + spring-aop + org.springframework + ${spring-framework.version} + + + spring-context-support + org.springframework + ${spring-framework.version} + + + spring-webmvc + org.springframework + ${spring-framework.version} + org.apache.linkis linkis-module ${linkis.version} provided + + org.apache.linkis + linkis-bml-client + ${linkis.version} + provided + mysql mysql-connector-java ${mysql.connector.version} - + + + org.apache.linkis + linkis-mybatis + ${linkis.version} + provided diff --git a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/alter/AlterContext.java b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/alter/AlterContext.java new file mode 100644 index 0000000000..f52879c14d --- /dev/null +++ b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/alter/AlterContext.java @@ -0,0 +1,16 @@ +package com.webank.wedatasphere.dss.common.alter; + +import com.webank.wedatasphere.dss.common.entity.Alter; + +public class AlterContext { + + private final ExceptionAlterSender exceptionAlterSender; + + public AlterContext(ExceptionAlterSender exceptionAlterSender) { + this.exceptionAlterSender = exceptionAlterSender; + } + + public void executeStrategy(Alter alter) { + exceptionAlterSender.sendAlter(alter); + } +} diff --git a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/alter/CustomAlterServiceImpl.java b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/alter/CustomAlterServiceImpl.java new file mode 100644 index 0000000000..a0a6420652 --- /dev/null +++ b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/alter/CustomAlterServiceImpl.java @@ -0,0 +1,11 @@ +package com.webank.wedatasphere.dss.common.alter; + +import com.webank.wedatasphere.dss.common.entity.Alter; + +public class CustomAlterServiceImpl implements ExceptionAlterSender { + + @Override + public void sendAlter(Alter alter) { + // Users can customize their own alarm methods + } +} diff --git a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/alter/ExceptionAlterSender.java b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/alter/ExceptionAlterSender.java new file mode 100644 index 0000000000..ae99e57c32 --- /dev/null +++ b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/alter/ExceptionAlterSender.java @@ -0,0 +1,10 @@ +package com.webank.wedatasphere.dss.common.alter; + +import com.webank.wedatasphere.dss.common.entity.Alter; + +public interface ExceptionAlterSender { + /** + * send alter to user when server exception, + */ + void sendAlter(Alter alter); +} diff --git a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/alter/ExecuteAlter.java b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/alter/ExecuteAlter.java new file mode 100644 index 0000000000..1333ed8577 --- /dev/null +++ b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/alter/ExecuteAlter.java @@ -0,0 +1,16 @@ +package com.webank.wedatasphere.dss.common.alter; + +import com.webank.wedatasphere.dss.common.conf.AlterConfiguration; +import com.webank.wedatasphere.dss.common.entity.Alter; +import org.springframework.stereotype.Component; + +@Component +public class ExecuteAlter { + + public void sendAlter(Alter alter) { + ExceptionAlterSender alterSender = AlterConfiguration.getAlter(); + AlterContext alterContext = new AlterContext(alterSender); + alterContext.executeStrategy(alter); + } + +} diff --git a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/auditlog/OperateTypeEnum.java b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/auditlog/OperateTypeEnum.java new file mode 100644 index 0000000000..6fc7bed4f2 --- /dev/null +++ b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/auditlog/OperateTypeEnum.java @@ -0,0 +1,41 @@ +package com.webank.wedatasphere.dss.common.auditlog; + +/** + * 审计日志操作类型枚举 + * Author: xlinliu + * Date: 2022/8/10 + */ +public enum OperateTypeEnum { + CREATE("create"), + UPDATE("update"), + DELETE("delete"), + COPY("copy"), + PUBLISH("publish"), + DISABLE("disable"), + ENABLE("enable"), + ADD_TO_FAVORITES("add_to_favorites"), + REM_FROM_FAVORITES("rem_from_favorites"), + UPDATE_ROLE_MENU("update_role_menu"), + UPDATE_ROLE_COMPONENT("update_role_component"), + ADD_USERS("add_users"), + UPDATE_USERS("update_users"), + KILL("kill"), + SEND_EMAIL("send_email"), + + + ; + + private String name; + + OperateTypeEnum(String name){ + this.name = name; + } + + public String getName() { + return name; + } + + public void setName(String name) { + this.name = name; + } +} diff --git a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/auditlog/TargetTypeEnum.java b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/auditlog/TargetTypeEnum.java new file mode 100644 index 0000000000..c0385102c6 --- /dev/null +++ b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/auditlog/TargetTypeEnum.java @@ -0,0 +1,64 @@ +package com.webank.wedatasphere.dss.common.auditlog; +/** + * 审计日志操作目标类型枚举 + * Author: xlinliu + * Date: 2022/8/10 + */ +public enum TargetTypeEnum { + /** + * 项目 + */ + PROJECT("project"), + /** + * 工作空间 + */ + WORKSPACE("workspace"), + /** + * API服务 + */ + APISERVICE("apiservice"), + /** + * 工作流 + */ + WORKFLOW("workflow"), + /** + * 编排 + */ + ORCHESTRATOR("orchestrator"), + /** + * context id + */ + CONTEXTID("contextid"), + /** + * 工作空间角色 + */ + WORKSPACE_ROLE("workspace_role"), + /** + * 数据库操作 + */ + DATAPIPE("datapipe"), + /** + * ec 引擎实例 + */ + EC_INSTANCE("ec_instance"), + + /** + * ec 自动释放规则 + */ + EC_KILL_STRATEGY("ec_kill_strategy"), + + ; + private String name; + + TargetTypeEnum(String name) { + this.name = name; + } + + public String getName() { + return name; + } + + public void setName(String name) { + this.name = name; + } +} diff --git a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/conf/AlterConfiguration.java b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/conf/AlterConfiguration.java new file mode 100644 index 0000000000..958999b027 --- /dev/null +++ b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/conf/AlterConfiguration.java @@ -0,0 +1,32 @@ +package com.webank.wedatasphere.dss.common.conf; + +import com.webank.wedatasphere.dss.common.alter.CustomAlterServiceImpl; +import com.webank.wedatasphere.dss.common.alter.ExceptionAlterSender; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class AlterConfiguration { + + private static final Logger logger = LoggerFactory.getLogger(AlterConfiguration.class); + + private static final ExceptionAlterSender ALTER = createAlter(); + + private static ExceptionAlterSender createAlter() { + + String alterClassName = DSSCommonConf.ALTER_CLASS.getValue(); + + try { + logger.info("Use user config Alter {}", alterClassName); + return (ExceptionAlterSender) AlterConfiguration.class.getClassLoader().loadClass(alterClassName).newInstance(); + } catch (Exception e) { + logger.warn("Use CustomAlter {}", alterClassName, e); + return new CustomAlterServiceImpl(); + } + + } + + public static ExceptionAlterSender getAlter() { + return ALTER; + } + +} diff --git a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/conf/DSSCommonConf.java b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/conf/DSSCommonConf.java index d5ba12e370..d1a08b564c 100644 --- a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/conf/DSSCommonConf.java +++ b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/conf/DSSCommonConf.java @@ -31,4 +31,24 @@ public class DSSCommonConf { */ public static final CommonVars DSS_TOKEN_TICKET_KEY = CommonVars.apply("wds.dss.user.ticket.key", "linkis_user_session_ticket_id_v1"); + public static final String[] SUPER_ADMIN_LIST = CommonVars.apply("wds.dss.super.admin", "").getValue().split(","); + + public static final CommonVars ALL_GLOBAL_LIMITS_PREFIX = CommonVars.apply("wds.dss.global.limits.prefix", "wds.dss.global.limits."); + + public static final CommonVars GLOBAL_LIMIT_PREFIX = CommonVars.apply("wds.dss.global.limit.prefix", "wds.dss.global.limit."); + + /** + * deploy DSS instance numbers + */ + public static final CommonVars DSS_INSTANCE_NUMBERS = CommonVars.apply("wds.dss.instance.numbers", 2); + /** + * check dss server is active period, The value can only be between 0 and 59 inclusive, or a multiple of 60 and its unit is 's'. + */ + public static final CommonVars DSS_CHECK_SERVER_ACTIVE_PERIOD = CommonVars.apply("wds.dss.check.server.active.period", 60); + /** + * The value can only be between 0 and 59 inclusive, or a multiple of 60 and its unit is 'second'. + */ + public static final CommonVars DSS_EC_KILL_PERIOD = CommonVars.apply("wds.dss.ec.kill.period", 60); + public static final CommonVars ALTER_CLASS = CommonVars.apply("wds.dss.alter.class", "com.webank.wedatasphere.dss.common.server.alter.ImsAlterServiceImpl"); + public static final CommonVars ALTER_RECEIVER = CommonVars.apply("wds.dss.alter.receiver", "burdezhang,mouhonghao"); } diff --git a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/conf/DSSCommonSpringConf.java b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/conf/DSSCommonSpringConf.java new file mode 100644 index 0000000000..a59687ce76 --- /dev/null +++ b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/conf/DSSCommonSpringConf.java @@ -0,0 +1,21 @@ +package com.webank.wedatasphere.dss.common.conf; + +import com.webank.wedatasphere.dss.common.utils.AssembleCronUtils; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; + +@Configuration +public class DSSCommonSpringConf { + + @Bean + public String getCheckInstanceIsActiveCron() { + Integer value = DSSCommonConf.DSS_CHECK_SERVER_ACTIVE_PERIOD.getValue(); + return AssembleCronUtils.getCron(value); + } + @Bean + public String getECInstanceReleaseCron(){ + Integer value=DSSCommonConf.DSS_EC_KILL_PERIOD.getValue(); + return AssembleCronUtils.getCron(value); + } + +} diff --git a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/Alter.java b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/Alter.java new file mode 100644 index 0000000000..d3fc7e39d3 --- /dev/null +++ b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/Alter.java @@ -0,0 +1,57 @@ +package com.webank.wedatasphere.dss.common.entity; + +public abstract class Alter { + + /** + * 告警标题,少于100个字符,必填项 + */ + private String alterTitle; + + private String alterInfo; + + private String alterLevel; + + private String alterReceiver; + + protected Alter() { + } + + protected Alter(String alterTitle, String alterInfo, String alterLevel, String alterReceiver) { + this.alterTitle = alterTitle; + this.alterInfo = alterInfo; + this.alterLevel = alterLevel; + this.alterReceiver = alterReceiver; + } + + public String getAlterTitle() { + return alterTitle; + } + + public void setAlterTitle(String alterTitle) { + this.alterTitle = alterTitle; + } + + public String getAlterInfo() { + return alterInfo; + } + + public void setAlterInfo(String alterInfo) { + this.alterInfo = alterInfo; + } + + public String getAlterLevel() { + return alterLevel; + } + + public void setAlterLevel(String alterLevel) { + this.alterLevel = alterLevel; + } + + public String getAlterReceiver() { + return alterReceiver; + } + + public void setAlterReceiver(String alterReceiver) { + this.alterReceiver = alterReceiver; + } +} diff --git a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/BmlResource.java b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/BmlResource.java new file mode 100644 index 0000000000..07cb6da99d --- /dev/null +++ b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/BmlResource.java @@ -0,0 +1,32 @@ +package com.webank.wedatasphere.dss.common.entity; + +public class BmlResource { + + private String resourceId; + + private String version; + + public BmlResource(String resourceId, String version) { + this.resourceId = resourceId; + this.version = version; + } + + public BmlResource() { + } + + public String getResourceId() { + return resourceId; + } + + public void setResourceId(String resourceId) { + this.resourceId = resourceId; + } + + public String getVersion() { + return version; + } + + public void setVersion(String version) { + this.version = version; + } +} diff --git a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/CustomAlter.java b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/CustomAlter.java new file mode 100644 index 0000000000..132a8d4a52 --- /dev/null +++ b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/CustomAlter.java @@ -0,0 +1,15 @@ +package com.webank.wedatasphere.dss.common.entity; + +public class CustomAlter extends Alter{ + + + public CustomAlter() { + } + + public CustomAlter(String alterTitle, String alterInfo, String alterLevel, String alterReceiver) { + super(alterTitle, alterInfo, alterLevel, alterReceiver); + + } + + +} diff --git a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/InputRelation.java b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/InputRelation.java deleted file mode 100644 index a6a02a4916..0000000000 --- a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/InputRelation.java +++ /dev/null @@ -1,75 +0,0 @@ -/* - * Copyright 2019 WeBank - * 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. - * - */ - -package com.webank.wedatasphere.dss.common.entity; - -public class InputRelation { - - private Long id; - private String type; - private IOEnv sourceEnv; - private Long sourceID; - private IOEnv targetEnv; - private Long targetID; - - public Long getId() { - return id; - } - - public void setId(Long id) { - this.id = id; - } - - public String getType() { - return type; - } - - public void setType(String type) { - this.type = type; - } - - public IOEnv getSourceEnv() { - return sourceEnv; - } - - public void setSourceEnv(IOEnv sourceEnv) { - this.sourceEnv = sourceEnv; - } - - public Long getSourceID() { - return sourceID; - } - - public void setSourceID(Long sourceID) { - this.sourceID = sourceID; - } - - public IOEnv getTargetEnv() { - return targetEnv; - } - - public void setTargetEnv(IOEnv targetEnv) { - this.targetEnv = targetEnv; - } - - public Long getTargetID() { - return targetID; - } - - public void setTargetID(Long targetID) { - this.targetID = targetID; - } -} diff --git a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/PageInfo.java b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/PageInfo.java new file mode 100644 index 0000000000..70115d901f --- /dev/null +++ b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/PageInfo.java @@ -0,0 +1,34 @@ +package com.webank.wedatasphere.dss.common.entity; + +import java.util.List; + +/** + * 分页结果通用类 + * Author: xlinliu + * Date: 2023/4/20 + */ +public class PageInfo { + List data; + long total; + + public PageInfo(List data, long total) { + this.data = data; + this.total = total; + } + + public List getData() { + return data; + } + + public void setData(List data) { + this.data = data; + } + + public long getTotal() { + return total; + } + + public void setTotal(long total) { + this.total = total; + } +} \ No newline at end of file diff --git a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/Resource.java b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/Resource.java index c814c7b027..b89d002beb 100644 --- a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/Resource.java +++ b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/Resource.java @@ -65,4 +65,13 @@ public boolean equals(Object o) { public int hashCode() { return Objects.hash(resourceId, version); } + + @Override + public String toString() { + return "Resource{" + + "fileName='" + fileName + '\'' + + ", resourceId='" + resourceId + '\'' + + ", version='" + version + '\'' + + '}'; + } } diff --git a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/node/DSSEdge.java b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/node/DSSEdge.java index 1d64eb7595..8fde23a298 100644 --- a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/node/DSSEdge.java +++ b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/node/DSSEdge.java @@ -20,7 +20,7 @@ public interface DSSEdge { String getSource(); - void setSource(); + void setSource(String source); String getTarget(); diff --git a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/node/DSSEdgeDefault.java b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/node/DSSEdgeDefault.java index c6c6389162..a6803ac646 100644 --- a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/node/DSSEdgeDefault.java +++ b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/node/DSSEdgeDefault.java @@ -28,7 +28,7 @@ public String getSource() { } @Override - public void setSource() { + public void setSource(String source) { this.source = source; } diff --git a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/node/DSSNode.java b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/node/DSSNode.java index c37438e53f..cbdf1484f3 100644 --- a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/node/DSSNode.java +++ b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/node/DSSNode.java @@ -18,6 +18,7 @@ import com.webank.wedatasphere.dss.common.entity.Resource; +import java.util.Date; import java.util.List; import java.util.Map; @@ -42,4 +43,8 @@ public interface DSSNode extends Node { String getUserProxy(); void setUserProxy(String userProxy); + + String getModifyUser(); + Long getModifyTime(); + String getDesc(); } diff --git a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/node/DSSNodeDefault.java b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/node/DSSNodeDefault.java index 72d24fefab..db5efa3111 100644 --- a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/node/DSSNodeDefault.java +++ b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/entity/node/DSSNodeDefault.java @@ -19,12 +19,14 @@ import com.webank.wedatasphere.dss.common.entity.Resource; import java.util.ArrayList; +import java.util.Date; import java.util.List; import java.util.Map; public class DSSNodeDefault implements DSSNode { private Layout layout; private String id; + private String key; private String jobType; private String lastUpdateTime; private Map params; @@ -33,6 +35,8 @@ public class DSSNodeDefault implements DSSNode { private String desc; private String createTime; private String userProxy; + private String modifyUser; + private Long modifyTime; /** * dependencys 是该Node的依赖节点 */ @@ -144,6 +148,21 @@ public List getDependencys() { return dependencys; } + @Override + public String getModifyUser() { + return modifyUser; + } + + @Override + public Long getModifyTime() { + return modifyTime; + } + + @Override + public String getDesc() { + return desc; + } + @Override public String toString() { return "dwsNode{" + diff --git a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/exception/Apply.java b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/exception/Apply.java new file mode 100644 index 0000000000..2009baa9a1 --- /dev/null +++ b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/exception/Apply.java @@ -0,0 +1,13 @@ +package com.webank.wedatasphere.dss.common.exception; + +/** + * @author enjoyyin + * @date 2022-07-20 + * @since 1.1.1 + */ +@FunctionalInterface +public interface Apply { + + void apply(); + +} diff --git a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/exception/ThrowingApply.java b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/exception/ThrowingApply.java new file mode 100644 index 0000000000..d1ddb0d167 --- /dev/null +++ b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/exception/ThrowingApply.java @@ -0,0 +1,13 @@ +package com.webank.wedatasphere.dss.common.exception; + +/** + * @author enjoyyin + * @date 2022-07-20 + * @since 1.1.1 + */ +@FunctionalInterface +public interface ThrowingApply { + + void apply() throws E; + +} diff --git a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/exception/ThrowingSupplier.java b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/exception/ThrowingSupplier.java new file mode 100644 index 0000000000..e9675612f7 --- /dev/null +++ b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/exception/ThrowingSupplier.java @@ -0,0 +1,13 @@ +package com.webank.wedatasphere.dss.common.exception; + +/** + * @author enjoyyin + * @date 2022-07-20 + * @since 1.1.1 + */ +@FunctionalInterface +public interface ThrowingSupplier { + + T get() throws E; + +} diff --git a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/protocol/JobStatus.java b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/protocol/JobStatus.java index 8409ea1f6f..afc778959c 100644 --- a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/protocol/JobStatus.java +++ b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/protocol/JobStatus.java @@ -16,6 +16,8 @@ package com.webank.wedatasphere.dss.common.protocol; +import java.util.Objects; + public enum JobStatus { /** @@ -30,14 +32,37 @@ public enum JobStatus { private String status; private int index; - private JobStatus(String status, int index){ + JobStatus(String status, int index){ this.status = status; this.index = index; } - public static boolean isSuccess(String status){ - return Success.status.equals(status); + public static JobStatus getJobStatusByStatus(String status) { + if (JobStatus.Inited.getStatus().equals(status)) { + return JobStatus.Inited; + } else if (JobStatus.Running.getStatus().equals(status)) { + return JobStatus.Running; + } else if (JobStatus.Success.getStatus().equals(status)) { + return JobStatus.Success; + } else { + return JobStatus.Failed; + } + } + + public String getStatus() { + return status; + } + + public void setStatus(String status) { + this.status = status; + } + + public int getIndex() { + return index; } + public void setIndex(int index) { + this.index = index; + } } diff --git a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/utils/AssembleCronUtils.java b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/utils/AssembleCronUtils.java new file mode 100644 index 0000000000..887491c78a --- /dev/null +++ b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/utils/AssembleCronUtils.java @@ -0,0 +1,30 @@ +package com.webank.wedatasphere.dss.common.utils; + +/** + * get cron parameter + */ +public class AssembleCronUtils { + + /** + * 该方法可以根据传入的数据,将其转换成cron表达式,最小周期间隔为1s,最大为每天零点执行一次 + * @param value cron表达式的执行周期 + * @return cron表达式 + */ + public static String getCron(Integer value) { + + if (value < 60) { + // 每隔多少秒执行一次,小于60s + return "0/" + value + " * * * * ?"; + } else if (value < 3600) { + // 每隔多少分钟执行一次,小于60min + return "0 0/" + value / 60 + " * * * ?"; + } else if (value < 86400){ + // 每隔多少小时执行一次,小于24h + return "0 0 0/" + value / 60 + " * * ?"; + } else { + // 每天零点执行一次 + return "0 0 0 * * ?"; + } + } + +} diff --git a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/utils/AuditLogUtils.java b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/utils/AuditLogUtils.java new file mode 100644 index 0000000000..e7ff8dad45 --- /dev/null +++ b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/utils/AuditLogUtils.java @@ -0,0 +1,51 @@ +package com.webank.wedatasphere.dss.common.utils; + +import com.google.gson.Gson; +import com.webank.wedatasphere.dss.common.auditlog.OperateTypeEnum; +import com.webank.wedatasphere.dss.common.auditlog.TargetTypeEnum; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import java.util.Date; + +/** + * 审计日志工具类 + */ +public class AuditLogUtils { + + private static final Logger LOGGER = LoggerFactory.getLogger(AuditLogUtils.class); + + /** + * 打印审计日志,id类的属性都是String + * @param user 执行操作的用户名 + * @param workspaceId 操作发生的工作空间id + * @param workspaceName 操作发生的工作空间名称 + * @param targetType 操作针对的对象类型 + * @param targetId 操作针对的对象id + * @param targetName 操作针对的对象名称 + * @param operateType 操作类型 + * @param params 操作相关的参数 + */ + public static void printLog(String user, String workspaceId, String workspaceName, TargetTypeEnum targetType, + String targetId, String targetName, OperateTypeEnum operateType,Object params) { + String detailInfo=new Gson().toJson(params); + LOGGER.info("[{}],[{}],[{}],[{}],[{}],[{}],[{}],[{}],[{}]", + new Date(),user, workspaceId,workspaceName,targetType.getName(), + targetId,targetName,operateType.getName(), detailInfo); + } + /** + * 打印审计日志,id类的属性都是Long类型 + * @param user 执行操作的用户名 + * @param workspaceId 操作发生的工作空间id + * @param workspaceName 操作发生的工作空间名称 + * @param targetType 操作针对的对象类型 + * @param targetId 操作针对的对象id + * @param targetName 操作针对的对象名称 + * @param operateType 操作类型 + * @param params 操作相关的参数 + */ + public static void printLog(String user, long workspaceId, String workspaceName, TargetTypeEnum targetType, + long targetId, String targetName, OperateTypeEnum operateType,Object params) { + printLog(user, String.valueOf(workspaceId), workspaceName, targetType, + String.valueOf(targetId), targetName, operateType, params); + } +} diff --git a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/utils/DSSCommonUtils.java b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/utils/DSSCommonUtils.java index b51032b545..61c766fd5a 100644 --- a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/utils/DSSCommonUtils.java +++ b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/utils/DSSCommonUtils.java @@ -17,14 +17,17 @@ package com.webank.wedatasphere.dss.common.utils; import com.fasterxml.jackson.databind.ObjectMapper; -import com.google.gson.Gson; -import com.google.gson.GsonBuilder; -import com.google.gson.JsonPrimitive; -import com.google.gson.JsonSerializer; +import com.google.common.reflect.TypeToken; +import com.google.gson.*; +import com.webank.wedatasphere.dss.common.entity.node.DSSNode; +import com.webank.wedatasphere.dss.common.entity.node.DSSNodeDefault; import com.webank.wedatasphere.dss.common.exception.DSSRuntimeException; import org.apache.linkis.common.conf.CommonVars; import org.apache.linkis.common.utils.JsonUtils; +import java.util.List; +import java.util.stream.Collectors; + public class DSSCommonUtils { public static final String FLOW_RESOURCE_NAME = "resources"; @@ -76,4 +79,25 @@ public static long parseToLong(Object val) { throw new DSSRuntimeException(90322, "parse the return of externalSystem failed, the value is null."); } + public static List getWorkFlowNodes(String workFlowJson) { + JsonParser parser = new JsonParser(); + JsonObject jsonObject = parser.parse(workFlowJson).getAsJsonObject(); + JsonArray nodeJsonArray = jsonObject.getAsJsonArray("nodes"); + List dwsNodes = DSSCommonUtils.COMMON_GSON.fromJson(nodeJsonArray, new TypeToken>() { + }.getType()); + return dwsNodes; + } + + public static List getWorkFlowNodesJson(String workFlowJson) { + JsonParser parser = new JsonParser(); + JsonObject jsonObject = parser.parse(workFlowJson).getAsJsonObject(); + JsonArray nodeJsonArray = jsonObject.getAsJsonArray("nodes"); + if (nodeJsonArray == null) { + return null; + } + List nodeJsonList = DSSCommonUtils.COMMON_GSON.fromJson(nodeJsonArray.toString(), new TypeToken>() { + }.getType()); + return nodeJsonList.stream().map(DSSCommonUtils.COMMON_GSON::toJson).collect(Collectors.toList()); + } + } diff --git a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/utils/DSSExceptionUtils.java b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/utils/DSSExceptionUtils.java index 3660a1d2fc..95f3777db0 100644 --- a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/utils/DSSExceptionUtils.java +++ b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/utils/DSSExceptionUtils.java @@ -16,18 +16,18 @@ package com.webank.wedatasphere.dss.common.utils; -import com.webank.wedatasphere.dss.common.exception.DSSRuntimeException; -import com.webank.wedatasphere.dss.common.exception.ThrowingConsumer; -import com.webank.wedatasphere.dss.common.exception.ThrowingFunction; +import com.webank.wedatasphere.dss.common.exception.*; import org.apache.commons.lang.exception.ExceptionUtils; import org.apache.linkis.common.exception.ErrorException; import org.apache.linkis.common.exception.WarnException; +import org.apache.linkis.server.Message; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import java.lang.reflect.Constructor; import java.util.function.Consumer; import java.util.function.Function; +import java.util.function.Supplier; public class DSSExceptionUtils { @@ -121,4 +121,45 @@ public static void dealWarnException(int errorCode, St throw warnException; } + public static Message getMessage(ThrowingSupplier supplier, Function function, String errorMsg) { + T result; + try { + result = supplier.get(); + } catch (ErrorException e) { + LOGGER.error(errorMsg, e); + return Message.error(errorMsg + e.getDesc()); + } catch (WarnException e) { + LOGGER.error(errorMsg, e); + return Message.error(errorMsg + e.getDesc()); + } catch (Exception e) { + LOGGER.error(errorMsg, e); + return Message.error(errorMsg + " 原因:" + ExceptionUtils.getRootCauseMessage(e)); + } + return function.apply(result); + } + + public static Message getMessage(ThrowingApply apply, Supplier supplier, String errorMsg) { + try { + apply.apply(); + } catch (ErrorException e) { + LOGGER.error(errorMsg, e); + return Message.error(errorMsg + e.getDesc()); + } catch (WarnException e) { + LOGGER.error(errorMsg, e); + return Message.error(errorMsg + e.getDesc()); + } catch (Exception e) { + LOGGER.error(errorMsg, e); + return Message.error(errorMsg + " 原因:" + ExceptionUtils.getRootCauseMessage(e)); + } + return supplier.get(); + } + + public static Message applyMessage(Supplier supplier, Function function, String errorMsg) { + return getMessage(supplier::get, function, errorMsg); + } + + public static Message applyMessage(Apply apply, Supplier supplier, String errorMsg) { + return getMessage(apply::apply, supplier, errorMsg); + } + } diff --git a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/utils/DomainUtils.java b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/utils/DomainUtils.java new file mode 100644 index 0000000000..8695e55734 --- /dev/null +++ b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/utils/DomainUtils.java @@ -0,0 +1,70 @@ +package com.webank.wedatasphere.dss.common.utils; + +import com.webank.wedatasphere.dss.common.conf.DSSCommonConf; + +import javax.servlet.http.HttpServletRequest; +import java.util.Arrays; +import java.util.regex.Matcher; +import java.util.regex.Pattern; + + +public class DomainUtils { + + private static final Pattern DOMAIN_REGEX = Pattern.compile("[a-zA-Z][a-zA-Z0-9\\.]+"); + private static final Pattern IP_REGEX = Pattern.compile("([^:]+):.+"); + + public static String getCookieDomain(HttpServletRequest request) { + String referer = request.getHeader("Referer"); + if(referer==null){ + referer=request.getServerName(); + } + return getCookieDomain(referer); + } + + /** + * "dss.com" -> ".dss.com" + * "127.0.0.1" -> "127.0.0.1" + * "127.0.0.1:8080" -> "127.0.0.1" + * @param host the Host in HttpRequest Headers + * @return + */ + public static String getCookieDomain(String host) { + int level = DSSCommonConf.DSS_DOMAIN_LEVEL.getValue(); + if(host.startsWith("https://")) { + host = host.substring(8); + } else if(host.startsWith("http://")) { + host = host.substring(7); + } + if (host.endsWith("/")) { + host = host.substring(0, host.length() - 1); + } + if(DOMAIN_REGEX.matcher(host).find()) { + String[] domains = host.split("\\."); + int index = level; + if (domains.length == level) { + index = level - 1; + } else if (domains.length < level) { + index = domains.length; + } + if (index < 0) { + return host; + } + String[] parsedDomains = Arrays.copyOfRange(domains, index, domains.length); + if (parsedDomains.length < level) { + return host; + } + String domain = String.join(".", parsedDomains); + if(domains.length >= level) { + return "." + domain; + } + return domain; + } + Matcher matcher = IP_REGEX.matcher(host); + if(matcher.find()) { + return matcher.group(1); + } else { + return host; + } + } + +} diff --git a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/utils/GlobalLimitsUtils.java b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/utils/GlobalLimitsUtils.java new file mode 100644 index 0000000000..3aca1bbf90 --- /dev/null +++ b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/utils/GlobalLimitsUtils.java @@ -0,0 +1,66 @@ +package com.webank.wedatasphere.dss.common.utils; + +import com.webank.wedatasphere.dss.common.conf.DSSCommonConf; +import org.apache.commons.collections4.MapUtils; +import org.apache.commons.lang.StringUtils; +import org.apache.linkis.common.conf.BDPConfiguration; +import org.apache.linkis.protocol.util.ImmutablePair; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.HashMap; +import java.util.Map; + +/** + * @author enjoyyin + * @date 2022-09-21 + * @since 0.5.0 + */ +public class GlobalLimitsUtils { + + private static final Logger logger = LoggerFactory.getLogger(GlobalLimitsUtils.class); + + private static volatile Map globalLimits; + private static final Map> globalLimitContents = new HashMap<>(); + + public static Map getAllGlobalLimits() { + if(globalLimits == null) { + synchronized (GlobalLimitsUtils.class) { + if(globalLimits == null) { + globalLimits = MapUtils.unmodifiableMap(getMap(DSSCommonConf.ALL_GLOBAL_LIMITS_PREFIX.acquireNew())); + logger.info("loaded global limits is {}.", globalLimits); + } + } + } + return globalLimits; + } + + public static Map getGlobalLimitMap(String globalLimitName) { + if(globalLimitContents.containsKey(globalLimitName)) { + return globalLimitContents.get(globalLimitName); + } + synchronized (globalLimitContents) { + if(!globalLimitContents.containsKey(globalLimitName)) { + Map globalLimitContent = getMap(DSSCommonConf.GLOBAL_LIMIT_PREFIX.acquireNew() + globalLimitName + "."); + logger.info("loaded global limit {}, the contents are {}.", globalLimitName, globalLimitContent); + globalLimitContents.put(globalLimitName, MapUtils.unmodifiableMap(globalLimitContent)); + } + } + return globalLimitContents.get(globalLimitName); + } + + private static Map getMap(String prefix) { + return BDPConfiguration.properties().entrySet().stream() + .filter(entry -> entry.getKey().toString().startsWith(prefix) && entry.getValue() != null && StringUtils.isNotBlank(entry.getValue().toString())) + .map(entry -> { + String key = ((String) entry.getKey()).substring(prefix.length()); + if("true".equals(entry.getValue()) || "false".equals(entry.getValue())) { + return new ImmutablePair<>(key, Boolean.parseBoolean((String) entry.getValue())); + } else { + return new ImmutablePair<>(key, entry.getValue()); + } + }).collect(HashMap::new, (map, pair) -> map.put(pair.getKey(), pair.getValue()), + HashMap::putAll); + } + +} diff --git a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/utils/IoUtils.java b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/utils/IoUtils.java index d73d9d51df..ab0610e618 100644 --- a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/utils/IoUtils.java +++ b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/utils/IoUtils.java @@ -17,7 +17,6 @@ package com.webank.wedatasphere.dss.common.utils; import com.webank.wedatasphere.dss.common.conf.DSSCommonConf; -import com.webank.wedatasphere.dss.common.entity.IOEnv; import com.webank.wedatasphere.dss.common.entity.IOType; import org.apache.commons.io.FileUtils; import org.slf4j.Logger; @@ -29,12 +28,23 @@ import java.util.Date; import java.util.Properties; +/** + * IO通用类。提供对一次IO操作的临时目录创建、文件创建、IO流创建功能。 + * 另外,提供一份IO properties记录本次IO的上下文信息,并以IO.properties文件的形式存放到临时目录下。通过提供的方法,可以方便的写入properties。 + */ public class IoUtils { private static Logger logger = LoggerFactory.getLogger(IoUtils.class); private static final String DATE_FORMAT = "yyyyMMddHHmmssSSS"; private static final String DEFAULT_IO_FILE_NAME = "IO.properties"; + /** + * 生成一个基础目录,这个目录用于某次任务的临时文件目录 + * @param userName 用户名 + * @param projectName 项目名 + * @param subDir 自定义的子目录名 + * @return 基础目录的全路径 + */ public static String generateIOPath(String userName,String projectName,String subDir) { String baseUrl = DSSCommonConf.DSS_EXPORT_URL.getValue(); String dataStr = new SimpleDateFormat(DATE_FORMAT).format(new Date()); @@ -45,6 +55,12 @@ private static String addFileSeparator(String... str) { return Arrays.stream(str).reduce((a, b) -> a + File.separator + b).orElse(""); } + /** + * 根据指定的文件路径,创建一个输出流 + * @param path 指定的文件路径 + * @return 输出流 + * @throws IOException 如果有io异常的话,直接抛出 + */ public static OutputStream generateExportOutputStream(String path) throws IOException { File file = new File(path); if (!file.getParentFile().exists()) { @@ -57,15 +73,33 @@ public static OutputStream generateExportOutputStream(String path) throws IOExce file.createNewFile(); return FileUtils.openOutputStream(file, true); } - + /** + * 根据指定的文件路径,创建一个输入流 + * @param path 指定的文件路径 + * @return 输入流 + * @throws IOException 如果有io异常的话,直接抛出 + */ public static InputStream generateInputInputStream(String path)throws IOException{ return new FileInputStream(path); } + /** + * 标记io类型到IO properties中 + * @param ioType + * @param basePath + * @throws IOException + */ public static void generateIOType(IOType ioType, String basePath) throws IOException { generateIOProperties("type",ioType.name(),basePath); } + /** + * 写一个properties键值对到IO properties中 + * @param key 键 + * @param value 值 + * @param basePath 目标目录 + * @throws IOException 有io异常,直接抛出 + */ public static void generateIOProperties(String key,String value,String basePath) throws IOException { Properties properties = new Properties(); properties.setProperty(key,value); @@ -83,6 +117,12 @@ public static void generateIOProperties(String key,String value,String basePath) } + /** + * 读取一个IO properties中的值 + * @param key 要读取的properties + * @param basepath 读取的目录 + * @return 读取的值 + */ public static String readIOProperties(String key,String basepath) throws IOException { try(FileInputStream inputStream = new FileInputStream(basepath + File.separator + DEFAULT_IO_FILE_NAME)){ Properties properties = new Properties(); @@ -91,26 +131,20 @@ public static String readIOProperties(String key,String basepath) throws IOExcep } } - + /** + * 标记当前环境label值到IO properties中 + * @param basePath + * @throws IOException + */ public static void generateIOEnv(String basePath) throws IOException { - generateIOProperties("env",getDSSServerEnv().name(),basePath); + generateIOProperties("env",getDSSServerEnv(),basePath); } - public static IOEnv getDSSServerEnv(){ + /** + * 获取当前服务器的环境 + */ + public static String getDSSServerEnv(){ //dssserverEnv 是当前dss服务启动的env环境 - return IOEnv.valueOf(DSSCommonConf.DSS_IO_ENV.getValue()); - } - - public static String addVersion(String version){ - String num = String.valueOf(Integer.valueOf(version.substring(1)) + 1); - int length = num.length(); - return version.substring(0,version.length() - length) + num; - } - - public static String subVersion(String version){ - String num = String.valueOf(Integer.valueOf(version.substring(1)) - 1); - int length = num.length(); - return version.substring(0,version.length() - length) + num; + return DSSCommonConf.DSS_IO_ENV.getValue(); } - } diff --git a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/utils/RpcAskUtils.java b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/utils/RpcAskUtils.java new file mode 100644 index 0000000000..83ea759b3d --- /dev/null +++ b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/utils/RpcAskUtils.java @@ -0,0 +1,22 @@ +package com.webank.wedatasphere.dss.common.utils; + +import org.apache.linkis.common.exception.LinkisRuntimeException; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +/** + * Process rpc exception + */ +public class RpcAskUtils { + + private static final Logger logger = LoggerFactory.getLogger(RpcAskUtils.class); + + public static T processAskException(Object o, Class responseClazz, Class requestClazz) { + if (o.getClass().equals(responseClazz)) { + return responseClazz.cast(o); + } else { + logger.error(requestClazz.getSimpleName() + " failed for " + o); + throw (LinkisRuntimeException) o; + } + } +} diff --git a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/utils/ZipHelper.java b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/utils/ZipHelper.java index fcc31ea1bc..62e0c54ced 100644 --- a/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/utils/ZipHelper.java +++ b/dss-commons/dss-common/src/main/java/com/webank/wedatasphere/dss/common/utils/ZipHelper.java @@ -39,6 +39,11 @@ public class ZipHelper { private static final String RECURSIVE = "-r"; private static final String ZIP_TYPE = ".zip"; + /** + * ZipHelper可以将传入的path进行打包,打包之后删除path目录 + * @param dirPath 需要打包的project路径,绝对路径 + * @return 打包之后的zip包全路径 + */ public static String zip(String dirPath) throws DSSErrorException { return zip(dirPath, true); } @@ -46,9 +51,13 @@ public static String zip(String dirPath) throws DSSErrorException { /** * ZipHelper可以将传入的path进行打包 * @param dirPath 需要打包的project路径,绝对路径 + * @param deleteOriginDir 打包之后,是否要删除path目录 * @return 打包之后的zip包全路径 */ public static String zip(String dirPath, boolean deleteOriginDir)throws DSSErrorException { + if(dirPath.endsWith(File.separator)) { + dirPath = dirPath.substring(0, dirPath.lastIndexOf(File.separator)); + } if(!FileHelper.checkDirExists(dirPath)){ logger.error("{} 不存在, 不能创建zip文件", dirPath); throw new DSSErrorException(90001,dirPath + " does not exist, can not zip"); @@ -112,7 +121,12 @@ public static String zip(String dirPath, boolean deleteOriginDir)throws DSSError return longZipFilePath; } - + /** + * 解压一个zip文件 + * @param dirPath zip文件的全路径名 + * @return 解压后的文件夹名 + * @throws DSSErrorException 解压出现异常 + */ public static String unzip(String dirPath)throws DSSErrorException { File file = new File(dirPath); @@ -182,11 +196,4 @@ private static boolean deleteDir(File dir) { // 目录此时为空,可以删除 return dir.delete(); } - - public static String zipExportProject(String projectPath) throws DSSErrorException { - if(projectPath.endsWith(File.separator)) { - projectPath = projectPath.substring(0, projectPath.lastIndexOf(File.separator)); - } - return ZipHelper.zip(projectPath); - } } diff --git a/dss-commons/dss-common/src/main/scala/com/webank/wedatasphere/dss/common/RestfulCatchAOP.scala b/dss-commons/dss-common/src/main/scala/com/webank/wedatasphere/dss/common/RestfulCatchAOP.scala deleted file mode 100644 index 0aa92d7b2e..0000000000 --- a/dss-commons/dss-common/src/main/scala/com/webank/wedatasphere/dss/common/RestfulCatchAOP.scala +++ /dev/null @@ -1,48 +0,0 @@ -/* - * Copyright 2019 WeBank - * - * 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. - */ - -package com.webank.wedatasphere.dss.common - -import org.apache.linkis.common.utils.Logging -import org.apache.linkis.server.{Message, catchIt} -import org.aspectj.lang.ProceedingJoinPoint -import org.aspectj.lang.annotation.{Around, Aspect, Pointcut} -import org.springframework.stereotype.Component - -@Aspect -@Component -class RestfulCatchAOP extends Logging { - - @Pointcut("@annotation(javax.ws.rs.Path) && execution(public org.apache.linkis.server.Message *(..))") - def restfulMessageCatch(): Unit = {} - - @Around("restfulMessageCatch()") - def dealMessageRestful(proceedingJoinPoint: ProceedingJoinPoint): Object = catchIt { - proceedingJoinPoint.proceed().asInstanceOf[Message] - } - - @Pointcut("@annotation(javax.ws.rs.Path) && execution(public javax.ws.rs.core.Response *(..)))") - def restfulResponseCatch(): Unit = {} - - @Around("restfulResponseCatch()") - def dealResponseRestful(proceedingJoinPoint: ProceedingJoinPoint): Object = { - val resp: Message = catchIt { - Message - return proceedingJoinPoint.proceed() - } - resp - } -} diff --git a/dss-commons/dss-common/src/main/scala/com/webank/wedatasphere/dss/common/protocol/DSSProtocolObject.scala b/dss-commons/dss-common/src/main/scala/com/webank/wedatasphere/dss/common/protocol/DSSProtocolObject.scala index b9da4e5c6d..41e6b87c8b 100644 --- a/dss-commons/dss-common/src/main/scala/com/webank/wedatasphere/dss/common/protocol/DSSProtocolObject.scala +++ b/dss-commons/dss-common/src/main/scala/com/webank/wedatasphere/dss/common/protocol/DSSProtocolObject.scala @@ -25,7 +25,7 @@ case class ResponseExportOrchestrator(resourceId: String, version: String, orcVersionId: Long) -case class ResponseImportOrchestrator(orcId: Long) +case class ResponseImportOrchestrator(orcId: Long,version:String) case class RequestUpdateWorkflow(userName: String, flowID: Long, diff --git a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/scala/com/webank/wedatasphere/dss/workflow/service/BMLService.scala b/dss-commons/dss-common/src/main/scala/com/webank/wedatasphere/dss/common/service/BMLService.scala similarity index 68% rename from dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/scala/com/webank/wedatasphere/dss/workflow/service/BMLService.scala rename to dss-commons/dss-common/src/main/scala/com/webank/wedatasphere/dss/common/service/BMLService.scala index 4a057bbea5..1c6738df60 100644 --- a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/scala/com/webank/wedatasphere/dss/workflow/service/BMLService.scala +++ b/dss-commons/dss-common/src/main/scala/com/webank/wedatasphere/dss/common/service/BMLService.scala @@ -14,28 +14,30 @@ * */ -package com.webank.wedatasphere.dss.workflow.service - -import java.io.{ByteArrayInputStream, InputStream} -import java.util -import java.util.UUID +package com.webank.wedatasphere.dss.common.service +import com.webank.wedatasphere.dss.common.entity.BmlResource import com.webank.wedatasphere.dss.common.exception.DSSErrorException import com.webank.wedatasphere.dss.common.utils.IoUtils +import org.apache.commons.io.IOUtils import org.apache.linkis.bml.client.{BmlClient, BmlClientFactory} import org.apache.linkis.bml.protocol.{BmlDownloadResponse, BmlUpdateResponse, BmlUploadResponse} import org.apache.linkis.common.utils.{JavaLog, Utils} -import org.apache.commons.io.IOUtils -import org.springframework.stereotype.Component +import java.io.{ByteArrayInputStream, InputStream} +import java.util +import java.util.UUID import scala.collection.JavaConversions._ - -@Component class BMLService extends JavaLog { - var bmlClient: BmlClient = _ + @volatile var bmlClient: BmlClient = _ + /** + * 单例,获取bml client + * @param userName + * @return + */ def getBmlClient(userName: String): BmlClient = { if (bmlClient == null) synchronized { if (bmlClient == null) { @@ -45,6 +47,13 @@ class BMLService extends JavaLog { bmlClient } + /** + * 上传一段文本 + * @param content 要上传的文本内容 + * @param fileName 指定生成的文件名 + * @param projectName 指定项目名 + * @return 返回BmlResource + */ def upload(userName: String, content: String, fileName: String, projectName: String): util.Map[String, Object] = { val inputStream = new ByteArrayInputStream(content.getBytes("utf-8")) val client: BmlClient = getBmlClient(userName) @@ -55,15 +64,27 @@ class BMLService extends JavaLog { map += "version" -> resource.version } - def upload(userName: String, inputStream: InputStream, fileName: String, projectName: String): util.Map[String, Object] = { + /** + * 上传二进制流 + * + * @param inputStream 要上传的二进制流 + * @param fileName 指定生成的文件名 + * @param projectName 指定项目名 + * @return 上传后的结果BmlResource + */ + def upload(userName: String, inputStream: InputStream, fileName: String, projectName: String): BmlResource = { val client: BmlClient = getBmlClient(userName) val resource: BmlUploadResponse = client.uploadShareResource(userName, projectName, fileName, inputStream) if (!resource.isSuccess) throw new DSSErrorException(911113, "上传失败") - val map = new util.HashMap[String, Object] - map += "resourceId" -> resource.resourceId - map += "version" -> resource.version + new BmlResource(resource.resourceId, resource.version) } + /** + * 将特定资源更新为指定二进制流 + * @param resourceId 要更新的资源 + * @param inputStream 二进制流 + * @return 更新后的结果BmlResource + */ def update(userName: String, resourceId: String, inputStream: InputStream): util.Map[String, Object] = { val client: BmlClient = getBmlClient(userName) val resource: BmlUpdateResponse = client.updateShareResource(userName, resourceId, "", inputStream) @@ -73,6 +94,13 @@ class BMLService extends JavaLog { map += "version" -> resource.version } + /** + * 将特定资源更新为指定文本 + * + * @param resourceId 要更新的资源 + * @param inputStream 文本内容 + * @return 更新后的结果BmlResource + */ def update(userName: String, resourceId: String, content: String): util.Map[String, Object] = { val inputStream = new ByteArrayInputStream(content.getBytes("utf-8")) val client: BmlClient = getBmlClient(userName) @@ -83,6 +111,9 @@ class BMLService extends JavaLog { map += "version" -> resource.version } + /** + * 读取一个resource并转为文本返回。 + */ def query(userName: String, resourceId: String, version: String): util.Map[String, Object] = { val client: BmlClient = getBmlClient(userName) var resource: BmlDownloadResponse = null @@ -97,6 +128,10 @@ class BMLService extends JavaLog { map += "string" -> inputstremToString(resource.inputStream) } + /** + * 下载一个资源 + * @return 资源的路径和输入流 + */ def download(userName: String, resourceId: String, version: String): util.Map[String, Object] = { val client: BmlClient = getBmlClient(userName) var resource: BmlDownloadResponse = null @@ -111,6 +146,10 @@ class BMLService extends JavaLog { map += "is" -> resource.inputStream } + /** + * 把资源下载到指定的本地文件,返 + * @param path 本地文件全路径名 + */ def downloadToLocalPath(userName: String, resourceId: String, version: String, path: String): String = { val result = download(userName, resourceId, version) val is = result.get("is").asInstanceOf[InputStream] @@ -122,44 +161,56 @@ class BMLService extends JavaLog { path } - def downloadAndGetFlowJson(userName: String, resourceId: String, version: String, path: String): String = { + def downloadAndGetText(userName: String, resourceId: String, version: String, path: String): String = { downloadToLocalPath(userName, resourceId, version, path) //因为下载到指定目录后返回的resource的Stream为null,只能从文件重新读取。 val is = IoUtils.generateInputInputStream(path) Utils.tryFinally(inputstremToString(is))(IOUtils.closeQuietly(is)) } + /** + * 读取本地文件,返回二进制输入流 + * + * @param userName 用户名 + * @param readPath 本地文件全路径名 + * @return 二进制输入流 + */ def readLocalResourceFile(userName: String, readPath: String): InputStream = { IoUtils.generateInputInputStream(readPath) } - def readLocalFlowJsonFile(userName: String, readPath: String): String = { + /** + * 读取本地指定文件名的一段文本 + * @param readPath 本地文件全路径名 + * @return 文件里的文本内容 + */ + def readLocalTextFile(userName: String, readPath: String): String = { var inputStream: InputStream = null - var flowJson: String = null + var text: String = null Utils.tryFinally { inputStream = IoUtils.generateInputInputStream(readPath) - flowJson = inputstremToString(inputStream) + text = inputstremToString(inputStream) } { IOUtils.closeQuietly(inputStream) } - flowJson + text } - def readFlowJsonFromBML(userName: String, resourceId: String, version: String): String = { + def readTextFromBML(userName: String, resourceId: String, version: String): String = { val result = download(userName, resourceId, version) val is = result.get("is").asInstanceOf[InputStream] var inputStream: InputStream = null - var flowJson: String = null + var text: String = null Utils.tryFinally { inputStream = is - flowJson = inputstremToString(inputStream) + text = inputstremToString(inputStream) } { IOUtils.closeQuietly(inputStream) } - flowJson + text } @@ -167,8 +218,16 @@ class BMLService extends JavaLog { scala.io.Source.fromInputStream(inputStream).mkString } - def createBmlProject(username: String, projectName: String, editUsers: util.List[String], - accessUsers: util.List[String]): Unit = { + + /** + * 创建BML工程 + * @param username 用户名 + * @param projectName 工程名 + * @param editUsers 编辑者 + * @param accessUsers 访问者 + */ + def createBmlProject(username:String, projectName:String, editUsers:util.List[String], + accessUsers:util.List[String] ): Unit ={ val client = getBmlClient(username) val response = client.createBmlProject(username, projectName, accessUsers, editUsers) if (response.isSuccess) { @@ -188,7 +247,13 @@ class BMLService extends JavaLog { } } - + /** + * 修改BML工程中的编辑者、访问者等权限 + * @param projectName 工程名 + * @param username 操作用户名 + * @param editUsers 新的编辑者名单 + * @param accessUsers 新的访问者名单 + */ def updateProjectPriv(projectName: String, username: String, editUsers: util.List[String], accessUsers: util.List[String]): Unit = { val client = getBmlClient(username) @@ -200,4 +265,28 @@ class BMLService extends JavaLog { } } + def deleteBmlResource(username: String, resourceId: String): Unit = { + val client = getBmlClient(username) + val response = client.deleteResource(username, resourceId) + if (response.isSuccess) { + logger.info(s"delete $username bml resource file success, resourceId: $resourceId") + } else { + logger.error(s"delete $username bml resource file failed, resourceId: $resourceId") + } + } + } + +object BMLService { + @volatile var bmlService: BMLService = _ + + def getInstance(): BMLService = { + if (bmlService == null) synchronized { + if (bmlService == null) { + bmlService = new BMLService() + } + } + bmlService + } + +} \ No newline at end of file diff --git a/dss-commons/dss-common/src/main/scala/com/webank/wedatasphere/dss/common/utils/MessageUtils.scala b/dss-commons/dss-common/src/main/scala/com/webank/wedatasphere/dss/common/utils/MessageUtils.scala deleted file mode 100644 index 27d979d808..0000000000 --- a/dss-commons/dss-common/src/main/scala/com/webank/wedatasphere/dss/common/utils/MessageUtils.scala +++ /dev/null @@ -1,21 +0,0 @@ -package com.webank.wedatasphere.dss.common.utils - -import org.apache.linkis.server.Message - -import javax.ws.rs.core.Response - -object MessageUtils { - implicit def messageToResponse(message: Message): Response = - Response.status(messageToHttpStatus(message)).entity(message).build() - - //implicit def responseToMessage(response: Response): Message = response.readEntity(classOf[Message]) - - def messageToHttpStatus(message: Message): Int = message.getStatus match { - case -1 => 401 - case 0 => 200 - case 1 => 400 - case 2 => 412 - case 3 => 403 - case 4 => 206 - } -} diff --git a/dss-commons/dss-common/src/main/scala/com/webank/wedatasphere/dss/common/utils/ScalaFunctionAdapter.scala b/dss-commons/dss-common/src/main/scala/com/webank/wedatasphere/dss/common/utils/ScalaFunctionAdapter.scala new file mode 100644 index 0000000000..589812ffff --- /dev/null +++ b/dss-commons/dss-common/src/main/scala/com/webank/wedatasphere/dss/common/utils/ScalaFunctionAdapter.scala @@ -0,0 +1,32 @@ +package com.webank.wedatasphere.dss.common.utils + +import java.util.function.{BiFunction, Consumer, Predicate, Supplier,Function} + +import org.apache.linkis.protocol.util.ImmutablePair + +/** + * java 调用scala时,传入函数到scala的适配器 + * @date 2022-09-14 + * @author enjoyyin + * @since 0.5.0 + */ +object ScalaFunctionAdapter { + + def doConsumer[P](consumer: Consumer[P]): (P) => Unit = p => consumer.accept(p) + + def doPredicate[P](predicate: Predicate[P]): (P) => Boolean = p => predicate.test(p) + + def doSupplier[R](supplier: Supplier[R]): () => R = () => supplier.get() + + def doFunction[P, R](function: Function[P, R]): (P) => R = p => function.apply(p) + + def doBiFunction[P1, P2, R](function: BiFunction[P1, P2, R]): (P1, P2) => R = + (p1, p2) => function.apply(p1, p2) + + def do3Function[P1, P2, P3, R](function: BiFunction[ImmutablePair[P1, P2], P3, R]): (P1, P2, P3) => R = + (p1, p2, p3) => function.apply(new ImmutablePair[P1, P2](p1, p2), p3) + + def do4Function[P1, P2, P3, P4, R](function: BiFunction[ImmutablePair[P1, P2], ImmutablePair[P3, P4], R]): (P1, P2, P3, P4) => R = + (p1, p2, p3, p4) => function.apply(new ImmutablePair[P1, P2](p1, p2), new ImmutablePair[P3, P4](p3, p4)) + +} diff --git a/dss-commons/dss-common/src/main/scala/com/webank/wedatasphere/dss/common/utils/VariableUtils.scala b/dss-commons/dss-common/src/main/scala/com/webank/wedatasphere/dss/common/utils/VariableUtils.scala deleted file mode 100644 index 63c72ca198..0000000000 --- a/dss-commons/dss-common/src/main/scala/com/webank/wedatasphere/dss/common/utils/VariableUtils.scala +++ /dev/null @@ -1,757 +0,0 @@ -/* - * Copyright 2019 WeBank - * 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. - * - */ - -package com.webank.wedatasphere.dss.common.utils - -import java.text.SimpleDateFormat -import java.util -import java.util.concurrent.{ExecutorService, Executors} -import java.util.{Calendar, Date} - -import org.apache.linkis.common.exception.LinkisCommonErrorException -import org.apache.linkis.common.utils.{Logging, Utils} -import org.apache.commons.lang.StringUtils -import org.apache.commons.lang.time.DateUtils - -import scala.collection.mutable -import scala.collection.convert.WrapAsScala._ -import scala.util.matching.Regex - -object VariableUtils extends Logging { - - val RUN_DATE = "run_date" - val RUN_TODAY = "run_today" - val RUN_MON = "run_mon" - - val dateFormatLocal = new ThreadLocal[SimpleDateFormat]() { - override protected def initialValue = new SimpleDateFormat("yyyyMMdd") - } - - val dateFormatStdLocal = new ThreadLocal[SimpleDateFormat]() { - override protected def initialValue = new SimpleDateFormat("yyyy-MM-dd") - } - - val dateFormatMonLocal = new ThreadLocal[SimpleDateFormat]() { - override protected def initialValue = new SimpleDateFormat("yyyyMM") - } - - val dateFormatMonStdLocal = new ThreadLocal[SimpleDateFormat]() { - override protected def initialValue = new SimpleDateFormat("yyyy-MM") - } - - val codeReg: Regex = "\\$\\{\\s*[A-Za-z][A-Za-z0-9_\\.]*\\s*[\\+\\-\\*/]?\\s*[A-Za-z0-9_\\.]*\\s*\\}".r - val calReg: Regex = "(\\s*[A-Za-z][A-Za-z0-9_\\.]*\\s*)([\\+\\-\\*/]?)(\\s*[A-Za-z0-9_\\.]*\\s*)".r - - def replace(replaceStr: String): String = replace(replaceStr, new util.HashMap[String, Any](0)) - - def replace(replaceStr: String, variables: util.Map[String, Any]): String = { - val nameAndType = mutable.Map[String, VariableType]() - var run_date: CustomDateType = null - variables foreach { - case (RUN_DATE, value) if nameAndType.get(RUN_DATE).isEmpty => - val run_date_str = value.asInstanceOf[String] - info(s"replaceStr:${run_date_str}") - if (StringUtils.isNotEmpty(run_date_str)) { - run_date = new CustomDateType(run_date_str, false) - nameAndType(RUN_DATE) = DateType(run_date) - } - case (key, value: String) if !nameAndType.contains(key) && StringUtils.isNotEmpty(value) => - nameAndType(key) = Utils.tryCatch[VariableType](DoubleValue(value.toDouble))(_ => StringType(value)) - case _ => - } - if (!nameAndType.contains(RUN_DATE) || null == run_date) { - val getYesterdayValue = getYesterday(false) - info(s"getYesterday:${getYesterdayValue}") - run_date = new CustomDateType(getYesterdayValue, false) - nameAndType(RUN_DATE) = DateType(new CustomDateType(run_date.toString, false)) - } - initAllDateVars(run_date, nameAndType) - parserVar(replaceStr, nameAndType) - } - - private def initAllDateVars(run_date: CustomDateType, nameAndType: mutable.Map[String, VariableType]): Unit = { - - nameAndType("run_date_std") = DateType(new CustomDateType(run_date.getStdDate)) - nameAndType("run_month_begin") = MonthType(new CustomMonthType(run_date.toString, false)) - nameAndType("run_month_begin_std") = MonthType(new CustomMonthType(run_date.toString)) - nameAndType("run_month_end") = MonthType(new CustomMonthType(run_date.toString, false, true)) - nameAndType("run_month_end_std") = MonthType(new CustomMonthType(run_date.toString, true, true)) - - /* - Variables based on run_today - */ - // if (nameAndType.get(RUN_TODAY).isEmpty || null == run_today) { - // run_today = new CustomDateType(getToday(false), false) - // nameAndType(RUN_TODAY) = DateType(new CustomDateType(run_today.toString, false)) - // } - - val run_today = new CustomDateType(getToday(false, run_date + 1), false) - nameAndType(RUN_TODAY) = DateType(new CustomDateType(run_today.toString, false)) - nameAndType("run_today_std") = DateType(new CustomDateType(run_today.getStdDate)) - nameAndType("run_month_now_begin") = MonthType(new CustomMonthType(new CustomMonthType(run_today.toString, false) - 1, false)) - nameAndType("run_month_now_begin_std") = MonthType(new CustomMonthType(new CustomMonthType(run_today.toString, false) - 1)) - nameAndType("run_month_now_end") = MonthType(new CustomMonthType(new CustomMonthType(run_today.toString, false) - 1, false, true)) - nameAndType("run_month_now_end_std") = MonthType(new CustomMonthType(new CustomMonthType(run_today.toString, false) - 1, true, true)) - - // if (nameAndType.get(RUN_MON).isEmpty || null == run_mon) { - // run_mon = new CustomMonType(getMonthDay(false), false) - // nameAndType(RUN_MON) = MonType(new CustomMonType(run_mon.toString, false)) - // } - - // calculate run_mon base on run_date - val run_mon = new CustomMonType(getMonthDay(false, run_date.getDate), false) - nameAndType("run_mon") = MonType(new CustomMonType(run_mon.toString, false)) - nameAndType("run_mon_std") = MonType(new CustomMonType(run_mon.toString, true, false)) - nameAndType("run_mon_start") = MonType(new CustomMonType(run_mon.toString, false, false)) - nameAndType("run_mon_start_std") = MonType(new CustomMonType(run_mon.toString, true, false)) - nameAndType("run_mon_end") = MonType(new CustomMonType(run_mon.toString, false, true)) - nameAndType("run_mon_end_std") = MonType(new CustomMonType(run_mon.toString, true, true)) - - - nameAndType("run_quarter_begin") = QuarterType(new CustomQuarterType(run_date.toString, false)) - nameAndType("run_quarter_begin_std") = QuarterType(new CustomQuarterType(run_date.toString)) - nameAndType("run_quarter_end") = QuarterType(new CustomQuarterType(run_date.toString, false, true)) - nameAndType("run_quarter_end_std") = QuarterType(new CustomQuarterType(run_date.toString, true, true)) - - nameAndType("run_half_year_begin") = HalfYearType(new CustomHalfYearType(run_date.toString, false)) - nameAndType("run_half_year_begin_std") = HalfYearType(new CustomHalfYearType(run_date.toString)) - nameAndType("run_half_year_end") = HalfYearType(new CustomHalfYearType(run_date.toString, false, true)) - nameAndType("run_half_year_end_std") = HalfYearType(new CustomHalfYearType(run_date.toString, true, true)) - - nameAndType("run_year_begin") = YearType(new CustomYearType(run_date.toString, false)) - nameAndType("run_year_begin_std") = YearType(new CustomYearType(run_date.toString)) - nameAndType("run_year_end") = YearType(new CustomYearType(run_date.toString, false, true)) - nameAndType("run_year_end_std") = YearType(new CustomYearType(run_date.toString, true, true)) - - } - - /** - * Parse and replace the value of the variable - * 1.Get the expression and calculations - * 2.Print user log - * 3.Assemble code - * - * @param replaceStr : replaceStr - * @param nameAndType : variable name and Type - * @return - */ - private def parserVar(replaceStr: String, nameAndType: mutable.Map[String, VariableType]): String = { - val parseCode = new StringBuilder - val codes = codeReg.split(replaceStr) - var i = 0 - codeReg.findAllIn(replaceStr).foreach { str => - i = i + 1 - calReg.findFirstMatchIn(str).foreach { ma => - val name = ma.group(1) - val signal = ma.group(2) - val bValue = ma.group(3) - if (StringUtils.isBlank(name)) { - throw new LinkisCommonErrorException(20041, s"[$str] with empty variable name.") - } - val replacedStr = nameAndType.get(name.trim).map { varType => - if (StringUtils.isNotBlank(signal)) { - if (StringUtils.isBlank(bValue)) { - throw new LinkisCommonErrorException(20042, s"[$str] expression is not right, please check.") - } - varType.calculator(signal.trim, bValue.trim) - } else varType.getValue - }.getOrElse { - warn(s"Use undefined variables or use the set method: [$str](使用了未定义的变量或者使用了set方式:[$str])") - str - } - parseCode ++= codes(i - 1) ++ replacedStr - } - } - if (i == codes.length - 1) { - parseCode ++= codes(i) - } - StringUtils.strip(parseCode.toString) - } - - /** - * Get Yesterday"s date - * - * @param std :2017-11-16 - * @return - */ - private def getYesterday(std: Boolean = true): String = { - val dateFormat = dateFormatLocal.get() - val dateFormat_std = dateFormatStdLocal.get() - val cal: Calendar = Calendar.getInstance() - cal.add(Calendar.DATE, -1) - if (std) { - dateFormat_std.format(cal.getTime) - } else { - dateFormat.format(cal.getTime) - } - } - - // /** - // * Get Month"s date - // * - // * @param std :2017-11-01 - // * @param isEnd :01 or 30,31 - // * @return - // */ - // private[utils] def getMonth(date: Date, std: Boolean = true, isEnd: Boolean = false): String = { - // val dateFormat = dateFormatLocal.get() - // val dateFormat_std = dateFormatStdLocal.get() - // val cal = Calendar.getInstance() - // cal.setTime(date) - // cal.set(Calendar.DATE, 1) - // if (isEnd) { - // cal.roll(Calendar.DATE, -1) - // } - // if (std) { - // dateFormat_std.format(cal.getTime) - // } else { - // dateFormat.format(cal.getTime) - // } - // } - - /** - * Get Today"s date - * - * @param std :2017-11-16 - * @return - */ - def getToday(std: Boolean = true, dateString: String = null): String = { - val dateFormat = dateFormatLocal.get() - val dateFormat_std = dateFormatStdLocal.get() - val cal: Calendar = Calendar.getInstance() - if (dateString != null) { - cal.setTime(dateFormat.parse(dateString)) - } - if (std) { - dateFormat_std.format(cal.getTime) - } else { - dateFormat.format(cal.getTime) - } - } - - - /** - * - * @param std 202106 - * @return - */ - def getMonthDay(std: Boolean = true, date: Date = null): String = { - val dateFormat = dateFormatMonLocal.get() - val dateFormat_std = dateFormatMonStdLocal.get() - if (std) { - dateFormat_std.format(date) - } else { - dateFormat.format(date) - } - } - - /** - * Get Month"s date - * - * @param std :2017-11-01 - * @param isEnd :01 or 30,31 - * @return - */ - def getMonth(std: Boolean = true, isEnd: Boolean = false, date: Date): String = { - val dateFormat = dateFormatLocal.get() - val dateFormat_std = dateFormatStdLocal.get() - val cal: Calendar = Calendar.getInstance() - cal.setTime(date) - cal.set(Calendar.DATE, 1) - if (isEnd) { - cal.roll(Calendar.DATE, -1) - } - if (std) { - dateFormat_std.format(cal.getTime) - } else { - dateFormat.format(cal.getTime) - } - } - - def getMon(std: Boolean = true, isEnd: Boolean = false, date: Date): String = { - val dateFormat = dateFormatMonLocal.get() - val dateFormat_std = dateFormatMonStdLocal.get() - val cal: Calendar = Calendar.getInstance() - cal.setTime(date) - if (isEnd) { - cal.set(Calendar.MONTH, 12) - } - if (std) { - dateFormat_std.format(cal.getTime) - } else { - dateFormat.format(cal.getTime) - } - } - - /** - * get 1st day or last day of a Quarter - * - * @param std - * @param isEnd - * @param date - * @return - */ - def getQuarter(std: Boolean = true, isEnd: Boolean = false, date: Date): String = { - val dateFormat = dateFormatLocal.get() - val dateFormat_std = dateFormatStdLocal.get() - val cal: Calendar = Calendar.getInstance() - cal.setTime(date) - cal.set(Calendar.DATE, 1) - val monthDigit: Int = cal.get(Calendar.MONTH) //get method with MONTH field returns 0-11 - if (0 <= monthDigit && monthDigit <= 2) { - cal.set(Calendar.MONTH, 0) - } else if (3 <= monthDigit && monthDigit <= 5) { - cal.set(Calendar.MONTH, 3) - } else if (6 <= monthDigit && monthDigit <= 8) { - cal.set(Calendar.MONTH, 6) - } else if (9 <= monthDigit && monthDigit <= 11) { - cal.set(Calendar.MONTH, 9) - } - if (isEnd) { - cal.add(Calendar.MONTH, 2) - cal.roll(Calendar.DATE, -1) - } - if (std) { - dateFormat_std.format(cal.getTime) - } else { - dateFormat.format(cal.getTime) - } - } - - /** - * get 1st day or last day of a HalfYear - * - * @param std - * @param isEnd - * @param date - * @return - */ - def getHalfYear(std: Boolean = true, isEnd: Boolean = false, date: Date): String = { - val dateFormat = dateFormatLocal.get() - val dateFormat_std = dateFormatStdLocal.get() - val cal: Calendar = Calendar.getInstance() - cal.setTime(date) - cal.set(Calendar.DATE, 1) - val monthDigit: Int = cal.get(Calendar.MONTH) //get method with MONTH field returns 0-11 - if (0 <= monthDigit && monthDigit <= 5) { - cal.set(Calendar.MONTH, 0) - } else if (6 <= monthDigit && monthDigit <= 11) { - cal.set(Calendar.MONTH, 6) - } - if (isEnd) { - cal.add(Calendar.MONTH, 5) - cal.roll(Calendar.DATE, -1) - } - if (std) { - dateFormat_std.format(cal.getTime) - } else { - dateFormat.format(cal.getTime) - } - } - - /** - * get 1st day or last day of a year - * - * @param std - * @param isEnd - * @param date - * @return - */ - def getYear(std: Boolean = true, isEnd: Boolean = false, date: Date): String = { - val dateFormat = dateFormatLocal.get() - val dateFormat_std = dateFormatStdLocal.get() - val cal: Calendar = Calendar.getInstance() - cal.setTime(date) - cal.set(Calendar.DATE, 1) - cal.set(Calendar.MONTH, 0) // set methods with field MONTH accepts 0-11 - if (isEnd) { - cal.add(Calendar.MONTH, 11) - cal.roll(Calendar.DATE, -1) - } - if (std) { - dateFormat_std.format(cal.getTime) - } else { - dateFormat.format(cal.getTime) - } - } - - - def main(args: Array[String]): Unit = { - val code = "--@set a=1\n--@set b=2\nselect ${a +2},${a + 1},${a},${a },${run_mon},${run_mon_std},${b}" - - val args: java.util.Map[String, Any] = new util.HashMap[String, Any]() - args.put(RUN_DATE, "20181030") - - val str = VariableUtils.replace(code, args) - println(str) - val fixedThreadPool = Executors.newFixedThreadPool(300) - - val task1 = new Runnable { - override def run(): Unit = { - // println("task1 running") - println(Thread.currentThread().getName) - val str = VariableUtils.replace(code, args) - println(str) - // println("task1 complete") - } - } - - - for(a <- 1 to 1000) { - fixedThreadPool.execute(task1) - } - - // println("************code**************") - // var preSQL = "" - // var endSQL = "" - // var sql = "select * from (select * from utf_8_test limit 20) t ;" - // if (sql.contains("limit")) { - // preSQL = sql.substring(0, sql.lastIndexOf("limit")).trim - // endSQL = sql.substring(sql.lastIndexOf("limit")).trim - // } else if (sql.contains("LIMIT")) { - // preSQL = sql.substring(0, sql.lastIndexOf("limit")).trim - // endSQL = sql.substring(sql.lastIndexOf("limit")).trim - // } - // println(preSQL) - // println(endSQL) - /* val yestd = new CustomDateType("2017-11-11",false) - println(yestd)*/ - } - -} - - -trait VariableType { - - def getValue: String - - def calculator(signal: String, bValue: String): String - -} - -case class DateType(value: CustomDateType) extends VariableType { - override def getValue: String = value.toString - - def calculator(signal: String, bValue: String): String = signal match { - case "+" => value + bValue.toInt - case "-" => value - bValue.toInt - case _ => throw new LinkisCommonErrorException(20046, s"Date class is not supported to uss:$signal.") - } -} - - -case class MonthType(value: CustomMonthType) extends VariableType { - override def getValue: String = value.toString - - def calculator(signal: String, bValue: String): String = { - signal match { - case "+" => value + bValue.toInt - case "-" => value - bValue.toInt - case _ => throw new LinkisCommonErrorException(20046, s"Date class is not supported to uss:${signal}") - } - } -} - -case class MonType(value: CustomMonType) extends VariableType { - override def getValue: String = value.toString - - def calculator(signal: String, bValue: String): String = { - signal match { - case "+" => value + bValue.toInt - case "-" => value - bValue.toInt - case _ => throw new LinkisCommonErrorException(20046, s"Date class is not supported to uss:${signal}") - } - } -} - -case class QuarterType(value: CustomQuarterType) extends VariableType { - override def getValue: String = value.toString - - def calculator(signal: String, bValue: String): String = { - signal match { - case "+" => value + bValue.toInt - case "-" => value - bValue.toInt - case _ => throw new LinkisCommonErrorException(20046, s"Date class is not supported to uss:${signal}") - } - } -} - -case class HalfYearType(value: CustomHalfYearType) extends VariableType { - override def getValue: String = value.toString - - def calculator(signal: String, bValue: String): String = { - signal match { - case "+" => value + bValue.toInt - case "-" => value - bValue.toInt - case _ => throw new LinkisCommonErrorException(20046, s"Date class is not supported to uss:${signal}") - } - } -} - -case class YearType(value: CustomYearType) extends VariableType { - override def getValue: String = value.toString - - def calculator(signal: String, bValue: String): String = { - signal match { - case "+" => value + bValue.toInt - case "-" => value - bValue.toInt - case _ => throw new LinkisCommonErrorException(20046, s"Date class is not supported to uss:${signal}") - } - } -} - -case class LongType(value: Long) extends VariableType { - override def getValue: String = value.toString - - def calculator(signal: String, bValue: String): String = signal match { - case "+" => val res = value + bValue.toLong; res.toString - case "-" => val res = value - bValue.toLong; res.toString - case "*" => val res = value * bValue.toLong; res.toString - case "/" => val res = value / bValue.toLong; res.toString - case _ => throw new LinkisCommonErrorException(20047, s"Int class is not supported to uss:$signal.") - } -} - -case class DoubleValue(value: Double) extends VariableType { - override def getValue: String = doubleOrLong(value).toString - - def calculator(signal: String, bValue: String): String = signal match { - case "+" => val res = value + bValue.toDouble; doubleOrLong(res).toString - case "-" => val res = value - bValue.toDouble; doubleOrLong(res).toString - case "*" => val res = value * bValue.toDouble; doubleOrLong(res).toString - case "/" => val res = value / bValue.toDouble; doubleOrLong(res).toString - case _ => throw new LinkisCommonErrorException(20047, s"Double class is not supported to uss:$signal.") - } - - private def doubleOrLong(d: Double): AnyVal = { - if (d.asInstanceOf[Long] == d) d.asInstanceOf[Long] else d - } - -} - -case class FloatType(value: Float) extends VariableType { - override def getValue: String = floatOrLong(value).toString - - def calculator(signal: String, bValue: String): String = signal match { - case "+" => val res = value + bValue.toFloat; floatOrLong(res).toString - case "-" => val res = value - bValue.toFloat; floatOrLong(res).toString - case "*" => val res = value * bValue.toFloat; floatOrLong(res).toString - case "/" => val res = value / bValue.toLong; floatOrLong(res).toString - case _ => throw new LinkisCommonErrorException(20048, s"Float class is not supported to use:$signal.") - } - - private def floatOrLong(f: Float): AnyVal = { - if (f.asInstanceOf[Long] == f) f.asInstanceOf[Long] else f - } - -} - -case class StringType(value: String) extends VariableType { - override def getValue: String = value.toString - - def calculator(signal: String, bValue: String): String = signal match { - case "+" => value + bValue - case _ => throw new LinkisCommonErrorException(20049, s"String class is not supported to uss:$signal.") - } -} - -import VariableUtils._ - -class CustomDateType(date: String, std: Boolean = true) { - - - - def -(days: Int): String = { - if (std) { - dateFormatStdLocal.get().format(DateUtils.addDays( dateFormatStdLocal.get().parse(date), -days)) - } else { - dateFormatLocal.get().format(DateUtils.addDays(dateFormatLocal.get().parse(date), -days)) - } - } - - def +(days: Int): String = { - if (std) { - dateFormatStdLocal.get().format(DateUtils.addDays( dateFormatStdLocal.get().parse(date), days)) - } else { - dateFormatLocal.get().format(DateUtils.addDays(dateFormatLocal.get().parse(date), days)) - } - } - - def getDate: Date = { - if (std) { - dateFormatStdLocal.get().parse(date) - } else { - dateFormatLocal.get().parse(date) - } - } - - def getStdDate: String = { - if (std) { - dateFormatStdLocal.get().format( dateFormatStdLocal.get().parse(date)) - } else { - dateFormatStdLocal.get().format(dateFormatLocal.get().parse(date)) - } - } - - override def toString: String = { - if (std) { - dateFormatStdLocal.get().format( dateFormatStdLocal.get().parse(date)) - } else { - dateFormatLocal.get().format(dateFormatLocal.get().parse(date)) - } - } -} - - -class CustomMonthType(date: String, std: Boolean = true, isEnd: Boolean = false) { - - def -(months: Int): String = { - if (std) { - VariableUtils.getMonth(std, isEnd, DateUtils.addMonths(dateFormatLocal.get().parse(date), -months)) - } else { - VariableUtils.getMonth(std, isEnd, DateUtils.addMonths(dateFormatLocal.get().parse(date), -months)) - } - } - - def +(months: Int): String = { - if (std) { - VariableUtils.getMonth(std, isEnd, DateUtils.addMonths(dateFormatLocal.get().parse(date), months)) - } else { - VariableUtils.getMonth(std, isEnd, DateUtils.addMonths(dateFormatLocal.get().parse(date), months)) - } - } - - override def toString: String = { - if (std) { - VariableUtils.getMonth(std, isEnd, dateFormatLocal.get().parse(date)) - } else { - val v = dateFormatLocal.get().parse(date) - VariableUtils.getMonth(std, isEnd, v) - } - } - -} - -class CustomMonType(date: String, std: Boolean = true, isEnd: Boolean = false) { - - def -(months: Int): String = { - if (std) { - VariableUtils.getMon(std, isEnd, DateUtils.addMonths(dateFormatMonLocal.get().parse(date), -months)) - } else { - VariableUtils.getMon(std, isEnd, DateUtils.addMonths(dateFormatMonLocal.get().parse(date), -months)) - } - } - - def +(months: Int): String = { - if (std) { - VariableUtils.getMon(std, isEnd, DateUtils.addMonths(dateFormatMonLocal.get().parse(date), months)) - } else { - VariableUtils.getMon(std, isEnd, DateUtils.addMonths(dateFormatMonLocal.get().parse(date), months)) - } - } - - override def toString: String = { - if (std) { - VariableUtils.getMon(std, isEnd, dateFormatMonLocal.get().parse(date)) - } else { - val v = dateFormatMonLocal.get().parse(date) - VariableUtils.getMon(std, isEnd, v) - } - } - -} - -/* - Given a Date, convert into Quarter - */ -class CustomQuarterType(date: String, std: Boolean = true, isEnd: Boolean = false) { - - - def getCurrentQuarter(date: String) = { - dateFormatLocal.get().parse(VariableUtils.getQuarter(false, false, dateFormatLocal.get().parse(date))) - } - - def -(quarters: Int): String = { - VariableUtils.getQuarter(std, isEnd, DateUtils.addMonths(getCurrentQuarter(date), -quarters * 3)) - } - - def +(quarters: Int): String = { - VariableUtils.getQuarter(std, isEnd, DateUtils.addMonths(getCurrentQuarter(date), quarters * 3)) - } - - override def toString: String = { - if (std) { - VariableUtils.getQuarter(std, isEnd, dateFormatLocal.get().parse(date)) - } else { - val v = dateFormatLocal.get().parse(date) - VariableUtils.getQuarter(std, isEnd, v) - } - } - -} - -/* - Given a Date, convert into HalfYear - */ -class CustomHalfYearType(date: String, std: Boolean = true, isEnd: Boolean = false) { - - - def getCurrentHalfYear(date: String) = { - dateFormatLocal.get().parse(VariableUtils.getHalfYear(false, false, dateFormatLocal.get().parse(date))) - } - - def -(halfYears: Int): String = { - VariableUtils.getHalfYear(std, isEnd, DateUtils.addMonths(getCurrentHalfYear(date), -halfYears * 6)) - } - - def +(halfYears: Int): String = { - VariableUtils.getHalfYear(std, isEnd, DateUtils.addMonths(getCurrentHalfYear(date), halfYears * 6)) - } - - override def toString: String = { - if (std) { - VariableUtils.getHalfYear(std, isEnd, dateFormatLocal.get().parse(date)) - } else { - val v = dateFormatLocal.get().parse(date) - VariableUtils.getHalfYear(std, isEnd, v) - } - } - -} - -/* - Given a Date convert into Year - */ -class CustomYearType(date: String, std: Boolean = true, isEnd: Boolean = false) { - - def -(years: Int): String = { - VariableUtils.getYear(std, isEnd, DateUtils.addYears(dateFormatLocal.get().parse(date), -years)) - } - - def +(years: Int): String = { - VariableUtils.getYear(std, isEnd, DateUtils.addYears(dateFormatLocal.get().parse(date), years)) - } - - override def toString: String = { - if (std) { - VariableUtils.getYear(std, isEnd, dateFormatLocal.get().parse(date)) - } else { - val v = dateFormatLocal.get().parse(date) - VariableUtils.getYear(std, isEnd, v) - } - } - -} diff --git a/dss-commons/dss-common/src/main/scala/org/apache/linkis/common/conf/DSSConfiguration.scala b/dss-commons/dss-common/src/main/scala/org/apache/linkis/common/conf/DSSConfiguration.scala index f1a788709c..afab964e49 100644 --- a/dss-commons/dss-common/src/main/scala/org/apache/linkis/common/conf/DSSConfiguration.scala +++ b/dss-commons/dss-common/src/main/scala/org/apache/linkis/common/conf/DSSConfiguration.scala @@ -1,9 +1,83 @@ package org.apache.linkis.common.conf +import org.apache.commons.io.IOUtils +import org.apache.linkis.common.utils.{Logging, Utils} + +import java.io.{File, FileInputStream, IOException, InputStream} +import java.util import java.util.Properties +import java.util.stream.Collectors +import scala.collection.JavaConverters._ + +object DSSConfiguration extends Logging { -object DSSConfiguration { + private val MAPPER_LOCATIONS = "wds.linkis.server.mybatis.mapperLocations" + private val TYPE_ALIASES_PACKAGE = "wds.linkis.server.mybatis.typeAliasesPackage" + private val BASE_PACKAGE = "wds.linkis.server.mybatis.BasePackage" + private val RESTFUL_SCAN_PACKAGES = "wds.linkis.server.restful.scan.packages" def getAllProperties: Properties = BDPConfiguration.properties + def addLegacyConfiguration(serverConfs: Array[String]): Unit = { + val mapperLocationList = new util.ArrayList[String](); + val typeAliasesPackageList = new util.ArrayList[String](); + val basePackageList = new util.ArrayList[String](); + val restfulScanPackagesList = new util.ArrayList[String](); + + serverConfs.foreach { serverConf => + val serverConfFileURL = getClass.getClassLoader.getResource(serverConf) + if (serverConfFileURL != null && new File(serverConfFileURL.getPath).exists) { + logger.info( + s"*********************** Notice: The DSS serverConf file is $serverConf ! ******************" + ) + val config = new Properties + initConfig(config, serverConfFileURL.getPath) + config.asScala.toMap.foreach { case (k, v) => + k match { + case MAPPER_LOCATIONS => mapperLocationList.add(v) + case TYPE_ALIASES_PACKAGE => typeAliasesPackageList.add(v) + case BASE_PACKAGE => basePackageList.add(v) + case RESTFUL_SCAN_PACKAGES => restfulScanPackagesList.add(v) + case _ => setConfig(k, v) + } + } + } else { + logger.warn( + s"**************** Notice: The DSS serverConf file $serverConf does not exist! *******************" + ) + } + } + setConfig(MAPPER_LOCATIONS, distinctStr(String.join(",", mapperLocationList))) + setConfig(TYPE_ALIASES_PACKAGE, distinctStr(String.join(",", typeAliasesPackageList))) + setConfig(BASE_PACKAGE, distinctStr(String.join(",", basePackageList))) + setConfig(RESTFUL_SCAN_PACKAGES, distinctStr(String.join(",", restfulScanPackagesList))) + } + + private def distinctStr(str: String): String = { + String.join(",", util.Arrays.stream(str.split(',')).collect(Collectors.toSet())) + } + + private def initConfig(config: Properties, filePath: String): Unit = { + var inputStream: InputStream = null + Utils.tryFinally { + Utils.tryCatch { + inputStream = new FileInputStream(filePath) + config.load(inputStream) + } { case e: IOException => + logger.error("Can't load " + filePath, e) + } + } { + IOUtils.closeQuietly(inputStream) + } + } + + def setConfig(key: String, value: String): Unit = { + logger.info(s"try to set a config, key:$key, value:$value") + BDPConfiguration.set(key, value) + } + + def setSpringApplicationName(name: String): Unit = { + BDPConfiguration.set("spring.spring.application.name", name) + } + } diff --git a/dss-commons/dss-contextservice/pom.xml b/dss-commons/dss-contextservice/pom.xml index 4d9c1d10f1..8b2ebf22e4 100644 --- a/dss-commons/dss-contextservice/pom.xml +++ b/dss-commons/dss-contextservice/pom.xml @@ -21,7 +21,8 @@ dss-commons com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 + ../pom.xml dss-contextservice diff --git a/dss-commons/dss-contextservice/src/main/java/com/webank/wedatasphere/dss/contextservice/service/impl/ContextServiceImpl.java b/dss-commons/dss-contextservice/src/main/java/com/webank/wedatasphere/dss/contextservice/service/impl/ContextServiceImpl.java index bd9a7d69fa..59471ed092 100644 --- a/dss-commons/dss-contextservice/src/main/java/com/webank/wedatasphere/dss/contextservice/service/impl/ContextServiceImpl.java +++ b/dss-commons/dss-contextservice/src/main/java/com/webank/wedatasphere/dss/contextservice/service/impl/ContextServiceImpl.java @@ -49,7 +49,7 @@ public class ContextServiceImpl implements ContextService { private static final Logger logger = LoggerFactory.getLogger(ContextServiceImpl.class); private static ContextClient contextClient = ContextClientFactory.getOrCreateContextClient(); - private static ContextService contextService = null; + private static volatile ContextService contextService = null; private ContextServiceImpl() {} @@ -105,13 +105,12 @@ public String checkAndCreateContextID(String jsonFlow, String flowVersion, Strin if (!DSSCommonConf.DSS_IO_ENV.getValue().equalsIgnoreCase(contextID.getEnv())) { updateContextId = true; } else if (StringUtils.isBlank(contextID.getProject()) - || StringUtils.isBlank(contextID.getFlow()) - || StringUtils.isBlank(contextID.getVersion())) { + || StringUtils.isBlank(contextID.getFlow())) { updateContextId = false; } else if ((null != contextID.getWorkSpace() && !contextID.getWorkSpace().equalsIgnoreCase(workspace)) || !contextID.getProject().equalsIgnoreCase(project) || !contextID.getFlow().equalsIgnoreCase(flow) - || !contextID.getVersion().equalsIgnoreCase(flowVersion)) { + || !flowVersion.equalsIgnoreCase(contextID.getVersion())) { updateContextId = true; } else { updateContextId = false; @@ -145,7 +144,7 @@ public void checkAndSaveContext(String jsonFlow,String parentFlowID) throws DSSE JsonObject flowObject = new Gson().fromJson(jsonFlow, JsonObject.class); if (!flowObject.has(CSCommonUtils.CONTEXT_ID_STR) || !flowObject.get(CSCommonUtils.CONTEXT_ID_STR).isJsonPrimitive()) { logger.error("Did not have invalid contextID, save context failed."); - return; + throw new DSSRuntimeException("does not have valid ContextID, save context failed(工作流格式错误,缺失有效的CS信息)"); } else { String contextIDStr = flowObject.get(CSCommonUtils.CONTEXT_ID_STR).getAsString(); // ①reset原有key 这里只清理 @@ -179,9 +178,9 @@ public void checkAndSaveContext(String jsonFlow,String parentFlowID) throws DSSE saveContextResource(contextIDStr, nodeRes, contextClient, CSCommonUtils.NODE_PREFIX, json.get(DSSCommonUtils.NODE_NAME_NAME).getAsString()); } - if (json.has(DSSCommonUtils.NODE_PROP_NAME)) { - JsonObject nodePropObj = json.get(DSSCommonUtils.NODE_PROP_NAME).getAsJsonObject(); - } +// if (json.has(DSSCommonUtils.NODE_PROP_NAME)) { +// JsonObject nodePropObj = json.get(DSSCommonUtils.NODE_PROP_NAME).getAsJsonObject(); +// } } } // 保存info信息 diff --git a/dss-commons/dss-sender-service/pom.xml b/dss-commons/dss-sender-service/pom.xml index 6be866d152..31d03b4548 100644 --- a/dss-commons/dss-sender-service/pom.xml +++ b/dss-commons/dss-sender-service/pom.xml @@ -21,7 +21,8 @@ dss-commons com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 + ../pom.xml 4.0.0 diff --git a/dss-commons/dss-sender-service/src/main/java/com/webank/wedatasphere/dss/sender/service/DSSSenderServiceFactory.java b/dss-commons/dss-sender-service/src/main/java/com/webank/wedatasphere/dss/sender/service/DSSSenderServiceFactory.java index d797ddd726..4a58f3aac9 100644 --- a/dss-commons/dss-sender-service/src/main/java/com/webank/wedatasphere/dss/sender/service/DSSSenderServiceFactory.java +++ b/dss-commons/dss-sender-service/src/main/java/com/webank/wedatasphere/dss/sender/service/DSSSenderServiceFactory.java @@ -17,7 +17,9 @@ package com.webank.wedatasphere.dss.sender.service; import com.webank.wedatasphere.dss.common.utils.ClassUtils; +import com.webank.wedatasphere.dss.sender.service.conf.DSSSenderServiceConf; import com.webank.wedatasphere.dss.sender.service.impl.DSSSenderServiceImpl; +import com.webank.wedatasphere.dss.sender.service.impl.DSSServerSenderServiceImpl; import org.slf4j.Logger; import org.slf4j.LoggerFactory; @@ -27,7 +29,11 @@ public class DSSSenderServiceFactory { private static final DSSSenderService service; static { - service = ClassUtils.getInstanceOrDefault(DSSSenderService.class, new DSSSenderServiceImpl()); + if (DSSSenderServiceConf.USE_DSS_SENDER.getValue()) { + service = ClassUtils.getInstanceOrDefault(DSSServerSenderServiceImpl.class, new DSSServerSenderServiceImpl()); + } else { + service = ClassUtils.getInstanceOrDefault(DSSSenderServiceImpl.class, new DSSSenderServiceImpl()); + } LOGGER.info("Use {} to instance a available DSSSenderService.", service.getClass().getName()); } diff --git a/dss-commons/dss-sender-service/src/main/java/com/webank/wedatasphere/dss/sender/service/conf/DSSSenderServiceConf.java b/dss-commons/dss-sender-service/src/main/java/com/webank/wedatasphere/dss/sender/service/conf/DSSSenderServiceConf.java index 5a551c5f72..ff33f617c0 100644 --- a/dss-commons/dss-sender-service/src/main/java/com/webank/wedatasphere/dss/sender/service/conf/DSSSenderServiceConf.java +++ b/dss-commons/dss-sender-service/src/main/java/com/webank/wedatasphere/dss/sender/service/conf/DSSSenderServiceConf.java @@ -28,4 +28,14 @@ public class DSSSenderServiceConf { public static final CommonVars PROJECT_SERVER_NAME = CommonVars.apply("wds.dss.project.sever.name", "dss-framework-project-server"); + public static final CommonVars DSS_SERVER_NAME = + CommonVars.apply("wds.dss.sever.name.dev", "dss-server-dev"); + + public static final CommonVars CURRENT_DSS_SERVER_NAME = + CommonVars.apply("spring.spring.application.name", "dss-server-dev"); + + //以服务合并后方式启动服务需要开启该参数,回退到老方式启动则需要关闭该参数 + public static final CommonVars USE_DSS_SENDER = + CommonVars.apply("wds.dss.server.use.dssSender", true); + } diff --git a/dss-commons/dss-sender-service/src/main/java/com/webank/wedatasphere/dss/sender/service/impl/DSSServerSenderServiceImpl.java b/dss-commons/dss-sender-service/src/main/java/com/webank/wedatasphere/dss/sender/service/impl/DSSServerSenderServiceImpl.java new file mode 100644 index 0000000000..4ebe11e5fe --- /dev/null +++ b/dss-commons/dss-sender-service/src/main/java/com/webank/wedatasphere/dss/sender/service/impl/DSSServerSenderServiceImpl.java @@ -0,0 +1,47 @@ +package com.webank.wedatasphere.dss.sender.service.impl; + +import com.webank.wedatasphere.dss.common.label.DSSLabel; +import com.webank.wedatasphere.dss.sender.service.DSSSenderService; +import com.webank.wedatasphere.dss.sender.service.conf.DSSSenderServiceConf; +import org.apache.linkis.rpc.Sender; + +import java.util.List; + +public class DSSServerSenderServiceImpl implements DSSSenderService { + private final Sender dssServerSender = Sender.getSender(DSSSenderServiceConf.DSS_SERVER_NAME.getValue()); + + @Override + public Sender getOrcSender() { + return dssServerSender; + } + + @Override + public Sender getOrcSender(List dssLabels) { + return dssServerSender; + } + + @Override + public Sender getScheduleOrcSender() { + return dssServerSender; + } + + @Override + public Sender getWorkflowSender(List dssLabels) { + return dssServerSender; + } + + @Override + public Sender getWorkflowSender() { + return dssServerSender; + } + + @Override + public Sender getSchedulerWorkflowSender() { + return dssServerSender; + } + + @Override + public Sender getProjectServerSender() { + return dssServerSender; + } +} diff --git a/dss-commons/pom.xml b/dss-commons/pom.xml index 1c4e33593b..f01ab0f5e9 100644 --- a/dss-commons/pom.xml +++ b/dss-commons/pom.xml @@ -21,7 +21,8 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 + ../pom.xml 4.0.0 diff --git a/dss-framework/dss-appconn-framework/pom.xml b/dss-framework/dss-appconn-framework/pom.xml index 9cea7ebf9c..2831390ac9 100644 --- a/dss-framework/dss-appconn-framework/pom.xml +++ b/dss-framework/dss-appconn-framework/pom.xml @@ -21,8 +21,8 @@ dss com.webank.wedatasphere.dss - 1.1.1 - + 1.1.2 + ../../pom.xml 4.0.0 @@ -36,7 +36,6 @@ test - org.apache.linkis linkis-mybatis @@ -55,6 +54,12 @@ ${dss.version} provided + + com.webank.wedatasphere.dss + dss-sender-service + ${dss.version} + provided + org.apache.linkis linkis-rpc diff --git a/dss-framework/dss-appconn-framework/src/main/java/com/webank/wedatasphere/dss/framework/appconn/conf/AppConnConf.java b/dss-framework/dss-appconn-framework/src/main/java/com/webank/wedatasphere/dss/framework/appconn/conf/AppConnConf.java index c0c1d17ef4..4aa3162433 100644 --- a/dss-framework/dss-appconn-framework/src/main/java/com/webank/wedatasphere/dss/framework/appconn/conf/AppConnConf.java +++ b/dss-framework/dss-appconn-framework/src/main/java/com/webank/wedatasphere/dss/framework/appconn/conf/AppConnConf.java @@ -9,12 +9,12 @@ public class AppConnConf { - public static final CommonVars IS_APPCONN_MANAGER = CommonVars.apply("wds.dss.appconn.framework.ismanager", true); - public static final CommonVars PROJECT_QUALITY_CHECKER_IGNORE_LIST = CommonVars.apply("wds.dss.appconn.checker.project.ignore.list", ""); public static final CommonVars DEVELOPMENT_QUALITY_CHECKER_IGNORE_LIST = CommonVars.apply("wds.dss.appconn.checker.development.ignore.list", ""); + public static final CommonVars APPCONN_UPLOAD_THREAD_NUM = CommonVars.apply("wds.dss.appconn.upload.thread.num", 2); + public static final List DISABLED_APP_CONNS = getDisabledAppConns(); private static List getDisabledAppConns() { diff --git a/dss-framework/dss-appconn-framework/src/main/java/com/webank/wedatasphere/dss/framework/appconn/dao/impl/appConnMapper.xml b/dss-framework/dss-appconn-framework/src/main/java/com/webank/wedatasphere/dss/framework/appconn/dao/impl/appConnMapper.xml index 2473ceae08..e8922867cc 100644 --- a/dss-framework/dss-appconn-framework/src/main/java/com/webank/wedatasphere/dss/framework/appconn/dao/impl/appConnMapper.xml +++ b/dss-framework/dss-appconn-framework/src/main/java/com/webank/wedatasphere/dss/framework/appconn/dao/impl/appConnMapper.xml @@ -29,18 +29,19 @@ + - `appconn_name`, `is_user_need_init`, `level`, `if_iframe`, `is_external`, + `appconn_name`, `is_user_need_init`, `level`, `if_iframe`, `is_external`,`is_micro_app`, `reference`, `class_name`, `resource` - `id`, `appconn_name`, `is_user_need_init`, `level`, `if_iframe`, `is_external`, + `id`, `appconn_name`, `is_user_need_init`, `level`, `if_iframe`, `is_external`,`is_micro_app`, `reference`, `class_name`, `resource` diff --git a/dss-framework/dss-appconn-framework/src/main/java/com/webank/wedatasphere/dss/framework/appconn/entity/AppConnBean.java b/dss-framework/dss-appconn-framework/src/main/java/com/webank/wedatasphere/dss/framework/appconn/entity/AppConnBean.java index 6fd4f6a48f..49ea8ba867 100644 --- a/dss-framework/dss-appconn-framework/src/main/java/com/webank/wedatasphere/dss/framework/appconn/entity/AppConnBean.java +++ b/dss-framework/dss-appconn-framework/src/main/java/com/webank/wedatasphere/dss/framework/appconn/entity/AppConnBean.java @@ -33,6 +33,7 @@ public class AppConnBean implements AppConnInfo, Serializable { private String level; private Boolean ifIframe; private Boolean isExternal; + private Boolean isMicroApp; // todo:目前通过这两个字段在classpath下加载类。 // 未来这两个字段的作用是在bml和默认应用的。 @@ -93,6 +94,14 @@ public void setExternal(Boolean external) { isExternal = external; } + public Boolean getIsMicroApp() { + return isMicroApp; + } + + public void setIsMicroApp(Boolean isMicroApp) { + this.isMicroApp = isMicroApp; + } + @Override public String getReference() { return reference; diff --git a/dss-framework/dss-appconn-framework/src/main/java/com/webank/wedatasphere/dss/framework/appconn/restful/AppConnManagerRestfulApi.java b/dss-framework/dss-appconn-framework/src/main/java/com/webank/wedatasphere/dss/framework/appconn/restful/AppConnManagerRestfulApi.java index 0247ced9b3..043c37985e 100644 --- a/dss-framework/dss-appconn-framework/src/main/java/com/webank/wedatasphere/dss/framework/appconn/restful/AppConnManagerRestfulApi.java +++ b/dss-framework/dss-appconn-framework/src/main/java/com/webank/wedatasphere/dss/framework/appconn/restful/AppConnManagerRestfulApi.java @@ -18,15 +18,19 @@ import com.webank.wedatasphere.dss.appconn.core.AppConn; import com.webank.wedatasphere.dss.appconn.manager.AppConnManager; +import com.webank.wedatasphere.dss.appconn.manager.conf.AppConnManagerCoreConf; import com.webank.wedatasphere.dss.appconn.manager.entity.AppConnInfo; import com.webank.wedatasphere.dss.appconn.manager.entity.AppInstanceInfo; import com.webank.wedatasphere.dss.appconn.manager.service.AppConnInfoService; +import com.webank.wedatasphere.dss.appconn.manager.utils.AppConnManagerUtils; +import com.webank.wedatasphere.dss.common.exception.DSSRuntimeException; import com.webank.wedatasphere.dss.common.utils.DSSExceptionUtils; import com.webank.wedatasphere.dss.framework.appconn.conf.AppConnConf; import com.webank.wedatasphere.dss.framework.appconn.service.AppConnQualityChecker; import com.webank.wedatasphere.dss.framework.appconn.service.AppConnResourceUploadService; -import org.apache.commons.lang.StringUtils; +import com.webank.wedatasphere.dss.sender.service.conf.DSSSenderServiceConf; import org.apache.commons.lang.exception.ExceptionUtils; +import org.apache.linkis.common.utils.Utils; import org.apache.linkis.server.Message; import org.slf4j.Logger; import org.slf4j.LoggerFactory; @@ -38,6 +42,11 @@ import javax.annotation.PostConstruct; import java.util.List; +import java.util.concurrent.CountDownLatch; +import java.util.concurrent.ExecutorService; +import java.util.concurrent.atomic.AtomicInteger; + +import static com.webank.wedatasphere.dss.framework.appconn.conf.AppConnConf.APPCONN_UPLOAD_THREAD_NUM; @RequestMapping(path = "/dss/framework/project/appconn", produces = {"application/json"}) @RestController @@ -51,9 +60,13 @@ public class AppConnManagerRestfulApi { @Autowired private List appConnQualityCheckers; + private ExecutorService uploadThreadPool = Utils.newFixedThreadPool(APPCONN_UPLOAD_THREAD_NUM.getValue(), "Upload-Appconn-Thread-", false); + + @PostConstruct - public void init() { - if (AppConnConf.IS_APPCONN_MANAGER.getValue()) { + public void init() throws InterruptedException { + //仅dss-server-dev的其中一个服务需要作为appconn-manager节点上传appconn包,其他服务都是client端 + if (AppConnManagerCoreConf.IS_APPCONN_MANAGER.getValue()) { LOGGER.info("First, try to load all AppConn..."); AppConnManager.getAppConnManager().listAppConns().forEach(appConn -> { LOGGER.info("Try to check the quality of AppConn {}.", appConn.getAppDesc().getAppName()); @@ -61,19 +74,34 @@ public void init() { }); LOGGER.info("All AppConn have loaded successfully."); LOGGER.info("Last, try to scan AppConn plugins and upload AppConn resources..."); - // reference不为空,说明是引用的其他appconn,不用上传appconn目录 - appConnInfoService.getAppConnInfos().stream().filter(l -> StringUtils.isBlank(l.getReference())) - .forEach(DSSExceptionUtils.handling(appConnInfo -> { - LOGGER.info("Try to scan AppConn {}.", appConnInfo.getAppConnName()); - appConnResourceUploadService.upload(appConnInfo.getAppConnName()); - })); + List uploadList = appConnInfoService.getAppConnInfos(); + CountDownLatch cdl = new CountDownLatch(uploadList.size()); + AtomicInteger failedCnt = new AtomicInteger(0); + uploadList.forEach(appConnInfo -> { + uploadThreadPool.submit(() -> { + LOGGER.info("Try to scan AppConn {}.", appConnInfo.getAppConnName()); + try { + appConnResourceUploadService.upload(appConnInfo.getAppConnName()); + } catch (Exception e) { + LOGGER.error("Error happened when uploading appconn:{}, error:", appConnInfo.getAppConnName(), e); + failedCnt.getAndIncrement(); + } + cdl.countDown(); + }); + }); + cdl.await(); + if (failedCnt.get() > 0) { + throw new DSSRuntimeException("Error happened when uploading appconn, service startup terminated."); + } LOGGER.info("All AppConn plugins has scanned."); + uploadThreadPool.shutdown(); } else { LOGGER.info("Not appConn manager, will not scan plugins."); + AppConnManagerUtils.autoLoadAppConnManager(); } } - @RequestMapping(path ="listAppConnInfos", method = RequestMethod.GET) + @RequestMapping(path = "listAppConnInfos", method = RequestMethod.GET) public Message listAppConnInfos() { List appConnInfos = appConnInfoService.getAppConnInfos(); Message message = Message.ok("Get AppConnInfo list succeed."); @@ -81,23 +109,25 @@ public Message listAppConnInfos() { return message; } - @RequestMapping(path ="{appConnName}/get", method = RequestMethod.GET) + @RequestMapping(path = "{appConnName}/get", method = RequestMethod.GET) public Message get(@PathVariable("appConnName") String appConnName) { + LOGGER.info("try to get appconn info:{}.", appConnName); AppConnInfo appConnInfo = appConnInfoService.getAppConnInfo(appConnName); Message message = Message.ok("Get AppConnInfo succeed."); message.data("appConnInfo", appConnInfo); return message; } - @RequestMapping(path ="{appConnName}/getAppInstances", method = RequestMethod.GET) + @RequestMapping(path = "{appConnName}/getAppInstances", method = RequestMethod.GET) public Message getAppInstancesByAppConnInfo(@PathVariable("appConnName") String appConnName) { + LOGGER.debug("try to get instances for appconn: {}.", appConnName); List appInstanceInfos = appConnInfoService.getAppInstancesByAppConnName(appConnName); Message message = Message.ok("Get AppInstance list succeed."); message.data("appInstanceInfos", appInstanceInfos); return message; } - @RequestMapping(path ="{appConnName}/load", method = RequestMethod.GET) + @RequestMapping(path = "{appConnName}/load", method = RequestMethod.GET) public Message load(@PathVariable("appConnName") String appConnName) { LOGGER.info("Try to reload AppConn {}.", appConnName); try { diff --git a/dss-framework/dss-appconn-framework/src/main/java/com/webank/wedatasphere/dss/framework/appconn/service/impl/AppConnInfoServiceImpl.java b/dss-framework/dss-appconn-framework/src/main/java/com/webank/wedatasphere/dss/framework/appconn/service/impl/AppConnInfoServiceImpl.java index 4a40e0b3c8..cdfa9d303e 100644 --- a/dss-framework/dss-appconn-framework/src/main/java/com/webank/wedatasphere/dss/framework/appconn/service/impl/AppConnInfoServiceImpl.java +++ b/dss-framework/dss-appconn-framework/src/main/java/com/webank/wedatasphere/dss/framework/appconn/service/impl/AppConnInfoServiceImpl.java @@ -29,6 +29,7 @@ import org.springframework.stereotype.Component; import java.util.List; +import java.util.stream.Collectors; @Component public class AppConnInfoServiceImpl implements AppConnInfoService { @@ -39,9 +40,10 @@ public class AppConnInfoServiceImpl implements AppConnInfoService { @Override public List getAppConnInfos() { - List appConnBeans = appConnMapper.getAllAppConnBeans(); - appConnBeans.stream().filter(appConnBean -> !AppConnConf.DISABLED_APP_CONNS.contains(appConnBean.getAppConnName())) - .forEach(appConnBean -> { + List appConnBeans = appConnMapper.getAllAppConnBeans().stream() + .filter(appConnBean -> !AppConnConf.DISABLED_APP_CONNS.contains(appConnBean.getAppConnName())) + .collect(Collectors.toList()); + appConnBeans.forEach(appConnBean -> { String resource = appConnBean.getResource(); if(StringUtils.isNotBlank(resource)) { appConnBean.setAppConnResource(AppConnServiceUtils.stringToResource(resource).getResource()); diff --git a/dss-framework/dss-appconn-framework/src/main/java/com/webank/wedatasphere/dss/framework/appconn/service/impl/AppConnResourceServiceImpl.java b/dss-framework/dss-appconn-framework/src/main/java/com/webank/wedatasphere/dss/framework/appconn/service/impl/AppConnResourceServiceImpl.java index b422a79de5..4fa36bca77 100644 --- a/dss-framework/dss-appconn-framework/src/main/java/com/webank/wedatasphere/dss/framework/appconn/service/impl/AppConnResourceServiceImpl.java +++ b/dss-framework/dss-appconn-framework/src/main/java/com/webank/wedatasphere/dss/framework/appconn/service/impl/AppConnResourceServiceImpl.java @@ -71,17 +71,18 @@ public String getAppConnHome(AppConnInfo appConnInfo) { @Override public void upload(String appConnName) throws DSSErrorException { File appConnPath = new File(AppConnUtils.getAppConnHomePath(), appConnName); + AppConnBean appConnBean = appConnMapper.getAppConnBeanByName(appConnName); if (!appConnPath.exists()) { - throw new AppConnNotExistsErrorException(20350, "AppConn home path " + appConnPath.getPath() + " not exists."); + //没有reference,必须有appconn目录 + if (StringUtils.isBlank(appConnBean.getReference())) { + throw new AppConnNotExistsErrorException(20350, "AppConn home path " + appConnPath.getPath() + " not exists."); + } else { + LOGGER.info("Appconn {} references other appConns and has no directory, so no upload is required", appConnName); + return; + } } else if (!appConnPath.isDirectory()) { throw new AppConnNotExistsErrorException(20350, "AppConn home path " + appConnPath.getPath() + " is not a directory."); } - File zipFile = new File(appConnPath.getPath() + ".zip"); - if (zipFile.exists() && !zipFile.delete()) { - throw new AppConnNotExistsErrorException(20001, "No permission to delete old zip file " + zipFile); - } - ZipHelper.zip(appConnPath.getPath(), false); - AppConnBean appConnBean = appConnMapper.getAppConnBeanByName(appConnName); AppConnResource appConnResource; File indexFile = null; if (appConnBean == null) { @@ -91,7 +92,7 @@ public void upload(String appConnName) throws DSSErrorException { appConnResource = AppConnServiceUtils.stringToResource(appConnBean.getResource()); indexFile = AppConnIndexFileUtils.getIndexFile(appConnPath); if (appConnPath.lastModified() == appConnResource.getLastModifiedTime() - && zipFile.length() == appConnResource.getSize() && AppConnIndexFileUtils.isLatestIndex(appConnPath, appConnResource.getResource())) { + && AppConnIndexFileUtils.isLatestIndex(appConnPath, appConnResource.getResource())) { LOGGER.info("No necessary to update the AppConn {}, since it's packages has no changes in path {}.", appConnName, appConnPath.getPath()); return; } @@ -99,17 +100,22 @@ public void upload(String appConnName) throws DSSErrorException { // If resource is not exists, this is the first time to upload this AppConn. appConnResource = new AppConnResource(); } + File zipFile = new File(appConnPath.getPath() + ".zip"); + if (zipFile.exists() && !zipFile.delete()) { + throw new AppConnNotExistsErrorException(20001, "No permission to delete old zip file " + zipFile); + } + ZipHelper.zip(appConnPath.getPath(), false); // At first, upload appConn file to BML Resource resource = new Resource(); InputStream inputStream = null; if (appConnResource.getResource() != null) { try { inputStream = new FileInputStream(zipFile.getPath()); - BmlUpdateResponse response = bmlClient.updateResource(Utils.getJvmUser(), appConnResource.getResource().getResourceId(), zipFile.getPath(),inputStream); + BmlUpdateResponse response = bmlClient.updateResource(Utils.getJvmUser(), appConnResource.getResource().getResourceId(), zipFile.getPath(), inputStream); resource.setResourceId(appConnResource.getResource().getResourceId()); resource.setVersion(response.version()); } catch (FileNotFoundException e) { - throw new AppConnNotExistsErrorException(20351, "AppConn update to bml failed"+e.getMessage()); + throw new AppConnNotExistsErrorException(20351, "AppConn update to bml failed" + e.getMessage()); } finally { IOUtils.closeQuietly(inputStream); } @@ -122,7 +128,7 @@ public void upload(String appConnName) throws DSSErrorException { resource.setResourceId(response.resourceId()); resource.setVersion(response.version()); } catch (FileNotFoundException e) { - throw new AppConnNotExistsErrorException(20352, "AppConn update to bml failed"+e.getMessage()); + throw new AppConnNotExistsErrorException(20352, "AppConn update to bml failed" + e.getMessage()); } finally { IOUtils.closeQuietly(inputStream); } @@ -130,28 +136,27 @@ public void upload(String appConnName) throws DSSErrorException { DSSCommonUtils.COMMON_GSON.toJson(resource)); } resource.setFileName(zipFile.getName()); + // update index file. + if (indexFile != null && !indexFile.delete()) { + throw new AppConnNotExistsErrorException(20350, "Delete index file " + indexFile.getName() + " failed, please ensure the permission is all right."); + } + indexFile = new File(appConnPath, AppConnIndexFileUtils.getIndexFileName(resource)); + try { + indexFile.createNewFile(); + } catch (IOException e) { + throw new AppConnNotExistsErrorException(20350, "create index file " + indexFile.getName() + " failed, please ensure the permission is all right.", e); + } // Then, insert into db. appConnResource.setLastModifiedTime(appConnPath.lastModified()); appConnResource.setSize(zipFile.length()); appConnResource.setResource(resource); String resourceStr = AppConnServiceUtils.resourceToString(appConnResource); - AppConnBean appConnBeanReLoad = new AppConnBean(); appConnBeanReLoad.setId(appConnBean.getId()); appConnBeanReLoad.setResource(resourceStr); appConnBeanReLoad.setAppConnName(appConnName); appConnBeanReLoad.setClassName(appConnBean.getClassName()); appConnMapper.updateResourceByName(appConnBeanReLoad); - // update index file. - if (indexFile != null && !indexFile.delete()) { - throw new AppConnNotExistsErrorException(20350, "Delete index file " + indexFile.getName() + " failed, please ensure the permission is all right."); - } - indexFile = new File(appConnPath, AppConnIndexFileUtils.getIndexFileName(resource)); - try { - indexFile.createNewFile(); - } catch (IOException e) { - throw new AppConnNotExistsErrorException(20350, "create index file " + indexFile.getName() + " failed, please ensure the permission is all right.", e); - } LOGGER.info("AppConn {} has updated resource to {}.", appConnName, resourceStr); } diff --git a/dss-framework/dss-framework-admin-service/pom.xml b/dss-framework/dss-framework-admin-service/pom.xml index 7bf00155cd..ddf9ce3a54 100644 --- a/dss-framework/dss-framework-admin-service/pom.xml +++ b/dss-framework/dss-framework-admin-service/pom.xml @@ -4,8 +4,8 @@ dss com.webank.wedatasphere.dss - 1.1.1 - + 1.1.2 + ../../pom.xml dss-framework-admin-service @@ -27,7 +27,7 @@ com.webank.wedatasphere.dss - dss-common + dss-framework-proxy-user-service ${dss.version} provided diff --git a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/common/exception/AdminException.java b/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/common/exception/AdminException.java deleted file mode 100644 index 6eb4da3dbc..0000000000 --- a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/common/exception/AdminException.java +++ /dev/null @@ -1,82 +0,0 @@ -package com.webank.wedatasphere.dss.framework.admin.common.exception; - - -import com.webank.wedatasphere.dss.framework.admin.common.domain.ResponseEnum; -//import lombok.Data; -//import lombok.NoArgsConstructor; - -//@Data -//@NoArgsConstructor -public class AdminException extends RuntimeException { - - public Integer getCode() { - return code; - } - - public void setCode(Integer code) { - this.code = code; - } - - @Override - public String getMessage() { - return message; - } - - public void setMessage(String message) { - this.message = message; - } - - /** - * 错误码 - */ - private Integer code; - /** - * 错误信息 - */ - private String message; - - /** - * @param message 错误消息 - */ - public AdminException(String message) { - this.message = message; - } - - /** - * @param message 错误消息 - * @param code 错误码 - */ - public AdminException(String message, Integer code) { - this.message = message; - this.code = code; - } - - /** - * @param message 错误消息 - * @param code 错误码 - * @param cause 原始异常对象 - */ - public AdminException(String message, Integer code, Throwable cause) { - super(cause); - this.message = message; - this.code = code; - } - - /** - * @param resultCodeEnum 接收枚举类型 - */ - public AdminException(ResponseEnum resultCodeEnum) { - this.message = resultCodeEnum.getMessage(); - this.code = resultCodeEnum.getStatus(); - } - - /** - * @param resultCodeEnum 接收枚举类型 - * @param cause 原始异常对象 - */ - public AdminException(ResponseEnum resultCodeEnum, Throwable cause) { - super(cause); - this.message = resultCodeEnum.getMessage(); - this.code = resultCodeEnum.getStatus(); - } -} diff --git a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/common/utils/SqlUtil.java b/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/common/utils/SqlUtil.java index 064ccfa898..692acd6199 100644 --- a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/common/utils/SqlUtil.java +++ b/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/common/utils/SqlUtil.java @@ -1,6 +1,6 @@ package com.webank.wedatasphere.dss.framework.admin.common.utils; -import com.webank.wedatasphere.dss.framework.admin.common.exception.AdminException; +import com.webank.wedatasphere.dss.framework.admin.exception.DSSAdminWarnException; public class SqlUtil { @@ -9,7 +9,7 @@ public static String escapeOrderBySql(String value) { if (StringUtils.isNotEmpty(value) && !isValidOrderBySql(value)) { - throw new AdminException("参数不符合规范,不能进行查询"); + throw new DSSAdminWarnException("参数不符合规范,不能进行查询"); } return value; } diff --git a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/conf/AdminConf.java b/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/conf/AdminConf.java index 197b69465d..dba77195b3 100644 --- a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/conf/AdminConf.java +++ b/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/conf/AdminConf.java @@ -18,6 +18,7 @@ package com.webank.wedatasphere.dss.framework.admin.conf; +import com.webank.wedatasphere.dss.common.conf.DSSCommonConf; import org.apache.linkis.common.conf.CommonVars; public interface AdminConf { @@ -26,12 +27,7 @@ public interface AdminConf { CommonVars LDAP_ADMIN_PASS = CommonVars.apply("wds.dss.ldap.admin.password", ""); CommonVars LDAP_URL = CommonVars.apply("wds.dss.ldap.url", ""); CommonVars LDAP_BASE_DN = CommonVars.apply("wds.dss.ldap.base.dn", ""); - CommonVars EXCHANGE_URL = CommonVars.apply("wds.dss.exchange.url", ""); - CommonVars EXCHANGE_ADMIN_COOKIE = CommonVars.apply("wds.dss.exchange.cookie", ""); - CommonVars DS_TRUST_TOKEN = CommonVars.apply("wds.dss.trust.token", ""); - CommonVars DS_PROXY_SELF_ENABLE = CommonVars.apply("wds.dss.proxy.self.enable", true); - CommonVars DSS_PROXY_ADMIN_NAME = CommonVars.apply("wds.dss.proxy.admin.name", "adminUser"); - String[] SUPER_ADMIN_LIST = CommonVars.apply("wds.dss.super.admin", "").getValue().split(","); + String[] SUPER_ADMIN_LIST = DSSCommonConf.SUPER_ADMIN_LIST; } diff --git a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/exception/DSSAdminErrorException.java b/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/exception/DSSAdminErrorException.java deleted file mode 100644 index 833a45e26e..0000000000 --- a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/exception/DSSAdminErrorException.java +++ /dev/null @@ -1,14 +0,0 @@ -package com.webank.wedatasphere.dss.framework.admin.exception; - -import com.webank.wedatasphere.dss.common.exception.DSSErrorException; - - -public class DSSAdminErrorException extends DSSErrorException { - public DSSAdminErrorException(int errCode, String desc) { - super(errCode, desc); - } - - public DSSAdminErrorException(int errCode, String desc, String ip, int port, String serviceKind) { - super(errCode, desc, ip, port, serviceKind); - } -} diff --git a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/exception/DSSAdminWarnException.java b/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/exception/DSSAdminWarnException.java new file mode 100644 index 0000000000..8b23157dd1 --- /dev/null +++ b/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/exception/DSSAdminWarnException.java @@ -0,0 +1,15 @@ +package com.webank.wedatasphere.dss.framework.admin.exception; + +import com.webank.wedatasphere.dss.common.exception.DSSRuntimeException; + + +public class DSSAdminWarnException extends DSSRuntimeException { + + public DSSAdminWarnException(int errCode, String desc) { + super(errCode, desc); + } + + public DSSAdminWarnException(String desc) { + this(50000, desc); + } +} diff --git a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/pojo/entity/DssExchangeTask.java b/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/pojo/entity/DssExchangeTask.java deleted file mode 100644 index 791c016131..0000000000 --- a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/pojo/entity/DssExchangeTask.java +++ /dev/null @@ -1,64 +0,0 @@ -package com.webank.wedatasphere.dss.framework.admin.pojo.entity; - -/** - * @Auther: Han Tang - * @Date: 2022/1/18-01-18-15:58 - */ -public class DssExchangeTask { - private int id; - private String jobName; - private String jobCorn; - private String jobDesc; - private String createTime; - private String jobStatus; - - public int getId() { - return id; - } - - public void setId(int id) { - this.id = id; - } - - public String getJobName() { - return jobName; - } - - public void setJobName(String jobName) { - this.jobName = jobName; - } - - public String getJobCorn() { - return jobCorn; - } - - public void setJobCorn(String jobCorn) { - this.jobCorn = jobCorn; - } - - public String getJobDesc() { - return jobDesc; - } - - public void setJobDesc(String jobDesc) { - this.jobDesc = jobDesc; - } - - public String getCreateTime() { - return createTime; - } - - public void setCreateTime(String createTime) { - this.createTime = createTime; - } - - public String getJobStatus() { - return jobStatus; - } - - public void setJobStatus(String jobStatus) { - this.jobStatus = jobStatus; - } - - -} diff --git a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/pojo/entity/DssExchangeTaskRes.java b/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/pojo/entity/DssExchangeTaskRes.java deleted file mode 100644 index 781a63bdc1..0000000000 --- a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/pojo/entity/DssExchangeTaskRes.java +++ /dev/null @@ -1,57 +0,0 @@ -package com.webank.wedatasphere.dss.framework.admin.pojo.entity; - -import java.util.List; - -/** - * @Auther: Han Tang - * @Date: 2022/1/18-01-18-18:06 - */ -public class DssExchangeTaskRes { - - private int page; - private int totalItems; - private int totalPages; - private int pageSize; - private List dssExchangeTaskList; - public int getPageSize() { - return pageSize; - } - - public void setPageSize(int pageSize) { - this.pageSize = pageSize; - } - - public int getPage() { - return page; - } - - public void setPage(int page) { - this.page = page; - } - - public int getTotalItems() { - return totalItems; - } - - public void setTotalItems(int totalItems) { - this.totalItems = totalItems; - } - - public int getTotalPages() { - return totalPages; - } - - public void setTotalPages(int totalPages) { - this.totalPages = totalPages; - } - - public List getDssExchangeTaskList() { - return dssExchangeTaskList; - } - - public void setDssExchangeTaskList(List dssExchangeTaskList) { - this.dssExchangeTaskList = dssExchangeTaskList; - } - - -} diff --git a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/pojo/entity/DssExchangisProject.java b/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/pojo/entity/DssExchangisProject.java deleted file mode 100644 index 84929b925c..0000000000 --- a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/pojo/entity/DssExchangisProject.java +++ /dev/null @@ -1,93 +0,0 @@ -package com.webank.wedatasphere.dss.framework.admin.pojo.entity; - -import java.util.List; - -/** - * @Auther: Han Tang - * @Date: 2022/1/18-01-18-14:05 - */ -public class DssExchangisProject { - private int id; - private String projectName; - private int parentId; - private String createUser; - private String createTime; - private String modifyUser; - private String modifyTime; - private int level; - private List children; - - public int getId() { - return id; - } - - public void setId(int id) { - this.id = id; - } - - public String getProjectName() { - return projectName; - } - - public void setProjectName(String projectName) { - this.projectName = projectName; - } - - public int getParentId() { - return parentId; - } - - public void setParentId(int parentId) { - this.parentId = parentId; - } - - public String getCreateUser() { - return createUser; - } - - public void setCreateUser(String createUser) { - this.createUser = createUser; - } - - public String getCreateTime() { - return createTime; - } - - public void setCreateTime(String createTime) { - this.createTime = createTime; - } - - public String getModifyUser() { - return modifyUser; - } - - public void setModifyUser(String modifyUser) { - this.modifyUser = modifyUser; - } - - public String getModifyTime() { - return modifyTime; - } - - public void setModifyTime(String modifyTime) { - this.modifyTime = modifyTime; - } - - public int getLevel() { - return level; - } - - public void setLevel(int level) { - this.level = level; - } - - public List getChildren() { - return children; - } - - public void setChildren(List children) { - this.children = children; - } - - -} diff --git a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/pojo/entity/DssProxyUser.java b/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/pojo/entity/DssProxyUser.java deleted file mode 100644 index d531af351b..0000000000 --- a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/pojo/entity/DssProxyUser.java +++ /dev/null @@ -1,45 +0,0 @@ -package com.webank.wedatasphere.dss.framework.admin.pojo.entity; - -import com.baomidou.mybatisplus.annotation.IdType; -import com.baomidou.mybatisplus.annotation.TableId; -import com.baomidou.mybatisplus.annotation.TableName; -import com.webank.wedatasphere.dss.framework.admin.common.domain.BaseEntity; - - -@TableName(value = "dss_proxy_user") -public class DssProxyUser extends BaseEntity { - private static final long serialVersionUID = 1L; - - @TableId(type = IdType.AUTO) - private Long id; - private String userName; - /**代理用户名*/ - private String proxyUserName; - - public Long getId() { - return id; - } - - public void setId(Long id) { - this.id = id; - } - - public String getUserName() { - return userName; - } - - public void setUserName(String userName) { - this.userName = userName; - } - - public String getProxyUserName() { - return proxyUserName; - } - - public void setProxyUserName(String proxyUserName) { - this.proxyUserName = proxyUserName; - } - - - -} diff --git a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/restful/BaseController.java b/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/restful/BaseController.java index be53326e45..258cf79b28 100644 --- a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/restful/BaseController.java +++ b/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/restful/BaseController.java @@ -12,7 +12,6 @@ import java.util.HashMap; import java.util.List; -import java.util.Map; public class BaseController { diff --git a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/restful/DSSFeatureAuthRestfulApi.java b/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/restful/DSSFeatureAuthRestfulApi.java new file mode 100644 index 0000000000..a794a2f937 --- /dev/null +++ b/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/restful/DSSFeatureAuthRestfulApi.java @@ -0,0 +1,30 @@ +package com.webank.wedatasphere.dss.framework.admin.restful; + +import com.webank.wedatasphere.dss.common.utils.GlobalLimitsUtils; +import org.apache.linkis.server.Message; +import org.springframework.web.bind.annotation.PathVariable; +import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RequestMethod; +import org.springframework.web.bind.annotation.RestController; + +/** + * @author enjoyyin + * @date 2022-03-29 + * @since 0.5.0 + */ +@RequestMapping(path = "/dss/framework/admin", produces = {"application/json"}) +@RestController +public class DSSFeatureAuthRestfulApi { + + @RequestMapping(value = "/globalLimits",method = RequestMethod.GET) + public Message globalLimits() { + return Message.ok().data("globalLimits", GlobalLimitsUtils.getAllGlobalLimits()); + } + + @RequestMapping(value = "/globalLimits/{globalLimitName}",method = RequestMethod.GET) + public Message globalLimit(@PathVariable("globalLimitName") String globalLimitName) { + return Message.ok().data("globalLimitName", globalLimitName) + .data("content", GlobalLimitsUtils.getGlobalLimitMap(globalLimitName)); + } + +} diff --git a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/restful/DssAuditController.java b/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/restful/DssAuditController.java deleted file mode 100644 index 8610242857..0000000000 --- a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/restful/DssAuditController.java +++ /dev/null @@ -1,61 +0,0 @@ -package com.webank.wedatasphere.dss.framework.admin.restful; -import com.baomidou.mybatisplus.extension.plugins.pagination.Page; -import com.github.pagehelper.PageHelper; -import com.github.pagehelper.PageInfo; -import com.webank.wedatasphere.dss.framework.admin.pojo.entity.DssScriptDownloadAudit; -import com.webank.wedatasphere.dss.framework.admin.service.DssScriptDownloadService; -import lombok.extern.slf4j.Slf4j; -import org.apache.linkis.server.security.SecurityFilter; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.validation.annotation.Validated; -import org.springframework.web.bind.annotation.*; -import org.apache.linkis.server.Message; - -import javax.servlet.http.HttpServletRequest; -import javax.validation.Valid; -import java.util.List; - -/** - * @Auther: Han Tang - * @Date: 2022/1/11-01-11-14:20 - */ - - -@RestController -@RequestMapping(path = "/dss/framework/admin/audit", produces = {"application/json"}) -@Slf4j -public class DssAuditController { - - private static final Logger LOGGER = LoggerFactory.getLogger(DssAuditController.class); - @Autowired - DssScriptDownloadService dssScriptDownloadService; - - @RequestMapping(path = "script/download/save", method = RequestMethod.POST) - public Message saveScriptDownload(@RequestBody @Valid DssScriptDownloadAudit dssScriptDownloadAudit, HttpServletRequest request) { - String userName = SecurityFilter.getLoginUsername(request); -// String userName = "demo"; - dssScriptDownloadAudit.setCreator(userName); - String sql = dssScriptDownloadAudit.getSql(); - if(sql.length()>2000){ - sql = sql.substring(0,2000); - dssScriptDownloadAudit.setSql(sql); - } - dssScriptDownloadService.save(dssScriptDownloadAudit); - Message message = Message.ok(); - return message; - } - - - @RequestMapping(path = "script/download/query", method = RequestMethod.GET) - public Message getScriptDownload(@RequestParam(required = false) String userName,@RequestParam(required = false) String startTime,@RequestParam(required = false) String endTime,@RequestParam Integer pn) { - PageHelper.startPage(pn, 10, true); - List userPage = dssScriptDownloadService.getDownloadAuditList(userName,startTime,endTime); - PageInfo pageInfo = new PageInfo<>(userPage); - Message message = Message.ok().data("data", pageInfo); - return message; - } -} - - diff --git a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/restful/DssFrameworkAdminDeptController.java b/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/restful/DssFrameworkAdminDeptController.java index 9c03342391..c2c20270f9 100644 --- a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/restful/DssFrameworkAdminDeptController.java +++ b/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/restful/DssFrameworkAdminDeptController.java @@ -6,7 +6,8 @@ import com.webank.wedatasphere.dss.framework.admin.common.utils.StringUtils; import com.webank.wedatasphere.dss.framework.admin.pojo.entity.DssAdminDept; import com.webank.wedatasphere.dss.framework.admin.service.DssAdminDeptService; -import org.springframework.stereotype.Component; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.*; @@ -16,13 +17,15 @@ @RequestMapping(path = "/dss/framework/admin/dept", produces = {"application/json"}) @RestController public class DssFrameworkAdminDeptController { + + private static final Logger LOGGER = LoggerFactory.getLogger(DssFrameworkAdminDeptController.class); + @Resource private DssAdminDeptService dssAdminDeptService; - @RequestMapping(path = "list", method = RequestMethod.GET) public Message listAll(@RequestParam(value = "parentId", required = false) Long parentId, @RequestParam(value = "deptName", required = false) String deptName) { - + LOGGER.info("begin to get dept list...parentId:{}, deptName:{}",parentId, deptName); DssAdminDept dept = new DssAdminDept(); dept.setParentId(parentId); dept.setDeptName(deptName); @@ -39,9 +42,8 @@ public Message add(@RequestBody DssAdminDept dssAdminDept) { } else if (dssAdminDeptService.checkDeptFinalStage(dssAdminDept.getParentId())) { return Message.error().message("新增部门'" + dssAdminDept.getDeptName() + "'失败,该部门是末级部门,不能新增下级部门"); } - + LOGGER.info("Add new dept {}", dssAdminDept.getDeptName()); int saveResult = dssAdminDeptService.insertDept(dssAdminDept); - System.out.println(saveResult); if (saveResult >= 1) { return Message.ok().message("保存成功"); } else { @@ -77,7 +79,7 @@ public Message edit(@Validated @RequestBody DssAdminDept dept) { && dssAdminDeptService.selectNormalChildrenDeptById(dept.getId()) > 0) { return Message.error().message("该部门包含未停用的子部门!"); } - + LOGGER.info("Modify dept name to {}",dept.getDeptName()); return Message.ok().data("修改成功", dssAdminDeptService.updateDept(dept)); } @@ -93,6 +95,7 @@ public Message remove(@PathVariable("deptId") Long deptId) { if (dssAdminDeptService.checkDeptExistUser(deptId)) { return Message.error().message("部门存在用户,不允许删除"); } + LOGGER.info("Delete dept {}", deptId); return Message.ok().data("删除成功", dssAdminDeptService.deleteDeptById(deptId)); } diff --git a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/restful/DssFrameworkAdminUserController.java b/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/restful/DssFrameworkAdminUserController.java index 2690e11d45..7a155a2955 100644 --- a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/restful/DssFrameworkAdminUserController.java +++ b/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/restful/DssFrameworkAdminUserController.java @@ -17,6 +17,8 @@ import com.webank.wedatasphere.dss.standard.sso.utils.SSOHelper; import org.apache.commons.codec.digest.DigestUtils; import org.apache.linkis.server.security.SecurityFilter; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.*; @@ -30,6 +32,9 @@ @RequestMapping(path = "/dss/framework/admin/user", produces = {"application/json"}) @RestController public class DssFrameworkAdminUserController extends BaseController { + + private static final Logger LOGGER = LoggerFactory.getLogger(DssFrameworkAdminUserController.class); + @Resource private DssAdminUserService dssAdminUserService; @Autowired @@ -38,7 +43,6 @@ public class DssFrameworkAdminUserController extends BaseController { DssUserMapper dssUserMapper; @RequestMapping(path = "list", method = RequestMethod.GET) -// public TableDataInfo list(DssAdminUser user) { public TableDataInfo list(@RequestParam(value = "userName", required = false) String userName, @RequestParam(value = "deptId", required = false) Long deptId, @RequestParam(value = "phonenumber", required = false) String phonenumber, @@ -54,6 +58,7 @@ public TableDataInfo list(@RequestParam(value = "userName", required = false) St user.setParams(params); startPage(); List userList = dssAdminUserService.selectUserList(user); + LOGGER.info("try to get DssAdminUser list, userList:{}", userList); return getDataTable(userList); } @@ -83,6 +88,7 @@ public Message add(@Validated @RequestBody DssAdminUser user, HttpServletRequest user.setCreateBy(SecurityFilter.getLoginUsername(req)); int rows = dssAdminUserService.insertUser(user, getWorkspace(req)); String userName = user.getUserName(); + LOGGER.info("Add new user {}", userName); ldapService.addUser(AdminConf.LDAP_ADMIN_NAME.getValue(), AdminConf.LDAP_ADMIN_PASS.getValue(), AdminConf.LDAP_URL.getValue(), AdminConf.LDAP_BASE_DN.getValue(), userName, pwd); return Message.ok().data("rows", rows).message("新增成功"); } catch (Exception exception) { @@ -94,6 +100,7 @@ public Message add(@Validated @RequestBody DssAdminUser user, HttpServletRequest private Workspace getWorkspace(HttpServletRequest req) { Workspace workspace = new Workspace(); try { + LOGGER.info("Put gateway url and cookies into workspace."); SSOHelper.addWorkspaceInfo(req, workspace); } catch (AppStandardWarnException ignored) {} // ignore it. return workspace; @@ -121,6 +128,7 @@ public Message edit(@Validated @RequestBody DssAdminUser user, HttpServletReques && UserConstants.NOT_UNIQUE.equals(dssAdminUserService.checkEmailUnique(user))) { return Message.error().message("修改用户'" + user.getUserName() + "'失败,邮箱账号已存在"); } + LOGGER.info("Modify user {} info", user.getUserName()); return Message.ok().data("修改用户成功。", dssAdminUserService.updateUser(user, getWorkspace(req))); } diff --git a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/restful/DssProxyUserController.java b/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/restful/DssProxyUserController.java deleted file mode 100644 index 822ee5c084..0000000000 --- a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/restful/DssProxyUserController.java +++ /dev/null @@ -1,115 +0,0 @@ -package com.webank.wedatasphere.dss.framework.admin.restful; - -import com.webank.wedatasphere.dss.common.utils.DSSExceptionUtils; -import com.webank.wedatasphere.dss.framework.admin.common.utils.StringUtils; -import com.webank.wedatasphere.dss.framework.admin.exception.DSSAdminErrorException; -import com.webank.wedatasphere.dss.framework.admin.pojo.entity.DssProxyUser; -import com.webank.wedatasphere.dss.framework.admin.service.DssProxyUserService; -import org.apache.commons.lang.exception.ExceptionUtils; -import org.apache.linkis.server.Message; -import org.apache.linkis.server.conf.ServerConfiguration; -import org.apache.linkis.server.security.ProxyUserSSOUtils; -import org.apache.linkis.server.security.SecurityFilter; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.web.bind.annotation.RequestBody; -import org.springframework.web.bind.annotation.RequestMapping; -import org.springframework.web.bind.annotation.RequestMethod; -import org.springframework.web.bind.annotation.RestController; -import scala.Tuple2; - -import javax.servlet.http.Cookie; -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletResponse; -import java.util.List; -import java.util.stream.Collectors; - -import static com.webank.wedatasphere.dss.framework.admin.conf.AdminConf.*; - -@RequestMapping(path = "/dss/framework/admin/user", produces = {"application/json"}) -@RestController -public class DssProxyUserController { - private Boolean sslEnable = (Boolean) ServerConfiguration.BDP_SERVER_SECURITY_SSL().getValue(); - private String PROXY_USER_TICKET_ID_STRING = ServerConfiguration.LINKIS_SERVER_SESSION_PROXY_TICKETID_KEY().getValue(); - protected final Logger LOGGER = LoggerFactory.getLogger(this.getClass()); - @Autowired - DssProxyUserService dssProxyUserService; - - @RequestMapping(path = "proxy/list", method = RequestMethod.GET) - public Message getProxyUserList(HttpServletRequest request) { - String username = SecurityFilter.getLoginUsername(request); - - List userList = dssProxyUserService.selectProxyUserList(username); - List proxyUserNameList=userList.stream().map(dssProxyUser -> dssProxyUser.getProxyUserName()).collect(Collectors.toList()); - if(DS_PROXY_SELF_ENABLE.getValue()) { - proxyUserNameList.add(username); - } - return Message.ok().data("proxyUserList", proxyUserNameList); - } - - @RequestMapping(path = "proxy/addUserCookie", method = RequestMethod.POST) - public Message setProxyUserCookie(@RequestBody DssProxyUser userRep, HttpServletRequest req, HttpServletResponse resp) { - - String username = SecurityFilter.getLoginUsername(req); - String trustCode = DS_TRUST_TOKEN.getValue(); - try { - if (userRep.getUserName().equals(username)) { - if (StringUtils.isEmpty(userRep.getUserName())) { - DSSExceptionUtils.dealErrorException(100101, "User name is empty", DSSAdminErrorException.class); - } else if (StringUtils.isEmpty(userRep.getProxyUserName())) { - DSSExceptionUtils.dealErrorException(100102, "Proxy user name is empty", DSSAdminErrorException.class); - } else if (dssProxyUserService.isExists(userRep.getUserName(), userRep.getProxyUserName())) { - for (Cookie cookie : req.getCookies()) { - if (null != cookie && cookie.getName().equalsIgnoreCase(PROXY_USER_TICKET_ID_STRING)) { - cookie.setValue(null); - cookie.setMaxAge(0); - resp.addCookie(cookie); - } - } - } - Tuple2 userTicketIdKv = ProxyUserSSOUtils.getProxyUserTicketKV(userRep.getProxyUserName(), trustCode); - Cookie cookie = new Cookie(userTicketIdKv._1, userTicketIdKv._2); - cookie.setMaxAge(-1); - if (sslEnable){ - cookie.setSecure(true); - } - cookie.setPath("/"); - resp.addCookie(cookie); - - } else { - DSSExceptionUtils.dealErrorException(100103,"The requested user name is not a login user",DSSAdminErrorException.class); - } - return Message.ok("Success to add proxy user into cookie"); - - } catch (Exception exception) { - LOGGER.error("Failed to set cookie for proxy user", exception); - return Message.error(ExceptionUtils.getRootCauseMessage(exception)); - } - - } - - @RequestMapping(path = "proxy/add", method = RequestMethod.POST) - public Message add(@RequestBody DssProxyUser userRep, HttpServletRequest req) { - String username = SecurityFilter.getLoginUsername(req); - - try { - if(!username.equals(DSS_PROXY_ADMIN_NAME.getValue())){ - DSSExceptionUtils.dealErrorException(100104, "Only administrators can add proxy users", DSSAdminErrorException.class); - } - if(StringUtils.isEmpty(userRep.getUserName())){ - DSSExceptionUtils.dealErrorException(100105, "User name is empty", DSSAdminErrorException.class); - }else if(StringUtils.isEmpty(userRep.getProxyUserName())){ - DSSExceptionUtils.dealErrorException(100106, "Proxy user name is empty", DSSAdminErrorException.class); - }else if (dssProxyUserService.isExists(userRep.getUserName(),userRep.getProxyUserName())) { - DSSExceptionUtils.dealErrorException(100107, "Failed to add proxy user,'userName:" + userRep.getUserName() + ",proxyName:"+userRep.getProxyUserName()+" already exists", DSSAdminErrorException.class); - } - dssProxyUserService.insertProxyUser(userRep); - return Message.ok("Success to add proxy user"); - } catch (Exception exception) { - LOGGER.error("Failed to add proxy user", exception); - return Message.error(ExceptionUtils.getRootCauseMessage(exception)); - } - - } -} diff --git a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/service/DssExchangeService.java b/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/service/DssExchangeService.java deleted file mode 100644 index eb7acae497..0000000000 --- a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/service/DssExchangeService.java +++ /dev/null @@ -1,115 +0,0 @@ -package com.webank.wedatasphere.dss.framework.admin.service; - -import com.google.gson.JsonArray; -import com.google.gson.JsonElement; -import com.google.gson.JsonObject; -import com.google.gson.JsonParser; -import com.squareup.okhttp.Request; -import com.squareup.okhttp.Response; -import com.webank.wedatasphere.dss.framework.admin.conf.AdminConf; -import com.webank.wedatasphere.dss.framework.admin.pojo.entity.DssExchangeTask; -import com.webank.wedatasphere.dss.framework.admin.pojo.entity.DssExchangeTaskRes; -import com.webank.wedatasphere.dss.framework.admin.pojo.entity.DssExchangisProject; - -import com.webank.wedatasphere.dss.framework.admin.util.OkHttpHelper; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.springframework.stereotype.Service; - -import java.util.ArrayList; -import java.util.List; - -/** - * @Auther: Han Tang - * @Date: 2022/1/18-01-18-15:10 - */ -@Service -public class DssExchangeService { - - private static final String PROJECT_TREE_PATH = "/api/v1/project/tree"; - private static final String TASK_TREE_PATH = "/api/v1/jobinfo/pageList"; - private static final Logger LOGGER = LoggerFactory.getLogger(OkHttpHelper.class); - - public List queryExchangeProject(String userName) throws Exception { - String url = AdminConf.EXCHANGE_URL.getValue() + PROJECT_TREE_PATH + "/"+userName; - Request getRequest = new Request.Builder() - .url(url) - .addHeader("Content-Type", "application/json") - .addHeader("Cookie", AdminConf.EXCHANGE_ADMIN_COOKIE.getValue()) - .build(); - Response response = OkHttpHelper.syncGet(getRequest); - JsonObject returnData = new JsonParser().parse(response.body().string()).getAsJsonObject(); - LOGGER.info(returnData.toString()); - JsonArray jsonArray = returnData.getAsJsonArray("data"); - List projectList = new ArrayList<>(); - List dssExchangisProjects = getExchangeProjectList(projectList, jsonArray); - return dssExchangisProjects; - } - - public DssExchangeTaskRes queryExchangeTask(int projectId, String userName, int pageNum, String fullName) throws Exception { - String url = AdminConf.EXCHANGE_URL.getValue() + TASK_TREE_PATH + "/" + userName + - "?projectId=" + projectId + "&page=" + pageNum + "&pageSize=10&fuzzyName=&jobId="; - Request getRequest = new Request.Builder() - .url(url) - .addHeader("Content-Type", "application/json") - .addHeader("Cookie", AdminConf.EXCHANGE_ADMIN_COOKIE.getValue()) - .get() - .build(); - Response response = OkHttpHelper.syncGet(getRequest); - JsonObject returnData = new JsonParser().parse(response.body().string()).getAsJsonObject(); - LOGGER.info(returnData.toString()); - DssExchangeTaskRes dssExchangeTaskRes = new DssExchangeTaskRes(); - JsonObject resJsonObject = returnData.getAsJsonObject("data"); - JsonArray resJsonArray = null; - if (resJsonObject != null) { - resJsonArray = resJsonObject.getAsJsonArray("data"); - } - - dssExchangeTaskRes.setPage(resJsonObject.get("page").getAsInt()); - dssExchangeTaskRes.setPageSize(resJsonObject.get("pageSize").getAsInt()); - dssExchangeTaskRes.setTotalPages(resJsonObject.get("totalPages").getAsInt()); - dssExchangeTaskRes.setTotalItems(resJsonObject.get("totalItems").getAsInt()); - List taskList = new ArrayList<>(); - if (resJsonArray != null) { - for (JsonElement jsonElement : resJsonArray) { - JsonObject jsonObject = jsonElement.getAsJsonObject(); - DssExchangeTask dssExchangeTask = new DssExchangeTask(); - dssExchangeTask.setId(jsonObject.get("id").getAsInt()); - dssExchangeTask.setJobName(jsonObject.get("jobName") == null ? null : jsonObject.get("jobName").getAsString()); - dssExchangeTask.setJobCorn(jsonObject.get("jobCorn") == null ? null : jsonObject.get("jobCorn").getAsString()); - dssExchangeTask.setJobDesc(jsonObject.get("jobDesc") == null ? null : jsonObject.get("jobDesc").getAsString()); - dssExchangeTask.setCreateTime(jsonObject.get("createTime") == null ? null : jsonObject.get("createTime").getAsString()); - dssExchangeTask.setJobStatus(jsonObject.get("jobStatus") == null ? null : jsonObject.get("jobStatus").getAsString()); - taskList.add(dssExchangeTask); - } - } - - dssExchangeTaskRes.setDssExchangeTaskList(taskList); - return dssExchangeTaskRes; - } - - - public String getSellScript(int taskId, int projectId) { - String shellScript = "str=`curl -X GET --data '{\"project_id\":" + projectId + ",\"task_id\":" + taskId + "}' " + - "--header 'Content-Type: application/json' --header 'Accept: application/json' " + - "--header 'Cookie:" + AdminConf.EXCHANGE_ADMIN_COOKIE.getValue() + "' " + AdminConf.EXCHANGE_URL.getValue() + - "/api/v1/jobinfo/runTask/" + taskId + "?userName=admin`;if [[ ${str} =~ 'job execution successed' ]];then exit 0;else exit 1;fi"; - return shellScript; - } - - public List getExchangeProjectList(List projectList, JsonArray jsonArray) { - if (jsonArray != null) { - for (JsonElement jsonElement : jsonArray) { - DssExchangisProject dssExchangisProject = new DssExchangisProject(); - dssExchangisProject.setProjectName(jsonElement.getAsJsonObject().get("projectName") == null ? null : jsonElement.getAsJsonObject().get("projectName").getAsString()); - dssExchangisProject.setId(jsonElement.getAsJsonObject().get("id") == null ? null : jsonElement.getAsJsonObject().get("id").getAsInt()); - projectList.add(dssExchangisProject); - JsonArray childJsonArray = jsonElement.getAsJsonObject().getAsJsonArray("children"); - if (childJsonArray != null && childJsonArray.size() > 0) { - getExchangeProjectList(projectList, childJsonArray); - } - } - } - return projectList; - } -} diff --git a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/service/DssProxyUserService.java b/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/service/DssProxyUserService.java deleted file mode 100644 index 5442a01bd5..0000000000 --- a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/service/DssProxyUserService.java +++ /dev/null @@ -1,25 +0,0 @@ -package com.webank.wedatasphere.dss.framework.admin.service; - - -import com.webank.wedatasphere.dss.framework.admin.pojo.entity.DssProxyUser; - -import java.util.List; - -public interface DssProxyUserService { - - /** - * 查询代理用户数据 - * - * @param userName 查询代理用户的用户名 - * @return 代理用户的集合 - */ - List selectProxyUserList(String userName); - - List getProxyUserNameList(String userName); - - int insertProxyUser(DssProxyUser dssProxyUser); - - boolean isExists(String userName,String proxyUserName); - - -} diff --git a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/service/DssScriptDownloadService.java b/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/service/DssScriptDownloadService.java deleted file mode 100644 index 26ba267910..0000000000 --- a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/service/DssScriptDownloadService.java +++ /dev/null @@ -1,17 +0,0 @@ -package com.webank.wedatasphere.dss.framework.admin.service; - -import com.baomidou.mybatisplus.extension.service.IService; -import com.webank.wedatasphere.dss.framework.admin.pojo.entity.DssScriptDownloadAudit; -import org.apache.ibatis.annotations.Param; - -import java.util.List; - -/** - * @Auther: Han Tang - * @Date: 2022/1/11-01-11-15:11 - */ -public interface DssScriptDownloadService extends IService { - - List getDownloadAuditList( String userName, String startIme, String endTime); - -} diff --git a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/service/impl/DssAdminDeptServiceImpl.java b/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/service/impl/DssAdminDeptServiceImpl.java index e653d459c8..4c85c5cda2 100644 --- a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/service/impl/DssAdminDeptServiceImpl.java +++ b/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/service/impl/DssAdminDeptServiceImpl.java @@ -1,13 +1,13 @@ package com.webank.wedatasphere.dss.framework.admin.service.impl; +import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; import com.webank.wedatasphere.dss.framework.admin.common.constant.UserConstants; -import com.webank.wedatasphere.dss.framework.admin.common.exception.AdminException; import com.webank.wedatasphere.dss.framework.admin.common.utils.StringUtils; +import com.webank.wedatasphere.dss.framework.admin.exception.DSSAdminWarnException; import com.webank.wedatasphere.dss.framework.admin.pojo.entity.DssAdminDept; -import com.webank.wedatasphere.dss.framework.admin.xml.DssAdminDeptMapper; import com.webank.wedatasphere.dss.framework.admin.pojo.entity.TreeSelect; import com.webank.wedatasphere.dss.framework.admin.service.DssAdminDeptService; -import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; +import com.webank.wedatasphere.dss.framework.admin.xml.DssAdminDeptMapper; import org.springframework.stereotype.Service; import javax.annotation.Resource; @@ -48,7 +48,7 @@ public int insertDept(DssAdminDept dept) { DssAdminDept info = dssAdminDeptMapper.selectDeptById(dept.getParentId()); // 如果父节点不为正常状态,则不允许新增子节点 if (!UserConstants.DEPT_NORMAL.equals(info.getStatus())) { - throw new AdminException("部门停用,不允许新增"); + throw new DSSAdminWarnException("部门停用,不允许新增"); } dept.setAncestors(info.getAncestors() + "," + dept.getParentId()); return dssAdminDeptMapper.insertDept(dept); diff --git a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/service/impl/DssProxyUserServiceImpl.java b/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/service/impl/DssProxyUserServiceImpl.java deleted file mode 100644 index 3e28e48b74..0000000000 --- a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/service/impl/DssProxyUserServiceImpl.java +++ /dev/null @@ -1,47 +0,0 @@ -package com.webank.wedatasphere.dss.framework.admin.service.impl; - -import com.webank.wedatasphere.dss.framework.admin.pojo.entity.DssProxyUser; -import com.webank.wedatasphere.dss.framework.admin.service.DssProxyUserService; -import com.webank.wedatasphere.dss.framework.admin.xml.DSSProxyUserMapper; -import org.springframework.stereotype.Service; -import javax.annotation.Resource; -import java.util.List; -import java.util.stream.Collectors; - -import static com.webank.wedatasphere.dss.framework.admin.conf.AdminConf.DS_PROXY_SELF_ENABLE; - -@Service -public class DssProxyUserServiceImpl implements DssProxyUserService { - @Resource - DSSProxyUserMapper dssProxyUserMapper; - - @Override - public List selectProxyUserList(String userName) { - return dssProxyUserMapper.selectProxyUserList(userName); - } - - @Override - public List getProxyUserNameList(String userName) { - List userList = dssProxyUserMapper.selectProxyUserList(userName);; - List proxyUserNameList=userList.stream().map(dssProxyUser -> dssProxyUser.getProxyUserName()).collect(Collectors.toList()); - return proxyUserNameList; - } - - @Override - public int insertProxyUser(DssProxyUser dssProxyUser) { - int rows = dssProxyUserMapper.insertUser(dssProxyUser); - return rows; - } - - @Override - public boolean isExists(String userName, String proxyUserName) { - List res= dssProxyUserMapper.getProxyUserList(userName,proxyUserName); - if(DS_PROXY_SELF_ENABLE.getValue() && userName.equalsIgnoreCase(proxyUserName)){ - return true; - }else if(res.size()==0){ - return false; - }else { - return true; - } - } -} diff --git a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/service/impl/LdapServiceImpl.java b/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/service/impl/LdapServiceImpl.java index 395717494e..55a80c535d 100644 --- a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/service/impl/LdapServiceImpl.java +++ b/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/service/impl/LdapServiceImpl.java @@ -25,7 +25,7 @@ public void addUser(String adminName,String adminPassword,String ldapUrl,String basicAttributes.put("cn", userName); basicAttributes.put("uid", userName); basicAttributes.put("userPassword", pwd); - ctx.createSubcontext("uid="+userName+","+baseDN, basicAttributes); + ctx.createSubcontext("uid=" + userName + "," + baseDN, basicAttributes); LdapUtils.closeContext(ctx); } @@ -46,13 +46,12 @@ public boolean exist(String adminName,String adminPassword,String ldapUrl,String LdapContext ctx = LdapUtils.connectLDAP(adminName, adminPassword, ldapUrl); SearchControls searchCtls = new SearchControls(); searchCtls.setSearchScope(SearchControls.SUBTREE_SCOPE); - String searchFilter = "uid="+userName; - String searchBase = baseDN; - String returnedAtts[] = { "cn" }; + String searchFilter = "uid=" + userName; + String[] returnedAtts = { "cn" }; searchCtls.setReturningAttributes(returnedAtts); boolean exist = false; - NamingEnumeration entries = ctx.search(searchBase, searchFilter, searchCtls); + NamingEnumeration entries = ctx.search(baseDN, searchFilter, searchCtls); if(entries.hasMore()){ exist = true; } @@ -61,9 +60,4 @@ public boolean exist(String adminName,String adminPassword,String ldapUrl,String } - - public static void main(String[] args) throws NamingException { - - } - } diff --git a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/xml/DSSProxyUserMapper.java b/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/xml/DSSProxyUserMapper.java deleted file mode 100644 index eadb667cd1..0000000000 --- a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/xml/DSSProxyUserMapper.java +++ /dev/null @@ -1,18 +0,0 @@ -package com.webank.wedatasphere.dss.framework.admin.xml; - -import com.baomidou.mybatisplus.core.mapper.BaseMapper; -import com.webank.wedatasphere.dss.framework.admin.pojo.entity.DssAdminUser; -import com.webank.wedatasphere.dss.framework.admin.pojo.entity.DssProxyUser; -import org.apache.ibatis.annotations.Mapper; -import org.apache.ibatis.annotations.Param; - -import java.util.List; - -@Mapper -public interface DSSProxyUserMapper extends BaseMapper { - public List selectProxyUserList(String userName); - public int insertUser(DssProxyUser user); - public List getProxyUserList(@Param("userName") String userName, @Param("proxyUserName") String proxyUserName); - - -} diff --git a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/xml/DssAuditMapper.java b/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/xml/DssAuditMapper.java deleted file mode 100644 index d0d49be845..0000000000 --- a/dss-framework/dss-framework-admin-service/src/main/java/com/webank/wedatasphere/dss/framework/admin/xml/DssAuditMapper.java +++ /dev/null @@ -1,16 +0,0 @@ -package com.webank.wedatasphere.dss.framework.admin.xml; - -import com.baomidou.mybatisplus.core.mapper.BaseMapper; -import com.webank.wedatasphere.dss.framework.admin.pojo.entity.DssScriptDownloadAudit; -import org.apache.ibatis.annotations.Param; -import org.apache.ibatis.annotations.Result; -import org.apache.ibatis.annotations.Results; -import org.apache.ibatis.annotations.Select; - -import java.util.List; - -public interface DssAuditMapper extends BaseMapper { - - List getDownloadAuditList(@Param("creator") String creator,@Param("startTime") String startIme,@Param("endTime") String endTime); - -} diff --git a/dss-framework/dss-framework-common/pom.xml b/dss-framework/dss-framework-common/pom.xml index 17bacb460e..65b5d525e7 100644 --- a/dss-framework/dss-framework-common/pom.xml +++ b/dss-framework/dss-framework-common/pom.xml @@ -21,8 +21,8 @@ dss com.webank.wedatasphere.dss - 1.1.1 - + 1.1.2 + ../../pom.xml 4.0.0 dss-framework-common diff --git a/dss-framework/dss-framework-orchestrator-server/pom.xml b/dss-framework/dss-framework-orchestrator-server/pom.xml index 3f3ef47b28..059f455f43 100644 --- a/dss-framework/dss-framework-orchestrator-server/pom.xml +++ b/dss-framework/dss-framework-orchestrator-server/pom.xml @@ -21,8 +21,8 @@ dss com.webank.wedatasphere.dss - 1.1.1 - + 1.1.2 + ../../pom.xml 4.0.0 @@ -66,6 +66,16 @@ ${dss.version} + + spring-aop + org.springframework + ${spring-framework.version} + + + spring-context + org.springframework + ${spring-framework.version} + org.apache.linkis linkis-rpc @@ -80,6 +90,14 @@ org.springframework.boot spring-boot-starter-log4j2 + + spring-aop + org.springframework + + + spring-context + org.springframework + @@ -164,6 +182,12 @@ org.apache.linkis linkis-storage ${linkis.version} + + + commons-text + org.apache.commons + + diff --git a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/conf/OrchestratorConf.java b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/conf/OrchestratorConf.java index 7c830dd65d..4ab4f98671 100644 --- a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/conf/OrchestratorConf.java +++ b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/conf/OrchestratorConf.java @@ -21,4 +21,8 @@ public class OrchestratorConf { public static final CommonVars DSS_UPLOAD_PATH = CommonVars.apply("wds.dss.file.upload.dir", "/appcom/tmp/uploads"); + public static final CommonVars DSS_PUBLISH_MAX_VERSION = CommonVars.apply("wds.dss.publish.max.remain.version", 30); + public static final CommonVars DSS_CS_CLEAR_ENV = CommonVars.apply("wds.dss.server.cs.clear.env", "DEV"); + public static final CommonVars DSS_CS_CLEAR_CRON = CommonVars.apply("wds.dss.server.scheduling.clear.cs.cron", ""); + } diff --git a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/conf/OrchestratorSpringConf.java b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/conf/OrchestratorSpringConf.java index e2df01daa3..1b0956224d 100644 --- a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/conf/OrchestratorSpringConf.java +++ b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/conf/OrchestratorSpringConf.java @@ -16,18 +16,38 @@ package com.webank.wedatasphere.dss.orchestrator.server.conf; +import com.webank.wedatasphere.dss.common.service.BMLService; +import com.webank.wedatasphere.dss.common.utils.AssembleCronUtils; import com.webank.wedatasphere.dss.contextservice.service.ContextService; import com.webank.wedatasphere.dss.contextservice.service.impl.ContextServiceImpl; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; +import java.util.concurrent.ScheduledThreadPoolExecutor; + @Configuration public class OrchestratorSpringConf { @Bean(name = "contextService") - public ContextService createContextService(){ + public ContextService createContextService() { return ContextServiceImpl.getInstance(); } + @Bean(name = "orchestratorBmlService") + public BMLService createBmlService() { + return BMLService.getInstance(); + } + + @Bean + public ScheduledThreadPoolExecutor scheduledExecutorService() { + ScheduledThreadPoolExecutor executor = new ScheduledThreadPoolExecutor(3); + return executor; + } + + @Bean + public String getBatchClearCsTaskCron() { + return OrchestratorConf.DSS_CS_CLEAR_CRON.getValue(); + } + } diff --git a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/constant/DSSOrchestratorConstant.java b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/constant/DSSOrchestratorConstant.java index e4a1dd6464..9fe2c841fd 100644 --- a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/constant/DSSOrchestratorConstant.java +++ b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/constant/DSSOrchestratorConstant.java @@ -17,13 +17,8 @@ package com.webank.wedatasphere.dss.orchestrator.server.constant; -import com.webank.wedatasphere.dss.orchestrator.publish.job.OrchestratorConversionJob; - -import java.util.Map; -import java.util.concurrent.ConcurrentHashMap; - public class DSSOrchestratorConstant { - public static final String PUBLISH_FLOW_REPORT_FORMATE = "工作流名:%s,版本号:%s,工作流内容为空,请自行修改或者删除"; - public static Map orchestratorConversionJobMap = new ConcurrentHashMap<>(); + //每次提交linkis清理数量控制在10个以下,超了接口就会返回超时 + public static final int MAX_CLEAR_SIZE = 10; } diff --git a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/request/AddOrchestratorRequest.java b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/request/AddOrchestratorRequest.java deleted file mode 100644 index 1ca831b51b..0000000000 --- a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/request/AddOrchestratorRequest.java +++ /dev/null @@ -1,71 +0,0 @@ -package com.webank.wedatasphere.dss.orchestrator.server.entity.request; - -import com.webank.wedatasphere.dss.common.label.LabelRouteVO; - -@Deprecated -public class AddOrchestratorRequest { - private String name; - private String workspaceName; - private String projectName; - private String type; - private String desc; - private Long projectID; - - private LabelRouteVO labels; - - public String getName() { - return name; - } - - public void setName(String name) { - this.name = name; - } - - public String getWorkspaceName() { - return workspaceName; - } - - public void setWorkspaceName(String workspaceName) { - this.workspaceName = workspaceName; - } - - public String getProjectName() { - return projectName; - } - - public void setProjectName(String projectName) { - this.projectName = projectName; - } - - public String getType() { - return type; - } - - public void setType(String type) { - this.type = type; - } - - public String getDesc() { - return desc; - } - - public void setDesc(String desc) { - this.desc = desc; - } - - public Long getProjectID() { - return projectID; - } - - public void setProjectID(Long projectID) { - this.projectID = projectID; - } - - public LabelRouteVO getLabels() { - return labels; - } - - public void setLabels(LabelRouteVO labels) { - this.labels = labels; - } -} diff --git a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/request/OpenOrchestratorRequest.java b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/request/OpenOrchestratorRequest.java index aee28c2f88..0569dbde95 100644 --- a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/request/OpenOrchestratorRequest.java +++ b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/request/OpenOrchestratorRequest.java @@ -32,4 +32,13 @@ public LabelRouteVO getLabels() { public void setLabels(LabelRouteVO labels) { this.labels = labels; } + + @Override + public String toString() { + return "OpenOrchestratorRequest{" + + "workspaceName='" + workspaceName + '\'' + + ", orchestratorId=" + orchestratorId + + ", labels=" + labels + + '}'; + } } diff --git a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/request/OrchestratorCopyRequest.java b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/request/OrchestratorCopyRequest.java new file mode 100644 index 0000000000..5c5c10f5ed --- /dev/null +++ b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/request/OrchestratorCopyRequest.java @@ -0,0 +1,136 @@ +package com.webank.wedatasphere.dss.orchestrator.server.entity.request; + +import com.webank.wedatasphere.dss.common.label.LabelRouteVO; + +import javax.validation.constraints.NotNull; +import javax.xml.bind.annotation.XmlRootElement; +import java.util.List; + +@XmlRootElement +public class OrchestratorCopyRequest { + + @NotNull(message = "源编排Id不能为空") + private Long sourceOrchestratorId; + + @NotNull(message = "源编排名称不能为空") + private String sourceOrchestratorName; + + @NotNull(message = "目标编排名称不能为空") + private String targetOrchestratorName; + + @NotNull(message = "源工程Id不能为空") + private Long sourceProjectId; + + @NotNull(message = "目标工程Id不能为空") + private Long targetProjectId; + + @NotNull(message = "源工程名不能为空") + private String sourceProjectName; + + @NotNull(message = "目标工程名不能为空") + private String targetProjectName; + + @NotNull(message = "工作空间Id不能为空") + private Long workspaceId; + + @NotNull(message = "目标工作流节点后缀") + private String workflowNodeSuffix; + + private LabelRouteVO labels; + + public LabelRouteVO getLabels() { + return labels; + } + + public void setLabels(LabelRouteVO labels) { + this.labels = labels; + } + + public Long getSourceOrchestratorId() { + return sourceOrchestratorId; + } + + public void setSourceOrchestratorId(Long sourceOrchestratorId) { + this.sourceOrchestratorId = sourceOrchestratorId; + } + + public String getSourceOrchestratorName() { + return sourceOrchestratorName; + } + + public void setSourceOrchestratorName(String sourceOrchestratorName) { + this.sourceOrchestratorName = sourceOrchestratorName; + } + + public String getTargetOrchestratorName() { + return targetOrchestratorName; + } + + public void setTargetOrchestratorName(String targetOrchestratorName) { + this.targetOrchestratorName = targetOrchestratorName; + } + + public Long getSourceProjectId() { + return sourceProjectId; + } + + public void setSourceProjectId(Long sourceProjectId) { + this.sourceProjectId = sourceProjectId; + } + + public Long getTargetProjectId() { + return targetProjectId; + } + + public void setTargetProjectId(Long targetProjectId) { + this.targetProjectId = targetProjectId; + } + + public String getSourceProjectName() { + return sourceProjectName; + } + + public void setSourceProjectName(String sourceProjectName) { + this.sourceProjectName = sourceProjectName; + } + + public String getTargetProjectName() { + return targetProjectName; + } + + public void setTargetProjectName(String targetProjectName) { + this.targetProjectName = targetProjectName; + } + + public Long getWorkspaceId() { + return workspaceId; + } + + public void setWorkspaceId(Long workspaceId) { + this.workspaceId = workspaceId; + } + + public String getWorkflowNodeSuffix() { + return workflowNodeSuffix; + } + + public void setWorkflowNodeSuffix(String workflowNodeSuffix) { + this.workflowNodeSuffix = workflowNodeSuffix; + } + + @Override + public String toString() { + return "OrchestratorCopyRequest{" + + "sourceOrchestratorId=" + sourceOrchestratorId + + ", sourceOrchestratorName='" + sourceOrchestratorName + '\'' + + ", targetOrchestratorName='" + targetOrchestratorName + '\'' + + ", sourceProjectId=" + sourceProjectId + + ", targetProjectId=" + targetProjectId + + ", sourceProjectName='" + sourceProjectName + '\'' + + ", targetProjectName='" + targetProjectName + '\'' + + ", workspaceId=" + workspaceId + + ", workflowNodeSuffix='" + workflowNodeSuffix + '\'' + + ", labels=" + labels + + '}'; + } +} diff --git a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/request/OrchestratorCreateRequest.java b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/request/OrchestratorCreateRequest.java index b543b0a8a2..d374b668cd 100644 --- a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/request/OrchestratorCreateRequest.java +++ b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/request/OrchestratorCreateRequest.java @@ -27,12 +27,6 @@ public class OrchestratorCreateRequest extends OrchestratorRequest { @NotNull(message = "编排名称不能为空") private String orchestratorName; - /** - * 编排模式,如工作流,组合编排等 - */ - @NotNull(message = "编排模式类型不能为空") - private String orchestratorMode; - /** * 编排方式 */ @@ -58,6 +52,20 @@ public class OrchestratorCreateRequest extends OrchestratorRequest { private String workspaceName; + public OrchestratorCreateRequest() { + } + + public OrchestratorCreateRequest(String orchestratorName, List orchestratorWays, String orchestratorLevel, List dssLabels, String uses, String description, String projectName, String workspaceName) { + this.orchestratorName = orchestratorName; + this.orchestratorWays = orchestratorWays; + this.orchestratorLevel = orchestratorLevel; + this.dssLabels = dssLabels; + this.uses = uses; + this.description = description; + this.projectName = projectName; + this.workspaceName = workspaceName; + } + public List getDssLabels() { return dssLabels; } @@ -74,16 +82,6 @@ public void setOrchestratorName(String orchestratorName) { this.orchestratorName = orchestratorName; } - @Override - public String getOrchestratorMode() { - return orchestratorMode; - } - - @Override - public void setOrchestratorMode(String orchestratorMode) { - this.orchestratorMode = orchestratorMode; - } - public List getOrchestratorWays() { return orchestratorWays; } @@ -138,7 +136,7 @@ public String toString() { "workspaceId=" + getWorkspaceId() + ", projectId=" + getProjectId() + ", arrangeName='" + orchestratorName + '\'' + - ", arrangeMode='" + orchestratorMode + '\'' + + ", arrangeMode='" + getOrchestratorMode() + '\'' + ", arrangeWays=" + orchestratorWays + ", uses='" + uses + '\'' + ", description='" + description + '\'' + diff --git a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/request/OrchestratorDeleteRequest.java b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/request/OrchestratorDeleteRequest.java index 1302926e35..fef1c32fbc 100644 --- a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/request/OrchestratorDeleteRequest.java +++ b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/request/OrchestratorDeleteRequest.java @@ -43,4 +43,12 @@ public Boolean getDeleteSchedulerWorkflow() { public void setDeleteSchedulerWorkflow(Boolean deleteSchedulerWorkflow) { this.deleteSchedulerWorkflow = deleteSchedulerWorkflow; } + + @Override + public String toString() { + return "OrchestratorDeleteRequest{" + + "id=" + id + + ", deleteSchedulerWorkflow=" + deleteSchedulerWorkflow + + '}'; + } } diff --git a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/request/OrchestratorModifyRequest.java b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/request/OrchestratorModifyRequest.java index b454d1d5b0..fb62a9fac5 100644 --- a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/request/OrchestratorModifyRequest.java +++ b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/request/OrchestratorModifyRequest.java @@ -29,12 +29,6 @@ public class OrchestratorModifyRequest extends OrchestratorRequest { @NotNull(message = "编排名称不能为空") private String orchestratorName; - /** - * 编排模式,如工作流,组合编排等 - */ - @NotNull(message = "编排模式不能为空") - private String orchestratorMode; - /** * 编排方式 */ @@ -69,14 +63,6 @@ public void setOrchestratorName(String orchestratorName) { this.orchestratorName = orchestratorName; } - public String getOrchestratorMode() { - return orchestratorMode; - } - - public void setOrchestratorMode(String orchestratorMode) { - this.orchestratorMode = orchestratorMode; - } - public List getOrchestratorWays() { return orchestratorWays; } @@ -116,7 +102,7 @@ public String toString() { ", workspaceId=" + getWorkspaceId() + ", projectId=" + getProjectId() + ", orchestratorName='" + orchestratorName + '\'' + - ", orchestratorMode='" + orchestratorMode + '\'' + + ", orchestratorMode='" + getOrchestratorMode() + '\'' + ", orchestratorWays=" + orchestratorWays + ", uses='" + uses + '\'' + ", description='" + description + '\'' + diff --git a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/request/OrchestratorUnlockRequest.java b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/request/OrchestratorUnlockRequest.java new file mode 100644 index 0000000000..4b76b847ea --- /dev/null +++ b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/request/OrchestratorUnlockRequest.java @@ -0,0 +1,55 @@ +/* + * Copyright 2019 WeBank + * 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. + * + */ + +package com.webank.wedatasphere.dss.orchestrator.server.entity.request; + +import javax.validation.constraints.NotNull; + +public class OrchestratorUnlockRequest extends OrchestratorRequest { + + @NotNull(message = "id不能为空") + private Long id; + + /** + * 是否确认解锁 + */ + private Boolean confirmDelete; + + public Long getId() { + return id; + } + + public void setId(Long id) { + this.id = id; + } + + + public Boolean getConfirmDelete() { + return confirmDelete; + } + + public void setConfirmDelete(Boolean confirmDelete) { + this.confirmDelete = confirmDelete; + } + + @Override + public String toString() { + return "OrchestratorUnlockRequest{" + + "id=" + id + + ", confirmDelete=" + confirmDelete + + '}' + super.toString(); + } +} diff --git a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/request/RollbackOrchestratorRequest.java b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/request/RollbackOrchestratorRequest.java index e26bb0e12e..23b8c758fe 100644 --- a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/request/RollbackOrchestratorRequest.java +++ b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/request/RollbackOrchestratorRequest.java @@ -49,4 +49,15 @@ public String getProjectName() { public void setProjectName(String projectName) { this.projectName = projectName; } + + @Override + public String toString() { + return "RollbackOrchestratorRequest{" + + "orchestratorId=" + orchestratorId + + ", version='" + version + '\'' + + ", projectId=" + projectId + + ", projectName='" + projectName + '\'' + + ", labels=" + labels + + '}'; + } } diff --git a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/vo/CommonOrchestratorVo.java b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/vo/CommonOrchestratorVo.java index f071985854..0f341085cc 100644 --- a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/vo/CommonOrchestratorVo.java +++ b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/vo/CommonOrchestratorVo.java @@ -27,6 +27,8 @@ public class CommonOrchestratorVo { private String orchestratorVersion; + private String orchestratorName; + private String type = WORK_FLOW; public Long getOrchestratorId() { @@ -45,6 +47,14 @@ public void setOrchestratorVersion(String orchestratorVersion) { this.orchestratorVersion = orchestratorVersion; } + public String getOrchestratorName() { + return orchestratorName; + } + + public void setOrchestratorName(String orchestratorName) { + this.orchestratorName = orchestratorName; + } + public String getType() { return type; } diff --git a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/vo/OrchestratorCopyHistory.java b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/vo/OrchestratorCopyHistory.java new file mode 100644 index 0000000000..0c6f756a8b --- /dev/null +++ b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/vo/OrchestratorCopyHistory.java @@ -0,0 +1,196 @@ +package com.webank.wedatasphere.dss.orchestrator.server.entity.vo; + + +public class OrchestratorCopyHistory { + + private String id; + + private String userName; + + private String workspaceName; + + private String sourceOrchestratorName; + + private String targetOrchestratorName; + + private String sourceProjectName; + + private String targetProjectName; + + /** + * 目标工作流节点后缀 + */ + private String workflowNodeSuffix; + + private String microserverName; + + private String exceptionInfo; + + /** + * 复制任务最终状态,1成功,0失败 + */ + private Integer status; + + /** + * 当前编排是否在被复制,1有,0没有 + */ + private Integer isCopying; + + /** + * 复制开始时间 + */ + private String startTime; + + /** + * 复制结束时间 + */ + private String endTime; + + public OrchestratorCopyHistory() { + } + + public OrchestratorCopyHistory(String id, String username, String workspaceName, String sourceOrchestratorName, String targetOrchestratorName, String sourceProjectName, String targetProjectName, String workflowNodeSuffix, String microserverName, String exceptionInfo, Integer status, Integer isCopying, String startTime, String endTime) { + this.id = id; + this.userName = username; + this.workspaceName = workspaceName; + this.sourceOrchestratorName = sourceOrchestratorName; + this.targetOrchestratorName = targetOrchestratorName; + this.sourceProjectName = sourceProjectName; + this.targetProjectName = targetProjectName; + this.workflowNodeSuffix = workflowNodeSuffix; + this.microserverName = microserverName; + this.exceptionInfo = exceptionInfo; + this.status = status; + this.isCopying = isCopying; + this.startTime = startTime; + this.endTime = endTime; + } + + public String getId() { + return id; + } + + public void setId(String id) { + this.id = id; + } + + public String getUserName() { return userName; } + + public void setUsername(String username) { this.userName = username; } + + public String getWorkspaceName() { + return workspaceName; + } + + public void setWorkspaceName(String workspaceName) { + this.workspaceName = workspaceName; + } + + public String getSourceOrchestratorName() { + return sourceOrchestratorName; + } + + public void setSourceOrchestratorName(String sourceOrchestratorName) { + this.sourceOrchestratorName = sourceOrchestratorName; + } + + public String getTargetOrchestratorName() { + return targetOrchestratorName; + } + + public void setTargetOrchestratorName(String targetOrchestratorName) { + this.targetOrchestratorName = targetOrchestratorName; + } + + public String getSourceProjectName() { + return sourceProjectName; + } + + public void setSourceProjectName(String sourceProjectName) { + this.sourceProjectName = sourceProjectName; + } + + public String getTargetProjectName() { + return targetProjectName; + } + + public void setTargetProjectName(String targetProjectName) { + this.targetProjectName = targetProjectName; + } + + public String getWorkflowNodeSuffix() { + return workflowNodeSuffix; + } + + public void setWorkflowNodeSuffix(String workflowNodeSuffix) { + this.workflowNodeSuffix = workflowNodeSuffix; + } + + public String getMicroserverName() { + return microserverName; + } + + public void setMicroserverName(String microserverName) { + this.microserverName = microserverName; + } + + public String getExceptionInfo() { + return exceptionInfo; + } + + public void setExceptionInfo(String exceptionInfo) { + this.exceptionInfo = exceptionInfo; + } + + public Integer getStatus() { + return status; + } + + public void setStatus(Integer status) { + this.status = status; + } + + public Integer getIsCopying() { + return isCopying; + } + + public void setIsCopying(Integer isCopying) { + this.isCopying = isCopying; + } + + public String getStartTime() { + return startTime; + } + + public void setStartTime(String startTime) { + this.startTime = startTime; + } + + public String getEndTime() { + return endTime; + } + + public void setEndTime(String endTime) { + this.endTime = endTime; + } + + @Override + public String toString() { + return "OrchestratorCopyHistory{" + + "id=" + id + + "username=" + userName + + ", workspaceName='" + workspaceName + '\'' + + ", sourceOrchestratorName='" + sourceOrchestratorName + '\'' + + ", targetOrchestratorName='" + targetOrchestratorName + '\'' + + ", sourceProjectName='" + sourceProjectName + '\'' + + ", targetProjectName='" + targetProjectName + '\'' + + ", workflowNodeSuffix='" + workflowNodeSuffix + '\'' + + ", microserverName='" + microserverName + '\'' + + ", exceptionInfo='" + exceptionInfo + '\'' + + ", status=" + status + + ", isCopying=" + isCopying + + ", startTime=" + startTime + + ", endTime=" + endTime + + '}'; + } +} diff --git a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/vo/OrchestratorCopyVo.java b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/vo/OrchestratorCopyVo.java new file mode 100644 index 0000000000..fd00691754 --- /dev/null +++ b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/vo/OrchestratorCopyVo.java @@ -0,0 +1,131 @@ +package com.webank.wedatasphere.dss.orchestrator.server.entity.vo; + +import com.webank.wedatasphere.dss.common.label.DSSLabel; +import com.webank.wedatasphere.dss.orchestrator.common.entity.DSSOrchestratorInfo; +import com.webank.wedatasphere.dss.standard.app.sso.Workspace; + +import java.io.Serializable; + +public class OrchestratorCopyVo implements Serializable { + + private static final long serialVersionUID = 1L; + + private final String username; + private final Long sourceProjectId; + private final String sourceProjectName; + private final Long targetProjectId; + private final String targetProjectName; + private final DSSOrchestratorInfo orchestrator; + private final String targetOrchestratorName; + private final String workflowNodeSuffix; + private final DSSLabel dssLabel; + private final Workspace workspace; + private final Long copyTaskId; + private final String instanceName; + + private OrchestratorCopyVo(Builder builder) { + this.username = builder.username; + this.sourceProjectId = builder.sourceProjectId; + this.sourceProjectName = builder.sourceProjectName; + this.targetProjectId = builder.targetProjectId; + this.targetProjectName = builder.targetProjectName; + this.orchestrator = builder.orchestrator; + this.targetOrchestratorName = builder.targetOrchestratorName; + this.workflowNodeSuffix = builder.workflowNodeSuffix; + this.dssLabel = builder.dssLabel; + this.workspace = builder.workspace; + this.copyTaskId = builder.copyTaskId; + this.instanceName = builder.instanceName; + } + + + public static class Builder { + + private final String username; + private final Long sourceProjectId; + private final String sourceProjectName; + private final Long targetProjectId; + private final String targetProjectName; + private final DSSOrchestratorInfo orchestrator; + private final String targetOrchestratorName; + private final String workflowNodeSuffix; + private final DSSLabel dssLabel; + private final Workspace workspace; + private Long copyTaskId; + private final String instanceName; + + public Builder(String username, Long sourceProjectId, String sourceProjectName, Long targetProjectId, + String targetProjectName, DSSOrchestratorInfo orchestrator, String targetOrchestratorName, + String workflowNodeSuffix, DSSLabel dssLabel, Workspace workspace, String instanceName) { + this.username = username; + this.sourceProjectId = sourceProjectId; + this.sourceProjectName = sourceProjectName; + this.targetProjectId = targetProjectId; + this.targetProjectName = targetProjectName; + this.orchestrator = orchestrator; + this.targetOrchestratorName = targetOrchestratorName; + this.workflowNodeSuffix = workflowNodeSuffix; + this.dssLabel = dssLabel; + this.workspace = workspace; + this.instanceName = instanceName; + } + + public Builder setCopyTaskId(Long copyTaskId){ + this.copyTaskId = copyTaskId; + return this; + } + + public OrchestratorCopyVo build(){ + return new OrchestratorCopyVo(this); + } + } + + public String getUsername() { + return username; + } + + public Long getSourceProjectId() { + return sourceProjectId; + } + + public String getSourceProjectName() { + return sourceProjectName; + } + + public Long getTargetProjectId() { + return targetProjectId; + } + + public String getTargetProjectName() { + return targetProjectName; + } + + public DSSOrchestratorInfo getOrchestrator() { + return orchestrator; + } + + public String getTargetOrchestratorName() { + return targetOrchestratorName; + } + + public String getWorkflowNodeSuffix() { + return workflowNodeSuffix; + } + + public DSSLabel getDssLabel() { + return dssLabel; + } + + public Workspace getWorkspace() { + return workspace; + } + + public Long getCopyTaskId() { + return copyTaskId; + } + + public String getInstanceName() { + return instanceName; + } +} + diff --git a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/vo/OrchestratorUnlockVo.java b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/vo/OrchestratorUnlockVo.java new file mode 100644 index 0000000000..90cb03bf21 --- /dev/null +++ b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/entity/vo/OrchestratorUnlockVo.java @@ -0,0 +1,46 @@ +package com.webank.wedatasphere.dss.orchestrator.server.entity.vo; + +public class OrchestratorUnlockVo extends CommonOrchestratorVo { + /** + * 已锁定用户 + */ + private String lockOwner; + /** + * 提示信息 + */ + private String confirmMessage; + /** + * 0:解锁成功,1:需用户二次确认解锁 + */ + private int status; + + public OrchestratorUnlockVo(String lockOwner, String confirmMessage, int status) { + this.lockOwner = lockOwner; + this.confirmMessage = confirmMessage; + this.status = status; + } + + public String getLockOwner() { + return lockOwner; + } + + public void setLockOwner(String lockOwner) { + this.lockOwner = lockOwner; + } + + public String getConfirmMessage() { + return confirmMessage; + } + + public void setConfirmMessage(String confirmMessage) { + this.confirmMessage = confirmMessage; + } + + public int getStatus() { + return status; + } + + public void setStatus(int status) { + this.status = status; + } +} diff --git a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/job/BatchClearCsTask.java b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/job/BatchClearCsTask.java new file mode 100644 index 0000000000..79b9c5c0cf --- /dev/null +++ b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/job/BatchClearCsTask.java @@ -0,0 +1,24 @@ +package com.webank.wedatasphere.dss.orchestrator.server.job; + +import com.webank.wedatasphere.dss.orchestrator.server.service.OrchestratorService; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.scheduling.annotation.EnableScheduling; +import org.springframework.scheduling.annotation.Scheduled; +import org.springframework.stereotype.Component; + +/** + * Description 定时清理ContextId的任务 + */ +@Component +@EnableScheduling +public class BatchClearCsTask { + + @Autowired + private OrchestratorService orchestratorService; + + @Scheduled(cron = "#{@getBatchClearCsTaskCron}") + public void batchClearCsTask(){ + orchestratorService.batchClearContextId(); + + } +} diff --git a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/job/CheckOrchestratorConversionJobTask.java b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/job/CheckOrchestratorConversionJobTask.java new file mode 100644 index 0000000000..572aa1c3fd --- /dev/null +++ b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/job/CheckOrchestratorConversionJobTask.java @@ -0,0 +1,89 @@ +package com.webank.wedatasphere.dss.orchestrator.server.job; + +import com.webank.wedatasphere.dss.common.alter.ExecuteAlter; +import com.webank.wedatasphere.dss.common.conf.DSSCommonConf; +import com.webank.wedatasphere.dss.common.entity.CustomAlter; +import com.webank.wedatasphere.dss.common.protocol.JobStatus; +import com.webank.wedatasphere.dss.common.utils.DSSCommonUtils; +import com.webank.wedatasphere.dss.orchestrator.db.dao.OrchestratorJobMapper; +import com.webank.wedatasphere.dss.orchestrator.common.entity.OrchestratorPublishJob; +import com.webank.wedatasphere.dss.sender.service.conf.DSSSenderServiceConf; +import org.apache.linkis.common.ServiceInstance; +import org.apache.linkis.rpc.Sender; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.scheduling.annotation.EnableScheduling; +import org.springframework.scheduling.annotation.Scheduled; +import org.springframework.stereotype.Component; + +import javax.annotation.PostConstruct; +import java.util.*; +import java.util.stream.Collectors; + +/** + * Description monitor workflow convert job if active + */ +@Component +@EnableScheduling +public class CheckOrchestratorConversionJobTask { + + private static final Logger LOGGER = LoggerFactory.getLogger(CheckOrchestratorConversionJobTask.class); + @Autowired + private OrchestratorJobMapper orchestratorJobMapper; + + @Autowired + private ExecuteAlter executeAlter; + + @PostConstruct + public void checkSelfExecuteTasks() { + LOGGER.info("CheckOrchestratorConversionJobTask: Start checking for tasks that are still running after instance exceptions"); + String thisInstance = Sender.getThisInstance(); + List maybeFailedJobs = orchestratorJobMapper.getPublishJobByJobStatuses + (Arrays.asList(JobStatus.Inited.getStatus(), JobStatus.Running.getStatus())); + List failedJobs = maybeFailedJobs.stream().filter(t -> Objects.equals(t.getInstanceName(), thisInstance)) + .peek(t -> { + t.setErrorMsg("执行发布的实例异常,请重新发布!"); + t.setStatus(JobStatus.Failed.getStatus()); + t.setUpdateTime(new Date()); + }).collect(Collectors.toList()); + if (failedJobs.size() > 0){ + LOGGER.warn("实例启动阶段,以下工作流发布任务因为该实例异常导致发布失败!{}", DSSCommonUtils.COMMON_GSON.toJson(failedJobs)); + orchestratorJobMapper.batchUpdatePublishJob(failedJobs); + } + } + + @Scheduled(cron = "#{@getCheckInstanceIsActiveCron}") + public void checkOrchestratorConversionJob() { + + ServiceInstance[] allActionInstances = Sender.getInstances(DSSSenderServiceConf.CURRENT_DSS_SERVER_NAME.getValue()); + List maybeFailedJobs = orchestratorJobMapper.getPublishJobByJobStatuses + (Arrays.asList(JobStatus.Inited.getStatus(), JobStatus.Running.getStatus())); + LOGGER.info("These tasks are maybe failed. " + DSSCommonUtils.COMMON_GSON.toJson(maybeFailedJobs)); + List activeInstance = Arrays.stream(allActionInstances).map(ServiceInstance::getInstance).collect(Collectors.toList()); + LOGGER.info("Active instances are " + activeInstance); + List failedJobs = new ArrayList<>(); + if (maybeFailedJobs.size() > 0) { + failedJobs = maybeFailedJobs.stream().filter(t -> !activeInstance.contains(t.getInstanceName())) + .peek(t -> { + t.setStatus(JobStatus.Failed.getStatus()); + t.setUpdateTime(new Date()); + t.setErrorMsg("执行发布的实例异常,请重新发布!"); + }).collect(Collectors.toList()); + } + + if (failedJobs.size() > 0) { + // update publish job status to failed + LOGGER.warn("以下工作流发布任务因为执行实例异常导致发布失败!{}", DSSCommonUtils.COMMON_GSON.toJson(failedJobs)); + orchestratorJobMapper.batchUpdatePublishJob(failedJobs); + List exceptionInstances = failedJobs.stream().map(OrchestratorPublishJob::getInstanceName).distinct().collect(Collectors.toList()); + List exceptionId = failedJobs.stream().map(OrchestratorPublishJob::getId).collect(Collectors.toList()); + failedJobs.clear(); + // send alter + CustomAlter customAlter = new CustomAlter("DSS exception of instance: " + exceptionInstances, + "以下Id工作流发布失败,请到表dss_orchestrator_job_info查看失败的工作流信息:" + exceptionId, + "1", DSSCommonConf.ALTER_RECEIVER.getValue()); + executeAlter.sendAlter(customAlter); + } + } +} diff --git a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/job/CheckOrchestratorCopyTask.java b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/job/CheckOrchestratorCopyTask.java new file mode 100644 index 0000000000..21e6db3d2b --- /dev/null +++ b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/job/CheckOrchestratorCopyTask.java @@ -0,0 +1,88 @@ +package com.webank.wedatasphere.dss.orchestrator.server.job; + +import com.webank.wedatasphere.dss.common.alter.ExecuteAlter; +import com.webank.wedatasphere.dss.common.conf.DSSCommonConf; +import com.webank.wedatasphere.dss.common.entity.CustomAlter; +import com.webank.wedatasphere.dss.common.utils.DSSCommonUtils; +import com.webank.wedatasphere.dss.orchestrator.common.entity.DSSOrchestratorCopyInfo; +import com.webank.wedatasphere.dss.orchestrator.db.dao.OrchestratorCopyJobMapper; +import com.webank.wedatasphere.dss.sender.service.conf.DSSSenderServiceConf; +import org.apache.linkis.common.ServiceInstance; +import org.apache.linkis.rpc.Sender; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.scheduling.annotation.EnableScheduling; +import org.springframework.scheduling.annotation.Scheduled; +import org.springframework.stereotype.Component; + +import javax.annotation.PostConstruct; +import java.util.*; +import java.util.stream.Collectors; + +/** + * Description monitor orchestrator copy job if active + */ +@Component +@EnableScheduling +public class CheckOrchestratorCopyTask { + + private static final Logger LOGGER = LoggerFactory.getLogger(CheckOrchestratorCopyTask.class); + @Autowired + private OrchestratorCopyJobMapper orchestratorCopyJobMapper; + + @Autowired + private ExecuteAlter executeAlter; + + @PostConstruct + public void checkSelfExecuteTasks() { + LOGGER.info("CheckOrchestratorCopyTask: Start checking for tasks that are still running after instance exceptions"); + String thisInstance = Sender.getThisInstance(); + List maybeFailedJobs = orchestratorCopyJobMapper.getRunningJob(); + List failedJobs = maybeFailedJobs.stream().filter(t -> Objects.equals(t.getInstanceName(), thisInstance)) + .peek(t -> { + t.setExceptionInfo("执行复制的实例异常,请稍后重试!"); + t.setStatus(0); + t.setEndTime(new Date()); + t.setIsCopying(0); + }).collect(Collectors.toList()); + if (failedJobs.size() > 0){ + LOGGER.warn("实例启动阶段,以下工作流复制任务因该实例异常导致失败!{}", DSSCommonUtils.COMMON_GSON.toJson(failedJobs)); + orchestratorCopyJobMapper.batchUpdateCopyJob(failedJobs); + } + } + + @Scheduled(cron = "#{@getCheckInstanceIsActiveCron}") + public void checkOrchestratorCopyJobTask() { + + ServiceInstance[] allActionInstances = Sender.getInstances(DSSSenderServiceConf.CURRENT_DSS_SERVER_NAME.getValue()); + List maybeFailedJobs = orchestratorCopyJobMapper.getRunningJob(); + LOGGER.info("These tasks are maybe failed. " + maybeFailedJobs.toString()); + List activeInstance = Arrays.stream(allActionInstances).map(ServiceInstance::getInstance).collect(Collectors.toList()); + LOGGER.info("Active instances are " + activeInstance); + List failedJobs = new ArrayList<>(); + if (maybeFailedJobs.size() > 0) { + failedJobs = maybeFailedJobs.stream().filter(t -> !activeInstance.contains(t.getInstanceName())) + .peek(t -> { + t.setExceptionInfo("执行复制的实例异常,请稍后重试!"); + t.setStatus(0); + t.setEndTime(new Date()); + t.setIsCopying(0); + }).collect(Collectors.toList()); + } + + // update copy job status to failed + if (failedJobs.size() > 0) { + LOGGER.warn("以下工作流复制任务因执行实例异常导致失败!{}", DSSCommonUtils.COMMON_GSON.toJson(failedJobs)); + orchestratorCopyJobMapper.batchUpdateCopyJob(failedJobs); + List exceptionInstances = failedJobs.stream().map(DSSOrchestratorCopyInfo::getInstanceName).distinct().collect(Collectors.toList()); + List exceptionId = failedJobs.stream().map(DSSOrchestratorCopyInfo::getId).collect(Collectors.toList()); + failedJobs.clear(); + // send alter + CustomAlter customAlter = new CustomAlter("DSS exception of instance: " + exceptionInstances, + "以下Id的工作流拷贝失败,请到表dss_orchestrator_copy_info查看失败的工作流信息:" + exceptionId, + "1", DSSCommonConf.ALTER_RECEIVER.getValue()); + executeAlter.sendAlter(customAlter); + } + } +} diff --git a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/job/OrchestratorCopyEnv.java b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/job/OrchestratorCopyEnv.java new file mode 100644 index 0000000000..6d751840f9 --- /dev/null +++ b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/job/OrchestratorCopyEnv.java @@ -0,0 +1,107 @@ +package com.webank.wedatasphere.dss.orchestrator.server.job; + +import com.webank.wedatasphere.dss.contextservice.service.ContextService; +import com.webank.wedatasphere.dss.common.service.BMLService; +import com.webank.wedatasphere.dss.orchestrator.db.dao.OrchestratorCopyJobMapper; +import com.webank.wedatasphere.dss.orchestrator.db.dao.OrchestratorMapper; +import com.webank.wedatasphere.dss.orchestrator.loader.OrchestratorManager; +import com.webank.wedatasphere.dss.orchestrator.publish.ExportDSSOrchestratorPlugin; +import com.webank.wedatasphere.dss.orchestrator.publish.ImportDSSOrchestratorPlugin; +import com.webank.wedatasphere.dss.orchestrator.server.service.OrchestratorFrameworkService; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.beans.factory.annotation.Qualifier; +import org.springframework.stereotype.Component; + +@Component +public class OrchestratorCopyEnv { + + @Autowired + private OrchestratorCopyJobMapper orchestratorCopyJobMapper; + + @Autowired + private OrchestratorMapper orchestratorMapper; + + @Autowired + @Qualifier("orchestratorBmlService") + private BMLService bmlService; + + @Autowired + private OrchestratorFrameworkService orchestratorFrameworkService; + + @Autowired + private ExportDSSOrchestratorPlugin exportDSSOrchestratorPlugin; + + @Autowired + private ImportDSSOrchestratorPlugin importDSSOrchestratorPlugin; + + @Autowired + private ContextService contextService; + + public ContextService getContextService() { + return contextService; + } + + public void setContextService(ContextService contextService) { + this.contextService = contextService; + } + + public OrchestratorManager getOrchestratorManager() { + return orchestratorManager; + } + + public void setOrchestratorManager(OrchestratorManager orchestratorManager) { + this.orchestratorManager = orchestratorManager; + } + + @Autowired + private OrchestratorManager orchestratorManager; + + public OrchestratorCopyJobMapper getOrchestratorCopyJobMapper() { + return orchestratorCopyJobMapper; + } + + public void setOrchestratorCopyJobMapper(OrchestratorCopyJobMapper orchestratorCopyJobMapper) { + this.orchestratorCopyJobMapper = orchestratorCopyJobMapper; + } + + public OrchestratorMapper getOrchestratorMapper() { + return orchestratorMapper; + } + + public void setOrchestratorMapper(OrchestratorMapper orchestratorMapper) { + this.orchestratorMapper = orchestratorMapper; + } + + public BMLService getBmlService() { + return bmlService; + } + + public void setBmlService(BMLService bmlService) { + this.bmlService = bmlService; + } + + + public OrchestratorFrameworkService getOrchestratorFrameworkService() { + return orchestratorFrameworkService; + } + + public void setOrchestratorFrameworkService(OrchestratorFrameworkService orchestratorFrameworkService) { + this.orchestratorFrameworkService = orchestratorFrameworkService; + } + + public ExportDSSOrchestratorPlugin getExportDSSOrchestratorPlugin() { + return exportDSSOrchestratorPlugin; + } + + public void setExportDSSOrchestratorPlugin(ExportDSSOrchestratorPlugin exportDSSOrchestratorPlugin) { + this.exportDSSOrchestratorPlugin = exportDSSOrchestratorPlugin; + } + + public ImportDSSOrchestratorPlugin getImportDSSOrchestratorPlugin() { + return importDSSOrchestratorPlugin; + } + + public void setImportDSSOrchestratorPlugin(ImportDSSOrchestratorPlugin importDSSOrchestratorPlugin) { + this.importDSSOrchestratorPlugin = importDSSOrchestratorPlugin; + } +} diff --git a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/job/OrchestratorCopyJob.java b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/job/OrchestratorCopyJob.java new file mode 100644 index 0000000000..33828ce96d --- /dev/null +++ b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/job/OrchestratorCopyJob.java @@ -0,0 +1,161 @@ +package com.webank.wedatasphere.dss.orchestrator.server.job; + +import com.google.common.collect.Lists; +import com.webank.wedatasphere.dss.common.exception.DSSErrorException; +import com.webank.wedatasphere.dss.common.label.DSSLabel; +import com.webank.wedatasphere.dss.common.utils.MapUtils; +import com.webank.wedatasphere.dss.orchestrator.common.entity.DSSOrchestratorCopyInfo; +import com.webank.wedatasphere.dss.orchestrator.common.entity.DSSOrchestratorInfo; +import com.webank.wedatasphere.dss.orchestrator.common.entity.DSSOrchestratorVersion; +import com.webank.wedatasphere.dss.orchestrator.common.ref.OrchestratorRefConstant; +import com.webank.wedatasphere.dss.orchestrator.core.DSSOrchestrator; +import com.webank.wedatasphere.dss.orchestrator.core.utils.OrchestratorUtils; +import com.webank.wedatasphere.dss.orchestrator.publish.utils.OrchestrationDevelopmentOperationUtils; +import com.webank.wedatasphere.dss.orchestrator.server.entity.vo.OrchestratorCopyVo; +import com.webank.wedatasphere.dss.standard.app.development.operation.RefCopyOperation; +import com.webank.wedatasphere.dss.standard.app.development.ref.CopyRequestRef; +import com.webank.wedatasphere.dss.standard.app.development.ref.RefJobContentResponseRef; +import com.webank.wedatasphere.dss.standard.app.development.service.RefCRUDService; +import com.webank.wedatasphere.dss.standard.app.development.standard.DevelopmentIntegrationStandard; +import com.webank.wedatasphere.dss.standard.app.sso.Workspace; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.BeanUtils; + +import java.util.*; + +public class OrchestratorCopyJob implements Runnable { + + private static final Logger LOGGER = LoggerFactory.getLogger(OrchestratorCopyJob.class); + + private OrchestratorCopyVo orchestratorCopyVo; + + protected OrchestratorCopyEnv orchestratorCopyEnv; + + private DSSOrchestratorCopyInfo orchestratorCopyInfo = new DSSOrchestratorCopyInfo(UUID.randomUUID().toString()); + + + @Override + public void run() { + try { + copyOrchestrator(); + } catch (Exception e) { + LOGGER.error("Copy {} for {} error ", orchestratorCopyVo.getOrchestrator().getName(), e); + } + } + + private void copyOrchestrator() { + //开始写入复制信息到编排复制任务历史表 + DSSOrchestratorInfo sourceOrchestrator = orchestratorCopyVo.getOrchestrator(); + orchestratorCopyInfo = new DSSOrchestratorCopyInfo(orchestratorCopyInfo.getId(), orchestratorCopyVo.getUsername(), sourceOrchestrator.getType(), orchestratorCopyVo.getWorkspace().getWorkspaceId(), + sourceOrchestrator.getId(), sourceOrchestrator.getName(), orchestratorCopyVo.getTargetOrchestratorName(), + orchestratorCopyVo.getSourceProjectName(), orchestratorCopyVo.getTargetProjectName(), orchestratorCopyVo.getWorkflowNodeSuffix(), + "Orchestrator server", 1, new Date(), new Date(), orchestratorCopyVo.getInstanceName()); + orchestratorCopyEnv.getOrchestratorCopyJobMapper().insertOrchestratorCopyInfo(orchestratorCopyInfo); + DSSOrchestratorVersion latestOrcVersion = orchestratorCopyEnv.getOrchestratorMapper().getLatestOrchestratorVersionById(sourceOrchestrator.getId()); + Long appId = latestOrcVersion.getAppId(); + DSSOrchestratorInfo newOrchestrator = new DSSOrchestratorInfo(); + BeanUtils.copyProperties(sourceOrchestrator, newOrchestrator); + newOrchestrator.setId(null); + newOrchestrator.setName(orchestratorCopyVo.getTargetOrchestratorName()); + newOrchestrator.setProjectId(orchestratorCopyVo.getTargetProjectId()); + newOrchestrator.setCreateTime(new Date()); + newOrchestrator.setCreator(orchestratorCopyVo.getUsername()); + newOrchestrator.setUUID(UUID.randomUUID().toString()); + newOrchestrator.setDesc("copy from " + sourceOrchestrator.getName()); + newOrchestrator.setUpdateTime(null); + newOrchestrator.setUpdateUser(null); + + try { + doOrchestratorCopy(orchestratorCopyVo.getUsername(), orchestratorCopyVo.getWorkspace(), newOrchestrator, + orchestratorCopyVo.getTargetProjectName(), Lists.newArrayList(orchestratorCopyVo.getDssLabel()), appId); + } catch (Exception e) { + //保存错误信息 + String errorMsg = "CopyOrcError: " + e.getMessage(); + if (errorMsg.length() > 1000) { + errorMsg = errorMsg.substring(0, 999); + } + orchestratorCopyInfo.setIsCopying(0); + orchestratorCopyInfo.setEndTime(new Date()); + orchestratorCopyInfo.setSuccessNode(Lists.newArrayList("ZERO")); + orchestratorCopyInfo.setStatus(0); + orchestratorCopyInfo.setExceptionInfo(errorMsg); + orchestratorCopyEnv.getOrchestratorCopyJobMapper().updateErrorMsgById(orchestratorCopyInfo); + LOGGER.error("copy orc error: sourceProjectName:{},targetProjectName:{}, sourceOrchestratorName:{}, targetOrchestratorName:{}. Exception:", + orchestratorCopyVo.getSourceProjectName(), orchestratorCopyVo.getTargetProjectName(), sourceOrchestrator.getName(), orchestratorCopyVo.getTargetOrchestratorName(), e); + throw new RuntimeException("error happened when copying orc.", e); + } + orchestratorCopyInfo.setIsCopying(0); + orchestratorCopyInfo.setEndTime(new Date()); + orchestratorCopyInfo.setSuccessNode(Lists.newArrayList("All")); + orchestratorCopyInfo.setStatus(1); + orchestratorCopyEnv.getOrchestratorCopyJobMapper().updateCopyStatus(orchestratorCopyInfo); + } + + private void doOrchestratorCopy(String userName, + Workspace workspace, + DSSOrchestratorInfo dssOrchestratorInfo, + String projectName, + List dssLabels, Long appId) throws DSSErrorException { + String copyInitVersion = OrchestratorUtils.generateNewCopyVersion(orchestratorCopyVo.getWorkflowNodeSuffix()); + String contextId = orchestratorCopyEnv.getContextService().createContextID(workspace.getWorkspaceName(), projectName, dssOrchestratorInfo.getName(), copyInitVersion, userName); + DSSOrchestratorVersion dssOrchestratorVersion = new DSSOrchestratorVersion(); + dssOrchestratorVersion.setComment("create with orchestrator copy"); + dssOrchestratorVersion.setProjectId(orchestratorCopyVo.getTargetProjectId()); + dssOrchestratorVersion.setSource("Orchestrator copy"); + dssOrchestratorVersion.setUpdater(userName); + dssOrchestratorVersion.setVersion(copyInitVersion); + dssOrchestratorVersion.setUpdateTime(new Date()); + dssOrchestratorVersion.setValidFlag(1); + dssOrchestratorVersion.setFormatContextId(contextId); + LOGGER.info("Create a new ContextId {} for orchestrator copy operation with name:{},version{}.", contextId, dssOrchestratorInfo.getName(), dssOrchestratorVersion.getVersion()); + DSSOrchestrator dssOrchestrator = orchestratorCopyEnv.getOrchestratorManager().getOrCreateOrchestrator(userName, workspace.getWorkspaceName(), dssOrchestratorInfo.getType(), + dssLabels); + RefJobContentResponseRef responseRef = OrchestrationDevelopmentOperationUtils.tryOrchestrationOperation(dssOrchestratorInfo, dssOrchestrator, userName, + workspace, dssLabels, DevelopmentIntegrationStandard::getRefCRUDService, + developmentService -> ((RefCRUDService) developmentService).getRefCopyOperation(), + dssContextRequestRef -> dssContextRequestRef.setContextId(contextId), + projectRefRequestRef -> projectRefRequestRef.setProjectName(projectName).setRefProjectId(dssOrchestratorVersion.getProjectId()), + (developmentOperation, developmentRequestRef) -> { + CopyRequestRef requestRef = (CopyRequestRef) developmentRequestRef; + Map refJobContent = MapUtils.newCommonMap(OrchestratorRefConstant.ORCHESTRATION_ID_KEY, appId, + OrchestratorRefConstant.ORCHESTRATION_DESCRIPTION, dssOrchestratorVersion.getComment()); + refJobContent.put(OrchestratorRefConstant.ORCHESTRATION_NAME, dssOrchestratorInfo.getName()); + refJobContent.put(OrchestratorRefConstant.ORCHESTRATION_NODE_SUFFIX, orchestratorCopyVo.getWorkflowNodeSuffix()); + requestRef.setNewVersion(dssOrchestratorVersion.getVersion()).setRefJobContent(refJobContent); + return ((RefCopyOperation) developmentOperation).copyRef(requestRef); + }, "copy"); + dssOrchestratorVersion.setAppId((Long) responseRef.getRefJobContent().get(OrchestratorRefConstant.ORCHESTRATION_ID_KEY)); + dssOrchestratorVersion.setContent((String) responseRef.getRefJobContent().get(OrchestratorRefConstant.ORCHESTRATION_CONTENT_KEY)); + + orchestratorCopyEnv.getOrchestratorMapper().addOrchestrator(dssOrchestratorInfo); + dssOrchestratorVersion.setOrchestratorId(dssOrchestratorInfo.getId()); + orchestratorCopyEnv.getOrchestratorMapper().addOrchestratorVersion(dssOrchestratorVersion); + } + + + public OrchestratorCopyVo getOrchestratorCopyVo() { + return orchestratorCopyVo; + } + + public void setOrchestratorCopyVo(OrchestratorCopyVo orchestratorCopyVo) { + this.orchestratorCopyVo = orchestratorCopyVo; + } + + public OrchestratorCopyEnv getOrchestratorCopyEnv() { + return orchestratorCopyEnv; + } + + public void setOrchestratorCopyEnv(OrchestratorCopyEnv orchestratorCopyEnv) { + this.orchestratorCopyEnv = orchestratorCopyEnv; + } + + public DSSOrchestratorCopyInfo getOrchestratorCopyInfo() { + return orchestratorCopyInfo; + } + + public void setOrchestratorCopyInfo(DSSOrchestratorCopyInfo orchestratorCopyInfo) { + this.orchestratorCopyInfo = orchestratorCopyInfo; + } +} diff --git a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/restful/DSSFrameworkOrchestratorRestful.java b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/restful/DSSFrameworkOrchestratorRestful.java index fbbaa17ebc..f982e439ea 100644 --- a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/restful/DSSFrameworkOrchestratorRestful.java +++ b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/restful/DSSFrameworkOrchestratorRestful.java @@ -17,26 +17,31 @@ package com.webank.wedatasphere.dss.orchestrator.server.restful; import com.webank.wedatasphere.dss.appconn.manager.utils.AppConnManagerUtils; +import com.webank.wedatasphere.dss.common.auditlog.OperateTypeEnum; +import com.webank.wedatasphere.dss.common.auditlog.TargetTypeEnum; +import com.webank.wedatasphere.dss.common.exception.DSSErrorException; +import com.webank.wedatasphere.dss.common.label.DSSLabel; +import com.webank.wedatasphere.dss.common.label.EnvDSSLabel; +import com.webank.wedatasphere.dss.common.utils.AuditLogUtils; +import com.webank.wedatasphere.dss.orchestrator.common.entity.DSSOrchestratorCopyInfo; +import com.webank.wedatasphere.dss.orchestrator.common.entity.OrchestratorVo; import com.webank.wedatasphere.dss.orchestrator.server.constant.OrchestratorLevelEnum; -import com.webank.wedatasphere.dss.orchestrator.server.entity.request.OrchestratorCreateRequest; -import com.webank.wedatasphere.dss.orchestrator.server.entity.request.OrchestratorDeleteRequest; -import com.webank.wedatasphere.dss.orchestrator.server.entity.request.OrchestratorModifyRequest; -import com.webank.wedatasphere.dss.orchestrator.server.entity.request.OrchestratorRequest; +import com.webank.wedatasphere.dss.orchestrator.server.entity.request.*; import com.webank.wedatasphere.dss.orchestrator.server.entity.vo.CommonOrchestratorVo; +import com.webank.wedatasphere.dss.orchestrator.server.entity.vo.OrchestratorCopyHistory; +import com.webank.wedatasphere.dss.orchestrator.server.entity.vo.OrchestratorUnlockVo; import com.webank.wedatasphere.dss.orchestrator.server.service.OrchestratorFrameworkService; import com.webank.wedatasphere.dss.orchestrator.server.service.OrchestratorService; import com.webank.wedatasphere.dss.standard.app.sso.Workspace; import com.webank.wedatasphere.dss.standard.sso.utils.SSOHelper; -import org.apache.commons.lang.exception.ExceptionUtils; +import org.apache.commons.math3.util.Pair; import org.apache.linkis.server.Message; import org.apache.linkis.server.security.SecurityFilter; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.web.bind.annotation.RequestBody; -import org.springframework.web.bind.annotation.RequestMapping; -import org.springframework.web.bind.annotation.RequestMethod; -import org.springframework.web.bind.annotation.RestController; +import org.springframework.web.bind.annotation.*; +import scala.tools.nsc.typechecker.Implicits; import javax.annotation.PostConstruct; import javax.servlet.http.HttpServletRequest; @@ -53,6 +58,8 @@ public class DSSFrameworkOrchestratorRestful { private OrchestratorFrameworkService orchestratorFrameworkService; @Autowired private OrchestratorService orchestratorService; + @Autowired + private HttpServletRequest httpServletRequest; @PostConstruct public void init() { @@ -62,33 +69,32 @@ public void init() { /** * 创建编排模式 * - * @param httpServletRequest * @param createRequest * @return */ @RequestMapping(path = "createOrchestrator", method = RequestMethod.POST) - public Message createOrchestrator(HttpServletRequest httpServletRequest, @RequestBody OrchestratorCreateRequest createRequest) throws Exception{ + public Message createOrchestrator(@RequestBody OrchestratorCreateRequest createRequest) throws Exception { String username = SecurityFilter.getLoginUsername(httpServletRequest); Workspace workspace = SSOHelper.getWorkspace(httpServletRequest); - LOGGER.info("workspace is {}", workspace.getWorkspaceName()); - //保存编排模式 //todo 先注释掉 // orchestratorService.saveOrchestrator(createRequest,null,username); CommonOrchestratorVo orchestratorVo = orchestratorFrameworkService.createOrchestrator(username, createRequest, workspace); + AuditLogUtils.printLog(username, workspace.getWorkspaceId(), workspace.getWorkspaceName(), TargetTypeEnum.ORCHESTRATOR, + orchestratorVo.getOrchestratorId(), createRequest.getOrchestratorName(), OperateTypeEnum.CREATE, createRequest); return Message.ok("创建工作流编排模式成功").data("orchestratorId", orchestratorVo.getOrchestratorId()); } /** * 查询所有的编排模式 * - * @param httpServletRequest * @param orchestratorRequest * @return */ @RequestMapping(path = "getAllOrchestrator", method = RequestMethod.POST) - public Message getAllOrchestrator(HttpServletRequest httpServletRequest, @RequestBody OrchestratorRequest orchestratorRequest) { + public Message getAllOrchestrator(@RequestBody OrchestratorRequest orchestratorRequest) { try { String username = SecurityFilter.getLoginUsername(httpServletRequest); + LOGGER.info("user {} begin to geyAllOrchestrator, requestBody:{}", username, orchestratorRequest); return Message.ok("获取编排模式成功").data("page", orchestratorService.getListByPage(orchestratorRequest, username)); } catch (Exception e) { LOGGER.error("getAllOrchestratorError ", e); @@ -99,39 +105,171 @@ public Message getAllOrchestrator(HttpServletRequest httpServletRequest, @Reques /** * 修改编排模式 * - * @param httpServletRequest * @param modifyRequest * @return */ @RequestMapping(path = "modifyOrchestrator", method = RequestMethod.POST) - public Message modifyOrchestrator(HttpServletRequest httpServletRequest, @RequestBody OrchestratorModifyRequest modifyRequest) throws Exception{ + public Message modifyOrchestrator(@RequestBody OrchestratorModifyRequest modifyRequest) throws Exception { String username = SecurityFilter.getLoginUsername(httpServletRequest); Workspace workspace = SSOHelper.getWorkspace(httpServletRequest); - LOGGER.info("workspace is {}", workspace.getWorkspaceName()); + if (orchestratorFrameworkService.getOrchestratorCopyStatus(modifyRequest.getId())) { + return Message.error("当前工作流正在被复制,不允许编辑"); + } CommonOrchestratorVo orchestratorVo = orchestratorFrameworkService.modifyOrchestrator(username, modifyRequest, workspace); + AuditLogUtils.printLog(username, workspace.getWorkspaceId(), workspace.getWorkspaceName(), TargetTypeEnum.ORCHESTRATOR, + orchestratorVo.getOrchestratorId(), modifyRequest.getOrchestratorName(), OperateTypeEnum.UPDATE, modifyRequest); return Message.ok("修改工作流编排模式成功").data("orchestratorId", orchestratorVo.getOrchestratorId()); } /** * 删除编排模式 * - * @param httpServletRequest * @param deleteRequest * @return */ @RequestMapping(path = "deleteOrchestrator", method = RequestMethod.POST) - public Message deleteOrchestrator(HttpServletRequest httpServletRequest, @RequestBody OrchestratorDeleteRequest deleteRequest) throws Exception { + public Message deleteOrchestrator(@RequestBody OrchestratorDeleteRequest deleteRequest) throws Exception { String username = SecurityFilter.getLoginUsername(httpServletRequest); Workspace workspace = SSOHelper.getWorkspace(httpServletRequest); - orchestratorFrameworkService.deleteOrchestrator(username, deleteRequest, workspace); + if (orchestratorFrameworkService.getOrchestratorCopyStatus(deleteRequest.getId())) { + return Message.error("当前工作流正在被复制,不允许删除"); + } + CommonOrchestratorVo orchestratorVo = orchestratorFrameworkService.deleteOrchestrator(username, deleteRequest, workspace); + AuditLogUtils.printLog(username, workspace.getWorkspaceId(), workspace.getWorkspaceName(), TargetTypeEnum.ORCHESTRATOR, + orchestratorVo.getOrchestratorId(), orchestratorVo.getOrchestratorName(), OperateTypeEnum.DELETE, deleteRequest); return Message.ok("删除工作流编排模式成功"); } + /** + * 复制编排模式 + * + * @param orchestratorCopyRequest + * @return + * @throws Exception + */ + @RequestMapping(path = "copyOrchestrator", method = RequestMethod.POST) + public Message copyOrchestrator(@RequestBody OrchestratorCopyRequest orchestratorCopyRequest) throws Exception { + String username = SecurityFilter.getLoginUsername(httpServletRequest); + Workspace workspace = SSOHelper.getWorkspace(httpServletRequest); + + if (orchestratorFrameworkService.getOrchestratorCopyStatus(orchestratorCopyRequest.getSourceOrchestratorId())) { + return Message.error("当前工作流正在被复制,不允许再次复制"); + } + + String copyJobId = orchestratorFrameworkService.copyOrchestrator(username, orchestratorCopyRequest, workspace); + AuditLogUtils.printLog(username, workspace.getWorkspaceId(), workspace.getWorkspaceName(), TargetTypeEnum.ORCHESTRATOR, + orchestratorCopyRequest.getSourceOrchestratorId(), orchestratorCopyRequest.getSourceOrchestratorName(), OperateTypeEnum.COPY, orchestratorCopyRequest); + + return Message.ok("复制工作流已经开始,正在后台复制中,复制状态可以从复制历史查看...").data("copyJobId", copyJobId); + } + + /** + * 获取编排复制任务状态 + * + * @return + * @throws Exception + */ + @RequestMapping(path = "/{id}/copyInfo", method = RequestMethod.GET) + public Message getCopyJobStatus(@PathVariable("id") String copyInfoId) throws Exception { + return Message.ok("获取编排复制任务状态成功").data("orchestratorCopyInfo", orchestratorFrameworkService.getOrchestratorCopyInfoById(copyInfoId)); + } + + /** + * 查询锁信息和解锁操作共用此接口。 + * confirmDelete参数为false时,返回编辑锁的owner等信息。confirmDelete为true时,执行解锁操作。 + * + * @return + * @throws Exception + */ + @RequestMapping(path = "unlockOrchestrator", method = RequestMethod.POST) + public Message unlockOrchestrator(@RequestBody OrchestratorUnlockRequest unlockRequest) throws Exception { + String username = SecurityFilter.getLoginUsername(httpServletRequest); + Workspace workspace = SSOHelper.getWorkspace(httpServletRequest); + OrchestratorUnlockVo orchestratorVo = orchestratorFrameworkService.unlockOrchestrator(username, workspace, unlockRequest); + AuditLogUtils.printLog(username, workspace.getWorkspaceId(), workspace.getWorkspaceName(), TargetTypeEnum.ORCHESTRATOR, + orchestratorVo.getOrchestratorId(), orchestratorVo.getOrchestratorName(), OperateTypeEnum.DELETE, unlockRequest); + return Message.ok().data("status", orchestratorVo.getStatus()) + .data("confirmMessage", orchestratorVo.getConfirmMessage()) + .data("lockOwner", orchestratorVo.getLockOwner()); + } + + /** + * 查看编排复制历史 + * + * @param orchestratorId + * @return + */ + @RequestMapping(path = "listOrchestratorCopyHistory", method = RequestMethod.GET) + public Message listOrchestratorCopyHistory(@RequestParam(required = false, name = "orchestratorId") Long orchestratorId, + @RequestParam(required = false, name = "currentPage") Integer currentPage, + @RequestParam(required = false, name = "pageSize") Integer pageSize) throws Exception { + String username = SecurityFilter.getLoginUsername(httpServletRequest); + Workspace workspace = SSOHelper.getWorkspace(httpServletRequest); + if (orchestratorId == null) { + return Message.error("请到父工作流查看复制历史信息!"); + } + Pair> result = orchestratorFrameworkService.getOrchestratorCopyHistory(username, workspace, orchestratorId, currentPage, pageSize); + return Message.ok("查找工作流复制历史成功").data("copyJobHistory", result.getSecond()).data("total", result.getFirst()); + } + @RequestMapping(path = "orchestratorLevels", method = RequestMethod.GET) - public Message getOrchestratorLevels(HttpServletRequest httpServletRequest) { + public Message getOrchestratorLevels() { String username = SecurityFilter.getLoginUsername(httpServletRequest); Workspace workspace = SSOHelper.getWorkspace(httpServletRequest); List levels = Arrays.asList(OrchestratorLevelEnum.values()); + LOGGER.info("user {} try to get OrchestratorLevels, workspaceId:{}, result:{}", username, workspace.getWorkspaceId(), levels); return Message.ok("获取编排重要级别列表成功").data("orchestratorLevels", levels); } + + @RequestMapping(path = "rollbackOrchestrator", method = RequestMethod.POST) + public Message rollbackOrchestrator(HttpServletRequest request, @RequestBody RollbackOrchestratorRequest rollbackOrchestratorRequest) { + String username = SecurityFilter.getLoginUsername(request); + Long orchestratorId = rollbackOrchestratorRequest.getOrchestratorId(); + String version = rollbackOrchestratorRequest.getVersion(); + Long projectId = rollbackOrchestratorRequest.getProjectId(); + String projectName = rollbackOrchestratorRequest.getProjectName(); + Workspace workspace = SSOHelper.getWorkspace(request); + DSSLabel envDSSLabel = new EnvDSSLabel(rollbackOrchestratorRequest.getLabels().getRoute()); + try { + LOGGER.info("user {} begin to rollbackOrchestrator, params:{}", username, rollbackOrchestratorRequest); + String newVersion = orchestratorService.rollbackOrchestrator(username, projectId, projectName, orchestratorId, version, envDSSLabel, workspace); + Message message = Message.ok("回滚版本成功").data("newVersion", newVersion); + return message; + } catch (final Throwable t) { + LOGGER.error("Failed to rollback orchestrator for user {} orchestratorId {}, projectId {} version {}", + username, orchestratorId, projectId, version, t); + return Message.error("回滚工作流版本失败"); + } + } + + @RequestMapping(path = "openOrchestrator", method = RequestMethod.POST) + public Message openOrchestrator(HttpServletRequest req, @RequestBody OpenOrchestratorRequest openOrchestratorRequest) throws Exception { + String openUrl = ""; + String userName = SecurityFilter.getLoginUsername(req); + Workspace workspace = SSOHelper.getWorkspace(req); + List dssLabelList = Arrays.asList(new EnvDSSLabel(openOrchestratorRequest.getLabels().getRoute())); + Long orchestratorId = openOrchestratorRequest.getOrchestratorId(); + if (orchestratorFrameworkService.getOrchestratorCopyStatus(orchestratorId)){ + return Message.error("当前工作流正在被复制,不允许被打开,请稍后"); + } + LOGGER.info("user {} try to openOrchestrator, params:{}", userName, openOrchestratorRequest); + openUrl = orchestratorService.openOrchestrator(userName, workspace, orchestratorId, dssLabelList); + OrchestratorVo orchestratorVo = orchestratorService.getOrchestratorVoById(orchestratorId); + LOGGER.info("open url is {}, orcId is {}, dssLabels is {}", openUrl, orchestratorId, dssLabelList); + return Message.ok().data("OrchestratorOpenUrl", openUrl). + data("OrchestratorVo", orchestratorVo); + } + + /** + * 获取编排模式下的所有版本号 + * + * @param queryOrchestratorVersion + * @return + * @throws Exception + */ + @RequestMapping(path = "getVersionByOrchestratorId", method = RequestMethod.POST) + public Message getVersionByOrchestratorId(@RequestBody QueryOrchestratorVersion queryOrchestratorVersion) throws Exception { + List list = orchestratorService.getVersionByOrchestratorId(queryOrchestratorVersion.getOrchestratorId()); + return Message.ok().data("list", list); + } } diff --git a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/restful/OrchestratorIERestful.java b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/restful/OrchestratorIERestful.java index 31293e1f4b..29f36fd914 100644 --- a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/restful/OrchestratorIERestful.java +++ b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/restful/OrchestratorIERestful.java @@ -16,19 +16,25 @@ package com.webank.wedatasphere.dss.orchestrator.server.restful; +import com.webank.wedatasphere.dss.common.auditlog.OperateTypeEnum; +import com.webank.wedatasphere.dss.common.auditlog.TargetTypeEnum; +import com.webank.wedatasphere.dss.common.entity.BmlResource; import com.webank.wedatasphere.dss.common.exception.DSSErrorException; import com.webank.wedatasphere.dss.common.exception.DSSRuntimeException; import com.webank.wedatasphere.dss.common.label.DSSLabel; import com.webank.wedatasphere.dss.common.label.EnvDSSLabel; import com.webank.wedatasphere.dss.common.label.LabelKeyConvertor; +import com.webank.wedatasphere.dss.common.utils.AuditLogUtils; import com.webank.wedatasphere.dss.common.utils.DSSCommonUtils; +import com.webank.wedatasphere.dss.orchestrator.common.entity.DSSOrchestratorVersion; import com.webank.wedatasphere.dss.orchestrator.common.entity.OrchestratorVo; import com.webank.wedatasphere.dss.orchestrator.common.protocol.RequestImportOrchestrator; import com.webank.wedatasphere.dss.orchestrator.core.DSSOrchestratorContext; -import com.webank.wedatasphere.dss.orchestrator.core.service.BMLService; +import com.webank.wedatasphere.dss.common.service.BMLService; import com.webank.wedatasphere.dss.orchestrator.publish.ExportDSSOrchestratorPlugin; import com.webank.wedatasphere.dss.orchestrator.publish.ImportDSSOrchestratorPlugin; import com.webank.wedatasphere.dss.orchestrator.server.service.OrchestratorService; +import com.webank.wedatasphere.dss.orchestrator.server.service.impl.OrchestratorFrameworkServiceImpl; import com.webank.wedatasphere.dss.standard.app.sso.Workspace; import com.webank.wedatasphere.dss.standard.sso.utils.SSOHelper; import org.apache.commons.io.IOUtils; @@ -40,6 +46,7 @@ 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.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.RequestMethod; import org.springframework.web.bind.annotation.RequestParam; @@ -59,6 +66,7 @@ public class OrchestratorIERestful { private static final Logger logger = LoggerFactory.getLogger(OrchestratorIERestful.class); @Autowired + @Qualifier("orchestratorBmlService") private BMLService bmlService; @Autowired private FsService fsService; @@ -67,6 +75,7 @@ public class OrchestratorIERestful { @Autowired private DSSOrchestratorContext orchestratorContext; + @RequestMapping(path ="importOrchestratorFile", method = RequestMethod.POST) public Message importOrcFile(HttpServletRequest req, @RequestParam(required = false, name = "projectName") String projectName, @@ -99,18 +108,20 @@ public Message importOrcFile(HttpServletRequest req, fileName = packageUri.substring(packageUri.lastIndexOf('/') + 1); } - Map resultMap = bmlService.upload(userName, inputStream, fileName, projectName); - Long importOrcId; + BmlResource resultMap = bmlService.upload(userName, inputStream, fileName, projectName); + DSSOrchestratorVersion dssOrchestratorVersion; try { RequestImportOrchestrator importRequest = new RequestImportOrchestrator(userName, projectName, - projectID, resultMap.get("resourceId").toString(), - resultMap.get("version").toString(), null, dssLabelList, workspace); - importOrcId = orchestratorContext.getDSSOrchestratorPlugin(ImportDSSOrchestratorPlugin.class).importOrchestrator(importRequest); + projectID, resultMap.getResourceId(), + resultMap.getVersion(), null, dssLabelList, workspace); + dssOrchestratorVersion = orchestratorContext.getDSSOrchestratorPlugin(ImportDSSOrchestratorPlugin.class).importOrchestrator(importRequest); + AuditLogUtils.printLog(userName, workspace.getWorkspaceId(), workspace.getWorkspaceName(), TargetTypeEnum.ORCHESTRATOR, + projectID, projectName, OperateTypeEnum.CREATE, importRequest); } catch (Exception e) { logger.error("Import orchestrator failed for ", e); throw new DSSErrorException(100789, "Import orchestrator failed for " + e.getMessage()); } - return Message.ok().data("importOrcId", importOrcId); + return Message.ok().data("importOrcId", dssOrchestratorVersion.getOrchestratorId()); } @RequestMapping(path ="exportOrchestrator", method = RequestMethod.GET) @@ -130,7 +141,7 @@ public void exportOrcFile(HttpServletRequest req, Workspace workspace = SSOHelper.getWorkspace(req); String userName = SecurityFilter.getLoginUsername(req); List dssLabelList = getDSSLabelList(labels); - Map res = null; + BmlResource res; OrchestratorVo orchestratorVo; if (orcVersionId != null) { orchestratorVo = orchestratorService.getOrchestratorVoByIdAndOrcVersionId(orchestratorId, orcVersionId); @@ -138,18 +149,22 @@ public void exportOrcFile(HttpServletRequest req, orchestratorVo = orchestratorService.getOrchestratorVoById(orchestratorId); } orcVersionId = orchestratorVo.getDssOrchestratorVersion().getId(); + //鉴权 + OrchestratorFrameworkServiceImpl.validateOperation(orchestratorVo.getDssOrchestratorInfo().getProjectId(), userName); logger.info("export orchestrator orchestratorId " + orchestratorId + ",orcVersionId:" + orcVersionId); try { res = orchestratorContext.getDSSOrchestratorPlugin(ExportDSSOrchestratorPlugin.class).exportOrchestrator(userName, - orchestratorId, orcVersionId, projectName, dssLabelList, addOrcVersion, workspace); + orchestratorId, orcVersionId, projectName, dssLabelList, addOrcVersion, workspace).getBmlResource(); + AuditLogUtils.printLog(userName, workspace.getWorkspaceId(), workspace.getWorkspaceName(), TargetTypeEnum.ORCHESTRATOR, + orcVersionId, orchestratorVo.getDssOrchestratorInfo().getName(), OperateTypeEnum.UPDATE, orchestratorVo); } catch (Exception e) { logger.error("export orchestrator failed for ", e); throw new DSSErrorException(100789, "export orchestrator failed for " + e.getMessage()); } if (null != res) { Map downRes = bmlService.download(userName, - res.get("resourceId").toString(), - res.get("version").toString()); + res.getResourceId(), + res.getVersion()); InputStream inputStream = (InputStream) downRes.get("is"); try { diff --git a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/restful/OrchestratorRestful.java b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/restful/OrchestratorRestful.java deleted file mode 100644 index 304aec3838..0000000000 --- a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/restful/OrchestratorRestful.java +++ /dev/null @@ -1,96 +0,0 @@ -/* - * Copyright 2019 WeBank - * 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. - * - */ - -package com.webank.wedatasphere.dss.orchestrator.server.restful; - -import com.webank.wedatasphere.dss.common.label.DSSLabel; -import com.webank.wedatasphere.dss.common.label.EnvDSSLabel; -import com.webank.wedatasphere.dss.orchestrator.common.entity.OrchestratorVo; -import com.webank.wedatasphere.dss.orchestrator.server.entity.request.OpenOrchestratorRequest; -import com.webank.wedatasphere.dss.orchestrator.server.entity.request.QueryOrchestratorVersion; -import com.webank.wedatasphere.dss.orchestrator.server.entity.request.RollbackOrchestratorRequest; -import com.webank.wedatasphere.dss.orchestrator.server.service.OrchestratorService; -import com.webank.wedatasphere.dss.standard.app.sso.Workspace; -import com.webank.wedatasphere.dss.standard.sso.utils.SSOHelper; -import org.apache.linkis.server.Message; -import org.apache.linkis.server.security.SecurityFilter; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.web.bind.annotation.RequestBody; -import org.springframework.web.bind.annotation.RequestMapping; -import org.springframework.web.bind.annotation.RequestMethod; -import org.springframework.web.bind.annotation.RestController; - -import javax.servlet.http.HttpServletRequest; -import java.util.Arrays; -import java.util.List; - - -@RequestMapping(path = "/dss/framework/orchestrator", produces = {"application/json"}) -@RestController -public class OrchestratorRestful { - private final static Logger LOGGER = LoggerFactory.getLogger(OrchestratorRestful.class); - @Autowired - OrchestratorService orchestratorService; - - @RequestMapping(path = "rollbackOrchestrator", method = RequestMethod.POST) - public Message rollbackOrchestrator(HttpServletRequest request, @RequestBody RollbackOrchestratorRequest rollbackOrchestratorRequest) { - String username = SecurityFilter.getLoginUsername(request); - Long orchestratorId = rollbackOrchestratorRequest.getOrchestratorId(); - String version = rollbackOrchestratorRequest.getVersion(); - Long projectId = rollbackOrchestratorRequest.getProjectId(); - String projectName = rollbackOrchestratorRequest.getProjectName(); - Workspace workspace = SSOHelper.getWorkspace(request); - DSSLabel envDSSLabel = new EnvDSSLabel(rollbackOrchestratorRequest.getLabels().getRoute()); - try { - String newVersion = orchestratorService.rollbackOrchestrator(username, projectId, projectName, orchestratorId, version, envDSSLabel, workspace); - Message message = Message.ok("回滚版本成功").data("newVersion", newVersion); - return message; - } catch (final Throwable t) { - LOGGER.error("Failed to rollback orchestrator for user {} orchestratorId {}, projectId {} version {}", - username, orchestratorId, projectId, version, t); - return Message.error("回滚工作流版本失败"); - } - } - - @RequestMapping(path = "openOrchestrator", method = RequestMethod.POST) - public Message openOrchestrator(HttpServletRequest req, @RequestBody OpenOrchestratorRequest openOrchestratorRequest) throws Exception { - String openUrl = ""; - String userName = SecurityFilter.getLoginUsername(req); - Workspace workspace = SSOHelper.getWorkspace(req); - List dssLabelList = Arrays.asList(new EnvDSSLabel(openOrchestratorRequest.getLabels().getRoute())); - Long orchestratorId = openOrchestratorRequest.getOrchestratorId(); - openUrl = orchestratorService.openOrchestrator(userName, workspace, orchestratorId, dssLabelList); - OrchestratorVo orchestratorVo = orchestratorService.getOrchestratorVoById(orchestratorId); - LOGGER.info("open url is {}, orcId is {}, dssLabels is {}", openUrl, orchestratorId, dssLabelList); - return Message.ok().data("OrchestratorOpenUrl", openUrl). - data("OrchestratorVo", orchestratorVo); - } - - /** - * 获取编排模式下的所有版本号 - * - * @param queryOrchestratorVersion - * @return - * @throws Exception - */ - @RequestMapping(path = "getVersionByOrchestratorId", method = RequestMethod.POST) - public Message getVersionByOrchestratorId(@RequestBody QueryOrchestratorVersion queryOrchestratorVersion) throws Exception { - List list = orchestratorService.getVersionByOrchestratorId(queryOrchestratorVersion.getOrchestratorId()); - return Message.ok().data("list", list); - } -} diff --git a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/service/OrchestratorFrameworkService.java b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/service/OrchestratorFrameworkService.java index e939b900a0..3d56192a91 100644 --- a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/service/OrchestratorFrameworkService.java +++ b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/service/OrchestratorFrameworkService.java @@ -16,11 +16,16 @@ package com.webank.wedatasphere.dss.orchestrator.server.service; -import com.webank.wedatasphere.dss.orchestrator.server.entity.request.OrchestratorCreateRequest; -import com.webank.wedatasphere.dss.orchestrator.server.entity.request.OrchestratorDeleteRequest; -import com.webank.wedatasphere.dss.orchestrator.server.entity.request.OrchestratorModifyRequest; +import com.webank.wedatasphere.dss.common.exception.DSSErrorException; +import com.webank.wedatasphere.dss.orchestrator.common.entity.DSSOrchestratorCopyInfo; +import com.webank.wedatasphere.dss.orchestrator.server.entity.request.*; import com.webank.wedatasphere.dss.orchestrator.server.entity.vo.CommonOrchestratorVo; +import com.webank.wedatasphere.dss.orchestrator.server.entity.vo.OrchestratorCopyHistory; +import com.webank.wedatasphere.dss.orchestrator.server.entity.vo.OrchestratorUnlockVo; import com.webank.wedatasphere.dss.standard.app.sso.Workspace; +import org.apache.commons.math3.util.Pair; + +import java.util.List; public interface OrchestratorFrameworkService { @@ -31,4 +36,13 @@ public interface OrchestratorFrameworkService { CommonOrchestratorVo deleteOrchestrator(String username, OrchestratorDeleteRequest orchestratorDeleteRequest, Workspace workspace) throws Exception; + OrchestratorUnlockVo unlockOrchestrator(String username, Workspace workspace, OrchestratorUnlockRequest request) throws DSSErrorException; + + String copyOrchestrator(String username, OrchestratorCopyRequest orchestratorCopyRequest, Workspace workspace) throws Exception; + + Pair> getOrchestratorCopyHistory(String username, Workspace workspace, Long orchestratorId, Integer currentPage, Integer pageSize) throws Exception; + + Boolean getOrchestratorCopyStatus(Long sourceOrchestratorId); + + DSSOrchestratorCopyInfo getOrchestratorCopyInfoById(String copyInfoId); } diff --git a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/service/OrchestratorPluginService.java b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/service/OrchestratorPluginService.java index f329849593..e15ae50503 100644 --- a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/service/OrchestratorPluginService.java +++ b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/service/OrchestratorPluginService.java @@ -21,6 +21,9 @@ public interface OrchestratorPluginService { + /** + * 把编排发布到调度系统,并处理一些本地db的元数据 + */ ResponseConvertOrchestrator convertOrchestration(RequestFrameworkConvertOrchestration requestConversionOrchestration); diff --git a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/service/OrchestratorService.java b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/service/OrchestratorService.java index a26fd2830f..711b79f65d 100644 --- a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/service/OrchestratorService.java +++ b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/service/OrchestratorService.java @@ -17,6 +17,7 @@ package com.webank.wedatasphere.dss.orchestrator.server.service; +import com.webank.wedatasphere.dss.common.exception.DSSErrorException; import com.webank.wedatasphere.dss.common.label.DSSLabel; import com.webank.wedatasphere.dss.framework.common.exception.DSSFrameworkErrorException; import com.webank.wedatasphere.dss.orchestrator.common.entity.DSSOrchestratorInfo; @@ -27,6 +28,7 @@ import com.webank.wedatasphere.dss.orchestrator.server.entity.request.OrchestratorModifyRequest; import com.webank.wedatasphere.dss.orchestrator.server.entity.request.OrchestratorRequest; import com.webank.wedatasphere.dss.orchestrator.server.entity.vo.OrchestratorBaseInfo; +import com.webank.wedatasphere.dss.orchestrator.server.entity.vo.OrchestratorUnlockVo; import com.webank.wedatasphere.dss.standard.app.sso.Workspace; import java.util.List; @@ -69,6 +71,21 @@ void deleteOrchestrator(String userName, Long orchestratorInfoId, List dssLabels) throws Exception; + /** + * 解锁编排对应的工作流 + * + * @param orchestratorInfoId 编排id + * @param confirmDelete 是否确认删除编辑锁 + * @param dssLabels + * @throws DSSErrorException + * @return + */ + OrchestratorUnlockVo unlockOrchestrator(String userName, + Workspace workspace, + String projectName, + Long orchestratorInfoId, + Boolean confirmDelete, + List dssLabels) throws DSSErrorException; /** * 返回一个编排,包含编排的基本信息和最新版本信息 @@ -119,4 +136,7 @@ String rollbackOrchestrator(String username, Long projectId, String projectName, List getListByPage(OrchestratorRequest orchestratorRequest, String username); ResponseOrchestratorInfos queryOrchestratorInfos(RequestOrchestratorInfos requestOrchestratorInfos); + + void batchClearContextId(); + } diff --git a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/service/impl/OrchestratorFrameworkServiceImpl.java b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/service/impl/OrchestratorFrameworkServiceImpl.java index 7d5c7701c9..5e9cbef17b 100644 --- a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/service/impl/OrchestratorFrameworkServiceImpl.java +++ b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/service/impl/OrchestratorFrameworkServiceImpl.java @@ -16,7 +16,8 @@ package com.webank.wedatasphere.dss.orchestrator.server.service.impl; -import com.webank.wedatasphere.dss.appconn.core.AppConn; +import com.google.common.collect.Lists; +import com.google.common.util.concurrent.ThreadFactoryBuilder; import com.webank.wedatasphere.dss.appconn.scheduler.SchedulerAppConn; import com.webank.wedatasphere.dss.appconn.scheduler.structure.orchestration.OrchestrationCreationOperation; import com.webank.wedatasphere.dss.appconn.scheduler.structure.orchestration.OrchestrationDeletionOperation; @@ -29,10 +30,15 @@ import com.webank.wedatasphere.dss.appconn.scheduler.utils.OrchestrationOperationUtils; import com.webank.wedatasphere.dss.common.constant.project.ProjectUserPrivEnum; import com.webank.wedatasphere.dss.common.entity.project.DSSProject; +import com.webank.wedatasphere.dss.common.exception.DSSErrorException; import com.webank.wedatasphere.dss.common.label.DSSLabel; import com.webank.wedatasphere.dss.common.label.EnvDSSLabel; import com.webank.wedatasphere.dss.common.protocol.project.*; +import com.webank.wedatasphere.dss.common.utils.DSSCommonUtils; import com.webank.wedatasphere.dss.common.utils.DSSExceptionUtils; +import com.webank.wedatasphere.dss.common.utils.RpcAskUtils; +import com.webank.wedatasphere.dss.framework.common.exception.DSSFrameworkErrorException; +import com.webank.wedatasphere.dss.orchestrator.common.entity.DSSOrchestratorCopyInfo; import com.webank.wedatasphere.dss.orchestrator.common.entity.DSSOrchestratorInfo; import com.webank.wedatasphere.dss.orchestrator.common.entity.DSSOrchestratorRefOrchestration; import com.webank.wedatasphere.dss.orchestrator.common.entity.OrchestratorVo; @@ -41,13 +47,16 @@ import com.webank.wedatasphere.dss.orchestrator.core.type.DSSOrchestratorRelation; import com.webank.wedatasphere.dss.orchestrator.core.type.DSSOrchestratorRelationManager; import com.webank.wedatasphere.dss.orchestrator.core.utils.OrchestratorUtils; +import com.webank.wedatasphere.dss.orchestrator.db.dao.OrchestratorCopyJobMapper; import com.webank.wedatasphere.dss.orchestrator.db.dao.OrchestratorMapper; -import com.webank.wedatasphere.dss.orchestrator.loader.LinkedAppConnResolver; import com.webank.wedatasphere.dss.orchestrator.loader.OrchestratorManager; -import com.webank.wedatasphere.dss.orchestrator.server.entity.request.OrchestratorCreateRequest; -import com.webank.wedatasphere.dss.orchestrator.server.entity.request.OrchestratorDeleteRequest; -import com.webank.wedatasphere.dss.orchestrator.server.entity.request.OrchestratorModifyRequest; +import com.webank.wedatasphere.dss.orchestrator.server.entity.request.*; import com.webank.wedatasphere.dss.orchestrator.server.entity.vo.CommonOrchestratorVo; +import com.webank.wedatasphere.dss.orchestrator.server.entity.vo.OrchestratorCopyHistory; +import com.webank.wedatasphere.dss.orchestrator.server.entity.vo.OrchestratorCopyVo; +import com.webank.wedatasphere.dss.orchestrator.server.entity.vo.OrchestratorUnlockVo; +import com.webank.wedatasphere.dss.orchestrator.server.job.OrchestratorCopyEnv; +import com.webank.wedatasphere.dss.orchestrator.server.job.OrchestratorCopyJob; import com.webank.wedatasphere.dss.orchestrator.server.service.OrchestratorFrameworkService; import com.webank.wedatasphere.dss.orchestrator.server.service.OrchestratorService; import com.webank.wedatasphere.dss.sender.service.DSSSenderServiceFactory; @@ -58,15 +67,19 @@ import com.webank.wedatasphere.dss.standard.common.entity.ref.ResponseRef; import com.webank.wedatasphere.dss.standard.common.exception.operation.ExternalOperationWarnException; import org.apache.commons.collections.CollectionUtils; +import org.apache.commons.lang.StringUtils; +import org.apache.commons.math3.util.Pair; import org.apache.linkis.protocol.util.ImmutablePair; +import org.apache.linkis.rpc.Sender; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.beans.factory.annotation.Autowired; -import java.util.Collections; -import java.util.Date; -import java.util.List; -import java.util.Optional; +import java.time.LocalDateTime; +import java.time.ZoneId; +import java.time.format.DateTimeFormatter; +import java.util.*; +import java.util.concurrent.*; import java.util.function.BiFunction; import java.util.function.Function; import java.util.stream.Collectors; @@ -76,14 +89,30 @@ public class OrchestratorFrameworkServiceImpl implements OrchestratorFrameworkSe protected final Logger LOGGER = LoggerFactory.getLogger(getClass()); + private static final DateTimeFormatter DTF = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss"); + @Autowired private OrchestratorMapper orchestratorMapper; @Autowired - private OrchestratorService newOrchestratorService; + private OrchestratorService orchestratorService; @Autowired private OrchestratorManager orchestratorManager; @Autowired - private LinkedAppConnResolver linkedAppConnResolver; + private OrchestratorCopyJobMapper orchestratorCopyJobMapper; + @Autowired + private OrchestratorCopyEnv orchestratorCopyEnv; + + private static final int MAX_DESC_LENGTH = 250; + private static final int MAX_NAME_LENGTH = 128; + + private final ThreadFactory orchestratorCopyThreadFactory = new ThreadFactoryBuilder() + .setNameFormat("dss-orchestrator—copy-thread-%d") + .setDaemon(false) + .build(); + + private final ExecutorService orchestratorCopyThreadPool = new ThreadPoolExecutor(0, 200, 60L, TimeUnit.SECONDS, + new LinkedBlockingQueue<>(1024), orchestratorCopyThreadFactory, new ThreadPoolExecutor.AbortPolicy()); + /** * 1.拿到的dss orchestrator的appconn * 2.然后创建 @@ -95,8 +124,15 @@ public class OrchestratorFrameworkServiceImpl implements OrchestratorFrameworkSe @SuppressWarnings("ConstantConditions") public CommonOrchestratorVo createOrchestrator(String username, OrchestratorCreateRequest orchestratorCreateRequest, Workspace workspace) throws Exception { + //检查desc字段长度 + if(orchestratorCreateRequest.getDescription().length() > MAX_DESC_LENGTH){ + DSSFrameworkErrorException.dealErrorException(60000, "描述过长,请限制在" + MAX_DESC_LENGTH + "以内"); + } + if(orchestratorCreateRequest.getOrchestratorName().length() > MAX_NAME_LENGTH){ + DSSFrameworkErrorException.dealErrorException(60000, "编排名称过长,请限制在" + MAX_NAME_LENGTH + "以内"); + } //是否存在相同的编排名称 - newOrchestratorService.isExistSameNameBeforeCreate(orchestratorCreateRequest.getWorkspaceId(), orchestratorCreateRequest.getProjectId(), orchestratorCreateRequest.getOrchestratorName()); + orchestratorService.isExistSameNameBeforeCreate(orchestratorCreateRequest.getWorkspaceId(), orchestratorCreateRequest.getProjectId(), orchestratorCreateRequest.getOrchestratorName()); //判断工程是否存在,并且取出工程名称和空间名称 DSSProject dssProject = validateOperation(orchestratorCreateRequest.getProjectId(), username); //1.创建编排实体bean @@ -126,7 +162,7 @@ public CommonOrchestratorVo createOrchestrator(String username, OrchestratorCrea OrchestrationService::getOrchestrationCreationOperation, (structureOperation, structureRequestRef) -> ((OrchestrationCreationOperation) structureOperation) .createOrchestration((DSSOrchestrationContentRequestRef) structureRequestRef), "create"); - OrchestratorVo orchestratorVo = newOrchestratorService.createOrchestrator(username, workspace, dssProject.getName(), + OrchestratorVo orchestratorVo = orchestratorService.createOrchestrator(username, workspace, dssProject.getName(), dssOrchestratorInfo.getProjectId(), dssOrchestratorInfo.getDesc(), dssOrchestratorInfo, dssLabels); Long orchestratorId = orchestratorVo.getDssOrchestratorInfo().getId(); Long orchestratorVersionId = orchestratorVo.getDssOrchestratorVersion().getId(); @@ -148,9 +184,9 @@ private V tryOrchestratio ImmutablePair orchestrationPair = getOrchestrationService(dssOrchestrator, userName, workspace, dssLabels); Long refProjectId, refOrchestrationId; if (askProjectSender) { - ProjectRefIdResponse projectRefIdResponse = (ProjectRefIdResponse) DSSSenderServiceFactory.getOrCreateServiceInstance().getProjectServerSender() - .ask(new ProjectRefIdRequest(Optional.ofNullable(orchestrationPair).map(ImmutablePair::getValue).map(AppInstance::getId).orElse(null), dssOrchestrator.getProjectId())); - refProjectId = projectRefIdResponse.getRefProjectId(); + ProjectRefIdResponse projectRefIdResponse = RpcAskUtils.processAskException(DSSSenderServiceFactory.getOrCreateServiceInstance().getProjectServerSender() + .ask(new ProjectRefIdRequest(Optional.ofNullable(orchestrationPair).map(ImmutablePair::getValue).map(AppInstance::getId).orElse(null), dssOrchestrator.getProjectId())), ProjectRefIdResponse.class, ProjectRefIdRequest.class); + refProjectId = projectRefIdResponse.getRefProjectId(); refOrchestrationId = null; } else { DSSOrchestratorRefOrchestration refOrchestration = orchestratorMapper.getRefOrchestrationId(dssOrchestrator.getId()); @@ -185,11 +221,18 @@ private V tryOrchestratio @Override public CommonOrchestratorVo modifyOrchestrator(String username, OrchestratorModifyRequest orchestratorModifyRequest, Workspace workspace) throws Exception { + //检查desc字段长度 + if(orchestratorModifyRequest.getDescription().length() > MAX_DESC_LENGTH){ + DSSFrameworkErrorException.dealErrorException(60000, "描述字段过长,请限制在" + MAX_DESC_LENGTH + "以内"); + } + if(orchestratorModifyRequest.getOrchestratorName().length() > MAX_NAME_LENGTH){ + DSSFrameworkErrorException.dealErrorException(60000, "编排名称过长,请限制在" + MAX_NAME_LENGTH + "以内"); + } //判断工程是否存在,并且取出工程名称和空间名称 DSSProject dssProject = validateOperation(orchestratorModifyRequest.getProjectId(), username); workspace.setWorkspaceName(dssProject.getWorkspaceName()); //是否存在相同的编排名称 //todo 返回orchestratorInfo而不是id - Long orchestratorId = newOrchestratorService.isExistSameNameBeforeUpdate(orchestratorModifyRequest); + Long orchestratorId = orchestratorService.isExistSameNameBeforeUpdate(orchestratorModifyRequest); LOGGER.info("{} begins to update a orchestrator {}.", username, orchestratorModifyRequest.getOrchestratorName()); List dssLabels = Collections.singletonList(new EnvDSSLabel(orchestratorModifyRequest.getLabels().getRoute())); DSSOrchestratorRelation dssOrchestratorRelation = DSSOrchestratorRelationManager.getDSSOrchestratorRelationByMode(orchestratorModifyRequest.getOrchestratorMode()); @@ -214,7 +257,7 @@ public CommonOrchestratorVo modifyOrchestrator(String username, OrchestratorModi OrchestrationService::getOrchestrationUpdateOperation, (structureOperation, structureRequestRef) -> ((OrchestrationUpdateOperation) structureOperation) .updateOrchestration((OrchestrationUpdateRequestRef) structureRequestRef), "update"); - newOrchestratorService.updateOrchestrator(username, workspace, dssOrchestratorInfo, dssLabels); + orchestratorService.updateOrchestrator(username, workspace, dssOrchestratorInfo, dssLabels); //3.将工程和orchestrator的关系存储到的数据库中 CommonOrchestratorVo orchestratorVo = new CommonOrchestratorVo(); orchestratorVo.setOrchestratorId(orchestratorId); @@ -228,17 +271,104 @@ public CommonOrchestratorVo deleteOrchestrator(String username, OrchestratorDele DSSOrchestratorInfo orchestratorInfo = orchestratorMapper.getOrchestrator(orchestratorDeleteRequest.getId()); LOGGER.info("{} begins to delete a orchestrator {}.", username, orchestratorInfo.getName()); List dssLabels = Collections.singletonList(new EnvDSSLabel(orchestratorDeleteRequest.getLabels().getRoute())); - tryOrchestrationOperation(dssLabels, false, username, dssProject.getName(), workspace, orchestratorInfo, - OrchestrationService::getOrchestrationDeletionOperation, - (structureOperation, structureRequestRef) -> ((OrchestrationDeletionOperation) structureOperation) - .deleteOrchestration((RefOrchestrationContentRequestRef) structureRequestRef), "delete"); + if(orchestratorDeleteRequest.getDeleteSchedulerWorkflow()) { + tryOrchestrationOperation(dssLabels, false, username, dssProject.getName(), workspace, orchestratorInfo, + OrchestrationService::getOrchestrationDeletionOperation, + (structureOperation, structureRequestRef) -> ((OrchestrationDeletionOperation) structureOperation) + .deleteOrchestration((RefOrchestrationContentRequestRef) structureRequestRef), "delete"); + } - newOrchestratorService.deleteOrchestrator(username, workspace, dssProject.getName(), orchestratorInfo.getId(), dssLabels); + orchestratorService.deleteOrchestrator(username, workspace, dssProject.getName(), orchestratorInfo.getId(), dssLabels); LOGGER.info("delete orchestrator {} by orchestrator framework succeed.", orchestratorInfo.getName()); CommonOrchestratorVo orchestratorVo = new CommonOrchestratorVo(); + orchestratorVo.setOrchestratorName(orchestratorInfo.getName()); + orchestratorVo.setOrchestratorId(orchestratorInfo.getId()); return orchestratorVo; } + @Override + public OrchestratorUnlockVo unlockOrchestrator(String username, Workspace workspace, OrchestratorUnlockRequest request) throws DSSErrorException { + //编辑权限校验 + DSSProject dssProject = validateOperation(request.getProjectId(), username); + DSSOrchestratorInfo orchestratorInfo = orchestratorMapper.getOrchestrator(request.getId()); + LOGGER.info("{} begins to unlock orchestrator {}.", username, orchestratorInfo.getName()); + List dssLabels = Collections.singletonList(new EnvDSSLabel(request.getLabels().getRoute())); + OrchestratorUnlockVo orchestratorUnlockVo = orchestratorService.unlockOrchestrator(username, workspace, + dssProject.getName(), orchestratorInfo.getId(), request.getConfirmDelete(), dssLabels); + orchestratorUnlockVo.setOrchestratorName(orchestratorInfo.getName()); + orchestratorUnlockVo.setOrchestratorId(orchestratorInfo.getId()); + return orchestratorUnlockVo; + } + + @Override + public String copyOrchestrator(String username, OrchestratorCopyRequest orchestratorCopyRequest, Workspace workspace) throws Exception{ + if(orchestratorCopyRequest.getTargetOrchestratorName().length() > MAX_NAME_LENGTH){ + DSSFrameworkErrorException.dealErrorException(60000, "编排名称过长,请限制在" + MAX_NAME_LENGTH + "以内"); + } + //校验编排名是可用 + orchestratorService.isExistSameNameBeforeCreate(workspace.getWorkspaceId(), orchestratorCopyRequest.getTargetProjectId(), orchestratorCopyRequest.getTargetOrchestratorName()); + //判断用户对项目是否有权限 + DSSProject sourceProject = validateOperation(orchestratorCopyRequest.getSourceProjectId(), username); + DSSProject targetProject = validateOperation(orchestratorCopyRequest.getTargetProjectId(), username); + + DSSOrchestratorInfo sourceOrchestratorInfo = orchestratorMapper.getOrchestrator(orchestratorCopyRequest.getSourceOrchestratorId()); + if (sourceOrchestratorInfo == null) { + LOGGER.error("orchestrator: {} not found.", orchestratorCopyRequest.getSourceOrchestratorName()); + DSSExceptionUtils.dealErrorException(6013, "orchestrator not found.", DSSOrchestratorErrorException.class); + } + if (StringUtils.isBlank(orchestratorCopyRequest.getWorkflowNodeSuffix())) { + orchestratorCopyRequest.setWorkflowNodeSuffix("copy"); + } else if (orchestratorCopyRequest.getWorkflowNodeSuffix().length() > 10) { + DSSExceptionUtils.dealErrorException(6014, "The node suffix length can not exceed 10. (节点后缀长度不能超过10)", DSSOrchestratorErrorException.class); + } + OrchestratorCopyVo orchestratorCopyVo = new OrchestratorCopyVo.Builder(username, sourceProject.getId(), sourceProject.getName(), targetProject.getId(), + targetProject.getName(), sourceOrchestratorInfo, orchestratorCopyRequest.getTargetOrchestratorName(), + orchestratorCopyRequest.getWorkflowNodeSuffix(), new EnvDSSLabel(DSSCommonUtils.ENV_LABEL_VALUE_DEV), + workspace, Sender.getThisInstance()).setCopyTaskId(null).build(); + OrchestratorCopyJob orchestratorCopyJob = new OrchestratorCopyJob(); + orchestratorCopyJob.setOrchestratorCopyVo(orchestratorCopyVo); + orchestratorCopyJob.setOrchestratorCopyEnv(orchestratorCopyEnv); + orchestratorCopyJob.getOrchestratorCopyInfo().setId(UUID.randomUUID().toString()); + + orchestratorCopyThreadPool.submit(orchestratorCopyJob); + + return orchestratorCopyJob.getOrchestratorCopyInfo().getId(); + } + + @Override + public Pair> getOrchestratorCopyHistory(String username, Workspace workspace, Long orchestratorId, Integer currentPage, Integer pageSize) throws Exception { + long total = 0L; + + List orchestratorCopyInfoList = orchestratorCopyJobMapper.getOrchestratorCopyInfoList(orchestratorId); + if (CollectionUtils.isEmpty(orchestratorCopyInfoList)) { + return new Pair<>(total, Lists.newArrayList()); + } + total = orchestratorCopyInfoList.size(); + + List < OrchestratorCopyHistory > orchestratorCopyHistoryList = new ArrayList<>(); + OrchestratorCopyHistory orchestratorCopyHistory; + for (DSSOrchestratorCopyInfo orchestratorCopyInfo: orchestratorCopyInfoList) { + orchestratorCopyHistory = new OrchestratorCopyHistory(orchestratorCopyInfo.getId(), orchestratorCopyInfo.getUsername(), workspace.getWorkspaceName(), + orchestratorCopyInfo.getSourceOrchestratorName(), orchestratorCopyInfo.getTargetOrchestratorName(), + orchestratorCopyInfo.getSourceProjectName(), orchestratorCopyInfo.getTargetProjectName(), orchestratorCopyInfo.getWorkflowNodeSuffix(), + orchestratorCopyInfo.getMicroserverName(), orchestratorCopyInfo.getExceptionInfo(), orchestratorCopyInfo.getStatus(), + orchestratorCopyInfo.getIsCopying(), DTF.format(LocalDateTime.ofInstant(orchestratorCopyInfo.getStartTime().toInstant(), ZoneId.systemDefault())), + DTF.format(LocalDateTime.ofInstant(orchestratorCopyInfo.getEndTime().toInstant(), ZoneId.systemDefault()))); + orchestratorCopyHistoryList.add(orchestratorCopyHistory); + } + return new Pair<>(total, orchestratorCopyHistoryList); + } + + @Override + public Boolean getOrchestratorCopyStatus(Long sourceOrchestratorId) { + return StringUtils.isNotBlank(orchestratorCopyJobMapper.getOrchestratorCopyStatus(sourceOrchestratorId)); + } + + @Override + public DSSOrchestratorCopyInfo getOrchestratorCopyInfoById(String copyInfoId) { + return orchestratorCopyJobMapper.getOrchestratorCopyInfoById(copyInfoId); + } + protected ImmutablePair getOrchestrationService(DSSOrchestratorInfo dssOrchestratorInfo, String user, Workspace workspace, @@ -248,16 +378,12 @@ protected ImmutablePair getOrchestrationServi dssOrchestratorInfo.setLinkedAppConnNames(dssOrchestrator.getLinkedAppConn().stream().map(appConn -> appConn.getAppDesc().getAppName()).collect(Collectors.toList())); } SchedulerAppConn appConn = dssOrchestrator.getSchedulerAppConn(); -// if (appConn == null) { -// throw new ExternalOperationWarnException(50322, "DSSOrchestrator " + dssOrchestrator.getName() + " has no SchedulerAppConn."); -// } if (appConn != null) { AppInstance appInstance = appConn.getAppDesc().getAppInstances().get(0); return new ImmutablePair<>(appConn.getOrCreateStructureStandard().getOrchestrationService(appInstance), appInstance); } else { return new ImmutablePair<>(null, null); } - } /** @@ -268,11 +394,11 @@ protected ImmutablePair getOrchestrationServi * @return * @throws DSSOrchestratorErrorException */ - private DSSProject validateOperation(long projectId, String username) throws DSSOrchestratorErrorException { + public static DSSProject validateOperation(long projectId, String username) throws DSSOrchestratorErrorException { ProjectInfoRequest projectInfoRequest = new ProjectInfoRequest(); projectInfoRequest.setProjectId(projectId); - DSSProject dssProject = (DSSProject) DSSSenderServiceFactory.getOrCreateServiceInstance().getProjectServerSender() - .ask(projectInfoRequest); + DSSProject dssProject = RpcAskUtils.processAskException(DSSSenderServiceFactory.getOrCreateServiceInstance().getProjectServerSender() + .ask(projectInfoRequest), DSSProject.class, ProjectInfoRequest.class); if (dssProject == null) { DSSExceptionUtils.dealErrorException(6003, "工程不存在", DSSOrchestratorErrorException.class); } @@ -282,9 +408,9 @@ private DSSProject validateOperation(long projectId, String username) throws DSS return dssProject; } - private boolean hasProjectEditPriv(Long projectId, String username) { - ProjectUserAuthResponse projectUserAuthResponse = (ProjectUserAuthResponse) DSSSenderServiceFactory.getOrCreateServiceInstance() - .getProjectServerSender().ask(new ProjectUserAuthRequest(projectId, username)); + private static boolean hasProjectEditPriv(Long projectId, String username) { + ProjectUserAuthResponse projectUserAuthResponse = RpcAskUtils.processAskException(DSSSenderServiceFactory.getOrCreateServiceInstance() + .getProjectServerSender().ask(new ProjectUserAuthRequest(projectId, username)), ProjectUserAuthResponse.class, ProjectUserAuthRequest.class); boolean hasEditPriv = false; if (!CollectionUtils.isEmpty(projectUserAuthResponse.getPrivList())) { hasEditPriv = projectUserAuthResponse.getPrivList().contains(ProjectUserPrivEnum.PRIV_EDIT.getRank()) || diff --git a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/service/impl/OrchestratorPluginServiceImpl.java b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/service/impl/OrchestratorPluginServiceImpl.java index 523fce553e..999b70bdd2 100644 --- a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/service/impl/OrchestratorPluginServiceImpl.java +++ b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/service/impl/OrchestratorPluginServiceImpl.java @@ -17,10 +17,14 @@ package com.webank.wedatasphere.dss.orchestrator.server.service.impl; import com.webank.wedatasphere.dss.common.entity.project.DSSProject; +import com.webank.wedatasphere.dss.common.exception.DSSErrorException; import com.webank.wedatasphere.dss.common.label.DSSLabel; import com.webank.wedatasphere.dss.common.label.DSSLabelUtil; +import com.webank.wedatasphere.dss.common.protocol.JobStatus; +import com.webank.wedatasphere.dss.common.protocol.project.ProjectInfoRequest; import com.webank.wedatasphere.dss.common.utils.DSSCommonUtils; import com.webank.wedatasphere.dss.common.utils.DSSExceptionUtils; +import com.webank.wedatasphere.dss.common.utils.RpcAskUtils; import com.webank.wedatasphere.dss.orchestrator.common.entity.DSSOrchestratorInfo; import com.webank.wedatasphere.dss.orchestrator.common.entity.DSSOrchestratorRefOrchestration; import com.webank.wedatasphere.dss.orchestrator.common.entity.DSSOrchestratorVersion; @@ -29,25 +33,31 @@ import com.webank.wedatasphere.dss.orchestrator.common.protocol.ResponseConvertOrchestrator; import com.webank.wedatasphere.dss.orchestrator.common.protocol.ResponseOperateOrchestrator; import com.webank.wedatasphere.dss.orchestrator.core.DSSOrchestratorContext; +import com.webank.wedatasphere.dss.orchestrator.core.plugin.DSSOrchestratorPlugin; import com.webank.wedatasphere.dss.orchestrator.db.dao.OrchestratorJobMapper; import com.webank.wedatasphere.dss.orchestrator.db.dao.OrchestratorMapper; +import com.webank.wedatasphere.dss.orchestrator.common.entity.OrchestratorPublishJob; +import com.webank.wedatasphere.dss.orchestrator.publish.ConversionDSSOrchestratorPlugin; import com.webank.wedatasphere.dss.orchestrator.publish.ExportDSSOrchestratorPlugin; +import com.webank.wedatasphere.dss.orchestrator.publish.conf.DSSOrchestratorConf; +import com.webank.wedatasphere.dss.orchestrator.publish.job.CommonUpdateConvertJobStatus; import com.webank.wedatasphere.dss.orchestrator.publish.job.ConversionJobEntity; import com.webank.wedatasphere.dss.orchestrator.publish.job.OrchestratorConversionJob; -import com.webank.wedatasphere.dss.orchestrator.server.constant.DSSOrchestratorConstant; import com.webank.wedatasphere.dss.orchestrator.server.service.OrchestratorPluginService; +import com.webank.wedatasphere.dss.sender.service.DSSSenderServiceFactory; import com.webank.wedatasphere.dss.standard.app.sso.Workspace; import org.apache.commons.collections.CollectionUtils; +import org.apache.commons.lang3.exception.ExceptionUtils; +import org.apache.linkis.common.utils.ByteTimeUtils; import org.apache.linkis.common.utils.Utils; import org.apache.linkis.manager.label.builder.factory.LabelBuilderFactoryContext; +import org.apache.linkis.rpc.Sender; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.transaction.annotation.Transactional; import java.util.*; import java.util.concurrent.ExecutorService; -import java.util.concurrent.atomic.AtomicInteger; public class OrchestratorPluginServiceImpl implements OrchestratorPluginService { @@ -63,11 +73,20 @@ public class OrchestratorPluginServiceImpl implements OrchestratorPluginService @Autowired private DSSOrchestratorContext dssOrchestratorContext; + @Autowired + private CommonUpdateConvertJobStatus commonUpdateConvertJobStatus; + private ExecutorService releaseThreadPool = Utils.newCachedThreadPool(50, "Convert-Orchestration-Thread-", true); - private AtomicInteger idGenerator = new AtomicInteger(); @Override public ResponseConvertOrchestrator convertOrchestration(RequestFrameworkConvertOrchestration requestConversionOrchestration) { + // 如果是先导入,后发布的分两步模式,则走专门的逻辑 + if(DSSOrchestratorConf.DSS_IMPORT_VALID_IMMEDIATELY.getValue() && + requestConversionOrchestration.isConvertAllOrcs() && + requestConversionOrchestration.getProjectId()!=null){ + ResponseOperateOrchestrator responseOperateOrchestrator = convertWholeProjectOrcs(requestConversionOrchestration); + return new ResponseConvertOrchestrator("-1", responseOperateOrchestrator); + } LOGGER.info("conversion request is called, request is {}.", DSSCommonUtils.COMMON_GSON.toJson(requestConversionOrchestration)); Long toPublishOrcId; if(requestConversionOrchestration.getOrcAppId() != null) { @@ -87,8 +106,8 @@ public ResponseConvertOrchestrator convertOrchestration(RequestFrameworkConvertO List publishedOrcIds; List labels = LabelBuilderFactoryContext.getLabelBuilderFactory().getLabels(requestConversionOrchestration.getLabels()); if(requestConversionOrchestration.isConvertAllOrcs()) { - //这个地方应该是要获取所有的已经发布过的orchestrator - publishedOrcIds = orchestratorMapper.getAllOrcIdsByProjectId(projectId); + //获取所有的已经发布过的orchestrator + publishedOrcIds = orchestratorMapper.getAllOrcIdsByProjectId(projectId,1); if (!publishedOrcIds.contains(toPublishOrcId)) { publishedOrcIds.add(toPublishOrcId); } @@ -144,9 +163,10 @@ public ResponseConvertOrchestrator convertOrchestration(RequestFrameworkConvertO return new ResponseConvertOrchestrator("no-necessary-id", ResponseOperateOrchestrator.failed("No suitable workflow(s) found, publish is ignored.")); } OrchestratorConversionJob job = new OrchestratorConversionJob(); - job.setId(generateId()); + job.setId(String.valueOf(UUID.randomUUID())); LOGGER.info("user {} try to submit a conversion job {}, the orchestrationIdMap is {}, the orcIdList is {}.", requestConversionOrchestration.getUserName(), job.getId(), orchestrationIdMap, publishedOrcIds); + ConversionJobEntity entity = new ConversionJobEntity(); entity.setResponse(ResponseOperateOrchestrator.inited()); entity.setCreateTime(new Date()); @@ -159,59 +179,161 @@ public ResponseConvertOrchestrator convertOrchestration(RequestFrameworkConvertO dssProject.setId(projectId); entity.setProject(dssProject); job.setConversionJobEntity(entity); + job.setCommonUpdateConvertJobStatus(commonUpdateConvertJobStatus); job.setConversionDSSOrchestratorPlugins(dssOrchestratorContext.getOrchestratorPlugins()); - job.afterConversion(response -> this.updateDBAfterConversion(toPublishOrcId, response, job, requestConversionOrchestration)); + job.afterConversion(response -> this.updateDBAfterConversion(toPublishOrcId, response, job.getConversionJobEntity(), requestConversionOrchestration)); + + OrchestratorPublishJob orchestratorPublishJob = new OrchestratorPublishJob(); + orchestratorPublishJob.setJobId(job.getId()); + orchestratorPublishJob.setStatus(JobStatus.Inited.getStatus()); + orchestratorPublishJob.setInstanceName(Sender.getThisInstance()); + orchestratorPublishJob.setCreateTime(new Date(System.currentTimeMillis())); + orchestratorPublishJob.setUpdateTime(new Date(System.currentTimeMillis())); + if (DSSCommonUtils.COMMON_GSON.toJson(entity).length() > 1024) { + orchestratorPublishJob.setConversionJobJson(DSSCommonUtils.COMMON_GSON.toJson(entity).substring(0, 1024)); + } else { + orchestratorPublishJob.setConversionJobJson(DSSCommonUtils.COMMON_GSON.toJson(entity)); + } + job.setOrchestratorPublishJob(orchestratorPublishJob); + orchestratorJobMapper.insertPublishJob(orchestratorPublishJob); //submit it releaseThreadPool.submit(job); - DSSOrchestratorConstant.orchestratorConversionJobMap.put(job.getId(), job); - // todo insert publishJob + + LOGGER.info("publish orchestrator success. publishedOrcIds:{} ",publishedOrcIds); return new ResponseConvertOrchestrator(job.getId(), entity.getResponse()); } - @Override - public ResponseConvertOrchestrator getConvertOrchestrationStatus(String id) { - OrchestratorConversionJob job = DSSOrchestratorConstant.orchestratorConversionJobMap.get(id); - // todo 找不到从db加载 - if(job.getConversionJobEntity().getResponse().isCompleted()) { - DSSOrchestratorConstant.orchestratorConversionJobMap.remove(job.getId()); + /** + * 发布整个项目中所有未发布过的工作流。 + * 使用在先导入,后发布两步走的场景中 + */ + private ResponseOperateOrchestrator convertWholeProjectOrcs(RequestFrameworkConvertOrchestration requestConversionOrchestration){ + Long projectId=requestConversionOrchestration.getProjectId(); + List labels = LabelBuilderFactoryContext.getLabelBuilderFactory().getLabels(requestConversionOrchestration.getLabels()); + + //这里的 key 为 DSS 具体编排(如 DSS 工作流)的 id;这里的 value 为 DSS 编排所对应的第三方调度系统的工作流 ID + //请注意:由于对接的 SchedulerAppConn 调度系统有可能没有实现 OrchestrationService, + //所以可能存在 DSS 在创建 DSS 编排时,无法同步去 SchedulerAppConn 创建工作流的情况,从而导致这个 Map 的所有 value 都为 null。 + Map orchestrationIdMap = new HashMap<>(); + List orcIds=new ArrayList<>(); + //所有之前发布过的编排 + List publishedOrcIds = orchestratorMapper.getAllOrcIdsByProjectId(projectId,1); + orcIds.addAll(publishedOrcIds); + //所有之前未发布过的编排 + List notPublished= orchestratorMapper.getAllOrcIdsByProjectId(projectId,0); + orcIds.addAll(notPublished); + for (Long publishedOrcId : orcIds) { + DSSOrchestratorVersion dssOrchestratorVersion = orchestratorMapper.getLatestOrchestratorVersionByIdAndValidFlag(publishedOrcId, 1); + if (dssOrchestratorVersion == null) { + continue; + } + DSSOrchestratorRefOrchestration dssOrchestratorRefOrchestration = orchestratorMapper.getRefOrchestrationId(publishedOrcId); + if(dssOrchestratorRefOrchestration != null) { + orchestrationIdMap.put(dssOrchestratorVersion.getAppId(), dssOrchestratorRefOrchestration.getRefOrchestrationId()); + } else { + orchestrationIdMap.put(dssOrchestratorVersion.getAppId(), null); + } + } + if(orchestrationIdMap.isEmpty()) { + LOGGER.info("the project {} has no suitable workflow, the publish by user {} is ignored.", projectId, + requestConversionOrchestration.getUserName()); + return ResponseOperateOrchestrator.failed("No suitable workflow(s) found, publish is ignored."); + + } + + LOGGER.info("user {} try to submit a conversion job , the orchestrationIdMap is {}, the orcIdList is {}.", + requestConversionOrchestration.getUserName(), + orchestrationIdMap, publishedOrcIds); + long startTime=System.currentTimeMillis(); + //进行发布到schedulis等调度系统 + LOGGER.info(" begin to convert project {} for user {} to scheduler, the orchestrationIds is {}.", + projectId,requestConversionOrchestration.getUserName() , orcIds); + ConversionDSSOrchestratorPlugin conversionDSSOrchestratorPlugin = null; + for (DSSOrchestratorPlugin plugin:dssOrchestratorContext.getOrchestratorPlugins()) { + if(plugin instanceof ConversionDSSOrchestratorPlugin) { + conversionDSSOrchestratorPlugin = (ConversionDSSOrchestratorPlugin) plugin; + } + } + if(conversionDSSOrchestratorPlugin==null){ + return ResponseOperateOrchestrator.failed("can not find plugin type of ConversionDSSOrchestratorPlugin,operate failed!"); + } + ProjectInfoRequest projectInfoRequest = new ProjectInfoRequest(); + projectInfoRequest.setProjectId(projectId); + ResponseOperateOrchestrator response; + try{ + DSSProject project = RpcAskUtils.processAskException(DSSSenderServiceFactory.getOrCreateServiceInstance().getProjectServerSender() + .ask(projectInfoRequest), DSSProject.class, ProjectInfoRequest.class); + response = conversionDSSOrchestratorPlugin.convert( + requestConversionOrchestration.getUserName(), + project, + (Workspace) requestConversionOrchestration.getWorkspace(), + orchestrationIdMap, + labels, + requestConversionOrchestration.getApproveId()); + if(response.isFailed()) { + String msg = response.getMessage() == null ? "Unknown reason, please ask admin for help!" : response.getMessage(); + throw new DSSErrorException(50000, msg); + } + final ResponseOperateOrchestrator finalResponse=response; + //最后,把所有之前未发布过的编排,都标记为已发布。 + ConversionJobEntity entity = new ConversionJobEntity(); + entity.setUserName(requestConversionOrchestration.getUserName()); + entity.setOrcIdList(publishedOrcIds); + entity.setLabels(labels); + entity.setProject(project); + notPublished.forEach(toPublishOrcId-> this.updateDBAfterConversion(toPublishOrcId, finalResponse, entity, requestConversionOrchestration) ); + } catch (final Exception t){ + LOGGER.error("convert for project failed"+projectId, t); + response = ResponseOperateOrchestrator.failed(ExceptionUtils.getRootCauseMessage(t)); } - return new ResponseConvertOrchestrator(job.getId(), job.getConversionJobEntity().getResponse()); + LOGGER.info("convert project {} for user {} to Orchestrators {}, costs {}.",projectId, + requestConversionOrchestration.getUserName(),orcIds , ByteTimeUtils.msDurationToString(System.currentTimeMillis()-startTime)); + + return response; } - private String generateId() { - return String.valueOf(idGenerator.getAndIncrement()); + @Override + public ResponseConvertOrchestrator getConvertOrchestrationStatus(String id) { + String jobId; + ResponseOperateOrchestrator responseOperateOrchestrator = new ResponseOperateOrchestrator(); + LOGGER.info("Convert to orchestrator id: {}", id); + OrchestratorPublishJob publishJob = orchestratorJobMapper.getPublishJobByJobId(id); + jobId = publishJob.getJobId(); + responseOperateOrchestrator.setJobStatus(JobStatus.getJobStatusByStatus(publishJob.getStatus())); + if (publishJob.getErrorMsg() != null) { + responseOperateOrchestrator.setMessage(publishJob.getErrorMsg()); + } + return new ResponseConvertOrchestrator(jobId, responseOperateOrchestrator); } - @Transactional(rollbackFor = Exception.class) private void updateDBAfterConversion(Long toPublishOrcId,ResponseOperateOrchestrator response, - OrchestratorConversionJob job, + ConversionJobEntity conversionJobEntity, RequestFrameworkConvertOrchestration requestConversionOrchestration) { - LOGGER.info("{} completed with status {}.", job.getId(), response.getJobStatus()); if (response.isSucceed()) { //1. 进行导出,用于升级版本,目的是为了复用原来的代码 - List orcIdList = job.getConversionJobEntity().getOrcIdList(); + List orcIdList = conversionJobEntity.getOrcIdList(); LOGGER.info("the orcIdList is:{}", orcIdList); // 开发环境才新增版本号 - if(DSSLabelUtil.isDevEnv(job.getConversionJobEntity().getLabels())){ + if(DSSLabelUtil.isDevEnv(conversionJobEntity.getLabels())){ orcIdList.forEach(DSSExceptionUtils.handling(orcId -> { DSSOrchestratorInfo dssOrchestratorInfo = orchestratorMapper.getOrchestrator(orcId); dssOrchestratorContext.getDSSOrchestratorPlugin(ExportDSSOrchestratorPlugin.class) - .orchestratorVersionIncrease(orcId, job.getConversionJobEntity().getUserName(), requestConversionOrchestration.getComment(), + .orchestratorVersionIncrease(orcId, conversionJobEntity.getUserName(), requestConversionOrchestration.getComment(), (Workspace) requestConversionOrchestration.getWorkspace(), dssOrchestratorInfo, - job.getConversionJobEntity().getProject().getName(), job.getConversionJobEntity().getLabels()); + conversionJobEntity.getProject().getName(), conversionJobEntity.getLabels()); })); } //2. 做一个标记表示已经发布过了 orcIdList.forEach(orchestratorMapper::setPublished); //3.更新当前的提交的发布编排模式版本的common - updateCurrentPublishOrchestratorCommon(toPublishOrcId,requestConversionOrchestration.getComment(),job.getConversionJobEntity().getLabels()); + updateCurrentPublishOrchestratorCommon(toPublishOrcId,requestConversionOrchestration.getComment(),conversionJobEntity.getLabels()); } } public void updateCurrentPublishOrchestratorCommon(Long orcId,String realComment,List dssLabels) { //DEV环境获取获取最新有效的版本即可 int validFlag = 1; - if (!DSSLabelUtil.isDevEnv(dssLabels)) { + if (!DSSOrchestratorConf.DSS_IMPORT_VALID_IMMEDIATELY.getValue()&& !DSSLabelUtil.isDevEnv(dssLabels)) { //PROD获取当前编排模式最新版本的ID validFlag = 0; } diff --git a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/service/impl/OrchestratorServiceImpl.java b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/service/impl/OrchestratorServiceImpl.java index ba1ec1d4b4..f66c567770 100644 --- a/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/service/impl/OrchestratorServiceImpl.java +++ b/dss-framework/dss-framework-orchestrator-server/src/main/java/com/webank/wedatasphere/dss/orchestrator/server/service/impl/OrchestratorServiceImpl.java @@ -16,15 +16,19 @@ package com.webank.wedatasphere.dss.orchestrator.server.service.impl; +import com.google.common.collect.Lists; import com.webank.wedatasphere.dss.appconn.core.AppConn; import com.webank.wedatasphere.dss.common.constant.project.ProjectUserPrivEnum; import com.webank.wedatasphere.dss.common.exception.DSSErrorException; import com.webank.wedatasphere.dss.common.label.DSSLabel; import com.webank.wedatasphere.dss.common.label.DSSLabelUtil; +import com.webank.wedatasphere.dss.common.label.EnvDSSLabel; +import com.webank.wedatasphere.dss.common.protocol.JobStatus; import com.webank.wedatasphere.dss.common.protocol.project.ProjectUserAuthRequest; import com.webank.wedatasphere.dss.common.protocol.project.ProjectUserAuthResponse; import com.webank.wedatasphere.dss.common.utils.DSSExceptionUtils; import com.webank.wedatasphere.dss.common.utils.MapUtils; +import com.webank.wedatasphere.dss.common.utils.RpcAskUtils; import com.webank.wedatasphere.dss.contextservice.service.ContextService; import com.webank.wedatasphere.dss.framework.common.exception.DSSFrameworkErrorException; import com.webank.wedatasphere.dss.orchestrator.common.entity.DSSOrchestratorInfo; @@ -40,9 +44,12 @@ import com.webank.wedatasphere.dss.orchestrator.db.dao.OrchestratorMapper; import com.webank.wedatasphere.dss.orchestrator.loader.OrchestratorManager; import com.webank.wedatasphere.dss.orchestrator.publish.utils.OrchestrationDevelopmentOperationUtils; +import com.webank.wedatasphere.dss.orchestrator.server.conf.OrchestratorConf; +import com.webank.wedatasphere.dss.orchestrator.server.constant.DSSOrchestratorConstant; import com.webank.wedatasphere.dss.orchestrator.server.entity.request.OrchestratorModifyRequest; import com.webank.wedatasphere.dss.orchestrator.server.entity.request.OrchestratorRequest; import com.webank.wedatasphere.dss.orchestrator.server.entity.vo.OrchestratorBaseInfo; +import com.webank.wedatasphere.dss.orchestrator.server.entity.vo.OrchestratorUnlockVo; import com.webank.wedatasphere.dss.orchestrator.server.service.OrchestratorService; import com.webank.wedatasphere.dss.sender.service.DSSSenderServiceFactory; import com.webank.wedatasphere.dss.standard.app.development.operation.*; @@ -56,7 +63,11 @@ import com.webank.wedatasphere.dss.standard.common.desc.AppInstance; import com.webank.wedatasphere.dss.standard.common.entity.ref.ResponseRef; import com.webank.wedatasphere.dss.standard.common.exception.operation.ExternalOperationWarnException; +import com.webank.wedatasphere.dss.workflow.common.protocol.*; import org.apache.commons.collections.CollectionUtils; +import org.apache.linkis.cs.client.ContextClient; +import org.apache.linkis.cs.client.builder.ContextClientFactory; +import org.apache.linkis.rpc.Sender; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.beans.BeanUtils; @@ -64,10 +75,12 @@ import org.springframework.stereotype.Service; import org.springframework.transaction.annotation.Transactional; +import java.time.LocalDateTime; import java.util.*; import java.util.function.BiFunction; import java.util.function.Consumer; import java.util.function.Function; +import java.util.stream.Collectors; @Service public class OrchestratorServiceImpl implements OrchestratorService { @@ -225,6 +238,40 @@ public void deleteOrchestrator(String userName, orchestratorMapper.deleteOrchestrator(orchestratorInfoId); } + @Override + public OrchestratorUnlockVo unlockOrchestrator(String userName, + Workspace workspace, + String projectName, + Long orchestratorInfoId, + Boolean confirmDelete, + List dssLabels) throws DSSErrorException { + DSSOrchestratorInfo dssOrchestratorInfo = orchestratorMapper.getOrchestrator(orchestratorInfoId); + if (null == dssOrchestratorInfo) { + LOGGER.error("Not exists orchestration {} in project {}.", orchestratorInfoId, projectName); + DSSExceptionUtils.dealErrorException(61123, + String.format("Not exists orchestration %s in project %s.", orchestratorInfoId, projectName), DSSErrorException.class); + } + DSSOrchestratorVersion dssOrchestratorVersion = orchestratorMapper.getLatestOrchestratorVersionByIdAndValidFlag(orchestratorInfoId, VALID_FLAG); + LOGGER.info("user {} try to unlock the project {} 's orchestration(such as DSS workflow) {} of orchestrator {} in version {}.", + userName, projectName, dssOrchestratorVersion.getAppId(), dssOrchestratorInfo.getName(), dssOrchestratorVersion.getVersion()); + RequestUnlockWorkflow requestUnlockWorkflow = new RequestUnlockWorkflow(userName, dssOrchestratorVersion.getAppId(), confirmDelete); + ResponseUnlockWorkflow responseUnlockWorkflow = RpcAskUtils.processAskException(DSSSenderServiceFactory.getOrCreateServiceInstance() + .getWorkflowSender(dssLabels).ask(requestUnlockWorkflow), ResponseUnlockWorkflow.class, RequestUnlockWorkflow.class); + switch (responseUnlockWorkflow.getUnlockStatus()) { + case ResponseUnlockWorkflow.NONEED_UNLOCK: + DSSExceptionUtils.dealErrorException(62001, String.format("解锁失败,当前工作流未被锁定:%s", dssOrchestratorInfo.getName()), DSSErrorException.class); + case ResponseUnlockWorkflow.NEED_SECOND_CONFIRM: + String lockOwner = responseUnlockWorkflow.getLockOwner(); + String confirmMsg = String.format("当前工作流已被%s锁定编辑,强制解锁工作流会导致%s已编辑的内容无法保存,请与%s确认后再解锁工作流。", lockOwner, lockOwner, lockOwner); + return new OrchestratorUnlockVo(lockOwner, confirmMsg, 1); + case ResponseUnlockWorkflow.UNLOCK_SUCCESS: + return new OrchestratorUnlockVo(null, "解锁成功", 0); + default: + DSSExceptionUtils.dealErrorException(62003, "unknown unlockStatus", DSSErrorException.class); + return null; + } + } + @Override public List getOrchestratorVoList(List orchestratorIds) { List orchestratorVoList = new ArrayList<>(); @@ -399,8 +446,9 @@ public List getListByPage(OrchestratorRequest orchestrator List retList = new ArrayList<>(list.size()); if (!CollectionUtils.isEmpty(list)) { //todo Is used in front-end? - ProjectUserAuthResponse projectUserAuthResponse = (ProjectUserAuthResponse) DSSSenderServiceFactory.getOrCreateServiceInstance() - .getProjectServerSender().ask(new ProjectUserAuthRequest(orchestratorRequest.getProjectId(), username)); + ProjectUserAuthResponse projectUserAuthResponse = RpcAskUtils.processAskException(DSSSenderServiceFactory.getOrCreateServiceInstance() + .getProjectServerSender().ask(new ProjectUserAuthRequest(orchestratorRequest.getProjectId(), username)), + ProjectUserAuthResponse.class, ProjectUserAuthRequest.class); boolean isReleasable = false, isEditable = false; if (!CollectionUtils.isEmpty(projectUserAuthResponse.getPrivList())) { isReleasable = projectUserAuthResponse.getPrivList().contains(ProjectUserPrivEnum.PRIV_RELEASE.getRank()); @@ -435,4 +483,61 @@ public ResponseOrchestratorInfos queryOrchestratorInfos(RequestOrchestratorInfos return new ResponseOrchestratorInfos(orchestratorInfos); } + @Override + public void batchClearContextId() { + LOGGER.info("--------------------{} start clear old contextId------------------------", LocalDateTime.now()); + try { + // 1、先去查询dss_orchestrator_version_info表,筛选出发布过n次及以上的编排,并获取老的发布记录。 + //为了不影响正常使用,需要频繁下载工作流bml文件,每次只拿50条工作流进行清理 + List historyOrcVersionList = orchestratorMapper.getHistoryOrcVersion(OrchestratorConf.DSS_PUBLISH_MAX_VERSION.getValue()); + while (historyOrcVersionList.size()>0) { + LOGGER.info("Clear historyOrcVersionList size is "+ historyOrcVersionList.size()); + if (historyOrcVersionList == null || historyOrcVersionList.isEmpty()) { + LOGGER.info("--------------------{} end clear old contextId------------------------", LocalDateTime.now()); + return; + } + List contextIdList = historyOrcVersionList.stream().map(DSSOrchestratorVersion::getContextId).collect(Collectors.toList()); + + // 2、根据appIds去查询子工作流contextId,必须保证标签绝对正确,否则可能清理了错误的工作流上下文ID + List dssLabels = Lists.newArrayList(new EnvDSSLabel(OrchestratorConf.DSS_CS_CLEAR_ENV.getValue())); + Sender sender = DSSSenderServiceFactory.getOrCreateServiceInstance().getWorkflowSender(dssLabels); + List workflowIdList = historyOrcVersionList.stream().map(orcInfo -> orcInfo.getAppId()).collect(Collectors.toList()); + ResponseSubFlowContextIds response = RpcAskUtils.processAskException(sender.ask(new RequestSubFlowContextIds(workflowIdList)), + ResponseSubFlowContextIds.class, RequestSubFlowContextIds.class); + if (response != null) { + List subContextIdList = response.getContextIdList(); + if (subContextIdList != null && subContextIdList.size() > 0) { + contextIdList.addAll(response.getContextIdList()); + } + } + LOGGER.info("Clear contextIdList size is "+ contextIdList.size()); + // 3、调用linkis接口批量删除contextId + ContextClient contextClient = ContextClientFactory.getOrCreateContextClient(); + // 每次处理1000条数据 + if (contextIdList.size() < DSSOrchestratorConstant.MAX_CLEAR_SIZE) { + LOGGER.info("clear old contextId, contextIds:{}", contextIdList.toString()); + contextClient.batchClearContextByHAID(contextIdList); + } else { + int len = DSSOrchestratorConstant.MAX_CLEAR_SIZE; + int size = contextIdList.size(); + int count = (size + len - 1) / len; + for (int i = 0; i < count; i++) { + List subList = contextIdList.subList(i * len, (Math.min((i + 1) * len, size))); + LOGGER.info("clear old contextId by batch, {} batch, contextIds:{}", i + 1, subList.toString()); + contextClient.batchClearContextByHAID(subList); + Thread.sleep(2000); + } + } + LOGGER.info("--------------------{} end clear old contextId------------------------", LocalDateTime.now()); + + orchestratorMapper.batchUpdateOrcInfo(historyOrcVersionList); + Thread.sleep(5000); + historyOrcVersionList=orchestratorMapper.getHistoryOrcVersion(OrchestratorConf.DSS_PUBLISH_MAX_VERSION.getValue()); + } + + } catch (Exception e) { + LOGGER.warn("execute linkis batch clear csId failed", e); + } + } + } \ No newline at end of file diff --git a/dss-framework/dss-framework-orchestrator-server/src/main/scala/com/webank/wedatasphere/dss/orchestrator/server/receiver/DSSOrchestratorReceiver.scala b/dss-framework/dss-framework-orchestrator-server/src/main/scala/com/webank/wedatasphere/dss/orchestrator/server/receiver/DSSOrchestratorReceiver.scala index 783dca8d06..25add1c6ed 100644 --- a/dss-framework/dss-framework-orchestrator-server/src/main/scala/com/webank/wedatasphere/dss/orchestrator/server/receiver/DSSOrchestratorReceiver.scala +++ b/dss-framework/dss-framework-orchestrator-server/src/main/scala/com/webank/wedatasphere/dss/orchestrator/server/receiver/DSSOrchestratorReceiver.scala @@ -21,9 +21,11 @@ import com.webank.wedatasphere.dss.common.protocol.{ResponseExportOrchestrator, import com.webank.wedatasphere.dss.orchestrator.common.entity.OrchestratorVo import com.webank.wedatasphere.dss.orchestrator.common.protocol._ import com.webank.wedatasphere.dss.orchestrator.core.DSSOrchestratorContext +import com.webank.wedatasphere.dss.orchestrator.publish.entity.OrchestratorExportResult import com.webank.wedatasphere.dss.orchestrator.publish.{ExportDSSOrchestratorPlugin, ImportDSSOrchestratorPlugin} import com.webank.wedatasphere.dss.orchestrator.server.service.{OrchestratorPluginService, OrchestratorService} import org.apache.linkis.rpc.{Receiver, Sender} +import org.slf4j.{Logger, LoggerFactory} import java.util import scala.concurrent.duration.Duration @@ -31,12 +33,14 @@ import scala.concurrent.duration.Duration class DSSOrchestratorReceiver(orchestratorService: OrchestratorService, orchestratorPluginService: OrchestratorPluginService, orchestratorContext: DSSOrchestratorContext) extends Receiver { + private val LOGGER = LoggerFactory.getLogger(classOf[DSSOrchestratorReceiver]) + override def receive(message: Any, sender: Sender): Unit = {} override def receiveAndReply(message: Any, sender: Sender): Any = message match { case reqExportOrchestrator: RequestExportOrchestrator => - val dssExportOrcResource: util.Map[String, AnyRef] = orchestratorContext.getDSSOrchestratorPlugin(classOf[ExportDSSOrchestratorPlugin]).exportOrchestrator( + val dssExportOrcResource: OrchestratorExportResult = orchestratorContext.getDSSOrchestratorPlugin(classOf[ExportDSSOrchestratorPlugin]).exportOrchestrator( reqExportOrchestrator.getUserName, reqExportOrchestrator.getOrchestratorId, reqExportOrchestrator.getOrcVersionId, @@ -44,13 +48,13 @@ class DSSOrchestratorReceiver(orchestratorService: OrchestratorService, orchestr reqExportOrchestrator.getDssLabels, reqExportOrchestrator.getAddOrcVersion, reqExportOrchestrator.getWorkspace) - ResponseExportOrchestrator(dssExportOrcResource.get("resourceId").toString, - dssExportOrcResource.get("version").toString, dssExportOrcResource.get("orcVersionId").asInstanceOf[Long] + ResponseExportOrchestrator(dssExportOrcResource.getBmlResource.getResourceId, + dssExportOrcResource.getBmlResource.getVersion, dssExportOrcResource.getOrcVersionId.toLong ) case requestImportOrchestrator: RequestImportOrchestrator => - val importOrcId = orchestratorContext.getDSSOrchestratorPlugin(classOf[ImportDSSOrchestratorPlugin]).importOrchestrator(requestImportOrchestrator) - ResponseImportOrchestrator(importOrcId) + val dssOrchestratorVersion = orchestratorContext.getDSSOrchestratorPlugin(classOf[ImportDSSOrchestratorPlugin]).importOrchestrator(requestImportOrchestrator) + ResponseImportOrchestrator(dssOrchestratorVersion.getOrchestratorId,dssOrchestratorVersion.getVersion) case addVersionAfterPublish: RequestAddVersionAfterPublish => orchestratorContext.getDSSOrchestratorPlugin(classOf[ExportDSSOrchestratorPlugin]).addVersionAfterPublish( @@ -68,6 +72,7 @@ class DSSOrchestratorReceiver(orchestratorService: OrchestratorService, orchestr case requestConversionOrchestration: RequestFrameworkConvertOrchestration => //发布调度,请注意 + LOGGER.info("received requestConversionOrchestration, the class is: {}", requestConversionOrchestration) orchestratorPluginService.convertOrchestration(requestConversionOrchestration) case requestConversionOrchestrationStatus: RequestFrameworkConvertOrchestrationStatus => orchestratorPluginService.getConvertOrchestrationStatus(requestConversionOrchestrationStatus.getId) diff --git a/dss-framework/dss-framework-project-server/pom.xml b/dss-framework/dss-framework-project-server/pom.xml index 911c0a2dc9..0ea3e3a485 100644 --- a/dss-framework/dss-framework-project-server/pom.xml +++ b/dss-framework/dss-framework-project-server/pom.xml @@ -21,8 +21,8 @@ dss com.webank.wedatasphere.dss - 1.1.1 - + 1.1.2 + ../../pom.xml 4.0.0 @@ -41,6 +41,12 @@ + + com.webank.wedatasphere.dss + dss-framework-proxy-user-service + ${dss.version} + + org.apache.commons commons-math3 diff --git a/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/conf/ProjectConf.java b/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/conf/ProjectConf.java index 34021996f3..1d00c56b00 100644 --- a/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/conf/ProjectConf.java +++ b/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/conf/ProjectConf.java @@ -22,5 +22,6 @@ public interface ProjectConf { CommonVars SUPPORT_ABILITY = CommonVars.apply("wds.dss.framework.project.support.ability", "import,export,publish"); + CommonVars MAX_DESC_LENGTH = CommonVars.apply("wds.dss.framework.project.max.desc.length", 250); } diff --git a/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/conf/ProjectSpringConf.java b/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/conf/ProjectSpringConf.java index 9b57855b22..b2f2a1b153 100644 --- a/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/conf/ProjectSpringConf.java +++ b/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/conf/ProjectSpringConf.java @@ -16,6 +16,7 @@ package com.webank.wedatasphere.dss.framework.project.conf; +import com.webank.wedatasphere.dss.common.service.BMLService; import com.webank.wedatasphere.dss.framework.project.service.DSSFrameworkProjectService; import com.webank.wedatasphere.dss.framework.project.service.DSSProjectUserService; import com.webank.wedatasphere.dss.framework.project.service.impl.DSSFrameworkProjectServiceImpl; @@ -39,4 +40,9 @@ public DSSProjectUserService createDSSProjectUserService() { public DSSFrameworkProjectService createDSSFrameworkProjectServiceImpl() { return new DSSFrameworkProjectServiceImpl(); } + + @Bean(name = "projectBmlService") + public BMLService createBmlService() { + return BMLService.getInstance(); + } } diff --git a/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/conf/ValidationExceptionMapper.java b/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/conf/ValidationExceptionMapper.java deleted file mode 100644 index d3c0ea6a82..0000000000 --- a/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/conf/ValidationExceptionMapper.java +++ /dev/null @@ -1,51 +0,0 @@ -/* - * Copyright 2019 WeBank - * 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. - * - */ - -package com.webank.wedatasphere.dss.framework.project.conf; - - -import com.webank.wedatasphere.dss.common.utils.MessageUtils; -import org.apache.linkis.server.Message; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import javax.validation.ConstraintViolation; -import javax.validation.ConstraintViolationException; -import javax.validation.ValidationException; -import javax.ws.rs.core.Response; -import javax.ws.rs.ext.ExceptionMapper; -import javax.ws.rs.ext.Provider; - - - -@Provider -public class ValidationExceptionMapper implements ExceptionMapper { - - private final Logger LOGGER = LoggerFactory.getLogger(getClass()); - - @Override - public Response toResponse(ValidationException e) { - if (LOGGER.isDebugEnabled()){ - LOGGER.debug("failed to validate request bean", e); - } - StringBuilder strBuilder = new StringBuilder(); - for (ConstraintViolation cv : ((ConstraintViolationException) e).getConstraintViolations()) { - strBuilder.append(cv.getMessage()).append(";"); - } - Message message = Message.error(strBuilder.toString()); - return MessageUtils.messageToResponse(message); - } -} diff --git a/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/dao/DSSProjectUserMapper.java b/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/dao/DSSProjectUserMapper.java index 32c6186365..14fc2069d8 100644 --- a/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/dao/DSSProjectUserMapper.java +++ b/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/dao/DSSProjectUserMapper.java @@ -49,4 +49,7 @@ public interface DSSProjectUserMapper extends BaseMapper { Long isWorkspaceUser(@Param("workspaceId")Long workspaceId,@Param("username")String username); List getPrivsByProjectId(@Param("projectId") Long projectId); + + @Select("SELECT admin_permission FROM dss_workspace WHERE id = #{id}") + Integer getWorkspaceAdminPermission(@Param("id") Long id); } diff --git a/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/entity/request/ProjectCreateRequest.java b/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/entity/request/ProjectCreateRequest.java index 4d9710881d..87f90a5b58 100644 --- a/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/entity/request/ProjectCreateRequest.java +++ b/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/entity/request/ProjectCreateRequest.java @@ -157,4 +157,22 @@ public List getOrchestratorModeList() { public void setOrchestratorModeList(List orchestratorModeList) { this.orchestratorModeList = orchestratorModeList; } + + @Override + public String toString() { + return "ProjectCreateRequest{" + + "name='" + name + '\'' + + ", applicationArea=" + applicationArea + + ", business='" + business + '\'' + + ", product='" + product + '\'' + + ", releaseUsers=" + releaseUsers + + ", editUsers=" + editUsers + + ", accessUsers=" + accessUsers + + ", description='" + description + '\'' + + ", workspaceId=" + workspaceId + + ", workspaceName='" + workspaceName + '\'' + + ", devProcessList=" + devProcessList + + ", orchestratorModeList=" + orchestratorModeList + + '}'; + } } diff --git a/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/restful/DSSFrameworkProjectRestfulApi.java b/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/restful/DSSFrameworkProjectRestfulApi.java index 866b800ad7..296ac92fb9 100644 --- a/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/restful/DSSFrameworkProjectRestfulApi.java +++ b/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/restful/DSSFrameworkProjectRestfulApi.java @@ -16,7 +16,11 @@ package com.webank.wedatasphere.dss.framework.project.restful; +import com.webank.wedatasphere.dss.common.auditlog.OperateTypeEnum; +import com.webank.wedatasphere.dss.common.auditlog.TargetTypeEnum; +import com.webank.wedatasphere.dss.common.utils.AuditLogUtils; import com.webank.wedatasphere.dss.common.utils.DSSExceptionUtils; +import com.webank.wedatasphere.dss.framework.project.conf.ProjectConf; import com.webank.wedatasphere.dss.framework.project.entity.DSSProjectDO; import com.webank.wedatasphere.dss.framework.project.entity.request.ProjectCreateRequest; import com.webank.wedatasphere.dss.framework.project.entity.request.ProjectDeleteRequest; @@ -27,35 +31,69 @@ import com.webank.wedatasphere.dss.framework.project.exception.DSSProjectErrorException; import com.webank.wedatasphere.dss.framework.project.service.DSSFrameworkProjectService; import com.webank.wedatasphere.dss.framework.project.service.DSSProjectService; +import com.webank.wedatasphere.dss.framework.project.service.ProjectHttpRequestHook; import com.webank.wedatasphere.dss.framework.project.utils.ApplicationArea; +import com.webank.wedatasphere.dss.framework.proxy.exception.DSSProxyUserErrorException; import com.webank.wedatasphere.dss.standard.app.sso.Workspace; import com.webank.wedatasphere.dss.standard.sso.utils.SSOHelper; +import org.apache.commons.lang3.StringUtils; +import org.apache.commons.lang3.exception.ExceptionUtils; +import org.apache.linkis.common.exception.WarnException; import org.apache.linkis.server.Message; import org.apache.linkis.server.security.SecurityFilter; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.web.bind.annotation.RequestBody; -import org.springframework.web.bind.annotation.RequestMapping; -import org.springframework.web.bind.annotation.RequestMethod; -import org.springframework.web.bind.annotation.RestController; +import org.springframework.web.bind.annotation.*; import javax.servlet.http.HttpServletRequest; import javax.validation.Valid; import java.util.ArrayList; import java.util.Arrays; import java.util.List; +import java.util.Objects; +import java.util.function.Consumer; +import java.util.function.Function; +import java.util.function.Supplier; +import java.util.stream.Collectors; @RequestMapping(path = "/dss/framework/project", produces = {"application/json"}) @RestController public class DSSFrameworkProjectRestfulApi { private static final Logger LOGGER = LoggerFactory.getLogger(DSSFrameworkProjectRestfulApi.class); + private static final int MAX_DESC_LENGTH = ProjectConf.MAX_DESC_LENGTH.getValue(); + private static final int MAX_BUSSINESS_SIZE = 200; @Autowired DSSFrameworkProjectService dssFrameworkProjectService; @Autowired private DSSProjectService projectService; @Autowired private DSSProjectService dssProjectService; + @Autowired + private List projectHttpRequestHooks; + + private Message executePreHook(Function function) { + String errorMsg = projectHttpRequestHooks.stream().map(function).filter(Objects::nonNull).map(Message::getMessage) + .collect(Collectors.joining(", ")); + if(StringUtils.isNotBlank(errorMsg)) { + return Message.error(errorMsg); + } else { + return null; + } + } + + private Message executeAfterHook(Consumer consumer, Supplier supplier) { + try { + projectHttpRequestHooks.forEach(consumer); + } catch (WarnException e) { + LOGGER.error("execute after hook failed!", e); + return Message.error("Execute hook failed, reason: " + e.getDesc()); + } catch (Exception e) { + LOGGER.error("execute after hook failed!", e); + return Message.error("Execute hook failed, reason: " + ExceptionUtils.getRootCauseMessage(e)); + } + return supplier.get(); + } /** * 获取所有工程或者单个工程 @@ -68,38 +106,77 @@ public class DSSFrameworkProjectRestfulApi { public Message getAllProjects(HttpServletRequest request, @RequestBody ProjectQueryRequest projectRequest) { String username = SecurityFilter.getLoginUsername(request); projectRequest.setUsername(username); + Message message = executePreHook(projectHttpRequestHook -> projectHttpRequestHook.beforeGetAllProjects(request, projectRequest)); + if(message != null) { + return message; + } + LOGGER.info("user {} begin to getAllProjects, projectId: {}.", username, projectRequest.getId()); List dssProjectVos = projectService.getListByParam(projectRequest); - Message message = Message.ok("获取工作空间的工程成功").data("projects", dssProjectVos); - return message; + return Message.ok("获取工作空间的工程成功").data("projects", dssProjectVos); } /** * 新建工程,通过和各个AppConn进行交互,将需要满足工程规范的所有的appconn进行创建工程 */ @RequestMapping(path = "createProject", method = RequestMethod.POST) - public Message createProject(HttpServletRequest request, @RequestBody ProjectCreateRequest projectCreateRequest) throws Exception { + public Message createProject(HttpServletRequest request, @RequestBody ProjectCreateRequest projectCreateRequest) throws DSSProxyUserErrorException { String username = SecurityFilter.getLoginUsername(request); Workspace workspace = SSOHelper.getWorkspace(request); + LOGGER.info("user {} begin to createProject, workspace {}, project entity: {}.", username, workspace.getWorkspaceName(), projectCreateRequest); + if(projectCreateRequest.getEditUsers() == null) { + projectCreateRequest.setEditUsers(new ArrayList<>()); + } + if(projectCreateRequest.getReleaseUsers() == null) { + projectCreateRequest.setReleaseUsers(new ArrayList<>()); + } + Message message = executePreHook(projectHttpRequestHook -> projectHttpRequestHook.beforeCreateProject(request, projectCreateRequest)); + if(message != null) { + return message; + } //將创建人默认为发布权限和編輯权限 if (!projectCreateRequest.getEditUsers().contains(username)) { projectCreateRequest.getEditUsers().add(username); } - List releaseUsers = projectCreateRequest.getReleaseUsers(); - if (releaseUsers == null) { - releaseUsers = new ArrayList<>(); - projectCreateRequest.setReleaseUsers(releaseUsers); - } - if (!releaseUsers.contains(username)) { - releaseUsers.add(username); - } - DSSProjectVo dssProjectVo = dssFrameworkProjectService.createProject(projectCreateRequest, username, workspace); - if (dssProjectVo != null) { - return Message.ok("创建工程成功").data("project", dssProjectVo); - } else { - return Message.error("创建工程失败"); + if (!projectCreateRequest.getReleaseUsers().contains(username)) { + projectCreateRequest.getReleaseUsers().add(username); } + return DSSExceptionUtils.getMessage(() -> { + DSSProjectVo dssProjectVo = dssFrameworkProjectService.createProject(projectCreateRequest, username, workspace); + AuditLogUtils.printLog(username, workspace.getWorkspaceId(), workspace.getWorkspaceName(), TargetTypeEnum.PROJECT, + dssProjectVo.getId(),dssProjectVo.getName(), OperateTypeEnum.CREATE,projectCreateRequest); + return dssProjectVo; + }, + dssProjectVo -> + executeAfterHook(projectHttpRequestHook -> projectHttpRequestHook.afterCreateProject(request, projectCreateRequest, dssProjectVo), + () -> Message.ok("创建工程成功.").data("project", dssProjectVo)), + String.format("用户 %s 创建工程 %s 失败. ", username, projectCreateRequest.getName())); + } + + @RequestMapping(path = "checkProjectName", method = RequestMethod.GET) + public Message checkProjectName(HttpServletRequest request, @RequestParam(name = "name") String name) { + String username = SecurityFilter.getLoginUsername(request); + Workspace workspace = SSOHelper.getWorkspace(request); + LOGGER.info("user {} begin to checkProjectName: {}", username, name); + return DSSExceptionUtils.getMessage(() -> dssFrameworkProjectService.checkProjectName(name, workspace, username), + () -> Message.ok().data("repeat", false), String.format("用户 %s,创建工程名 %s 失败. ", username, name)); } + @RequestMapping(path = "getProjectInfo", method = RequestMethod.GET) + public Message getProjectInfoByName(@RequestParam(name = "projectName") String projectName){ + DSSProjectDO dbProject = dssProjectService.getProjectByName(projectName); + Message message; + if(dbProject==null){ + String msg = String.format("project %s does not exist.", projectName); + message = Message.error(msg); + }else{ + DSSProjectVo dssProjectVo = new DSSProjectVo(); + dssProjectVo.setDescription(dbProject.getDescription()); + dssProjectVo.setId(dbProject.getId()); + dssProjectVo.setName(dbProject.getName()); + message = Message.ok().data("project", dssProjectVo); + } + return message; + } /** * 编辑工程 * @@ -109,14 +186,27 @@ public Message createProject(HttpServletRequest request, @RequestBody ProjectCre */ @RequestMapping(path = "modifyProject", method = RequestMethod.POST) public Message modifyProject(HttpServletRequest request, @RequestBody ProjectModifyRequest projectModifyRequest) throws Exception { + if (projectModifyRequest.getId() == null || projectModifyRequest.getId() < 0) { + return Message.error("project id is null, cannot modify it."); + } String username = SecurityFilter.getLoginUsername(request); Workspace workspace = SSOHelper.getWorkspace(request); + LOGGER.info("user {} begin to modifyProject, workspace {}, project entity: {}.", username, workspace.getWorkspaceName(), projectModifyRequest); + Message message = executePreHook(projectHttpRequestHook -> projectHttpRequestHook.beforeModifyProject(request, projectModifyRequest)); + if(message != null) { + return message; + } + if (projectModifyRequest.getDescription().length() > MAX_DESC_LENGTH) { + return Message.error("The project description information is too long, exceeding the maximum length:" + MAX_DESC_LENGTH); + } + if(org.apache.commons.lang.StringUtils.isNotEmpty(projectModifyRequest.getBusiness()) && projectModifyRequest.getBusiness().length() > MAX_BUSSINESS_SIZE){ + return Message.error("The project bussiness is too long, exceeding the maximum length:" + MAX_BUSSINESS_SIZE); + } DSSProjectDO dbProject = dssProjectService.getProjectById(projectModifyRequest.getId()); //工程不存在 if (dbProject == null) { - LOGGER.error("{} project id is null, can not modify", projectModifyRequest.getName()); - DSSExceptionUtils.dealErrorException(60021, - String.format("%s project id is null, can not modify", projectModifyRequest.getName()), DSSProjectErrorException.class); + LOGGER.error("project {} is not exists.", projectModifyRequest.getName()); + return Message.error(String.format("project %s is not exists.", projectModifyRequest.getName())); } String createUsername = dbProject.getUsername(); //將创建人默认为发布权限和編輯权限 @@ -131,8 +221,15 @@ public Message modifyProject(HttpServletRequest request, @RequestBody ProjectMod if (!releaseUsers.contains(createUsername)) { releaseUsers.add(createUsername); } - dssFrameworkProjectService.modifyProject(projectModifyRequest, dbProject, username, workspace); - return Message.ok("修改工程成功"); + return DSSExceptionUtils.getMessage(() -> { + dssFrameworkProjectService.modifyProject(projectModifyRequest, dbProject, username, workspace); + AuditLogUtils.printLog(username, workspace.getWorkspaceId(), workspace.getWorkspaceName(), TargetTypeEnum.PROJECT, + projectModifyRequest.getId(),projectModifyRequest.getName(), OperateTypeEnum.UPDATE,projectModifyRequest); + }, + () -> + executeAfterHook(projectHttpRequestHook -> projectHttpRequestHook.afterModifyProject(request, projectModifyRequest), + () -> Message.ok("修改工程成功.")), + String.format("用户 %s 修改工程 %s 失败. ", username, projectModifyRequest.getName())); } /** @@ -143,13 +240,25 @@ public Message modifyProject(HttpServletRequest request, @RequestBody ProjectMod * @return */ @RequestMapping(path = "deleteProject", method = RequestMethod.POST) - public Message deleteProject(HttpServletRequest request, @RequestBody ProjectDeleteRequest projectDeleteRequest) throws Exception { + public Message deleteProject(HttpServletRequest request, @RequestBody ProjectDeleteRequest projectDeleteRequest) { String username = SecurityFilter.getLoginUsername(request); Workspace workspace = SSOHelper.getWorkspace(request); - // 检查是否具有删除项目权限 - projectService.isDeleteProjectAuth(projectDeleteRequest.getId(), username); - projectService.deleteProject(username, projectDeleteRequest, workspace); - return Message.ok("删除工程成功"); + Message message = executePreHook(projectHttpRequestHook -> projectHttpRequestHook.beforeDeleteProject(request, projectDeleteRequest)); + if(message != null) { + return message; + } + LOGGER.info("user {} begin to deleteProject, workspace {}, project entity: {}.", username, workspace.getWorkspaceName(), projectDeleteRequest); + return DSSExceptionUtils.getMessage(() -> { + // 检查是否具有删除项目权限 + projectService.isDeleteProjectAuth(projectDeleteRequest.getId(), username); + DSSProjectDO dssProjectDO= projectService.getProjectById(projectDeleteRequest.getId()); + projectService.deleteProject(username, projectDeleteRequest, workspace,dssProjectDO); + AuditLogUtils.printLog(username, workspace.getWorkspaceId(), workspace.getWorkspaceName(), TargetTypeEnum.PROJECT, + dssProjectDO.getId(),dssProjectDO.getName(), OperateTypeEnum.DELETE,projectDeleteRequest); }, + () -> + executeAfterHook(projectHttpRequestHook -> projectHttpRequestHook.afterDeleteProject(request, projectDeleteRequest), + () -> Message.ok("删除工程成功.")), + String.format("用户 %s 删除工程失败. ", username)); } @RequestMapping(path = "listApplicationAreas", method = RequestMethod.GET) @@ -188,9 +297,12 @@ public Message getProjectAbilities(HttpServletRequest request) { public Message getDeletedProjects(HttpServletRequest request, @Valid @RequestBody ProjectQueryRequest projectRequest) { String username = SecurityFilter.getLoginUsername(request); projectRequest.setUsername(username); + Message message = executePreHook(projectHttpRequestHook -> projectHttpRequestHook.beforeGetDeletedProject(request, projectRequest)); + if(message != null) { + return message; + } List dssProjectVos = projectService.getDeletedProjects(projectRequest); - Message message = Message.ok("获取工作空间已删除的工程成功").data("projects", dssProjectVos); - return message; + return Message.ok("获取工作空间已删除的工程成功").data("projects", dssProjectVos); } diff --git a/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/service/DSSFrameworkProjectService.java b/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/service/DSSFrameworkProjectService.java index a55fe76198..46c57a0816 100644 --- a/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/service/DSSFrameworkProjectService.java +++ b/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/service/DSSFrameworkProjectService.java @@ -21,6 +21,7 @@ import com.webank.wedatasphere.dss.framework.project.entity.request.ProjectModifyRequest; import com.webank.wedatasphere.dss.framework.project.entity.vo.DSSProjectDetailVo; import com.webank.wedatasphere.dss.framework.project.entity.vo.DSSProjectVo; +import com.webank.wedatasphere.dss.framework.project.exception.DSSProjectErrorException; import com.webank.wedatasphere.dss.standard.app.sso.Workspace; public interface DSSFrameworkProjectService { @@ -38,4 +39,6 @@ public interface DSSFrameworkProjectService { DSSProjectVo createProject(ProjectCreateRequest projectCreateRequest, String username, Workspace workspace) throws Exception; void modifyProject(ProjectModifyRequest projectModifyRequest, DSSProjectDO dbProject, String username, Workspace workspace) throws Exception; + + void checkProjectName(String name, Workspace workspace,String username) throws DSSProjectErrorException; } diff --git a/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/service/DSSProjectService.java b/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/service/DSSProjectService.java index 477a08cbae..a8781eca50 100644 --- a/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/service/DSSProjectService.java +++ b/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/service/DSSProjectService.java @@ -66,7 +66,7 @@ public interface DSSProjectService extends IService { Long getAppConnProjectId(Long appInstanceId, Long dssProjectId); - void deleteProject(String username, ProjectDeleteRequest projectDeleteRequest, Workspace workspace) throws Exception; + void deleteProject(String username, ProjectDeleteRequest projectDeleteRequest, Workspace workspace, DSSProjectDO dssProjectDO) throws Exception; List getProjectAbilities(String username); diff --git a/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/service/ProjectHttpRequestHook.java b/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/service/ProjectHttpRequestHook.java new file mode 100644 index 0000000000..2b1572971d --- /dev/null +++ b/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/service/ProjectHttpRequestHook.java @@ -0,0 +1,84 @@ +package com.webank.wedatasphere.dss.framework.project.service; + +import com.webank.wedatasphere.dss.framework.project.entity.request.ProjectCreateRequest; +import com.webank.wedatasphere.dss.framework.project.entity.request.ProjectDeleteRequest; +import com.webank.wedatasphere.dss.framework.project.entity.request.ProjectModifyRequest; +import com.webank.wedatasphere.dss.framework.project.entity.request.ProjectQueryRequest; +import com.webank.wedatasphere.dss.framework.project.entity.vo.DSSProjectVo; +import org.apache.linkis.server.Message; + +import javax.servlet.http.HttpServletRequest; + +/** + * @author enjoyyin + * @date 2022-09-21 + * @since 0.5.0 + */ +public interface ProjectHttpRequestHook { + + /** + * 如果前置操作失败,则返回 Message,否则返回 null即可 + * @param request + * @param projectRequest + * @return 前置操作失败,请返回 Message,否则返回 null + */ + Message beforeGetAllProjects(HttpServletRequest request, ProjectQueryRequest projectRequest); + + /** + * 如果前置操作失败,则返回 Message,否则返回 null即可 + * @param request + * @param projectCreateRequest + * @return 前置操作失败,请返回 Message,否则返回 null + */ + Message beforeCreateProject(HttpServletRequest request, ProjectCreateRequest projectCreateRequest); + + /** + * 创建成功后的后置操作。 + * 请注意:我们忽略了创建失败的后置操作。 + * @param request + * @param projectCreateRequest + * @param dssProjectVo + */ + default void afterCreateProject(HttpServletRequest request, ProjectCreateRequest projectCreateRequest, DSSProjectVo dssProjectVo){ + } + + /** + * 如果前置操作失败,则返回 Message,否则返回 null即可 + * @param request + * @param projectModifyRequest + * @return 前置操作失败,请返回 Message,否则返回 null + */ + Message beforeModifyProject(HttpServletRequest request, ProjectModifyRequest projectModifyRequest); + + /** + * 更新成功后的后置操作。 + * 请注意:我们忽略了更新失败的后置操作。 + * @param request + * @param projectModifyRequest + */ + default void afterModifyProject(HttpServletRequest request, ProjectModifyRequest projectModifyRequest) { + } + /** + * 如果前置操作失败,请返回 Message,否则返回 null即可 + * @param request + * @param projectDeleteRequest + * @return 前置操作失败,请返回 Message,否则返回 null + */ + Message beforeDeleteProject(HttpServletRequest request, ProjectDeleteRequest projectDeleteRequest); + + /** + * 更新成功后的后置操作。 + * 请注意:我们忽略了更新失败的后置操作。 + * @param request + * @param projectDeleteRequest + */ + default void afterDeleteProject(HttpServletRequest request, ProjectDeleteRequest projectDeleteRequest) { + } + /** + * 如果前置操作失败,则返回 Message,否则返回 null即可 + * @param request + * @param projectRequest + * @return 前置操作失败,请返回 Message,否则返回 null + */ + Message beforeGetDeletedProject(HttpServletRequest request, ProjectQueryRequest projectRequest); +} diff --git a/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/service/impl/DSSFrameworkProjectServiceImpl.java b/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/service/impl/DSSFrameworkProjectServiceImpl.java index 9b496ea148..d9b0ab3b1c 100644 --- a/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/service/impl/DSSFrameworkProjectServiceImpl.java +++ b/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/service/impl/DSSFrameworkProjectServiceImpl.java @@ -20,6 +20,7 @@ import com.webank.wedatasphere.dss.appconn.core.ext.OnlyStructureAppConn; import com.webank.wedatasphere.dss.common.entity.project.DSSProject; import com.webank.wedatasphere.dss.common.utils.DSSExceptionUtils; +import com.webank.wedatasphere.dss.framework.project.conf.ProjectConf; import com.webank.wedatasphere.dss.framework.project.contant.ProjectServerResponse; import com.webank.wedatasphere.dss.framework.project.entity.DSSProjectDO; import com.webank.wedatasphere.dss.framework.project.entity.DSSProjectUser; @@ -41,8 +42,10 @@ import com.webank.wedatasphere.dss.standard.common.desc.AppInstance; import com.webank.wedatasphere.dss.standard.common.exception.operation.ExternalOperationFailedException; import org.apache.commons.collections4.CollectionUtils; +import org.apache.commons.lang.StringUtils; import org.apache.commons.lang.exception.ExceptionUtils; import org.apache.linkis.common.conf.CommonVars; +import org.apache.linkis.common.exception.WarnException; import org.apache.linkis.protocol.util.ImmutablePair; import org.slf4j.Logger; import org.slf4j.LoggerFactory; @@ -59,8 +62,9 @@ public class DSSFrameworkProjectServiceImpl implements DSSFrameworkProjectService { private static final Logger LOGGER = LoggerFactory.getLogger(DSSFrameworkProjectServiceImpl.class); - public static final int MAX_PROJECT_NAME_SIZE = 150; - public static final int MAX_PROJECT_DESC_SIZE = 2048; + public static final int MAX_PROJECT_NAME_SIZE = 64; + public static final int MAX_PROJECT_DESC_SIZE = ProjectConf.MAX_DESC_LENGTH.getValue(); + private static final int MAX_PROJECT_BUSSINESS_SIZE = 200; @Autowired private DSSProjectService dssProjectService; @Autowired @@ -85,27 +89,15 @@ public DSSProjectVo createProject(ProjectCreateRequest projectCreateRequest, Str DSSExceptionUtils.dealErrorException(ProjectServerResponse.PROJECT_USER_NOT_IN_WORKSPACE.getCode(), ProjectServerResponse.PROJECT_USER_NOT_IN_WORKSPACE.getMsg(), DSSProjectErrorException.class); } //增加名称长度限制 - if(projectCreateRequest.getName().length()> MAX_PROJECT_NAME_SIZE || projectCreateRequest.getDescription().length() > MAX_PROJECT_DESC_SIZE){ - DSSExceptionUtils.dealErrorException(60021,"project name or desc is too long for project name length is " + projectCreateRequest.getName().length() - + ", project desc length is " + projectCreateRequest.getDescription().length(), DSSProjectErrorException.class); + if(projectCreateRequest.getName().length()> MAX_PROJECT_NAME_SIZE){ + DSSExceptionUtils.dealErrorException(60021,"project name is too long. the length must be less than " + MAX_PROJECT_NAME_SIZE, DSSProjectErrorException.class); + } else if(projectCreateRequest.getDescription().length() > MAX_PROJECT_DESC_SIZE) { + DSSExceptionUtils.dealErrorException(60021,"project description is too long. the length must be less than " + MAX_PROJECT_DESC_SIZE, DSSProjectErrorException.class); + } else if(StringUtils.isNotEmpty(projectCreateRequest.getBusiness()) && projectCreateRequest.getBusiness().length() > MAX_PROJECT_BUSSINESS_SIZE){ + DSSExceptionUtils.dealErrorException(60021,"project bussiness is too long. the length must be less than " + MAX_PROJECT_BUSSINESS_SIZE, DSSProjectErrorException.class); } - //判断工程是否存在相同的名称 - DSSProjectDO dbProject = dssProjectService.getProjectByName(projectCreateRequest.getName()); - if (dbProject != null) { - DSSExceptionUtils.dealErrorException(60022, String.format("project name %s has already been exists.", projectCreateRequest.getName()), DSSProjectErrorException.class); - } - - List appConnNameList = new ArrayList<>(1); - //判断已有组件是否已经存在相同的工程名称 - try { - isExistSameProjectName(projectCreateRequest, workspace, appConnNameList, username); - } catch (Exception e) { - throw new DSSProjectErrorException(71000, "向第三方应用发起检查工程名是否重复失败,原因:" + ExceptionUtils.getRootCauseMessage(e), e); - } - if (!appConnNameList.isEmpty()) { - throw new DSSProjectErrorException(71000, String.join(", ", appConnNameList) + " 已存在相同项目名称,请重新命名!"); - } + this.checkProjectName(projectCreateRequest.getName(), workspace, username); Map projectMap = createAppConnProject(projectCreateRequest, workspace, username); //3.保存dss_project @@ -124,6 +116,29 @@ public DSSProjectVo createProject(ProjectCreateRequest projectCreateRequest, Str } + @Override + public void checkProjectName(String name, Workspace workspace, String username) throws DSSProjectErrorException { + //判断工程是否存在相同的名称 + DSSProjectDO dbProject = dssProjectService.getProjectByName(name); + if (dbProject != null) { + DSSExceptionUtils.dealErrorException(60022, String.format("project name %s has already been exists.", name), DSSProjectErrorException.class); + } + List appConnNameList = new ArrayList<>(1); + //判断已有组件是否已经存在相同的工程名称 + ProjectCreateRequest projectCreateRequest = new ProjectCreateRequest(); + projectCreateRequest.setName(name); + try { + isExistSameProjectName(projectCreateRequest, workspace, appConnNameList, username); + } catch (WarnException e) { + throw new DSSProjectErrorException(71000, "向第三方应用发起检查工程名是否重复失败. " + e.getDesc(), e); + } catch (Exception e) { + throw new DSSProjectErrorException(71000, "向第三方应用发起检查工程名是否重复失败. 原因:" + ExceptionUtils.getRootCauseMessage(e), e); + } + if (!appConnNameList.isEmpty()) { + throw new DSSProjectErrorException(71000, String.join(", ", appConnNameList) + " 已存在相同项目名称,请重新命名!"); + } + } + @Override public void modifyProject(ProjectModifyRequest projectModifyRequest, DSSProjectDO dbProject, String username, Workspace workspace) throws Exception { //如果不是工程的创建人,则校验是否管理员 @@ -140,6 +155,10 @@ public void modifyProject(ProjectModifyRequest projectModifyRequest, DSSProjectD } //调用第三方的工程修改接口 dbProject.setUsername(username); + dbProject.setApplicationArea(Integer.valueOf(projectModifyRequest.getApplicationArea())); + dbProject.setDescription(projectModifyRequest.getDescription()); + dbProject.setBusiness(projectModifyRequest.getBusiness()); + dbProject.setProduct(projectModifyRequest.getProduct()); modifyThirdProject(projectModifyRequest, dbProject, workspace); //1.统一修改各个接入的第三方的系统的工程状态信息 @@ -218,7 +237,7 @@ private void isExistSameProjectName(ProjectCreateRequest dssProjectCreateRequest if(responseRef.getRefProjectId() != null && responseRef.getRefProjectId() > 0 && projectService.isProjectNameUnique()) { appConnNameList.add(pair.getLeft().getAppDesc().getAppName()); } - }, "check project name " + dssProjectCreateRequest.getName() + " whether third-party refProject is exists"); + }, "check project name " + dssProjectCreateRequest.getName() + " whether it is exists"); } // 1.新建DSS工程,这样才能进行回滚,如果后面去DSS工程,可能会由于DSS工程建立失败了,但是仍然无法去回滚第三方系统的工程 新增dss_project调用 @@ -246,20 +265,34 @@ private Map createAppConnProject(ProjectCreateRequest dssProj appConnListMap.get(pair.left).add(pair.right); }, "create refProject " + dssProjectCreateRequest.getName()); } catch (RuntimeException e) { - LOGGER.error("create appconn project failed:", e); + LOGGER.error("create AppConn project failed!", e); if(!STRICT_PROJECT_CREATE_MODE) { throw e; } + LOGGER.warn("Strict project create mode is opened, now try to delete the projects {} created in external AppConns.", dssProjectCreateRequest.getName()); // 如果创建失败并且是严格创建模式 // 如果一个AppInstance实例是失败的,那么我们将所有已经建的工程给撤销掉 - appConnListMap.forEach((key, value) -> value.forEach(appInstance -> { - StructureOperationUtils.tryProjectOperation(() -> ((OnlyStructureAppConn) key).getOrCreateStructureStandard().getProjectService(appInstance), - ProjectService::getProjectDeletionOperation, null, - refProjectContentRequestRef -> refProjectContentRequestRef.setRefProjectId(projectMap.get(appInstance)) - .setProjectName(dssProjectCreateRequest.getName()).setWorkspace(workspace).setUserName(username), - (structureOperation, structureRequestRef) -> ((ProjectDeletionOperation) structureOperation).deleteProject((RefProjectContentRequestRef) structureRequestRef), - "delete refProject " + dssProjectCreateRequest.getName()); - })); + try { + appConnListMap.forEach((key, value) -> value.forEach(appInstance -> { + LOGGER.warn("{} try to delete the create-failed project {} with AppInstance {}!", key.getAppDesc().getAppName(), dssProjectCreateRequest.getName(), appInstance.getBaseUrl()); + StructureOperationUtils.tryProjectOperation(() -> ((OnlyStructureAppConn) key).getOrCreateStructureStandard().getProjectService(appInstance), + ProjectService::getProjectDeletionOperation, null, + refProjectContentRequestRef -> refProjectContentRequestRef.setRefProjectId(projectMap.get(appInstance)) + .setProjectName(dssProjectCreateRequest.getName()).setWorkspace(workspace).setUserName(username), + (structureOperation, structureRequestRef) -> ((ProjectDeletionOperation) structureOperation).deleteProject((RefProjectContentRequestRef) structureRequestRef), + key.getAppDesc().getAppName() + " try to delete refProject " + dssProjectCreateRequest.getName()); + })); + } catch (RuntimeException e1) { + // 原则上,如果删除失败,只是会存在一个无用的工程,不会对用户的使用造成影响 + LOGGER.error("try to delete the create-failed project {} in AppConn failed!", dssProjectCreateRequest.getName(), e1); + String errorMsg; + if(e instanceof WarnException) { + errorMsg = String.format("%s. please notify the admin that try to rollback the created projects in external AppConn failed.", ((WarnException) e).getDesc()); + } else { + errorMsg = String.format("create project failed, Reason: %s. please notify the admin that try to rollback the created projects in external AppConn failed.", ExceptionUtils.getRootCauseMessage(e)); + } + throw new ExternalOperationFailedException(50009, errorMsg, e); + } throw e; } return projectMap; diff --git a/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/service/impl/DSSProjectServiceImpl.java b/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/service/impl/DSSProjectServiceImpl.java index 09c895930f..17187a6117 100644 --- a/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/service/impl/DSSProjectServiceImpl.java +++ b/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/service/impl/DSSProjectServiceImpl.java @@ -151,8 +151,14 @@ public DSSProjectDO getProjectById(Long id) { public List getListByParam(ProjectQueryRequest projectRequest) { //根据dss_project、dss_project_user查询出所在空间登录用户相关的工程 List list; + //判断工作空间是否设置了管理员能否查看该工作空间下所有项目的权限 + Integer workspaceAdminPermission = projectUserMapper.getWorkspaceAdminPermission(projectRequest.getWorkspaceId()); if (isWorkspaceAdmin(projectRequest.getWorkspaceId(), projectRequest.getUsername())) { - list = projectMapper.getListForAdmin(projectRequest); + if (workspaceAdminPermission == 1){ + list = projectMapper.getListForAdmin(projectRequest); + } else { + list = projectMapper.getListByParam(projectRequest); + } } else { list = projectMapper.getListByParam(projectRequest); } @@ -161,11 +167,12 @@ public List getListByParam(ProjectQueryRequest projectRequest) } List projectResponseList = new ArrayList<>(); + ProjectResponse projectResponse; for (QueryProjectVo projectVo : list) { if (projectVo.getVisible() == 0) { continue; } - ProjectResponse projectResponse = new ProjectResponse(); + projectResponse = new ProjectResponse(); projectResponse.setApplicationArea(projectVo.getApplicationArea()); projectResponse.setId(projectVo.getId()); projectResponse.setBusiness(projectVo.getBusiness()); @@ -254,8 +261,7 @@ public Long getAppConnProjectId(Long appInstanceId, Long dssProjectId) { } @Override - public void deleteProject(String username, ProjectDeleteRequest projectDeleteRequest, Workspace workspace) throws Exception { - DSSProjectDO dssProjectDO = projectMapper.selectById(projectDeleteRequest.getId()); + public void deleteProject(String username, ProjectDeleteRequest projectDeleteRequest, Workspace workspace, DSSProjectDO dssProjectDO) throws Exception { if (dssProjectDO == null) { throw new DSSErrorException(600001, "工程不存在!"); } diff --git a/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/service/impl/DSSProjectUserServiceImpl.java b/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/service/impl/DSSProjectUserServiceImpl.java index c6bb7c1659..548c375bf5 100644 --- a/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/service/impl/DSSProjectUserServiceImpl.java +++ b/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/service/impl/DSSProjectUserServiceImpl.java @@ -27,7 +27,7 @@ import com.webank.wedatasphere.dss.framework.project.entity.request.ProjectCreateRequest; import com.webank.wedatasphere.dss.framework.project.entity.request.ProjectModifyRequest; import com.webank.wedatasphere.dss.framework.project.exception.DSSProjectErrorException; -import com.webank.wedatasphere.dss.framework.project.server.service.BMLService; +import com.webank.wedatasphere.dss.common.service.BMLService; import com.webank.wedatasphere.dss.framework.project.service.DSSProjectUserService; import com.webank.wedatasphere.dss.framework.project.utils.ProjectUserUtils; import com.webank.wedatasphere.dss.standard.app.sso.Workspace; @@ -48,7 +48,7 @@ public class DSSProjectUserServiceImpl implements DSSProjectUserService { @Autowired private DSSProjectUserMapper projectUserMapper; @Autowired - @Qualifier("projectServerBMLService") + @Qualifier("projectBmlService") private BMLService bmlService; @Autowired private DSSProjectMapper dssProjectMapper; diff --git a/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/service/impl/ProxyUserProjectHttpRequestHook.java b/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/service/impl/ProxyUserProjectHttpRequestHook.java new file mode 100644 index 0000000000..8122693901 --- /dev/null +++ b/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/service/impl/ProxyUserProjectHttpRequestHook.java @@ -0,0 +1,122 @@ +package com.webank.wedatasphere.dss.framework.project.service.impl; + +import com.webank.wedatasphere.dss.framework.project.entity.request.ProjectCreateRequest; +import com.webank.wedatasphere.dss.framework.project.entity.request.ProjectDeleteRequest; +import com.webank.wedatasphere.dss.framework.project.entity.request.ProjectModifyRequest; +import com.webank.wedatasphere.dss.framework.project.entity.request.ProjectQueryRequest; +import com.webank.wedatasphere.dss.framework.project.entity.response.ProjectResponse; +import com.webank.wedatasphere.dss.framework.project.service.DSSProjectService; +import com.webank.wedatasphere.dss.framework.project.service.ProjectHttpRequestHook; +import com.webank.wedatasphere.dss.framework.proxy.conf.ProxyUserConfiguration; +import com.webank.wedatasphere.dss.framework.proxy.exception.DSSProxyUserErrorException; +import com.webank.wedatasphere.dss.framework.proxy.service.DssProxyUserService; +import com.webank.wedatasphere.dss.standard.app.sso.Workspace; +import com.webank.wedatasphere.dss.standard.sso.utils.SSOHelper; +import org.apache.commons.collections4.CollectionUtils; +import org.apache.commons.lang3.StringUtils; +import org.apache.linkis.server.Message; +import org.apache.linkis.server.security.SecurityFilter; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Component; + +import javax.servlet.http.HttpServletRequest; +import java.util.Arrays; +import java.util.List; +import java.util.function.Consumer; +import java.util.function.Function; +import java.util.stream.Collectors; + +import org.slf4j.Logger; + +/** + * @author enjoyyin + * @date 2022-09-21 + * @since 0.5.0 + */ +@Component +public class ProxyUserProjectHttpRequestHook implements ProjectHttpRequestHook { + private static final Logger LOGGER= LoggerFactory.getLogger(ProxyUserProjectHttpRequestHook.class); + @Autowired + private DSSProjectService dssProjectService; + @Autowired + private DssProxyUserService dssProxyUserService; + + private Message doProxyUserFunction(HttpServletRequest request, Function function) { + if(!ProxyUserConfiguration.isProxyUserEnable()) { + return null; + } + String proxyUser; + try { + proxyUser = dssProxyUserService.getProxyUser(request); + } catch (DSSProxyUserErrorException e) { + LOGGER.error("getProxyUser Failed,cookie is :{}", Arrays.stream(request.getCookies()) + .map(cookie->String.format("%s=%s",cookie.getName(),cookie.getValue())).collect(Collectors.joining(","))); + LOGGER.error("getProxyUser failed.",e); + return Message.error(e.getMessage()); + } + return function.apply(proxyUser); + } + + private Message doProxyUserConsumer(HttpServletRequest request, Consumer consumer) { + return doProxyUserFunction(request, proxyUser -> { + consumer.accept(proxyUser); + return null; + }); + } + + @Override + public Message beforeGetAllProjects(HttpServletRequest request, ProjectQueryRequest projectRequest) { + return doProxyUserConsumer(request, projectRequest::setUsername); + } + + @Override + public Message beforeCreateProject(HttpServletRequest request, ProjectCreateRequest projectCreateRequest) { + return doProxyUserFunction(request, proxyUser -> { + if(CollectionUtils.isNotEmpty(projectCreateRequest.getAccessUsers()) || + CollectionUtils.isNotEmpty(projectCreateRequest.getEditUsers()) || CollectionUtils.isNotEmpty(projectCreateRequest.getReleaseUsers())) { + return Message.error("This environment is not allowed to set accessUsers, editUsers or ReleaseUsers(本环境不允许设置发布权限、编辑权限和查看权限,请删除相关权限后再重试)."); + } + String userName= SecurityFilter.getLoginUsername(request); + if(userName.equals(proxyUser) + &&!StringUtils.startsWithIgnoreCase(proxyUser,"WTSS_") + &&!StringUtils.startsWithIgnoreCase(proxyUser,"hduser")){ + return Message.error("only ops proxy user can create project(只允许代理用户创建工程)."); + } + projectCreateRequest.getEditUsers().add(proxyUser); + projectCreateRequest.getReleaseUsers().add(proxyUser); + return null; + }); + } + + @Override + public Message beforeModifyProject(HttpServletRequest request, ProjectModifyRequest projectModifyRequest) { + Workspace workspace = SSOHelper.getWorkspace(request); + return doProxyUserFunction(request, proxyUser -> { + ProjectQueryRequest projectQueryRequest = new ProjectQueryRequest(); + projectQueryRequest.setId(projectModifyRequest.getId()); + // 这里直接使用代理用户来查询 + projectQueryRequest.setUsername(proxyUser); + projectQueryRequest.setWorkspaceId(workspace.getWorkspaceId()); + List projectResponseList = dssProjectService.getListByParam(projectQueryRequest); + if(CollectionUtils.isEmpty(projectResponseList)) { + return Message.error("You have no permission to modify this project."); + } else if(!CollectionUtils.isEqualCollection(projectModifyRequest.getEditUsers(), projectResponseList.get(0).getEditUsers()) || + !CollectionUtils.isEqualCollection(projectModifyRequest.getReleaseUsers(), projectResponseList.get(0).getReleaseUsers()) || + CollectionUtils.isNotEmpty(projectModifyRequest.getAccessUsers())) { + return Message.error("This environment is not allowed to set accessUsers, editUsers or ReleaseUsers(本环境不允许设置发布权限、编辑权限和查看权限,请删除相关权限后再重试)."); + } + return null; + }); + } + + @Override + public Message beforeDeleteProject(HttpServletRequest request, ProjectDeleteRequest projectDeleteRequest) { + return null; + } + + @Override + public Message beforeGetDeletedProject(HttpServletRequest request, ProjectQueryRequest projectRequest) { + return doProxyUserConsumer(request, projectRequest::setUsername); + } +} diff --git a/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/utils/ProjectOperationUtils.java b/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/utils/ProjectOperationUtils.java index 8bf07bc2d3..5ef95a427c 100644 --- a/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/utils/ProjectOperationUtils.java +++ b/dss-framework/dss-framework-project-server/src/main/java/com/webank/wedatasphere/dss/framework/project/utils/ProjectOperationUtils.java @@ -52,7 +52,7 @@ public static void tryPr (structureOperation, structureRequestRef) -> { structureRequestRef.setDSSLabels(appInstance.getLabels()).setWorkspace(workspace); return responseRefConsumer.apply(structureOperation, (K) structureRequestRef); - }, errorMsg); + }, appConn.getAppDesc().getAppName() + " try to " + errorMsg); if(dealResponseRefConsumer != null) { dealResponseRefConsumer.accept(new ImmutablePair<>(appConn, appInstance), responseRef); } diff --git a/dss-framework/dss-framework-project-server/src/main/scala/com/webank/wedatasphere/dss/framework/project/server/rpc/ProjectReceiver.scala b/dss-framework/dss-framework-project-server/src/main/scala/com/webank/wedatasphere/dss/framework/project/server/rpc/ProjectReceiver.scala index 4dd0d108af..e98000bbdb 100644 --- a/dss-framework/dss-framework-project-server/src/main/scala/com/webank/wedatasphere/dss/framework/project/server/rpc/ProjectReceiver.scala +++ b/dss-framework/dss-framework-project-server/src/main/scala/com/webank/wedatasphere/dss/framework/project/server/rpc/ProjectReceiver.scala @@ -19,9 +19,7 @@ package com.webank.wedatasphere.dss.framework.project.server.rpc import java.util import com.webank.wedatasphere.dss.common.entity.project.DSSProject -import com.webank.wedatasphere.dss.common.protocol.{ProxyUserCheckRequest, ResponseProxyUserCheck} -import com.webank.wedatasphere.dss.common.protocol.project.{ProjectInfoRequest, ProjectRefIdRequest, ProjectRefIdResponse, ProjectRelationRequest, ProjectRelationResponse, ProjectUserAuthRequest, ProjectUserAuthResponse} -import com.webank.wedatasphere.dss.framework.admin.service.DssProxyUserService +import com.webank.wedatasphere.dss.common.protocol.project._ import com.webank.wedatasphere.dss.framework.project.entity.DSSProjectDO import com.webank.wedatasphere.dss.framework.project.entity.vo.ProjectInfoVo import com.webank.wedatasphere.dss.framework.project.service.{DSSProjectService, DSSProjectUserService} @@ -38,8 +36,7 @@ import scala.concurrent.duration.Duration @Component class ProjectReceiver(projectService: DSSProjectService, dssWorkspaceUserService: DSSWorkspaceUserService, - projectUserService: DSSProjectUserService, - dssProxyUserService: DssProxyUserService) extends Receiver { + projectUserService: DSSProjectUserService) extends Receiver { override def receive(message: Any, sender: Sender): Unit = { @@ -54,7 +51,7 @@ class ProjectReceiver(projectService: DSSProjectService, val appConnProjectId = projectService.getAppConnProjectId(dssProjectId, appConnName, dssLabels) new ProjectRelationResponse(dssProjectId, appConnName, dssLabels, appConnProjectId) case projectRefIdRequest: ProjectRefIdRequest => - val refProjectId = projectService.getAppConnProjectId(projectRefIdRequest.getAppInstanceId, projectRefIdRequest.getDssProjectId); + val refProjectId = projectService.getAppConnProjectId(projectRefIdRequest.getAppInstanceId, projectRefIdRequest.getDssProjectId) new ProjectRefIdResponse(projectRefIdRequest.getAppInstanceId, projectRefIdRequest.getDssProjectId, refProjectId); case requestUserWorkspace: RequestUserWorkspace => val userWorkspaceIds: util.List[Integer] = dssWorkspaceUserService.getUserWorkspaceIds(requestUserWorkspace.getUserName) @@ -67,22 +64,18 @@ class ProjectReceiver(projectService: DSSProjectService, case projectInfoRequest: ProjectInfoRequest => val dssProjectDO: DSSProjectDO = projectService.getProjectById(projectInfoRequest.getProjectId) val projectInfoVo: ProjectInfoVo = projectService.getProjectInfoById(projectInfoRequest.getProjectId) - val DSSProject = new DSSProject() - BeanUtils.copyProperties(dssProjectDO, DSSProject) - DSSProject.setWorkspaceName(projectInfoVo.getWorkspaceName) - DSSProject + val dssProject = new DSSProject() + BeanUtils.copyProperties(dssProjectDO, dssProject) + dssProject.setWorkspaceId(dssProjectDO.getWorkspaceId.intValue()) + dssProject.setWorkspaceName(projectInfoVo.getWorkspaceName) + dssProject - case projectUserAuthRequest: ProjectUserAuthRequest => { + case projectUserAuthRequest: ProjectUserAuthRequest => val projectId = projectUserAuthRequest.getProjectId val userName = projectUserAuthRequest.getUserName val projectDo: DSSProjectDO = projectService.getProjectById(projectId) val privList = projectUserService.getProjectUserPriv(projectId, userName).map(_.getPriv) new ProjectUserAuthResponse(projectId, userName, privList, projectDo.getCreateBy) - } - case proxyUserCheckRequest: ProxyUserCheckRequest => { - val isExists = dssProxyUserService.isExists(proxyUserCheckRequest.userName, proxyUserCheckRequest.proxyUser) - ResponseProxyUserCheck(isExists,dssProxyUserService.getProxyUserNameList(proxyUserCheckRequest.userName)) - } } } diff --git a/dss-framework/dss-framework-project-server/src/main/scala/com/webank/wedatasphere/dss/framework/project/server/rpc/ProjectReceiverChooser.scala b/dss-framework/dss-framework-project-server/src/main/scala/com/webank/wedatasphere/dss/framework/project/server/rpc/ProjectReceiverChooser.scala index 9841c0700c..6555d78c0a 100644 --- a/dss-framework/dss-framework-project-server/src/main/scala/com/webank/wedatasphere/dss/framework/project/server/rpc/ProjectReceiverChooser.scala +++ b/dss-framework/dss-framework-project-server/src/main/scala/com/webank/wedatasphere/dss/framework/project/server/rpc/ProjectReceiverChooser.scala @@ -18,7 +18,6 @@ package com.webank.wedatasphere.dss.framework.project.server.rpc import com.webank.wedatasphere.dss.common.protocol.ProxyUserCheckRequest import com.webank.wedatasphere.dss.common.protocol.project.{ProjectInfoRequest, ProjectRelationRequest} -import com.webank.wedatasphere.dss.framework.admin.service.DssProxyUserService import com.webank.wedatasphere.dss.framework.project.service.{DSSProjectService, DSSProjectUserService} import com.webank.wedatasphere.dss.framework.workspace.service.DSSWorkspaceUserService import com.webank.wedatasphere.dss.orchestrator.common.protocol.{RequestProjectImportOrchestrator, RequestProjectUpdateOrcVersion} @@ -41,13 +40,10 @@ class ProjectReceiverChooser extends ReceiverChooser { @Autowired private var dssProjectUserService: DSSProjectUserService = _ - @Autowired - private var dssProxyUserService : DssProxyUserService= _ - private var receiver: Option[ProjectReceiver] = _ @PostConstruct - def init(): Unit = receiver = Some(new ProjectReceiver(projectService, dssWorkspaceUserService, dssProjectUserService,dssProxyUserService)) + def init(): Unit = receiver = Some(new ProjectReceiver(projectService, dssWorkspaceUserService, dssProjectUserService)) override def chooseReceiver(event: RPCMessageEvent): Option[Receiver] = event.message match { case _: ProjectRelationRequest => receiver diff --git a/dss-framework/dss-framework-project-server/src/main/scala/com/webank/wedatasphere/dss/framework/project/server/service/BMLService.scala b/dss-framework/dss-framework-project-server/src/main/scala/com/webank/wedatasphere/dss/framework/project/server/service/BMLService.scala deleted file mode 100644 index f7e946c62c..0000000000 --- a/dss-framework/dss-framework-project-server/src/main/scala/com/webank/wedatasphere/dss/framework/project/server/service/BMLService.scala +++ /dev/null @@ -1,183 +0,0 @@ -/* - * Copyright 2019 WeBank - * 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. - * - */ - -package com.webank.wedatasphere.dss.framework.project.server.service - -import java.io.{ByteArrayInputStream, InputStream} -import java.util -import java.util.UUID - -import com.webank.wedatasphere.dss.common.exception.DSSErrorException -import com.webank.wedatasphere.dss.common.utils.IoUtils -import org.apache.linkis.bml.client.{BmlClient, BmlClientFactory} -import org.apache.linkis.bml.protocol.{BmlDownloadResponse, BmlUpdateResponse, BmlUploadResponse} -import org.apache.linkis.common.utils.{JavaLog, Logging, Utils} -import org.apache.commons.io.IOUtils -import org.springframework.stereotype.Component - -import scala.collection.JavaConversions._ - - -@Component("projectServerBMLService") -class BMLService extends Logging{ - - def upload(userName: String, content: String, fileName: String, projectName:String): util.Map[String, Object] = { - val inputStream = new ByteArrayInputStream(content.getBytes("utf-8")) - val client: BmlClient = createBMLClient(userName) - val resource: BmlUploadResponse = client.uploadShareResource(userName, projectName, fileName, inputStream) - if (!resource.isSuccess) throw new DSSErrorException(911113, "上传失败") - val map = new util.HashMap[String, Object] - map += "resourceId" -> resource.resourceId - map += "version" -> resource.version - } - - def upload(userName: String, inputStream: InputStream, fileName: String, projectName:String): util.Map[String, Object] = { - val client: BmlClient = createBMLClient(userName) - val resource: BmlUploadResponse = client.uploadShareResource(userName, projectName, fileName, inputStream) - if (!resource.isSuccess) throw new DSSErrorException(911113, "上传失败") - val map = new util.HashMap[String, Object] - map += "resourceId" -> resource.resourceId - map += "version" -> resource.version - } - - def update(userName: String, resourceId: String, inputStream: InputStream): util.Map[String, Object] = { - val client: BmlClient = createBMLClient(userName) - val resource: BmlUpdateResponse = client.updateShareResource(userName, resourceId, "", inputStream) - if (!resource.isSuccess) throw new DSSErrorException(911114, "更新失败") - val map = new util.HashMap[String, Object] - map += "resourceId" -> resource.resourceId - map += "version" -> resource.version - } - - def update(userName: String, resourceId: String, content: String): util.Map[String, Object] = { - val inputStream = new ByteArrayInputStream(content.getBytes("utf-8")) - val client: BmlClient = createBMLClient(userName) - val resource: BmlUpdateResponse = client.updateShareResource(userName, resourceId, UUID.randomUUID().toString+".json", inputStream) - if (!resource.isSuccess) throw new DSSErrorException(911114, "更新失败") - val map = new util.HashMap[String, Object] - map += "resourceId" -> resource.resourceId - map += "version" -> resource.version - } - - def query(userName: String, resourceId: String, version: String): util.Map[String, Object] = { - val client: BmlClient = createBMLClient(userName) - var resource: BmlDownloadResponse = null - if (version == null) { - resource = client.downloadShareResource(userName, resourceId) - } else { - resource = client.downloadShareResource(userName, resourceId, version) - } - if (!resource.isSuccess) throw new DSSErrorException(911115, "下载失败") - val map = new util.HashMap[String, Object] - map += "path" -> resource.fullFilePath - map += "string" -> inputstremToString(resource.inputStream) - } - - def download(userName: String, resourceId: String, version: String): util.Map[String, Object] = { - val client: BmlClient = createBMLClient(userName) - var resource: BmlDownloadResponse = null - if (version == null) { - resource = client.downloadShareResource(userName, resourceId) - } else { - resource = client.downloadShareResource(userName, resourceId, version) - } - if (!resource.isSuccess) throw new DSSErrorException(911115, "下载失败") - val map = new util.HashMap[String, Object] - map += "path" -> resource.fullFilePath - map += "is" -> resource.inputStream - } - - def downloadToLocalPath(userName: String, resourceId: String, version: String, path: String): String = { - val result = download(userName,resourceId,version) - val is = result.get("is").asInstanceOf[InputStream] - val os = IoUtils.generateExportOutputStream(path) - Utils.tryFinally(IOUtils.copy(is,os)){ - IOUtils.closeQuietly(os) - IOUtils.closeQuietly(is) - } - path - } - - def downloadAndGetFlowJson(userName: String, resourceId: String, version: String, path: String): String = { - downloadToLocalPath(userName,resourceId,version,path) - //因为下载到指定目录后返回的resource的Stream为null,只能从文件重新读取。 - val is = IoUtils.generateInputInputStream(path) - Utils.tryFinally(inputstremToString(is))(IOUtils.closeQuietly(is)) - } - - def readLocalResourceFile(userName: String,readPath: String): InputStream ={ - IoUtils.generateInputInputStream(readPath) - } - - def readLocalFlowJsonFile(userName: String,readPath: String): String ={ - var inputStream:InputStream = null - var flowJson:String = null - Utils.tryFinally{ - inputStream = IoUtils.generateInputInputStream(readPath) - flowJson = inputstremToString(inputStream) - }{ - IOUtils.closeQuietly(inputStream) - } - flowJson - } - - private def inputstremToString(inputStream: InputStream): String = { - scala.io.Source.fromInputStream(inputStream).mkString - } - - private def createBMLClient(userName: String): BmlClient = { - if (userName == null) - BmlClientFactory.createBmlClient() - else - BmlClientFactory.createBmlClient(userName) - } - - def createBmlProject(username:String, projectName:String, editUsers:util.List[String], - accessUsers:util.List[String] ): Unit ={ - val client = createBMLClient(username) - val response = client.createBmlProject(username, projectName, accessUsers, editUsers) - if (response.isSuccess){ - logger.info(s"for user $username create bml project $projectName success") - }else{ - logger.error(s"for user $username create bml project $projectName failed") - } - } - - def attachResourceAndProject(username:String, projectName:String, resourceId:String):Unit = { - val client = createBMLClient(username) - val response = client.attachResourceAndProject(projectName, resourceId) - if (response.isSuccess){ - logger.info(s"attach $username and $projectName success") - }else{ - logger.error(s"attach $username and $projectName failed") - } - } - - - def updateProjectPriv(projectName:String, username:String, editUsers:util.List[String], - accessUsers:util.List[String]): Unit ={ - val client = createBMLClient(username) - val response = client.updateProjectPriv(username, projectName, editUsers, accessUsers) - if (response.isSuccess){ - logger.info(s"attach $username and $projectName success") - }else{ - logger.error(s"attach $username and $projectName failed") - } - } - - - -} diff --git a/dss-framework/dss-framework-proxy-user-service/pom.xml b/dss-framework/dss-framework-proxy-user-service/pom.xml new file mode 100644 index 0000000000..84c1fbceb5 --- /dev/null +++ b/dss-framework/dss-framework-proxy-user-service/pom.xml @@ -0,0 +1,102 @@ + + + + dss-framework + com.webank.wedatasphere.dss + 1.1.2 + + 4.0.0 + + dss-framework-proxy-user-service + + 2.9.2 + 1.9.2 + 3.9 + + + + org.apache.linkis + linkis-module + ${linkis.version} + provided + + + + + com.webank.wedatasphere.dss + dss-common + ${dss.version} + provided + + + + org.apache.commons + commons-lang3 + ${commons.lang3.version} + provided + + + org.apache.linkis + linkis-httpclient + ${linkis.version} + + + linkis-common + org.apache.linkis + + + + + + com.webank.wedatasphere.dss + dss-framework-common + ${dss.version} + + + com.webank.wedatasphere.dss + dss-sso-integration-standard + ${dss.version} + provided + + + + + + + + + + org.apache.maven.plugins + maven-deploy-plugin + + + + net.alchim31.maven + scala-maven-plugin + + + org.apache.maven.plugins + maven-jar-plugin + + + org.apache.maven.plugins + maven-compiler-plugin + + 8 + 8 + + + + + + src/main/java + + **/*.xml + + + + + + \ No newline at end of file diff --git a/dss-framework/dss-framework-proxy-user-service/src/main/java/com/webank/wedatasphere/dss/framework/proxy/conf/ProxyUserConfiguration.java b/dss-framework/dss-framework-proxy-user-service/src/main/java/com/webank/wedatasphere/dss/framework/proxy/conf/ProxyUserConfiguration.java new file mode 100644 index 0000000000..99af58b302 --- /dev/null +++ b/dss-framework/dss-framework-proxy-user-service/src/main/java/com/webank/wedatasphere/dss/framework/proxy/conf/ProxyUserConfiguration.java @@ -0,0 +1,20 @@ +package com.webank.wedatasphere.dss.framework.proxy.conf; + +import com.webank.wedatasphere.dss.common.conf.DSSCommonConf; +import org.apache.linkis.common.conf.CommonVars; + +/** + * @author enjoyyin + * @date 2022-09-05 + * @since 0.5.0 + */ +public class ProxyUserConfiguration { + + public static boolean isProxyUserEnable() { + return CommonVars.apply(DSSCommonConf.ALL_GLOBAL_LIMITS_PREFIX.acquireNew() + "proxyEnable", false).acquireNew(); + } + + public static final CommonVars DS_TRUST_TOKEN = CommonVars.apply("wds.dss.trust.token", ""); + public static final CommonVars DS_PROXY_SELF_ENABLE = CommonVars.apply("wds.dss.proxy.self.enable", true); + +} diff --git a/dss-framework/dss-framework-proxy-user-service/src/main/java/com/webank/wedatasphere/dss/framework/proxy/exception/DSSProxyUserErrorException.java b/dss-framework/dss-framework-proxy-user-service/src/main/java/com/webank/wedatasphere/dss/framework/proxy/exception/DSSProxyUserErrorException.java new file mode 100644 index 0000000000..92aa011e26 --- /dev/null +++ b/dss-framework/dss-framework-proxy-user-service/src/main/java/com/webank/wedatasphere/dss/framework/proxy/exception/DSSProxyUserErrorException.java @@ -0,0 +1,16 @@ +package com.webank.wedatasphere.dss.framework.proxy.exception; + +import com.webank.wedatasphere.dss.common.exception.DSSErrorException; + + +public class DSSProxyUserErrorException extends DSSErrorException { + + public DSSProxyUserErrorException(int errCode, String desc) { + super(errCode, desc); + } + + public DSSProxyUserErrorException(int errCode, String desc, String ip, int port, String serviceKind) { + super(errCode, desc, ip, port, serviceKind); + } + +} diff --git a/dss-framework/dss-framework-proxy-user-service/src/main/java/com/webank/wedatasphere/dss/framework/proxy/pojo/entity/DssProxyUser.java b/dss-framework/dss-framework-proxy-user-service/src/main/java/com/webank/wedatasphere/dss/framework/proxy/pojo/entity/DssProxyUser.java new file mode 100644 index 0000000000..fb66ef78fc --- /dev/null +++ b/dss-framework/dss-framework-proxy-user-service/src/main/java/com/webank/wedatasphere/dss/framework/proxy/pojo/entity/DssProxyUser.java @@ -0,0 +1,10 @@ +package com.webank.wedatasphere.dss.framework.proxy.pojo.entity; + + +public interface DssProxyUser { + + String getUserName(); + + String getProxyUserName(); + +} diff --git a/dss-framework/dss-framework-proxy-user-service/src/main/java/com/webank/wedatasphere/dss/framework/proxy/pojo/entity/DssProxyUserImpl.java b/dss-framework/dss-framework-proxy-user-service/src/main/java/com/webank/wedatasphere/dss/framework/proxy/pojo/entity/DssProxyUserImpl.java new file mode 100644 index 0000000000..11771b6f05 --- /dev/null +++ b/dss-framework/dss-framework-proxy-user-service/src/main/java/com/webank/wedatasphere/dss/framework/proxy/pojo/entity/DssProxyUserImpl.java @@ -0,0 +1,34 @@ +package com.webank.wedatasphere.dss.framework.proxy.pojo.entity; + + +public class DssProxyUserImpl implements DssProxyUser { + + private String userName; + private String proxyUserName; + + public DssProxyUserImpl() { + } + + public DssProxyUserImpl(String userName, String proxyUserName) { + this.userName = userName; + this.proxyUserName = proxyUserName; + } + + @Override + public String getUserName() { + return userName; + } + + public void setUserName(String userName) { + this.userName = userName; + } + + @Override + public String getProxyUserName() { + return proxyUserName; + } + + public void setProxyUserName(String proxyUserName) { + this.proxyUserName = proxyUserName; + } +} diff --git a/dss-framework/dss-framework-proxy-user-service/src/main/java/com/webank/wedatasphere/dss/framework/proxy/restful/DssProxyUserController.java b/dss-framework/dss-framework-proxy-user-service/src/main/java/com/webank/wedatasphere/dss/framework/proxy/restful/DssProxyUserController.java new file mode 100644 index 0000000000..9cceb7f1ec --- /dev/null +++ b/dss-framework/dss-framework-proxy-user-service/src/main/java/com/webank/wedatasphere/dss/framework/proxy/restful/DssProxyUserController.java @@ -0,0 +1,126 @@ +package com.webank.wedatasphere.dss.framework.proxy.restful; + +import com.webank.wedatasphere.dss.common.utils.DSSExceptionUtils; +import com.webank.wedatasphere.dss.common.utils.DomainUtils; +import com.webank.wedatasphere.dss.common.utils.ScalaFunctionAdapter; +import com.webank.wedatasphere.dss.framework.proxy.conf.ProxyUserConfiguration; +import com.webank.wedatasphere.dss.framework.proxy.pojo.entity.DssProxyUser; +import com.webank.wedatasphere.dss.framework.proxy.pojo.entity.DssProxyUserImpl; +import com.webank.wedatasphere.dss.framework.proxy.service.DssProxyUserService; +import com.webank.wedatasphere.dss.standard.app.sso.Workspace; +import com.webank.wedatasphere.dss.standard.common.exception.AppStandardWarnException; +import com.webank.wedatasphere.dss.standard.sso.utils.SSOHelper; +import org.apache.commons.lang.exception.ExceptionUtils; +import org.apache.commons.lang3.StringUtils; +import org.apache.linkis.server.Message; +import org.apache.linkis.server.conf.ServerConfiguration; +import org.apache.linkis.server.security.ProxyUserSSOUtils; +import org.apache.linkis.server.security.SecurityFilter; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.web.bind.annotation.RequestBody; +import org.springframework.web.bind.annotation.RequestMapping; +import org.springframework.web.bind.annotation.RequestMethod; +import scala.Tuple2; + +import javax.servlet.http.Cookie; +import javax.servlet.http.HttpServletRequest; +import javax.servlet.http.HttpServletResponse; +import java.util.List; +import java.util.stream.Collectors; + +import static com.webank.wedatasphere.dss.framework.proxy.conf.ProxyUserConfiguration.DS_PROXY_SELF_ENABLE; +import static com.webank.wedatasphere.dss.framework.proxy.conf.ProxyUserConfiguration.DS_TRUST_TOKEN; + + +public class DssProxyUserController { + + private Boolean sslEnable = (Boolean) ServerConfiguration.BDP_SERVER_SECURITY_SSL().getValue(); + protected final Logger LOGGER = LoggerFactory.getLogger(this.getClass()); + + @Autowired + protected DssProxyUserService dssProxyUserService; + + @RequestMapping(path = "list", method = RequestMethod.GET) + public Message getProxyUserList(HttpServletRequest request) { + if(!ProxyUserConfiguration.isProxyUserEnable()) { + return Message.error("proxy user service is not enable, please ask admin for help."); + } + String username = SecurityFilter.getLoginUsername(request); + Workspace workspace = new Workspace(); + try { + workspace = SSOHelper.getWorkspace(request); + } catch (AppStandardWarnException ignored) {} + Workspace finalWorkspace = workspace; + return DSSExceptionUtils.applyMessage(() -> dssProxyUserService.selectProxyUserList(username, finalWorkspace), + userList -> { + LOGGER.info("user {} got proxy list. userList: {}.", username, userList); + List proxyUserNameList = userList.stream().map(DssProxyUser::getProxyUserName).collect(Collectors.toList()); + if(DS_PROXY_SELF_ENABLE.getValue()&&!proxyUserNameList.contains(username)) { + proxyUserNameList.add(username); + } + return Message.ok().data("proxyUserList", proxyUserNameList); + }, "fetch proxy user list failed."); + } + + @RequestMapping(path = "getProxyUser", method = RequestMethod.GET) + public Message getProxyUser(HttpServletRequest request) { + if(!ProxyUserConfiguration.isProxyUserEnable()) { + return Message.error("proxy user service is not enable, please ask admin for help."); + } + String username = SecurityFilter.getLoginUsername(request); + LOGGER.info("user {} try to get proxy user.", username); + return DSSExceptionUtils.getMessage(() -> dssProxyUserService.getProxyUser(request), + proxyUser -> Message.ok().data("userName", username).data("proxyUser", proxyUser), ""); + } + + @RequestMapping(path = "setProxyUser", method = RequestMethod.POST) + public Message setProxyUserCookie(@RequestBody DssProxyUserImpl userRep, + HttpServletRequest req, + HttpServletResponse resp) { + if(!ProxyUserConfiguration.isProxyUserEnable()) { + return Message.error("proxy user service is not enable, please ask admin for help."); + } + String username = SecurityFilter.getLoginUsername(req); + if (StringUtils.isEmpty(userRep.getUserName())) { + return Message.error("userName is null."); + } else if (StringUtils.isEmpty(userRep.getProxyUserName())) { + return Message.error("proxyUserName is null."); + } else if(!username.equals(userRep.getUserName()) && !username.equals(userRep.getProxyUserName())) { + return Message.error("The requested user name is not a login user."); + } + LOGGER.info("user {} try to add user proxy cookie, params:{}.", username, userRep); + // 兼容linkis端直接为登陆用户名加_c后缀的场景 + if (userRep.getProxyUserName().equals(username)) { + userRep.setUserName(userRep.getProxyUserName()); + } + Workspace workspace = new Workspace(); + try { + workspace = SSOHelper.getWorkspace(req); + } catch (AppStandardWarnException ignored) {} // ignore error + try { + if ((ProxyUserConfiguration.DS_PROXY_SELF_ENABLE.getValue() && userRep.getUserName().equalsIgnoreCase(userRep.getProxyUserName())) || + dssProxyUserService.isExists(userRep.getUserName(), userRep.getProxyUserName(), workspace)) { + ProxyUserSSOUtils.removeProxyUser(ScalaFunctionAdapter.doSupplier(req::getCookies)); + } else { + LOGGER.info("user {} have no permission to proxy to user {}.", userRep.getUserName(), userRep.getProxyUserName()); + return Message.error("user " + userRep.getUserName() + " have no permission to proxy to user " + userRep.getProxyUserName()); + } + } catch (Exception exception) { + LOGGER.error("Failed to set proxy user to cookie.", exception); + return Message.error(ExceptionUtils.getRootCauseMessage(exception)); + } + Tuple2 userTicketIdKv = ProxyUserSSOUtils.getProxyUserTicketKV(userRep.getProxyUserName(), DS_TRUST_TOKEN.getValue()); + Cookie cookie = new Cookie(userTicketIdKv._1, userTicketIdKv._2); + cookie.setMaxAge(-1); + if (sslEnable){ + cookie.setSecure(true); + } + cookie.setDomain(DomainUtils.getCookieDomain(req)); + cookie.setPath("/"); + resp.addCookie(cookie); + return Message.ok("Success to add proxy user into cookie."); + } + +} diff --git a/dss-framework/dss-framework-proxy-user-service/src/main/java/com/webank/wedatasphere/dss/framework/proxy/service/DssProxyUserService.java b/dss-framework/dss-framework-proxy-user-service/src/main/java/com/webank/wedatasphere/dss/framework/proxy/service/DssProxyUserService.java new file mode 100644 index 0000000000..e9cb7fd44e --- /dev/null +++ b/dss-framework/dss-framework-proxy-user-service/src/main/java/com/webank/wedatasphere/dss/framework/proxy/service/DssProxyUserService.java @@ -0,0 +1,40 @@ +package com.webank.wedatasphere.dss.framework.proxy.service; + + +import com.webank.wedatasphere.dss.common.entity.DSSWorkspace; +import com.webank.wedatasphere.dss.framework.proxy.exception.DSSProxyUserErrorException; +import com.webank.wedatasphere.dss.framework.proxy.pojo.entity.DssProxyUser; +import org.apache.linkis.server.security.ProxyUserSSOUtils; +import scala.Option; + +import javax.servlet.http.HttpServletRequest; +import java.util.List; + +public interface DssProxyUserService { + + /** + * 查询代理用户数据 + * + * @param userName 查询代理用户的用户名 + * @return 代理用户的集合 + */ + List selectProxyUserList(String userName, DSSWorkspace workspace); + + default boolean isExists(String userName, String proxyUserName, DSSWorkspace workspace) { + List proxyUserList = selectProxyUserList(userName, workspace); + if(proxyUserList == null || proxyUserList.isEmpty()) { + return false; + } + return proxyUserList.stream().anyMatch(proxyUser -> proxyUser.getProxyUserName().equals(proxyUserName)); + } + + default String getProxyUser(HttpServletRequest request) throws DSSProxyUserErrorException { + Option proxyUser = ProxyUserSSOUtils.getProxyUserUsername(request); + if(proxyUser.isEmpty()) { + throw new DSSProxyUserErrorException(60050, "proxy user is not exists in cookies."); + } + return proxyUser.get(); + } + + +} diff --git a/dss-framework/dss-framework-workspace-server/pom.xml b/dss-framework/dss-framework-workspace-server/pom.xml index 2892482135..6741aeaa60 100644 --- a/dss-framework/dss-framework-workspace-server/pom.xml +++ b/dss-framework/dss-framework-workspace-server/pom.xml @@ -21,8 +21,8 @@ dss com.webank.wedatasphere.dss - 1.1.1 - + 1.1.2 + ../../pom.xml 4.0.0 diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/DSSApplication.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/DSSApplication.java deleted file mode 100644 index e69de29bb2..0000000000 diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/DSSHomepage.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/DSSHomepage.java deleted file mode 100644 index 48ab10de3c..0000000000 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/DSSHomepage.java +++ /dev/null @@ -1,48 +0,0 @@ -/* - * Copyright 2019 WeBank - * 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. - * - */ - -package com.webank.wedatasphere.dss.framework.workspace.bean; - - -public class DSSHomepage { - private int id; - private String name; - private String url; - - public int getId() { - return id; - } - - public void setId(int id) { - this.id = id; - } - - public String getName() { - return name; - } - - public void setName(String name) { - this.name = name; - } - - public String getUrl() { - return url; - } - - public void setUrl(String url) { - this.url = url; - } -} diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/DSSWorkspace.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/DSSWorkspace.java index a8248371f9..bf9829176e 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/DSSWorkspace.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/DSSWorkspace.java @@ -32,6 +32,15 @@ public class DSSWorkspace { private Date lastUpdateTime; private String lastUpdateUser; private String workspaceType; + private int adminPermission; + + public int getAdminPermission() { + return adminPermission; + } + + public void setAdminPermission(int adminPermission) { + this.adminPermission = adminPermission; + } public String getWorkspaceType() { return workspaceType; diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/DSSWorkspaceAssociateDepartments.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/DSSWorkspaceAssociateDepartments.java new file mode 100644 index 0000000000..8760c29dc8 --- /dev/null +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/DSSWorkspaceAssociateDepartments.java @@ -0,0 +1,79 @@ +package com.webank.wedatasphere.dss.framework.workspace.bean; + +import java.util.Date; + +public class DSSWorkspaceAssociateDepartments { + private Long id; + private Long workspaceId; + private String departments; + private String roleIds; + private Date createTime; + private Date updateTime; + private String createBy; + private String updateBy; + + + public Long getId() { + return id; + } + + public void setId(Long id) { + this.id = id; + } + + public Long getWorkspaceId() { + return workspaceId; + } + + public void setWorkspaceId(Long workspaceId) { + this.workspaceId = workspaceId; + } + + public String getDepartments() { + return departments; + } + + public void setDepartments(String departments) { + this.departments = departments; + } + + public String getRoleIds() { + return roleIds; + } + + public void setRoleIds(String roleIds) { + this.roleIds = roleIds; + } + + public Date getCreateTime() { + return createTime; + } + + public void setCreateTime(Date createTime) { + this.createTime = createTime; + } + + public Date getUpdateTime() { + return updateTime; + } + + public void setUpdateTime(Date updateTime) { + this.updateTime = updateTime; + } + + public String getCreateBy() { + return createBy; + } + + public void setCreateBy(String createBy) { + this.createBy = createBy; + } + + public String getUpdateBy() { + return updateBy; + } + + public void setUpdateBy(String updateBy) { + this.updateBy = updateBy; + } +} diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/DSSWorkspaceMenuComponentUrl.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/DSSWorkspaceMenuComponentUrl.java deleted file mode 100644 index 23348c5bf0..0000000000 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/DSSWorkspaceMenuComponentUrl.java +++ /dev/null @@ -1,86 +0,0 @@ -/* - * Copyright 2019 WeBank - * 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. - * - */ - -package com.webank.wedatasphere.dss.framework.workspace.bean; - -import java.util.Date; - - -public class DSSWorkspaceMenuComponentUrl { - private int id; - private int menuId; - private int dssApplicationId; - private String url; - private String manulUrl; - private String operationUrl; - private Date updateTime; - - public int getId() { - return id; - } - - public void setId(int id) { - this.id = id; - } - - public int getMenuId() { - return menuId; - } - - public void setMenuId(int menuId) { - this.menuId = menuId; - } - - public int getDssApplicationId() { - return dssApplicationId; - } - - public void setDssApplicationId(int dssApplicationId) { - this.dssApplicationId = dssApplicationId; - } - - public String getUrl() { - return url; - } - - public void setUrl(String url) { - this.url = url; - } - - public String getManulUrl() { - return manulUrl; - } - - public void setManulUrl(String manulUrl) { - this.manulUrl = manulUrl; - } - - public String getOperationUrl() { - return operationUrl; - } - - public void setOperationUrl(String operationUrl) { - this.operationUrl = operationUrl; - } - - public Date getUpdateTime() { - return updateTime; - } - - public void setUpdateTime(Date updateTime) { - this.updateTime = updateTime; - } -} diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/DSSWorkspaceUser.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/DSSWorkspaceUser.java index 4252966448..1f2b476ed6 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/DSSWorkspaceUser.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/DSSWorkspaceUser.java @@ -31,6 +31,36 @@ public class DSSWorkspaceUser { private Date joinTime; + private String roleIds; + + private Date updateTime; + + private String updateUser; + + public Date getUpdateTime() { + return updateTime; + } + + public void setUpdateTime(Date updateTime) { + this.updateTime = updateTime; + } + + public String getUpdateUser() { + return updateUser; + } + + public void setUpdateUser(String updateUser) { + this.updateUser = updateUser; + } + + public String getRoleIds() { + return roleIds; + } + + public void setRoleIds(String roleIds) { + this.roleIds = roleIds; + } + public Long getId() { return id; } @@ -71,6 +101,8 @@ public void setJoinTime(Date joinTime) { this.joinTime = joinTime; } + + @Override public String toString() { return "DSSWorkspaceUser{" + @@ -79,6 +111,7 @@ public String toString() { ", workspaceId=" + workspaceId + ", creator='" + creator + '\'' + ", joinTime=" + joinTime + + ", roleIds=" + roleIds + '}'; } } diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/NoticeContent.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/NoticeContent.java new file mode 100644 index 0000000000..058b9852f1 --- /dev/null +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/NoticeContent.java @@ -0,0 +1,53 @@ +package com.webank.wedatasphere.dss.framework.workspace.bean; + +import com.baomidou.mybatisplus.annotation.TableName; + +import java.io.Serializable; +import java.util.Date; + +/** + * 首页公告 + * Author: xlinliu + * Date: 2023/3/13 + */ +@TableName(value = "dss_notice") +public class NoticeContent implements Serializable { + /** + * 公告内容 + */ + private String content; + /** + * 公告生效时间 + */ + private Date startTime; + /** + * 公告失效时间 + */ + private Date endTime; + + + + public String getContent() { + return content; + } + + public void setContent(String content) { + this.content = content; + } + + public Date getStartTime() { + return startTime; + } + + public void setStartTime(Date startTime) { + this.startTime = startTime; + } + + public Date getEndTime() { + return endTime; + } + + public void setEndTime(Date endTime) { + this.endTime = endTime; + } +} diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/StaffInfo.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/StaffInfo.java index ea422578e5..719c6554ed 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/StaffInfo.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/StaffInfo.java @@ -16,31 +16,29 @@ package com.webank.wedatasphere.dss.framework.workspace.bean; -import org.codehaus.jackson.annotate.JsonIgnoreProperties; -import org.codehaus.jackson.annotate.JsonProperty; +import com.google.gson.annotations.SerializedName; -@JsonIgnoreProperties(ignoreUnknown = true) public class StaffInfo { - @JsonProperty(value = "ID") + @SerializedName(value = "ID") private String id; - @JsonProperty(value = "StaffID") + @SerializedName(value = "StaffID") private String staffId; - @JsonProperty(value = "OID") + @SerializedName(value = "OID") private String oId; - @JsonProperty(value = "ChineseName") + @SerializedName(value = "ChineseName") private String chineseName; - @JsonProperty(value = "EnglishName") + @SerializedName(value = "EnglishName") private String englishName; - @JsonProperty(value = "FullName") + @SerializedName(value = "FullName") private String fullName; - @JsonProperty(value = "OrgName") + @SerializedName(value = "OrgName") private String orgName; - @JsonProperty(value = "OrgFullName") + @SerializedName(value = "OrgFullName") private String orgFullName; - @JsonProperty(value = "Status") + @SerializedName(value = "Status") private String status; - @JsonProperty(value = "PersonGroup") + @SerializedName(value = "PersonGroup") private String personGroup; public StaffInfo(){ diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/dto/response/HomepageDemoInstanceVo.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/dto/response/HomepageDemoInstanceVo.java deleted file mode 100644 index 66096fde20..0000000000 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/dto/response/HomepageDemoInstanceVo.java +++ /dev/null @@ -1,93 +0,0 @@ -/* - * Copyright 2019 WeBank - * 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. - * - */ -package com.webank.wedatasphere.dss.framework.workspace.bean.dto.response; - -@Deprecated -public class HomepageDemoInstanceVo { - - private Long id; - private Long menuId; - private String name; - private String url; - private String title; - private String description; - private String icon; - private Integer order; - - public Long getId() { - return id; - } - - public void setId(Long id) { - this.id = id; - } - - public Long getMenuId() { - return menuId; - } - - public void setMenuId(Long menuId) { - this.menuId = menuId; - } - - public String getName() { - return name; - } - - public void setName(String name) { - this.name = name; - } - - public String getUrl() { - return url; - } - - public void setUrl(String url) { - this.url = url; - } - - public String getTitle() { - return title; - } - - public void setTitle(String title) { - this.title = title; - } - - public String getDescription() { - return description; - } - - public void setDescription(String description) { - this.description = description; - } - - public String getIcon() { - return icon; - } - - public void setIcon(String icon) { - this.icon = icon; - } - - public Integer getOrder() { - return order; - } - - public void setOrder(Integer order) { - this.order = order; - } -} diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/dto/response/WorkspaceMenuAppconnVo.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/dto/response/WorkspaceMenuAppconnVo.java index 20219fa0ed..596d54b064 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/dto/response/WorkspaceMenuAppconnVo.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/dto/response/WorkspaceMenuAppconnVo.java @@ -25,22 +25,16 @@ public class WorkspaceMenuAppconnVo { private String title; private String description; private String labels; - // private String accessButton; -// private String accessButtonUrl; -// private String manualButton; -// private String manualButtonUrl; -// private String projectUrl; private String name; private Boolean isActive; private Boolean accessable; private String icon; private Integer order; - // private Map nameAndUrls; - //图片 private String image; private Boolean ifIframe; private Boolean isExternal; + private Boolean isMicroApp; private List appInstances; @@ -83,38 +77,6 @@ public String getLabels() { public void setLabels(String labels) { this.labels = labels; } -// -// public String getAccessButton() { -// return accessButton; -// } -// -// public void setAccessButton(String accessButton) { -// this.accessButton = accessButton; -// } -// -// public String getAccessButtonUrl() { -// return accessButtonUrl; -// } -// -// public void setAccessButtonUrl(String accessButtonUrl) { -// this.accessButtonUrl = accessButtonUrl; -// } -// -// public String getManualButton() { -// return manualButton; -// } -// -// public void setManualButton(String manualButton) { -// this.manualButton = manualButton; -// } -// -// public String getManualButtonUrl() { -// return manualButtonUrl; -// } -// -// public void setManualButtonUrl(String manualButtonUrl) { -// this.manualButtonUrl = manualButtonUrl; -// } public String getIcon() { return icon; @@ -131,14 +93,6 @@ public Integer getOrder() { public void setOrder(Integer order) { this.order = order; } -// -// public String getProjectUrl() { -// return projectUrl; -// } -// -// public void setProjectUrl(String projectUrl) { -// this.projectUrl = projectUrl; -// } public Boolean getActive() { return isActive; @@ -156,14 +110,6 @@ public void setName(String name) { this.name = name; } -// public Map getNameAndUrls() { -// return nameAndUrls; -// } -// -// public void setNameAndUrls(Map nameAndUrls) { -// this.nameAndUrls = nameAndUrls; -// } - public Boolean getAccessable() { return accessable; } @@ -199,4 +145,12 @@ public Boolean getExternal() { public void setExternal(Boolean external) { isExternal = external; } + + public Boolean getIsMicroApp() { + return isMicroApp; + } + + public void setIsMicroApp(Boolean isMicroApp) { + this.isMicroApp = isMicroApp; + } } diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/request/AddWorkspaceRoleRequest.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/request/AddWorkspaceRoleRequest.java index c0cee240e3..6ee50c1a4a 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/request/AddWorkspaceRoleRequest.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/request/AddWorkspaceRoleRequest.java @@ -41,4 +41,14 @@ public List getComponentIds() { public void setComponentIds(List componentIds) { this.componentIds = componentIds; } + + @Override + public String toString() { + return "AddWorkspaceRoleRequest{" + + "workspaceId=" + workspaceId + + ", roleName='" + roleName + '\'' + + ", menuIds=" + menuIds + + ", componentIds=" + componentIds + + '}'; + } } diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/request/CreateWorkspaceRequest.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/request/CreateWorkspaceRequest.java index 5cbafc1d53..d1456830a4 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/request/CreateWorkspaceRequest.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/request/CreateWorkspaceRequest.java @@ -50,4 +50,15 @@ public String getProductName() { public void setProductName(String productName) { this.productName = productName; } + + @Override + public String toString() { + return "CreateWorkspaceRequest{" + + "workspaceName='" + workspaceName + '\'' + + ", department='" + department + '\'' + + ", description='" + description + '\'' + + ", tags='" + tags + '\'' + + ", productName='" + productName + '\'' + + '}'; + } } diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/request/DeleteWorkspaceUserRequest.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/request/DeleteWorkspaceUserRequest.java index 8fc2a2cf22..fc13ca8aba 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/request/DeleteWorkspaceUserRequest.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/request/DeleteWorkspaceUserRequest.java @@ -22,4 +22,12 @@ public String getUserName() { public void setUserName(String username) { this.userName = username; } + + @Override + public String toString() { + return "DeleteWorkspaceUserRequest{" + + "workspaceId=" + workspaceId + + ", userName='" + userName + '\'' + + '}'; + } } diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/request/UpdateRoleComponentPrivRequest.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/request/UpdateRoleComponentPrivRequest.java index 8976c2e3e3..01841fcb50 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/request/UpdateRoleComponentPrivRequest.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/request/UpdateRoleComponentPrivRequest.java @@ -33,4 +33,13 @@ public Map getComponentPrivs() { public void setComponentPrivs(Map componentPrivs) { this.componentPrivs = componentPrivs; } + + @Override + public String toString() { + return "UpdateRoleComponentPrivRequest{" + + "componentId=" + componentId + + ", workspaceId=" + workspaceId + + ", componentPrivs=" + componentPrivs + + '}'; + } } diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/request/UpdateRoleMenuPrivRequest.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/request/UpdateRoleMenuPrivRequest.java index 697b3b5be2..93e612d998 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/request/UpdateRoleMenuPrivRequest.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/request/UpdateRoleMenuPrivRequest.java @@ -33,4 +33,13 @@ public Map getMenuPrivs() { public void setMenuPrivs(Map menuPrivs) { this.menuPrivs = menuPrivs; } + + @Override + public String toString() { + return "UpdateRoleMenuPrivRequest{" + + "menuId=" + menuId + + ", workspaceId=" + workspaceId + + ", menuPrivs=" + menuPrivs + + '}'; + } } diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/request/UpdateWorkspaceUserRequest.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/request/UpdateWorkspaceUserRequest.java index 3aff4a07f9..0ff86fbca7 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/request/UpdateWorkspaceUserRequest.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/request/UpdateWorkspaceUserRequest.java @@ -43,4 +43,14 @@ public List getRoles() { public void setRoles(List roles) { this.roles = roles; } + + @Override + public String toString() { + return "UpdateWorkspaceUserRequest{" + + "workspaceId=" + workspaceId + + ", userName='" + userName + '\'' + + ", roles=" + roles + + ", userId='" + userId + '\'' + + '}'; + } } diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/vo/DSSWorkspaceUserVO.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/vo/DSSWorkspaceUserVO.java index a3aa5775f9..dbf529a950 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/vo/DSSWorkspaceUserVO.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/vo/DSSWorkspaceUserVO.java @@ -30,11 +30,27 @@ public class DSSWorkspaceUserVO extends AbstractDSSVO{ private String office; private String creator; private Date joinTime; + private Date updateTime; + private String updateUser; public DSSWorkspaceUserVO() { } + public Date getUpdateTime() { + return updateTime; + } + + public void setUpdateTime(Date updateTime) { + this.updateTime = updateTime; + } + public String getUpdateUser() { + return updateUser; + } + + public void setUpdateUser(String updateUser) { + this.updateUser = updateUser; + } public int getId() { return id; diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/vo/DepartmentVO.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/vo/DepartmentVO.java deleted file mode 100644 index 3e5a705826..0000000000 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/bean/vo/DepartmentVO.java +++ /dev/null @@ -1,58 +0,0 @@ -/* - * Copyright 2019 WeBank - * 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. - * - */ - -package com.webank.wedatasphere.dss.framework.workspace.bean.vo; - - -public class DepartmentVO { - private int id; - private String name; - private String frontName; - - public DepartmentVO(){ - - } - - public DepartmentVO(int id, String name, String frontName) { - this.id = id; - this.name = name; - this.frontName = frontName; - } - - public int getId() { - return id; - } - - public void setId(int id) { - this.id = id; - } - - public String getName() { - return name; - } - - public void setName(String name) { - this.name = name; - } - - public String getFrontName() { - return frontName; - } - - public void setFrontName(String frontName) { - this.frontName = frontName; - } -} diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/conf/DSSWorkspaceRoleCheckConfiguration.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/conf/DSSWorkspaceRoleCheckConfiguration.java new file mode 100644 index 0000000000..68f367fe7a --- /dev/null +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/conf/DSSWorkspaceRoleCheckConfiguration.java @@ -0,0 +1,18 @@ +package com.webank.wedatasphere.dss.framework.workspace.conf; + +import com.webank.wedatasphere.dss.framework.workspace.service.DSSWorkspaceRoleCheckService; +import com.webank.wedatasphere.dss.framework.workspace.service.impl.DSSWorkspaceRoleCheckServiceImpl; +import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; + +@Configuration +public class DSSWorkspaceRoleCheckConfiguration { + + @Bean + @ConditionalOnMissingBean + public DSSWorkspaceRoleCheckService createDSSWorkspaceRoleCheckService(){ + return new DSSWorkspaceRoleCheckServiceImpl(); + } + +} diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/constant/ApplicationConf.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/constant/ApplicationConf.java index 50feb60e84..df18f0a84c 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/constant/ApplicationConf.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/constant/ApplicationConf.java @@ -22,21 +22,9 @@ public class ApplicationConf { - public static final CommonVars FAQ = CommonVars.apply("wds.linkis.application.dws.params","http://127.0.0.1:8088/wiki/scriptis/manual/feature_overview_cn.html"); - public static final CommonVars SCHEDULIS_URL = - CommonVars.apply("wds.linkis.schedulis.url", "http://127.0.0.1:8088"); - public static final CommonVars HOMEPAGE_MODULE_NAME = CommonVars.apply("wds.linkis.special.homepage.module.name", "apiServices"); public static final CommonVars HOMEPAGE_URL = CommonVars.apply("wds.linkis.special.homepage.module.url", "/newHome?workspaceId="); - - public static final CommonVars DSS_ENV_PROD_LABEL = - CommonVars.apply("wds.dss.env.prod.label", "PROD"); - - public static final String SCHEDULER_APP_CONN_NAME = CommonVars.apply("wds.dss.appconn.scheduler.name", "schedulis").getValue(); - - public static final String ESB_APPID = CommonVars.apply("wds.dss.esb.appid", "").getValue(); - } diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/dao/DSSWorkspaceInfoMapper.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/dao/DSSWorkspaceInfoMapper.java deleted file mode 100644 index 4c197851f1..0000000000 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/dao/DSSWorkspaceInfoMapper.java +++ /dev/null @@ -1,41 +0,0 @@ -/* - * Copyright 2019 WeBank - * 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. - * - */ - -package com.webank.wedatasphere.dss.framework.workspace.dao; - - -import com.webank.wedatasphere.dss.framework.workspace.bean.DSSWorkspace; -import org.apache.ibatis.annotations.*; - - -@Mapper -public interface DSSWorkspaceInfoMapper { - - @Select("select name from dss_workspace where id = #{workspaceId} ") - String getWorkspaceNameById(@Param("workspaceId") int workspaceId); - - @Select("select id from dss_workspace where name = #{workspaceName}") - Integer getWorkspaceIdByName(@Param("workspaceName") String workspaceName); - - @Select("select * from dss_workspace where id = #{workspaceId}") - @Results({ - @Result(property = "createBy", column = "create_by"), - @Result(property = "createTime", column = "create_time"), - @Result(property = "lastUpdateTime", column = "last_update_time"), - @Result(property = "lastUpdateUser", column = "last_update_user"), - }) - DSSWorkspace getWorkspace(@Param("workspaceId") int workspaceId); -} diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/dao/DSSWorkspaceMapper.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/dao/DSSWorkspaceMapper.java index 8c038deb67..3396ceccf2 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/dao/DSSWorkspaceMapper.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/dao/DSSWorkspaceMapper.java @@ -24,36 +24,40 @@ @Mapper public interface DSSWorkspaceMapper { + @Select("select name from dss_workspace where id = #{workspaceId} ") + String getWorkspaceNameById(@Param("workspaceId") long workspaceId); + + @Select("select id from dss_workspace where name = #{workspaceName}") + Integer getWorkspaceIdByName(@Param("workspaceName") String workspaceName); + + @Select("select * from dss_workspace where id = #{workspaceId}") + @Results({ + @Result(property = "createBy", column = "create_by"), + @Result(property = "createTime", column = "create_time"), + @Result(property = "lastUpdateTime", column = "last_update_time"), + @Result(property = "lastUpdateUser", column = "last_update_user"), + }) + DSSWorkspace getWorkspace(@Param("workspaceId") int workspaceId); void createWorkSpace(DSSWorkspace dssWorkspace); List getWorkspaces(String username); - List getUserMenuAppConnId(@Param("username")String username, @Param("workspaceId")Long workspaceId); + List getUserMenuAppConnId(@Param("username") String username, @Param("workspaceId") Long workspaceId); + List getMenuId(int roleId, String workspaceId); List getDSSWorkspaceMenuPriv(String workspaceId); @Select("select -1 as workspaceId, id as menu_id, 1 as role_id, 1 as priv from dss_workspace_menu") @Results({ - @Result(property = "workspaceId",column = "workspace_id"), - @Result(property = "menuId",column = "menu_id"), - @Result(property = "roleId",column = "role_id"), - @Result(property = "priv",column = "priv") + @Result(property = "workspaceId", column = "workspace_id"), + @Result(property = "menuId", column = "menu_id"), + @Result(property = "roleId", column = "role_id"), + @Result(property = "priv", column = "priv") }) List getDefaultWorkspaceMenuPriv(); - @Insert({ - "" - }) - void setDefaultComponentRoles(@Param("privs") List dssWorkspaceComponentPrivs); - @Select("select count(1) from dss_workspace_user_favorites_appconn where menu_appconn_id=#{menuAppId} and workspace_id=#{workspaceId}" + " and username=#{userName} and type=#{type}") int getByMenuAppIdAndUser(@Param("menuAppId") Long menuAppId, @Param("workspaceId") Long workspaceId, @@ -62,4 +66,17 @@ int getByMenuAppIdAndUser(@Param("menuAppId") Long menuAppId, @Param("workspaceI @Select("select id from dss_workspace_menu_appconn where title_en=#{appName}") Long getMenuAppIdByName(@Param("appName") String appName); + @Insert("insert into dss_workspace_associate_departments(workspace_id,departments,role_ids,create_time,create_by) values(#{workspaceId},#{departments},#{roleIds},now(),#{user})") + void addDepartmentsForWorkspace(@Param("workspaceId") Long workspaceId, @Param("departments") String departments, + @Param("roleIds") String roleIds, @Param("user") String user); + + @Update("update dss_workspace_associate_departments set departments=#{departments},role_ids=#{roleIds},update_time=now(),update_by=#{user} where workspace_id=#{workspaceId}") + void updateDepartmentsForWorkspace(@Param("workspaceId") Long workspaceId, @Param("departments") String departments, + @Param("roleIds") String roleIds, @Param("user") String user); + + @Select("select * from dss_workspace_associate_departments where workspace_id=#{workspaceId}") + DSSWorkspaceAssociateDepartments getAssociateDepartmentsByWorkspaceId(@Param("workspaceId") Long workspaceId); + + @Select("select * from dss_workspace_associate_departments") + List getWorkspaceAssociateDepartments(); } diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/dao/DSSWorkspacePrivMapper.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/dao/DSSWorkspacePrivMapper.java index f46f41b9a6..ff90c03b08 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/dao/DSSWorkspacePrivMapper.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/dao/DSSWorkspacePrivMapper.java @@ -40,19 +40,19 @@ void insertMenuRolePriv(@Param("workspaceId") int workspaceId, @Param("menuId") @Param("username") String username); @Update("update dss_workspace_appconn_role set priv = #{priv} , update_time = now()" + - "where workspace_id = #{workspaceId} and appconn_id = #{componentId} and role_id = #{roleId}") - void updateRoleComponentPriv(@Param("workspaceId") int workspaceId, @Param("componentId") int componentId, + "where workspace_id = #{workspaceId} and appconn_id = #{appconnId} and role_id = #{roleId}") + void updateRoleComponentPriv(@Param("workspaceId") int workspaceId, @Param("appconnId") int appconnId, @Param("roleId") int roleId, @Param("priv") int priv); @Select("select id from dss_workspace_role where workspace_id = #{workspaceId} and name = #{key}") Integer getRoleId(@Param("workspaceId") int workspaceId, @Param("key") String key); - @Select("select count(*) from dss_workspace_appconn_role where workspace_id = #{workspaceId} and appconn_id = #{componentId} and role_id = #{roleId}") - int queryCntOfRCP(@Param("workspaceId") int workspaceId, @Param("componentId") int componentId, @Param("roleId") int roleId); + @Select("select count(*) from dss_workspace_appconn_role where workspace_id = #{workspaceId} and appconn_id = #{appconnId} and role_id = #{roleId}") + int queryCntOfRCP(@Param("workspaceId") int workspaceId, @Param("appconnId") int appconnId, @Param("roleId") int roleId); @Select("insert into dss_workspace_appconn_role (`workspace_id`, `appconn_id`, `role_id`, `priv`, `update_time`, `updateby`) " + - "values(#{workspaceId}, #{componentId}, #{roleId}, #{priv}, now(), #{username})") - void insertRolComponentPriv(@Param("workspaceId") int workspaceId, @Param("componentId") int componentId, @Param("roleId") int roleId, + "values(#{workspaceId}, #{appconnId}, #{roleId}, #{priv}, now(), #{username})") + void insertRolComponentPriv(@Param("workspaceId") int workspaceId, @Param("appconnId") int appconnId, @Param("roleId") int roleId, @Param("priv") int priv, @Param("username") String username); } diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/dao/DSSWorkspaceRoleMapper.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/dao/DSSWorkspaceRoleMapper.java index c757fa8835..63da6fb665 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/dao/DSSWorkspaceRoleMapper.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/dao/DSSWorkspaceRoleMapper.java @@ -99,7 +99,7 @@ void updateRoleComponent(@Param("roleId") int roleId, @Param("workspaceId") int Integer getDefaultWorkspaceId(@Param("defaultWorkspaceName") String defaultWorkspaceName); @Select("select id from dss_workspace_role where workspace_id = #{workspaceId} and name = #{apiUser}") - int getRoleId(@Param("apiUser") String apiUser, @Param("workspaceId") int workspaceId); + Integer getRoleId(@Param("apiUser") String apiUser, @Param("workspaceId") int workspaceId); @Select("Select count(*) from dss_workspace_appconn_role where workspace_id = #{workspaceId} and role_id = #{roleId} and appconn_id = #{componentId}") int getCount(@Param("workspaceId") Integer workspaceId, @Param("componentId") int componentId, @Param("roleId") int roleId); diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/dao/DSSWorkspaceUserMapper.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/dao/DSSWorkspaceUserMapper.java index 17e701a228..151b18a976 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/dao/DSSWorkspaceUserMapper.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/dao/DSSWorkspaceUserMapper.java @@ -20,6 +20,7 @@ import com.webank.wedatasphere.dss.framework.workspace.bean.DSSWorkspaceUser; import org.apache.ibatis.annotations.*; +import java.util.Date; import java.util.List; @@ -30,10 +31,16 @@ public interface DSSWorkspaceUserMapper { String getUserName(Long userID); - @Insert("insert into dss_workspace_user_role(workspace_id, username, role_id, create_time, created_by,user_id)" + - "values(#{workspaceId}, #{username}, #{roleId}, now(), #{createdBy}, #{userId})") + @Insert("insert into dss_workspace_user_role(workspace_id, username, role_id, create_time, created_by,user_id, update_user, update_time)" + + "values(#{workspaceId}, #{username}, #{roleId}, now(), #{createdBy}, #{userId}, #{updateUser}, now())") void setUserRoleInWorkspace(@Param("workspaceId") int workspaceId, @Param("roleId") int roleId, - @Param("username") String username, @Param("createdBy") String createdBy, @Param("userId") Long userId); + @Param("username") String username, @Param("createdBy") String createdBy, @Param("userId") Long userId, + @Param("updateUser") String updateUser); + + @Insert("insert into dss_workspace_user_role(workspace_id, username, role_id, create_time, created_by, user_id, update_user, update_time)" + + "values(#{workspaceId}, #{username}, #{roleId}, #{createTime}, #{createdBy}, #{userId}, #{updateUser}, now())") + void insertUserRoleInWorkspace(@Param("workspaceId") int workspaceId, @Param("roleId") int roleId, @Param("createTime") Date createTime, + @Param("username") String username, @Param("createdBy") String createdBy, @Param("userId") Long userId,@Param("updateUser") String updateUser); @Select("select role_id from dss_workspace_user_role where workspace_id = #{workspaceId} and username = #{username}") List getRoleInWorkspace(@Param("workspaceId") int workspaceId, @Param("username") String username); @@ -57,12 +64,14 @@ void setUserRoleInWorkspace(@Param("workspaceId") int workspaceId, @Param("roleI @Select({ "" }) - List getWorkspaceUsers(@Param("workspaceId") String workspaceId,@Param("username") String username); + List getWorkspaceUsers(@Param("workspaceId") String workspaceId,@Param("username") String username, @Param("roleId") String roleId); @Select("select distinct created_by as creator, username as username, create_time as joinTime,workspace_id as workspaceId " + " from dss_workspace_user_role where role_id = #{roleId} and workspace_id = #{workspaceId}") @@ -75,4 +84,16 @@ void setUserRoleInWorkspace(@Param("workspaceId") int workspaceId, @Param("roleI @Select("select count(1) from dss_workspace_user_role where workspace_id = #{workspaceId} and username = #{username}") Long getCountByUsername(@Param("username") String username, @Param("workspaceId") int workspaceId); + @Select("select distinct workspace_id, role_id as roleIds " + + "from dss_workspace_user_role where username = #{username} ") + List getWorkspaceRoleByUsername(@Param("username") String username); + + @Delete("delete from dss_workspace_user_role where username = #{username} ") + void deleteUserRolesByUserName(@Param("username") String username); + + @Delete("delete from dss_user where username = #{username} ") + void deleteUserByUserName(@Param("username") String username); + + @Delete("delete from dss_proxy_user where username = #{username} ") + void deleteProxyUserByUserName(@Param("username") String username); } diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/dao/NoticeMapper.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/dao/NoticeMapper.java new file mode 100644 index 0000000000..268f3c77cc --- /dev/null +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/dao/NoticeMapper.java @@ -0,0 +1,9 @@ +package com.webank.wedatasphere.dss.framework.workspace.dao; + +import com.baomidou.mybatisplus.core.mapper.BaseMapper; +import com.webank.wedatasphere.dss.framework.workspace.bean.NoticeContent; +import org.apache.ibatis.annotations.Mapper; + +@Mapper +public interface NoticeMapper extends BaseMapper { +} diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/dao/impl/WorkspaceMapper.xml b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/dao/impl/WorkspaceMapper.xml index 269cadd722..5d86da48ea 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/dao/impl/WorkspaceMapper.xml +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/dao/impl/WorkspaceMapper.xml @@ -60,7 +60,8 @@ source, last_update_time, last_update_user, - workspace_type + workspace_type, + admin_permission FROM dss_workspace WHERE id = #{workspaceId} @@ -117,7 +118,7 @@ SELECT m.`id`,m.`title_cn` AS `title`, m.`desc_cn` AS `description`, m.`labels_cn` AS `labels`, m.image, m.`is_active`, - m.`icon`,m.`order`, app.`appconn_name` as `name`, app.`if_iframe`, app.`is_external` + m.`icon`,m.`order`, app.`appconn_name` as `name`, app.`if_iframe`, app.`is_external`,app.`is_micro_app` FROM dss_workspace_menu_appconn m JOIN dss_appconn app ON m.`appconn_id` = app.`id` @@ -131,7 +132,7 @@ SELECT m.`id`,m.`title_en` AS `title`, m.`desc_en` AS `description`, m.`labels_en` AS `labels`, m.image, m.`is_active`, - m.`icon`,m.`order`, app.`appconn_name` as `name`, app.`if_iframe`, app.`is_external` + m.`icon`,m.`order`, app.`appconn_name` as `name`, app.`if_iframe`, app.`is_external`,app.`is_micro_app` FROM dss_workspace_menu_appconn m JOIN dss_appconn app ON m.`appconn_id` = app.`id` diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/restful/DSSDictionaryRestful.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/restful/DSSDictionaryRestful.java index 8bba9f7e0d..d1b04a994b 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/restful/DSSDictionaryRestful.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/restful/DSSDictionaryRestful.java @@ -52,6 +52,7 @@ public class DSSDictionaryRestful { */ @RequestMapping(path ="getDicList", method = RequestMethod.POST) public Message getDevFlowList(HttpServletRequest request, @RequestBody DSSDictionaryRequestVO dictionaryRequestVO){ + LOGGER.info("begin to getDicList for dickey:{}, parantKey:{} ", dictionaryRequestVO.getDicKey(), dictionaryRequestVO.getParentKey()); try{ if(dictionaryRequestVO.getWorkspaceId()==null){ return Message.error("workspaceId(空间id)不能为空"); @@ -73,6 +74,7 @@ public Message getDevFlowList(HttpServletRequest request, @RequestBody DSSDictio @RequestMapping(path ="getDicSecondList", method = RequestMethod.POST) public Message getDicSecondList(HttpServletRequest request, @RequestBody DSSDictionaryRequestVO dictionaryRequestVO){ + LOGGER.info("begin to getDicSecondList for dickey:{}, parantKey:{} ", dictionaryRequestVO.getDicKey(), dictionaryRequestVO.getParentKey()); try{ if(dictionaryRequestVO.getWorkspaceId()==null){ return Message.error("workspaceId(空间id)不能为空"); diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/restful/DSSSideInfoRestful.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/restful/DSSSideInfoRestful.java index da58c2f5cf..e2f1965081 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/restful/DSSSideInfoRestful.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/restful/DSSSideInfoRestful.java @@ -42,6 +42,7 @@ public class DSSSideInfoRestful { @RequestMapping(path ="getSideInfos", method = RequestMethod.GET) public Message getSideInfos(HttpServletRequest request, @RequestParam(required = false, name = "workspaceID") Long workspaceId){ String username = SecurityFilter.getLoginUsername(request); + LOGGER.info("Begin to getSideInfos for user:{}", username); try{ boolean isEnglish = "en".equals(request.getHeader("Content-language")); return Message.ok("获取侧边栏成功").data("presentations", dssSideInfoService.getSidebarVOList(username, workspaceId,isEnglish)); diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/restful/DSSWorkspacePrivRestful.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/restful/DSSWorkspacePrivRestful.java index eb98251c7b..df26205493 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/restful/DSSWorkspacePrivRestful.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/restful/DSSWorkspacePrivRestful.java @@ -17,6 +17,10 @@ package com.webank.wedatasphere.dss.framework.workspace.restful; +import com.webank.wedatasphere.dss.common.auditlog.OperateTypeEnum; +import com.webank.wedatasphere.dss.common.auditlog.TargetTypeEnum; +import com.webank.wedatasphere.dss.common.exception.DSSErrorException; +import com.webank.wedatasphere.dss.common.utils.AuditLogUtils; import com.webank.wedatasphere.dss.framework.workspace.bean.request.UpdateRoleComponentPrivRequest; import com.webank.wedatasphere.dss.framework.workspace.bean.request.UpdateRoleMenuPrivRequest; import com.webank.wedatasphere.dss.framework.workspace.bean.vo.DSSWorkspaceHomepageSettingVO; @@ -24,9 +28,14 @@ import com.webank.wedatasphere.dss.framework.workspace.service.DSSWorkspacePrivService; import com.webank.wedatasphere.dss.framework.workspace.service.DSSWorkspaceService; import com.webank.wedatasphere.dss.framework.workspace.util.WorkspaceDBHelper; +import com.webank.wedatasphere.dss.framework.workspace.util.WorkspaceUtils; +import com.webank.wedatasphere.dss.standard.app.sso.Workspace; +import com.webank.wedatasphere.dss.standard.sso.utils.SSOHelper; import org.apache.commons.math3.util.Pair; import org.apache.linkis.server.Message; import org.apache.linkis.server.security.SecurityFilter; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.web.bind.annotation.*; @@ -42,35 +51,47 @@ @RestController public class DSSWorkspacePrivRestful { + private static final Logger LOGGER = LoggerFactory.getLogger(DSSWorkspacePrivRestful.class); + @Autowired DSSWorkspaceService dssWorkspaceService; @Autowired DSSWorkspacePrivService dssWorkspacePrivService; @Autowired WorkspaceDBHelper workspaceDBHelper; + @Autowired + HttpServletRequest httpServletRequest; @RequestMapping(path ="getWorkspaceMenuPrivs", method = RequestMethod.GET) - public Message getWorkspaceMenuPrivs(HttpServletRequest request, @RequestParam(WORKSPACE_ID_STR) String workspaceId){ + public Message getWorkspaceMenuPrivs(@RequestParam(WORKSPACE_ID_STR) String workspaceId){ //todo 返回工作空间中角色对菜单的访问权限 DSSWorkspacePrivVO workspaceMenuPrivs = dssWorkspaceService.getWorkspaceMenuPrivs(workspaceId); return Message.ok().data("workspaceMenuPrivs", workspaceMenuPrivs); } @RequestMapping(path ="getWorkspaceHomepageSettings", method = RequestMethod.GET) - public Message getWorkspaceHomepageSettings(HttpServletRequest request, @RequestParam(WORKSPACE_ID_STR) int workspaceId){ - String username = SecurityFilter.getLoginUsername(request); + public Message getWorkspaceHomepageSettings(@RequestParam(WORKSPACE_ID_STR) int workspaceId){ DSSWorkspaceHomepageSettingVO dssWorkspaceHomepageSettingVO = dssWorkspaceService.getWorkspaceHomepageSettings(workspaceId); return Message.ok().data("homepageSettings", dssWorkspaceHomepageSettingVO); } - @RequestMapping(path ="updateRoleMenuPriv", method = RequestMethod.POST) - public Message updateRoleMenuPriv(HttpServletRequest request,@RequestBody UpdateRoleMenuPrivRequest updateRoleMenuPrivRequest){ - String updater = SecurityFilter.getLoginUsername(request); + /** + * 更新角色菜单权限 + * @param updateRoleMenuPrivRequest + * @return + */ + @RequestMapping(path = "updateRoleMenuPriv", method = RequestMethod.POST) + public Message updateRoleMenuPriv(@RequestBody UpdateRoleMenuPrivRequest updateRoleMenuPrivRequest) throws Exception{ + String updater = SecurityFilter.getLoginUsername(httpServletRequest); int menuId = updateRoleMenuPrivRequest.getMenuId(); int workspaceId = updateRoleMenuPrivRequest.getWorkspaceId(); + WorkspaceUtils.validateWorkspace(workspaceId, httpServletRequest); + if (!dssWorkspaceService.checkAdminByWorkspace(updater, workspaceId)) { + return Message.error("无权限进行该操作"); + } Map menuPrivs = updateRoleMenuPrivRequest.getMenuPrivs(); List> pairs = new ArrayList<>(); - for(String key : menuPrivs.keySet()){ + for (String key : menuPrivs.keySet()) { Integer roleId = dssWorkspacePrivService.getRoleId(workspaceId, key); if (roleId == null) { roleId = workspaceDBHelper.getRoleIdByName(key); @@ -78,30 +99,40 @@ public Message updateRoleMenuPriv(HttpServletRequest request,@RequestBody Update pairs.add(new Pair(roleId, menuPrivs.get(key))); } dssWorkspacePrivService.updateRoleMenuPriv(workspaceId, menuId, updater, pairs); + String workspaceName= dssWorkspaceService.getWorkspaceName((long)workspaceId); + AuditLogUtils.printLog(updater,workspaceId, workspaceName, TargetTypeEnum.WORKSPACE,workspaceId,workspaceName, + OperateTypeEnum.UPDATE_ROLE_MENU, updateRoleMenuPrivRequest); return Message.ok("更新角色对于菜单的权限成功"); } @RequestMapping(path ="updateRoleComponentPriv", method = RequestMethod.POST) - public Message updateRoleComponentPriv(HttpServletRequest request,@RequestBody UpdateRoleComponentPrivRequest updateRoleComponentPrivRequest){ + public Message updateRoleComponentPriv(@RequestBody UpdateRoleComponentPrivRequest updateRoleComponentPrivRequest)throws DSSErrorException{ //todo 更新工作空间中角色对于component的权限 - String username = SecurityFilter.getLoginUsername(request); - int menuId = updateRoleComponentPrivRequest.getComponentId(); + String username = SecurityFilter.getLoginUsername(httpServletRequest); + int appconnId = updateRoleComponentPrivRequest.getComponentId(); int workspaceId = updateRoleComponentPrivRequest.getWorkspaceId(); - Map componentPrivs = updateRoleComponentPrivRequest.getComponentPrivs(); + WorkspaceUtils.validateWorkspace(workspaceId, httpServletRequest); + if (!dssWorkspaceService.checkAdminByWorkspace(username, workspaceId)) { + return Message.error("无权限进行该操作"); + } + Map componentPrivs = updateRoleComponentPrivRequest.getComponentPrivs(); List> pairs = new ArrayList<>(); - for (String key : componentPrivs.keySet()){ + for (String key : componentPrivs.keySet()) { Integer roleId = dssWorkspacePrivService.getRoleId(workspaceId, key); if (roleId == null) { roleId = workspaceDBHelper.getRoleIdByName(key); } pairs.add(new Pair(roleId, componentPrivs.get(key))); } - dssWorkspacePrivService.updateRoleComponentPriv(workspaceId, menuId, username, pairs); - return Message.ok().data("updateRoleComponentPriv","更新组件权限成功"); + dssWorkspacePrivService.updateRoleComponentPriv(workspaceId, appconnId, username, pairs); + String workspaceName = dssWorkspaceService.getWorkspaceName((long) workspaceId); + AuditLogUtils.printLog(username, workspaceId, workspaceName, TargetTypeEnum.WORKSPACE, + workspaceId, workspaceName, OperateTypeEnum.UPDATE_ROLE_MENU, updateRoleComponentPrivRequest); + return Message.ok().data("updateRoleComponentPriv", "更新组件权限成功"); } @RequestMapping(path ="updateRoleHomepage", method = RequestMethod.POST) - public Message updateRoleHomepage(HttpServletRequest request){ + public Message updateRoleHomepage(){ return null; } } diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/restful/DSSWorkspaceRestful.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/restful/DSSWorkspaceRestful.java index 659eb716d7..cc8ac0bab7 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/restful/DSSWorkspaceRestful.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/restful/DSSWorkspaceRestful.java @@ -16,68 +16,128 @@ package com.webank.wedatasphere.dss.framework.workspace.restful; +import com.webank.wedatasphere.dss.common.auditlog.OperateTypeEnum; +import com.webank.wedatasphere.dss.common.auditlog.TargetTypeEnum; +import com.webank.wedatasphere.dss.common.exception.DSSErrorException; +import com.webank.wedatasphere.dss.common.utils.AuditLogUtils; +import com.webank.wedatasphere.dss.common.utils.DSSCommonUtils; import com.webank.wedatasphere.dss.framework.admin.service.DssAdminUserService; import com.webank.wedatasphere.dss.framework.workspace.bean.DSSWorkspace; +import com.webank.wedatasphere.dss.framework.workspace.bean.NoticeContent; +import com.webank.wedatasphere.dss.framework.workspace.bean.dto.response.WorkspaceFavoriteVo; +import com.webank.wedatasphere.dss.framework.workspace.bean.dto.response.WorkspaceMenuVo; import com.webank.wedatasphere.dss.framework.workspace.bean.request.CreateWorkspaceRequest; import com.webank.wedatasphere.dss.framework.workspace.bean.vo.DSSWorkspaceHomePageVO; import com.webank.wedatasphere.dss.framework.workspace.bean.vo.DSSWorkspaceOverviewVO; import com.webank.wedatasphere.dss.framework.workspace.bean.vo.DSSWorkspaceVO; -import com.webank.wedatasphere.dss.framework.workspace.bean.vo.DepartmentVO; +import com.webank.wedatasphere.dss.framework.workspace.service.DSSWorkspaceRoleService; import com.webank.wedatasphere.dss.framework.workspace.service.DSSWorkspaceService; +import com.webank.wedatasphere.dss.framework.workspace.service.NoticeService; import com.webank.wedatasphere.dss.framework.workspace.util.WorkspaceDBHelper; +import com.webank.wedatasphere.dss.framework.workspace.util.WorkspaceUtils; import com.webank.wedatasphere.dss.standard.app.sso.Workspace; import com.webank.wedatasphere.dss.standard.common.exception.AppStandardWarnException; import com.webank.wedatasphere.dss.standard.sso.utils.SSOHelper; import org.apache.linkis.common.exception.ErrorException; import org.apache.linkis.server.Message; import org.apache.linkis.server.security.SecurityFilter; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.util.CollectionUtils; import org.springframework.web.bind.annotation.*; import javax.servlet.http.HttpServletRequest; -import java.util.ArrayList; -import java.util.List; +import javax.servlet.http.HttpServletResponse; +import java.util.*; +import java.util.stream.Collectors; import static com.webank.wedatasphere.dss.framework.workspace.util.DSSWorkspaceConstant.WORKSPACE_ID_STR; @RequestMapping(path = "/dss/framework/workspace", produces = {"application/json"}) @RestController public class DSSWorkspaceRestful { + + private static final Logger LOGGER = LoggerFactory.getLogger(DSSWorkspaceRestful.class); + @Autowired private DSSWorkspaceService dssWorkspaceService; @Autowired private DssAdminUserService dssUserService; @Autowired private WorkspaceDBHelper workspaceDBHelper; + @Autowired + private DSSWorkspaceRoleService dssWorkspaceRoleService; - @RequestMapping(path ="createWorkspace", method = RequestMethod.POST) - public Message createWorkspace(HttpServletRequest request, @RequestBody CreateWorkspaceRequest createWorkspaceRequest)throws ErrorException { - String userName = SecurityFilter.getLoginUsername(request); - if (!dssWorkspaceService.checkAdmin(userName)){ - return Message.error("您好,您不是管理员,没有权限建立工作空间"); + @Autowired + private HttpServletRequest httpServletRequest; + @Autowired + private HttpServletResponse httpServletResponse; + + @Autowired + private NoticeService noticeService; + + @RequestMapping(path = "createWorkspace", method = RequestMethod.POST) + public Message createWorkspace(@RequestBody CreateWorkspaceRequest createWorkspaceRequest) throws ErrorException { + String userName = SecurityFilter.getLoginUsername(httpServletRequest); + if (!dssWorkspaceService.checkAdmin(userName)) { + return Message.error("您好,您不是管理员,没有权限建立工作空间"); } String workSpaceName = createWorkspaceRequest.getWorkspaceName(); String department = createWorkspaceRequest.getDepartment(); String description = createWorkspaceRequest.getDescription(); String stringTags = createWorkspaceRequest.getTags(); String productName = createWorkspaceRequest.getProductName(); - int workspaceId = dssWorkspaceService.createWorkspace(workSpaceName, stringTags, userName, description, department, productName,""); - return Message.ok().data("workspaceId", workspaceId).data("workspaceName",workSpaceName); + int workspaceId = dssWorkspaceService.createWorkspace(workSpaceName, stringTags, userName, description, department, productName, ""); + AuditLogUtils.printLog(userName, workspaceId, workSpaceName, TargetTypeEnum.WORKSPACE, workspaceId, workSpaceName, + OperateTypeEnum.CREATE, createWorkspaceRequest); + return Message.ok().data("workspaceId", workspaceId).data("workspaceName", workSpaceName); + } + + /** + * 返回所有office,格式为:a-b-c + * + * @param workspaceId + * @return + */ + @RequestMapping(path = "listAllDepartments", method = RequestMethod.GET) + public Message listAllDepartments(@RequestParam(value = WORKSPACE_ID_STR, required = false) String workspaceId) { + List departments = dssWorkspaceService.getAllDepartmentWithOffices(); + return Message.ok().data("departmentWithOffices", departments); + } + + /** + * 绑定工作空间和部门-科室关系 + * + * @return + */ + @PostMapping(value = "associateDepartments") + public Message associateDepartments(@RequestBody Map params) throws Exception { + String userName = SecurityFilter.getLoginUsername(httpServletRequest); + Long workspaceId = Long.valueOf((Integer) params.get("workspaceId")); + WorkspaceUtils.validateWorkspace(workspaceId, httpServletRequest); + ArrayList departmentWithOffices = (ArrayList) params.get("departmentWithOffices"); + ArrayList roles = (ArrayList) params.get("roles"); + if (CollectionUtils.isEmpty(departmentWithOffices) || CollectionUtils.isEmpty(roles)) { + return Message.error("参数不能为空!(params can not be null!)"); + } + String roleStr = roles.stream().map(String::valueOf).collect(Collectors.joining(",")); + dssWorkspaceService.associateDepartments(workspaceId, String.join(",", departmentWithOffices), roleStr, userName); + return Message.ok(); } - @RequestMapping(path ="listDepartments", method = RequestMethod.GET) - public Message listDepartments(HttpServletRequest request, @RequestParam(WORKSPACE_ID_STR) String workspaceId){ - //todo 要从um中获取 - List departments = dssWorkspaceService.getDepartments(); - return Message.ok().data("departments", departments); + @GetMapping(value = "{workspaceId}/associateDepartmentsInfo") + public Message getAssociateDepartments(@PathVariable("workspaceId") Long workspaceId) throws Exception { + WorkspaceUtils.validateWorkspace(workspaceId, httpServletRequest); + return Message.ok().data("associateDepartments", dssWorkspaceService.getAssociateDepartmentsInfo(workspaceId)); } - @RequestMapping(path ="getWorkspaces", method = RequestMethod.GET) - public Message getWorkspaces(HttpServletRequest request){ - String username = SecurityFilter.getLoginUsername(request); + @RequestMapping(path = "getWorkspaces", method = RequestMethod.GET) + public Message getWorkspaces() { + String username = SecurityFilter.getLoginUsername(httpServletRequest); List workspaces = dssWorkspaceService.getWorkspaces(username); List dssWorkspaceVOS = new ArrayList<>(); - for (DSSWorkspace workspace:workspaces ){ + for (DSSWorkspace workspace : workspaces) { String name = workspace.getName(); int id = workspace.getId(); String labels = workspace.getLabel(); @@ -94,33 +154,193 @@ public Message getWorkspaces(HttpServletRequest request){ return Message.ok().data("workspaces", dssWorkspaceVOS); } - @RequestMapping(path ="getWorkspaceHomePage", method = RequestMethod.GET) - public Message getWorkspaceHomePage(HttpServletRequest request, @RequestParam(required = false, name = "micro_module") String moduleName) throws Exception{ + @RequestMapping(path = "getWorkspaceHomePage", method = RequestMethod.GET) + public Message getWorkspaceHomePage(@RequestParam(required = false, name = "micro_module") String moduleName) throws Exception { //如果用户的工作空间大于两个,那么就直接返回/workspace页面 - String username = SecurityFilter.getLoginUsername(request); + String username = SecurityFilter.getLoginUsername(httpServletRequest); Workspace workspace = new Workspace(); try { - SSOHelper.addWorkspaceInfo(request, workspace); - } catch (AppStandardWarnException ignored) {} // ignore it. + LOGGER.info("Put gateway url and cookies into workspace."); + SSOHelper.addWorkspaceInfo(httpServletRequest, workspace); + } catch (AppStandardWarnException ignored) { + } // ignore it. dssUserService.insertOrUpdateUser(username, workspace); - DSSWorkspaceHomePageVO dssWorkspaceHomePageVO = dssWorkspaceService.getWorkspaceHomePage(username,moduleName); + DSSWorkspaceHomePageVO dssWorkspaceHomePageVO = dssWorkspaceService.getWorkspaceHomePage(username, moduleName); return Message.ok().data("workspaceHomePage", dssWorkspaceHomePageVO); } - @RequestMapping(path ="getOverview", method = RequestMethod.GET) - public Message getOverview(HttpServletRequest request, @RequestParam(WORKSPACE_ID_STR) int workspaceId){ - String username = SecurityFilter.getLoginUsername(request); - String language = request.getHeader("Content-language"); + @RequestMapping(path = "getOverview", method = RequestMethod.GET) + public Message getOverview(@RequestParam(WORKSPACE_ID_STR) int workspaceId) { + String username = SecurityFilter.getLoginUsername(httpServletRequest); + String language = httpServletRequest.getHeader("Content-language"); boolean isEnglish = "en".equals(language); DSSWorkspaceOverviewVO dssWorkspaceOverviewVO = dssWorkspaceService.getOverview(username, workspaceId, isEnglish); return Message.ok().data("overview", dssWorkspaceOverviewVO); } - @RequestMapping(path ="refreshCache", method = RequestMethod.GET) - public Message refreshCache(HttpServletRequest request){ + @RequestMapping(path = "refreshCache", method = RequestMethod.GET) + public Message refreshCache() { workspaceDBHelper.retrieveFromDB(); return Message.ok("refresh ok"); } + @RequestMapping(path = "workspaces", method = RequestMethod.GET) + public Message getAllWorkspaces() { + String username = SecurityFilter.getLoginUsername(httpServletRequest); + List workspaces = dssWorkspaceService.getWorkspaces(username); + return Message.ok().data("workspaces", workspaces); + } + + /** + * 获取所有工程或者单个工程 + * + * @return + */ + @RequestMapping(path = "getWorkSpaceStr", method = RequestMethod.GET) + public Message getWorkSpaceStr(@RequestParam(name = "workspaceName") String workspaceName) { + String username = SecurityFilter.getLoginUsername(httpServletRequest); + DSSWorkspace workspaceEntity; + try { + workspaceEntity = dssWorkspaceService.getWorkspacesByName(workspaceName, username); + } catch (DSSErrorException e) { + LOGGER.error("User {} get workspace {} failed.", username, workspaceName, e); + return Message.error(e); + } + Workspace workspace = SSOHelper.setAndGetWorkspace(httpServletRequest, httpServletResponse, workspaceEntity.getId(), workspaceName); + return Message.ok("succeed.").data("workspaceStr", DSSCommonUtils.COMMON_GSON.toJson(workspace)); + } + + @RequestMapping(path = "/workspaces/{id}", method = RequestMethod.GET) + public Message getWorkspacesById(@PathVariable("id") Long workspaceId) { + String username = SecurityFilter.getLoginUsername(httpServletRequest); + DSSWorkspace workspace = null; + try { + workspace = dssWorkspaceService.getWorkspacesById(workspaceId, username); + } catch (DSSErrorException e) { + LOGGER.error("User {} get workspace {} failed.", username, workspaceId, e); + return Message.error(e); + } + SSOHelper.setAndGetWorkspace(httpServletRequest, httpServletResponse, workspace.getId(), workspace.getName()); + List roles = dssWorkspaceRoleService.getRoleInWorkspace(username, workspaceId.intValue()); + if (roles == null || roles.isEmpty()) { + LOGGER.error("username {}, in workspace {} roles are null or empty", username, workspaceId); + return Message.error("can not get roles information"); + } + //判断如果是没有权限的,那么就直接干掉 + if (roles.contains("apiUser")) { + int priv = dssWorkspaceRoleService.getApiPriv(username, workspaceId.intValue(), "apiUser", "apiService"); + if (priv <= 0) { + roles.remove("apiUser"); + } + } + Message retMessage = Message.ok(); + //如果其他的角色也是有这个api权限的,那么就加上这个apiUser + boolean flag = false; + for (String role : roles) { + int priv = dssWorkspaceRoleService.getApiPriv(username, workspaceId.intValue(), role, "apiService"); + if (priv >= 1) { + flag = true; + break; + } + } + if (flag && !roles.contains("apiUser")) { + roles.add("apiUser"); + } + return retMessage.data("roles", roles).data("workspace", workspace); + } + + @RequestMapping(path = "/workspaces/exists", method = RequestMethod.GET) + public Message getUsernameExistence(@RequestParam(required = false, name = "name") String name) { + boolean exists = dssWorkspaceService.existWorkspaceName(name); + return Message.ok().data("workspaceNameExists", exists); + } + + @RequestMapping(path = "/workspaces", method = RequestMethod.POST) + public Message addWorkspace(@RequestBody Map json) throws ErrorException { + String userName = SecurityFilter.getLoginUsername(httpServletRequest); + Workspace workspace = SSOHelper.getWorkspace(httpServletRequest); + if (!dssWorkspaceService.checkAdmin(userName)) { + return Message.error("您好,您不是管理员,没有权限建立工作空间"); + + } + String workspaceName = json.get("name"); + if (dssWorkspaceService.existWorkspaceName(workspaceName)) { + return Message.error("工作空间名重复"); + } + String department = json.get("department"); + String label = json.get("label"); + String description = json.get("description"); + String workspaceType = json.get("workspace_type"); + + String productName = "DSS"; + int workspaceId = dssWorkspaceService.createWorkspace(workspaceName, label, userName, description, department, productName, workspaceType); + AuditLogUtils.printLog(userName, workspaceId, workspaceName, TargetTypeEnum.WORKSPACE, workspaceId, workspaceName, + OperateTypeEnum.CREATE, json); + return Message.ok().data("workspaceId", workspaceId); + } + + @RequestMapping(path = "workspaces/{workspaceId}/appconns", method = RequestMethod.GET) + public Message getWorkspaceAppConns(@PathVariable("workspaceId") Long workspaceId) { + String header = httpServletRequest.getHeader("Content-language").trim(); + boolean isChinese = "zh-CN".equals(header); + String username = SecurityFilter.getLoginUsername(httpServletRequest); + Workspace workspace = SSOHelper.getWorkspace(httpServletRequest); + List menuAppconnVos; + try { + menuAppconnVos = dssWorkspaceService.getWorkspaceAppConns(workspace, workspaceId, username, isChinese); + } catch (DSSErrorException e) { + LOGGER.warn("{} get appconns from workspace {} failed.", username, workspaceId, e); + return Message.error(e); + } + return Message.ok().data("menus", menuAppconnVos); + } + + + @RequestMapping(path = "/workspaces/{workspaceId}/favorites", method = RequestMethod.GET) + public Message getWorkspaceFavorites(@PathVariable("workspaceId") Long workspaceId, @RequestParam(value = "type", required = false) String type) { + String header = httpServletRequest.getHeader("Content-language").trim(); + boolean isChinese = "zh-CN".equals(header); + String username = SecurityFilter.getLoginUsername(httpServletRequest); + List favorites = dssWorkspaceService.getWorkspaceFavorites(workspaceId, username, isChinese, type == null ? "" : type); + Set favoriteVos = new HashSet<>(favorites); + return Message.ok().data("favorites", favoriteVos); + } + + /** + * 应用加入收藏,返回收藏后id + * + * @param json + * @return + */ + @RequestMapping(path = "/workspaces/{workspaceId}/favorites", method = RequestMethod.POST) + public Message addFavorite(@PathVariable("workspaceId") Long workspaceId, @RequestBody Map json) { + String username = SecurityFilter.getLoginUsername(httpServletRequest); + Long menuApplicationId = Long.valueOf(json.get("menuApplicationId")); + String type = json.get("type") == null ? "" : json.get("type"); + String workspaceName = dssWorkspaceService.getWorkspaceName(workspaceId); + Long favoriteId = dssWorkspaceService.addFavorite(username, workspaceId, menuApplicationId, type); + AuditLogUtils.printLog(username, workspaceId, workspaceName, TargetTypeEnum.WORKSPACE, workspaceId, workspaceName, + OperateTypeEnum.ADD_TO_FAVORITES, json); + return Message.ok().data("favoriteId", favoriteId); + } + + + @RequestMapping(path = "/workspaces/{workspaceId}/favorites/{appconnId}", method = RequestMethod.POST) + public Message deleteFavorite(@PathVariable("workspaceId") Long workspaceId, @PathVariable("appconnId") Long appconnId, @RequestBody Map json) { + String username = SecurityFilter.getLoginUsername(httpServletRequest); + String type = json.get("type") == null ? "" : json.get("type"); + String workspaceName = dssWorkspaceService.getWorkspaceName(workspaceId); + Long favoriteId = dssWorkspaceService.deleteFavorite(username, appconnId, workspaceId, type); + AuditLogUtils.printLog(username, workspaceId, workspaceName, TargetTypeEnum.WORKSPACE, workspaceId, workspaceName, + OperateTypeEnum.REM_FROM_FAVORITES, json); + return Message.ok().data("favoriteId", favoriteId); + } + + @GetMapping("getNotice") + public Message getNotice(){ + List noticeContent= noticeService.getNoticeContent(); + return Message.ok("公告获取成功").data("notices", noticeContent); + } + } diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/restful/DSSWorkspaceRoleRestful.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/restful/DSSWorkspaceRoleRestful.java index 5f383e0cca..321ff42aed 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/restful/DSSWorkspaceRoleRestful.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/restful/DSSWorkspaceRoleRestful.java @@ -17,6 +17,10 @@ package com.webank.wedatasphere.dss.framework.workspace.restful; +import com.webank.wedatasphere.dss.common.auditlog.OperateTypeEnum; +import com.webank.wedatasphere.dss.common.auditlog.TargetTypeEnum; +import com.webank.wedatasphere.dss.common.utils.AuditLogUtils; +import com.webank.wedatasphere.dss.framework.workspace.bean.DSSWorkspaceRole; import com.webank.wedatasphere.dss.framework.workspace.bean.request.AddWorkspaceRoleRequest; import com.webank.wedatasphere.dss.framework.workspace.bean.vo.DSSWorkspaceRoleVO; import com.webank.wedatasphere.dss.framework.workspace.service.DSSWorkspaceRoleService; @@ -45,7 +49,7 @@ public class DSSWorkspaceRoleRestful { private DSSWorkspaceRoleService dssWorkspaceRoleService; @RequestMapping(path = "getWorkspaceRoles", method = RequestMethod.GET) - public Message getWorkspaceRoles(HttpServletRequest request, @RequestParam(WORKSPACE_ID_STR) int workspaceId) { + public Message getWorkspaceRoles(@RequestParam(WORKSPACE_ID_STR) int workspaceId) { //todo 获取工作空间中所有的角色 List workspaceRoles = dssWorkspaceService.getWorkspaceRoles(workspaceId); return Message.ok().data("workspaceRoles", workspaceRoles); @@ -61,74 +65,17 @@ public Message addWorkspaceRole(HttpServletRequest request, @RequestBody AddWork if (!dssWorkspaceService.checkAdmin(username) || !dssWorkspaceService.checkAdminByWorkspace(username, workspaceId)) { return Message.error("无权限进行该操作"); } - dssWorkspaceRoleService.addWorkspaceRole(roleName, workspaceId, menuIds, componentIds, username); + DSSWorkspaceRole dssRole = dssWorkspaceRoleService.addWorkspaceRole(roleName, workspaceId, menuIds, componentIds, username); + String workspaceName = dssWorkspaceService.getWorkspaceName((long) workspaceId); + AuditLogUtils.printLog(username, workspaceId,workspaceName, TargetTypeEnum.WORKSPACE_ROLE,dssRole.getId(),roleName, + OperateTypeEnum.CREATE, addWorkspaceRoleRequest); return Message.ok("创建角色成功"); } @RequestMapping(path = "deleteWorkspaceRole", method = RequestMethod.POST) - public Message deleteWorkspaceRole(HttpServletRequest request) { + public Message deleteWorkspaceRole() { return null; } -// @RequestMapping(path = "getWorkspaceBaseInfo", method = RequestMethod.GET) -// @Deprecated -// public Message getWorkspaceInfo(HttpServletRequest request, -// HttpServletResponse response, -// @RequestParam(WORKSPACE_ID_STR) Integer workspaceId) { -// String username = SecurityFilter.getLoginUsername(request); -// //如果workspaceId为null的话,那么就找到这个用户工作空间 -// if (workspaceId == null || workspaceId <= 0) { -// workspaceId = dssWorkspaceRoleService.getWorkspaceIdByUser(username); -// } -// DSSWorkspace workspace; -// try { -// workspace = dssWorkspaceService.getWorkspacesById(workspaceId.longValue(), username); -// } catch (DSSErrorException e) { -// return Message.error(ExceptionUtils.getRootCauseMessage(e)); -// } -// //将workspaceId作为cookie写入 -// SSOHelper.setAndGetWorkspace(request, response, workspaceId, workspace.getName()); -// List roles = dssWorkspaceRoleService.getRoleInWorkspace(username, workspaceId); -// if (roles == null || roles.isEmpty()) { -// LOGGER.error("username {}, in workspace {} roles are null or empty", username, workspaceId); -// return Message.error("can not get roles information"); -// } -// //判断如果是没有权限的,那么就直接干掉 -// if (roles.contains("apiUser")) { -// int priv = dssWorkspaceRoleService.getApiPriv(username, workspaceId, "apiUser", "apiService"); -// if (priv <= 0) { -// roles.remove("apiUser"); -// } -// } -// Message retMessage = Message.ok(); -// //工作空间中,加上用户在顶部的菜单 -// if (roles.contains("analyser")) { -// retMessage.data("topName", "Scriptis"); -// retMessage.data("topUrl", "/home"); -// } else if (roles.contains("developer")) { -// retMessage.data("topName", "Scriptis"); -// retMessage.data("topUrl", "/home"); -// } else if (roles.contains("apiUser") && roles.size() == 1) { -// retMessage.data("topName", "Scriptis"); -// retMessage.data("topUrl", "/home"); -// } else { -// retMessage.data("topName", "Scriptis"); -// retMessage.data("topUrl", "/home"); -// } -// //如果其他的角色也是有这个api权限的,那么就加上这个apiUser -// boolean flag = false; -// for (String role : roles) { -// int priv = dssWorkspaceRoleService.getApiPriv(username, workspaceId, role, "apiService"); -// if (priv >= 1) { -// flag = true; -// break; -// } -// } -// if (flag && !roles.contains("apiUser")) { -// roles.add("apiUser"); -// } -// return retMessage.data("roles", roles).data("workspaceId", workspaceId); -// } - } diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/restful/DSSWorkspaceUserRestful.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/restful/DSSWorkspaceUserRestful.java index caab1675cd..1cdea452f2 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/restful/DSSWorkspaceUserRestful.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/restful/DSSWorkspaceUserRestful.java @@ -16,20 +16,27 @@ package com.webank.wedatasphere.dss.framework.workspace.restful; +import com.webank.wedatasphere.dss.common.auditlog.OperateTypeEnum; +import com.webank.wedatasphere.dss.common.auditlog.TargetTypeEnum; +import com.webank.wedatasphere.dss.common.utils.AuditLogUtils; import com.webank.wedatasphere.dss.framework.workspace.bean.request.DeleteWorkspaceUserRequest; import com.webank.wedatasphere.dss.framework.workspace.bean.request.UpdateWorkspaceUserRequest; import com.webank.wedatasphere.dss.framework.workspace.bean.vo.DSSWorkspaceRoleVO; import com.webank.wedatasphere.dss.framework.workspace.bean.vo.DSSWorkspaceUserVO; import com.webank.wedatasphere.dss.framework.workspace.bean.vo.DSSWorkspaceUsersVo; import com.webank.wedatasphere.dss.framework.workspace.bean.vo.StaffInfoVO; +import com.webank.wedatasphere.dss.framework.workspace.service.DSSWorkspaceRoleCheckService; import com.webank.wedatasphere.dss.framework.workspace.service.DSSWorkspaceService; import com.webank.wedatasphere.dss.framework.workspace.service.DSSWorkspaceUserService; import com.webank.wedatasphere.dss.framework.workspace.util.WorkspaceDBHelper; import com.webank.wedatasphere.dss.standard.app.sso.Workspace; +import com.webank.wedatasphere.dss.standard.common.exception.AppStandardWarnException; import com.webank.wedatasphere.dss.standard.sso.utils.SSOHelper; import org.apache.commons.lang.StringUtils; +import org.apache.linkis.common.conf.CommonVars; import org.apache.linkis.server.Message; import org.apache.linkis.server.security.SecurityFilter; +import org.apache.linkis.server.utils.ModuleUserUtils; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.beans.factory.annotation.Autowired; @@ -37,7 +44,9 @@ import javax.servlet.http.HttpServletRequest; import java.util.ArrayList; +import java.util.Arrays; import java.util.List; +import java.util.Map; import java.util.stream.Collectors; import static com.webank.wedatasphere.dss.framework.workspace.util.DSSWorkspaceConstant.WORKSPACE_ID_STR; @@ -48,19 +57,29 @@ public class DSSWorkspaceUserRestful { private static final Logger LOGGER = LoggerFactory.getLogger(DSSWorkspaceUserRestful.class); + private static final String HPMS_USER_TOKEN = CommonVars.apply("wds.dss.workspace.hpms.user.token", "HPMS-KhFGSQkdaaCPBYfE").getValue(); + @Autowired private DSSWorkspaceService dssWorkspaceService; @Autowired private WorkspaceDBHelper workspaceDBHelper; @Autowired private DSSWorkspaceUserService dssWorkspaceUserService; + @Autowired + private HttpServletRequest httpServletRequest; + @Autowired + private DSSWorkspaceRoleCheckService roleCheckService; @RequestMapping(path = "getWorkspaceUsers", method = RequestMethod.GET) - public Message getWorkspaceUsers(HttpServletRequest request, @RequestParam(WORKSPACE_ID_STR) String workspaceId, + public Message getWorkspaceUsers( @RequestParam(WORKSPACE_ID_STR) String workspaceId, @RequestParam(required = false, name = "pageNow") Integer pageNow, @RequestParam(required = false, name = "pageSize") Integer pageSize, @RequestParam(required = false, name = "department") String department, @RequestParam(required = false, name = "userName") String username, @RequestParam(required = false, name = "roleName") String roleName) { //todo 获取工作空间中所有的用户以及他们的角色信息 + if(pageNow==null&&pageSize==null){ + pageSize=Integer.MAX_VALUE; + pageNow=1; + } if (pageNow == null) { pageNow = 1; } @@ -68,27 +87,18 @@ public Message getWorkspaceUsers(HttpServletRequest request, @RequestParam(WORKS //默认改成20 pageSize = 20; } - if (StringUtils.isNotEmpty(roleName)) { - //如果roleName不是空的话,就按照roleName来吧 - List totals = new ArrayList<>(); - List workspaceUsers = - dssWorkspaceService.getWorkspaceUsersByRole(Integer.parseInt(workspaceId), roleName, totals, pageNow, pageSize); - List dssRoles = workspaceDBHelper.getRoleVOs(Integer.parseInt(workspaceId)); - return Message.ok().data("roles", dssRoles).data("workspaceUsers", workspaceUsers).data("total", totals.get(0)); - } else { - List totals = new ArrayList<>(); - List workspaceUsers = - dssWorkspaceService.getWorkspaceUsers(workspaceId, department, username, roleName, pageNow, pageSize, totals); - List dssRoles = workspaceDBHelper.getRoleVOs(Integer.parseInt(workspaceId)); - return Message.ok().data("roles", dssRoles).data("workspaceUsers", workspaceUsers).data("total", totals.get(0)); - } + List totals = new ArrayList<>(); + List workspaceUsers = + dssWorkspaceService.getWorkspaceUsers(workspaceId, department, username, roleName, pageNow, pageSize, totals); + List dssRoles = workspaceDBHelper.getRoleVOs(Integer.parseInt(workspaceId)); + return Message.ok().data("roles", dssRoles).data("workspaceUsers", workspaceUsers).data("total", totals.get(0)); } @RequestMapping(path = "getAllWorkspaceUsers", method = RequestMethod.GET) - public Message getAllWorkspaceUsers(HttpServletRequest request) { + public Message getAllWorkspaceUsers() { DSSWorkspaceUsersVo dssWorkspaceUsersVo = new DSSWorkspaceUsersVo(); // workspaceId改为从cookie取 - int workspaceId = (int) SSOHelper.getWorkspace(request).getWorkspaceId(); + int workspaceId = (int) SSOHelper.getWorkspace(httpServletRequest).getWorkspaceId(); dssWorkspaceUsersVo.setAccessUsers(dssWorkspaceUserService.getAllWorkspaceUsers(workspaceId)); // dssWorkspaceUsersVo.setEditUsers(dssWorkspaceUserService.getWorkspaceEditUsers(workspaceId)); // dssWorkspaceUsersVo.setReleaseUsers(dssWorkspaceUserService.getWorkspaceReleaseUsers(workspaceId)); @@ -99,8 +109,8 @@ public Message getAllWorkspaceUsers(HttpServletRequest request) { @RequestMapping(path = "existUserInWorkspace", method = RequestMethod.GET) - public Message existUserInWorkspace(HttpServletRequest request, @RequestParam(WORKSPACE_ID_STR) int workspaceId, @RequestParam("queryUserName") String queryUserName) { - String username = SecurityFilter.getLoginUsername(request); + public Message existUserInWorkspace(@RequestParam(WORKSPACE_ID_STR) int workspaceId, @RequestParam("queryUserName") String queryUserName) { + String username = SecurityFilter.getLoginUsername(httpServletRequest); List users = dssWorkspaceUserService.getAllWorkspaceUsers(workspaceId); boolean existFlag = users.stream().anyMatch(user -> user.equalsIgnoreCase(queryUserName)); LOGGER.info("Check exist user result:" + existFlag + ", query user is " + queryUserName + ",workSpace id is " + workspaceId); @@ -109,11 +119,23 @@ public Message existUserInWorkspace(HttpServletRequest request, @RequestParam(WO @RequestMapping(path = "addWorkspaceUser", method = RequestMethod.POST) - public Message addWorkspaceUser(HttpServletRequest request, @RequestBody UpdateWorkspaceUserRequest updateWorkspaceUserRequest) { + public Message addWorkspaceUser(@RequestBody UpdateWorkspaceUserRequest updateWorkspaceUserRequest) { //todo 工作空间添加用户 - String creator = SecurityFilter.getLoginUsername(request); + String creator = SecurityFilter.getLoginUsername(httpServletRequest); List roles = updateWorkspaceUserRequest.getRoles(); - Workspace workspace = SSOHelper.getWorkspace(request); + Workspace workspace; + //兼容外部系统通过接口调用场景,cookie未设置workspaceName + if (Arrays.stream(httpServletRequest.getCookies()).noneMatch(l -> l.getName().equals("workspaceName"))) { + workspace = new Workspace(); + try { + workspace = SSOHelper.getWorkspace(httpServletRequest); + } catch (AppStandardWarnException appStandardWarnException) { + workspace.setWorkspaceId(updateWorkspaceUserRequest.getWorkspaceId()); + workspace.setWorkspaceName(String.valueOf(updateWorkspaceUserRequest.getWorkspaceId())); + } + } else { + workspace = SSOHelper.getWorkspace(httpServletRequest); + } int workspaceId = updateWorkspaceUserRequest.getWorkspaceId(); if (workspace.getWorkspaceId() != workspaceId) { return Message.error("cookie 中的 workspaceId 与请求添加用户的 workspace 不同!"); @@ -124,48 +146,56 @@ public Message addWorkspaceUser(HttpServletRequest request, @RequestBody UpdateW if (count != null && count > 0) { return Message.error("用户已经存在该工作空间,不需要重复添加!"); } - if (!dssWorkspaceService.isAdminUser((long) workspaceId, creator)) { + if (!roleCheckService.checkRolesOperation(workspaceId, creator, userName, roles)) { return Message.error("无权限进行该操作"); } dssWorkspaceService.addWorkspaceUser(roles, workspace, userName, creator, userId); + AuditLogUtils.printLog(userName,workspaceId, workspace.getWorkspaceName(), TargetTypeEnum.WORKSPACE,workspaceId, + workspace.getWorkspaceName(), OperateTypeEnum.ADD_USERS,updateWorkspaceUserRequest); return Message.ok(); } @RequestMapping(path = "updateWorkspaceUser", method = RequestMethod.POST) - public Message updateWorkspaceUser(HttpServletRequest request, @RequestBody UpdateWorkspaceUserRequest updateWorkspaceUserRequest) { - String creator = SecurityFilter.getLoginUsername(request); + public Message updateWorkspaceUser( @RequestBody UpdateWorkspaceUserRequest updateWorkspaceUserRequest) { + String creator = SecurityFilter.getLoginUsername(httpServletRequest); List roles = updateWorkspaceUserRequest.getRoles(); int workspaceId = updateWorkspaceUserRequest.getWorkspaceId(); - if (!dssWorkspaceService.isAdminUser(Long.valueOf(workspaceId), creator)) { + String workspaceName= dssWorkspaceService.getWorkspaceName((long)workspaceId); + String userName = updateWorkspaceUserRequest.getUserName(); + if (!roleCheckService.checkRolesOperation(workspaceId, creator, userName, roles)) { return Message.error("无权限进行该操作"); } - String userName = updateWorkspaceUserRequest.getUserName(); dssWorkspaceUserService.updateWorkspaceUser(roles, workspaceId, userName, creator); + AuditLogUtils.printLog(userName,workspaceId, workspaceName, TargetTypeEnum.WORKSPACE,workspaceId, + workspaceName, OperateTypeEnum.UPDATE_USERS,updateWorkspaceUserRequest); return Message.ok(); } @RequestMapping(path = "deleteWorkspaceUser", method = RequestMethod.POST) - public Message deleteWorkspaceUser(HttpServletRequest request, @RequestBody DeleteWorkspaceUserRequest deleteWorkspaceUserRequest) { + public Message deleteWorkspaceUser( @RequestBody DeleteWorkspaceUserRequest deleteWorkspaceUserRequest) { //todo 删除工作空间中的用户 String userName = deleteWorkspaceUserRequest.getUserName(); int workspaceId = deleteWorkspaceUserRequest.getWorkspaceId(); - String creator = SecurityFilter.getLoginUsername(request); - if (!dssWorkspaceService.checkAdmin(creator) || !dssWorkspaceService.checkAdminByWorkspace(creator, workspaceId)) { + String workspaceName= dssWorkspaceService.getWorkspaceName((long)workspaceId); + String creator = SecurityFilter.getLoginUsername(httpServletRequest); + if (!roleCheckService.checkRolesOperation(workspaceId, creator, userName, new ArrayList<>())) { return Message.error("无权限进行该操作"); } dssWorkspaceUserService.deleteWorkspaceUser(userName, workspaceId); + AuditLogUtils.printLog(userName,workspaceId, workspaceName, TargetTypeEnum.WORKSPACE,workspaceId, + workspaceName, OperateTypeEnum.UPDATE_USERS,deleteWorkspaceUserRequest); return Message.ok(); } @RequestMapping(path = "listAllUsers", method = RequestMethod.GET) - public Message listAllUsers(HttpServletRequest request) { + public Message listAllUsers() { List dssUsers = dssWorkspaceUserService.listAllDSSUsers(); return Message.ok().data("users", dssUsers); } @RequestMapping(path = "getWorkspaceIdByUserName", method = RequestMethod.GET) - public Message getWorkspaceIdByUserName(HttpServletRequest request, @RequestParam(required = false, name = "userName") String userName) { - String loginUserName = SecurityFilter.getLoginUsername(request); + public Message getWorkspaceIdByUserName( @RequestParam(required = false, name = "userName") String userName) { + String loginUserName = SecurityFilter.getLoginUsername(httpServletRequest); String queryUserName = userName; if (StringUtils.isEmpty(userName)) { queryUserName = loginUserName; @@ -174,4 +204,35 @@ public Message getWorkspaceIdByUserName(HttpServletRequest request, @RequestPara String userWorkspaceIdStr = userWorkspaceIds.stream().map(x -> x.toString()).collect(Collectors.joining(",")); return Message.ok().data("userWorkspaceIds", userWorkspaceIdStr); } + + @RequestMapping(path = "getUserRole", method = RequestMethod.GET) + public Message getWorkspaceUserRole(@RequestParam(name = "userName") String username) { + String token = ModuleUserUtils.getToken(httpServletRequest); + if (StringUtils.isNotBlank(token)) { + if(!token.equals(HPMS_USER_TOKEN)){ + return Message.error("Token:" + token + " has no permission to get user info."); + } + }else { + return Message.error("User:" + username + " has no permission to get user info."); + } + List> userRoles = dssWorkspaceUserService.getUserRoleByUserName(username); + return Message.ok().data("userName", username).data("roleInfo", userRoles); + } + + @RequestMapping(path = "/clearUser", method = RequestMethod.GET) + public Message clearUser(@RequestParam("userName") String userName) { + String token = ModuleUserUtils.getToken(httpServletRequest); + if (StringUtils.isNotBlank(token)) { + if(!token.equals(HPMS_USER_TOKEN)){ + return Message.error("Token:" + token + " has no permission to clear user."); + } + }else { + return Message.error("User:" + userName + " has no permission to clear user."); + } + boolean clearResult = dssWorkspaceUserService.clearUserByUserName(userName); + AuditLogUtils.printLog(userName,null, null, TargetTypeEnum.WORKSPACE_ROLE,null, + null, OperateTypeEnum.DELETE,null); + return clearResult ? Message.ok("清理成功") : Message.error("userName不是实名用户,不会清理此用户"); + + } } diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/restful/WorkspaceRestfulApi.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/restful/WorkspaceRestfulApi.java deleted file mode 100644 index c24b5e26fd..0000000000 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/restful/WorkspaceRestfulApi.java +++ /dev/null @@ -1,227 +0,0 @@ -/* - * Copyright 2019 WeBank - * 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. - * - */ - -package com.webank.wedatasphere.dss.framework.workspace.restful; - -import com.webank.wedatasphere.dss.common.exception.DSSErrorException; -import com.webank.wedatasphere.dss.common.utils.DSSCommonUtils; -import com.webank.wedatasphere.dss.framework.workspace.bean.DSSWorkspace; -import com.webank.wedatasphere.dss.framework.workspace.bean.dto.response.WorkspaceFavoriteVo; -import com.webank.wedatasphere.dss.framework.workspace.bean.dto.response.WorkspaceMenuVo; -import com.webank.wedatasphere.dss.framework.workspace.bean.vo.DepartmentVO; -import com.webank.wedatasphere.dss.framework.workspace.service.DSSWorkspaceRoleService; -import com.webank.wedatasphere.dss.framework.workspace.service.DSSWorkspaceService; -import com.webank.wedatasphere.dss.standard.app.sso.Workspace; -import com.webank.wedatasphere.dss.standard.sso.utils.SSOHelper; -import org.apache.linkis.common.exception.ErrorException; -import org.apache.linkis.server.Message; -import org.apache.linkis.server.security.SecurityFilter; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.web.bind.annotation.*; - -import javax.servlet.http.HttpServletRequest; -import javax.servlet.http.HttpServletResponse; -import java.util.HashSet; -import java.util.List; -import java.util.Map; -import java.util.Set; - -@RestController -@RequestMapping(path = "/dss/framework/workspace", produces = {"application/json"}) -public class WorkspaceRestfulApi { - private static final Logger LOGGER = LoggerFactory.getLogger(WorkspaceRestfulApi.class); - - @Autowired - private DSSWorkspaceService dssWorkspaceService; - - @Autowired - private DSSWorkspaceRoleService dssWorkspaceRoleService; - - @RequestMapping(path = "workspaces", method = RequestMethod.GET) - public Message getAllWorkspaces(HttpServletRequest req) { - String username = SecurityFilter.getLoginUsername(req); - List workspaces = dssWorkspaceService.getWorkspaces(username); - return Message.ok().data("workspaces", workspaces); - } - - /** - * 获取所有工程或者单个工程 - * - * @param request - * @return - */ - @RequestMapping(path = "getWorkSpaceStr", method = RequestMethod.GET) - public Message getWorkSpaceStr(HttpServletRequest request, - HttpServletResponse response, - @RequestParam(name = "workspaceName") String workspaceName) { - String username = SecurityFilter.getLoginUsername(request); - DSSWorkspace workspaceEntity; - try { - workspaceEntity = dssWorkspaceService.getWorkspacesByName(workspaceName, username); - } catch (DSSErrorException e) { - LOGGER.error("User {} get workspace {} failed.", username, workspaceName, e); - return Message.error(e); - } - Workspace workspace = SSOHelper.setAndGetWorkspace(request, response, workspaceEntity.getId(), workspaceName); - return Message.ok("succeed.").data("workspaceStr", DSSCommonUtils.COMMON_GSON.toJson(workspace)); - } - - @RequestMapping(path = "/workspaces/{id}", method = RequestMethod.GET) - public Message getWorkspacesById(HttpServletRequest req, - HttpServletResponse resp, - @PathVariable("id") Long workspaceId) { - String username = SecurityFilter.getLoginUsername(req); - DSSWorkspace workspace = null; - try { - workspace = dssWorkspaceService.getWorkspacesById(workspaceId, username); - } catch (DSSErrorException e) { - LOGGER.error("User {} get workspace {} failed.", username, workspaceId, e); - return Message.error(e); - } - SSOHelper.setAndGetWorkspace(req, resp, workspace.getId(), workspace.getName()); - List roles = dssWorkspaceRoleService.getRoleInWorkspace(username, workspaceId.intValue()); - if (roles == null || roles.isEmpty()) { - LOGGER.error("username {}, in workspace {} roles are null or empty", username, workspaceId); - return Message.error("can not get roles information"); - } - //判断如果是没有权限的,那么就直接干掉 - if (roles.contains("apiUser")) { - int priv = dssWorkspaceRoleService.getApiPriv(username, workspaceId.intValue(), "apiUser", "apiService"); - if (priv <= 0) { - roles.remove("apiUser"); - } - } - Message retMessage = Message.ok(); - //如果其他的角色也是有这个api权限的,那么就加上这个apiUser - boolean flag = false; - for (String role : roles) { - int priv = dssWorkspaceRoleService.getApiPriv(username, workspaceId.intValue(), role, "apiService"); - if (priv >= 1) { - flag = true; - break; - } - } - if (flag && !roles.contains("apiUser")) { - roles.add("apiUser"); - } - return retMessage.data("roles", roles).data("workspace", workspace); - } - - @RequestMapping(path = "/workspaces/departments", method = RequestMethod.GET) - public Message getAllWorkspaceDepartments(HttpServletRequest req) { - List departments = dssWorkspaceService.getDepartments(); - return Message.ok().data("departments", departments); - } - - @RequestMapping(path = "/workspaces/exists", method = RequestMethod.GET) - public Message getUsernameExistence(HttpServletRequest req, @RequestParam(required = false, name = "name") String name) { - boolean exists = dssWorkspaceService.existWorkspaceName(name); - return Message.ok().data("workspaceNameExists", exists); - } - - @RequestMapping(path = "/workspaces", method = RequestMethod.POST) - public Message addWorkspace(HttpServletRequest req, @RequestBody Map json) throws ErrorException { - String userName = SecurityFilter.getLoginUsername(req); - if (!dssWorkspaceService.checkAdmin(userName)) { - return Message.error("您好,您不是管理员,没有权限建立工作空间"); - - } - String name = json.get("name"); - if (dssWorkspaceService.existWorkspaceName(name)) { - return Message.error("工作空间名重复"); - } - String department = json.get("department"); - String label = json.get("label"); - String description = json.get("description"); - String workspaceType = json.get("workspace_type"); - - String productName = "DSS"; - int workspaceId = dssWorkspaceService.createWorkspace(name, label, userName, description, department, productName, workspaceType); - return Message.ok().data("workspaceId", workspaceId); - } - - @RequestMapping(path = "workspaces/{workspaceId}/appconns", method = RequestMethod.GET) - public Message getWorkspaceAppConns(HttpServletRequest req, @PathVariable("workspaceId") Long workspaceId) { - String header = req.getHeader("Content-language").trim(); - boolean isChinese = "zh-CN".equals(header); - String username = SecurityFilter.getLoginUsername(req); - Workspace workspace = SSOHelper.getWorkspace(req); - List appconns; - try { - appconns = dssWorkspaceService.getWorkspaceAppConns(workspace, workspaceId, username, isChinese); - } catch (DSSErrorException e) { - LOGGER.warn("{} get appconns from workspace {} failed.", username, workspaceId, e); - return Message.error(e); - } - return Message.ok().data("menus", appconns); - } - - - @RequestMapping(path = "/workspaces/{workspaceId}/favorites", method = RequestMethod.GET) - public Message getWorkspaceFavorites(HttpServletRequest req, @PathVariable("workspaceId") Long workspaceId, @RequestParam(value = "type", required = false) String type) { - String header = req.getHeader("Content-language").trim(); - boolean isChinese = "zh-CN".equals(header); - String username = SecurityFilter.getLoginUsername(req); - List favorites = dssWorkspaceService.getWorkspaceFavorites(workspaceId, username, isChinese, type == null ? "" : type); - Set favoriteVos = new HashSet<>(favorites); - return Message.ok().data("favorites", favoriteVos); - } - - /** - * 应用加入收藏,返回收藏后id - * - * @param req - * @param json - * @return - */ - @RequestMapping(path = "/workspaces/{workspaceId}/favorites", method = RequestMethod.POST) - public Message addFavorite(HttpServletRequest req, @PathVariable("workspaceId") Long workspaceId, @RequestBody Map json) { - String username = SecurityFilter.getLoginUsername(req); -// Long menuApplicationId = json.get("menuApplicationId").getLongValue(); - Long menuApplicationId = Long.valueOf(json.get("menuApplicationId")); - String type = json.get("type") == null ? "" : json.get("type"); - Long favoriteId = dssWorkspaceService.addFavorite(username, workspaceId, menuApplicationId, type); - return Message.ok().data("favoriteId", favoriteId); - } - - - @RequestMapping(path = "/workspaces/{workspaceId}/favorites/{applicationId}", method = RequestMethod.POST) - public Message deleteFavorite(HttpServletRequest req, @PathVariable("workspaceId") Long workspaceId, @PathVariable("applicationId") Long applicationId, @RequestBody Map json) { - String username = SecurityFilter.getLoginUsername(req); - String type = json.get("type") == null ? "" : json.get("type"); - Long favoriteId = dssWorkspaceService.deleteFavorite(username, applicationId, workspaceId, type); - return Message.ok().data("favoriteId", favoriteId); - } - - - @RequestMapping(path = "/workspaces/{workspaceId}/favorites/{applicationId}", method = RequestMethod.GET) - public Message deleteFavorite1(HttpServletRequest req, @PathVariable("workspaceId") Long workspaceId, @PathVariable("applicationId") Long applicationId, @RequestParam(value = "type", required = false) String type) { - String username = SecurityFilter.getLoginUsername(req); - Long favoriteId = dssWorkspaceService.deleteFavorite(username, applicationId, workspaceId, type); - return Message.ok().data("favoriteId", favoriteId); - } - - @RequestMapping(path = "/workspaces/{workspaceId}/favorites/{favouritesId}", method = RequestMethod.DELETE) - public Message deleteFavorite2(HttpServletRequest req, @PathVariable("workspaceId") Long workspaceId, @PathVariable("favouritesId") Long favouritesId, @RequestParam(value = "type", required = false) String type) { - String username = SecurityFilter.getLoginUsername(req); - Long favoriteId = dssWorkspaceService.deleteFavorite(username, favouritesId, workspaceId, type); - return Message.ok().data("favoriteId", favoriteId); - } - - -} diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/DSSDictionaryService.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/DSSDictionaryService.java index ab71a8185d..48b646b1bd 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/DSSDictionaryService.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/DSSDictionaryService.java @@ -19,7 +19,6 @@ import com.webank.wedatasphere.dss.framework.workspace.bean.DSSDictionary; import com.webank.wedatasphere.dss.framework.workspace.bean.vo.DSSDictionaryRequestVO; -import com.webank.wedatasphere.dss.framework.workspace.bean.vo.DepartmentVO; import java.util.List; import java.util.Map; @@ -35,11 +34,4 @@ public interface DSSDictionaryService { * @return */ public Map getDicSecondList(DSSDictionaryRequestVO dictionaryRequestVO) ; - - /** - * 获取空间默认部门 - * @return - */ - public List getDefaultDepartmentVOList(); - } diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/DSSWorkspaceMenuService.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/DSSWorkspaceMenuService.java deleted file mode 100644 index dfdf49c508..0000000000 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/DSSWorkspaceMenuService.java +++ /dev/null @@ -1,29 +0,0 @@ -///* -// * Copyright 2019 WeBank -// * 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. -// * -// */ -// -//package com.webank.wedatasphere.dss.framework.workspace.service; -// -// -// -//import com.webank.wedatasphere.dss.framework.workspace.bean.DSSMenu; -// -// -//import java.util.List; -// -//public interface DSSWorkspaceMenuService { -// @Deprecated -// List getRealComponents(List subMenus, String workspaceId, String username); -//} diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/DSSWorkspaceRoleCheckService.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/DSSWorkspaceRoleCheckService.java new file mode 100644 index 0000000000..9ac67cc63f --- /dev/null +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/DSSWorkspaceRoleCheckService.java @@ -0,0 +1,17 @@ +package com.webank.wedatasphere.dss.framework.workspace.service; + +import java.util.List; + +public interface DSSWorkspaceRoleCheckService { + + /** + * 权限管理界面判断登录人是否拥有用户操作权限 + * @param workspaceId 工作空间id + * @param loginUser 当前登录用户 + * @param username 将要修改的用户 + * @param roles 将要修改的角色 + * @return + */ + boolean checkRolesOperation(int workspaceId, String loginUser, String username, List roles); + +} diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/DSSWorkspaceRoleService.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/DSSWorkspaceRoleService.java index c8de173f92..c3da20b121 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/DSSWorkspaceRoleService.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/DSSWorkspaceRoleService.java @@ -16,11 +16,13 @@ package com.webank.wedatasphere.dss.framework.workspace.service; +import com.webank.wedatasphere.dss.framework.workspace.bean.DSSWorkspaceRole; + import java.util.List; public interface DSSWorkspaceRoleService { - void addWorkspaceRole(String roleName, int workspaceId, List menuIds, List componentIds, String username); + DSSWorkspaceRole addWorkspaceRole(String roleName, int workspaceId, List menuIds, List componentIds, String username); List getRoleInWorkspace(String username, int workspaceId); diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/DSSWorkspaceService.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/DSSWorkspaceService.java index ba2c1587a3..596dae6dc2 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/DSSWorkspaceService.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/DSSWorkspaceService.java @@ -18,6 +18,7 @@ import com.webank.wedatasphere.dss.common.exception.DSSErrorException; import com.webank.wedatasphere.dss.framework.workspace.bean.DSSWorkspace; +import com.webank.wedatasphere.dss.framework.workspace.bean.DSSWorkspaceAssociateDepartments; import com.webank.wedatasphere.dss.framework.workspace.bean.dto.response.WorkspaceMenuVo; import com.webank.wedatasphere.dss.framework.workspace.bean.dto.response.WorkspaceDepartmentVo; import com.webank.wedatasphere.dss.framework.workspace.bean.dto.response.WorkspaceFavoriteVo; @@ -27,7 +28,6 @@ import com.webank.wedatasphere.dss.framework.workspace.bean.vo.DSSWorkspacePrivVO; import com.webank.wedatasphere.dss.framework.workspace.bean.vo.DSSWorkspaceRoleVO; import com.webank.wedatasphere.dss.framework.workspace.bean.vo.DSSWorkspaceUserVO; -import com.webank.wedatasphere.dss.framework.workspace.bean.vo.DepartmentVO; import com.webank.wedatasphere.dss.standard.app.sso.Workspace; import org.apache.linkis.common.exception.ErrorException; @@ -48,6 +48,10 @@ public interface DSSWorkspaceService { List getWorkspaceUsers(String workspaceId, String department, String username, String roleName, int pageNow, int pageSize, List total); + /** + * 获取空间内所有成员,已用户名返回 + */ + List getWorkspaceUsers(String workspaceId); List getWorkspaceRoles(int workspaceId); @@ -57,11 +61,28 @@ List getWorkspaceUsers(String workspaceId, String department DSSWorkspaceHomepageSettingVO getWorkspaceHomepageSettings(int workspaceId); - String getWorkspaceName(String workspaceId); + String getWorkspaceName(Long workspaceId); + /** + * 工作空间管理员权限判断(is_admin=1的用户才拥有工作空间操作权限) + * @param userName + * @return + */ boolean checkAdmin(String userName); - List getDepartments(); + /** + * 判断是否拥有该工作空间管理员权限 + * @param workspaceId + * @param username + * @return + */ + public boolean isAdminUser(Long workspaceId, String username); + + List getAllDepartmentWithOffices(); + + void associateDepartments(Long workspaceId, String departments, String roles,String user) throws DSSErrorException; + + DSSWorkspaceAssociateDepartments getAssociateDepartmentsInfo(Long workspaceId); List getWorkspaceUsersByRole(int workspaceId, String roleName, List totals, int pageNow, int pageSize); @@ -83,12 +104,9 @@ List getWorkspaceAppConns(Workspace workspace, Long workspaceId Long addFavorite(String username, Long workspaceId, Long menuApplicationId,String type); - Long deleteFavorite(String username, Long applicationId, Long workspaceId,String type); + Long deleteFavorite(String username, Long appconnId, Long workspaceId,String type); boolean checkAdminByWorkspace(String username, int workspaceId); - //是否为超级管理员 - public boolean isAdminUser(Long workspaceId, String username); - } diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/DSSWorkspaceUserService.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/DSSWorkspaceUserService.java index 30bcb89416..23d23222f3 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/DSSWorkspaceUserService.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/DSSWorkspaceUserService.java @@ -20,6 +20,7 @@ import com.webank.wedatasphere.dss.framework.workspace.bean.vo.StaffInfoVO; import java.util.List; +import java.util.Map; public interface DSSWorkspaceUserService { @@ -39,4 +40,8 @@ public interface DSSWorkspaceUserService { List getWorkspaceReleaseUsers(int workspaceId); Long getCountByUsername(String username,int workspaceId); + + List> getUserRoleByUserName(String userName); + + boolean clearUserByUserName(String userName); } diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/NoticeService.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/NoticeService.java new file mode 100644 index 0000000000..24e03763f5 --- /dev/null +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/NoticeService.java @@ -0,0 +1,11 @@ +package com.webank.wedatasphere.dss.framework.workspace.service; + +import com.webank.wedatasphere.dss.framework.workspace.bean.NoticeContent; + +import java.util.List; + +public interface NoticeService { + + List getNoticeContent(); + +} diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/impl/DSSDictionaryServiceImpl.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/impl/DSSDictionaryServiceImpl.java index 88e3980bd9..04fbe11e34 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/impl/DSSDictionaryServiceImpl.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/impl/DSSDictionaryServiceImpl.java @@ -19,11 +19,8 @@ import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; import com.webank.wedatasphere.dss.framework.workspace.bean.DSSDictionary; import com.webank.wedatasphere.dss.framework.workspace.bean.vo.DSSDictionaryRequestVO; -import com.webank.wedatasphere.dss.framework.workspace.bean.vo.DepartmentVO; import com.webank.wedatasphere.dss.framework.workspace.dao.DSSDictionaryMapper; import com.webank.wedatasphere.dss.framework.workspace.service.DSSDictionaryService; -import com.webank.wedatasphere.dss.framework.workspace.util.DSSDictionaryConstant; -import org.apache.commons.collections.CollectionUtils; import org.apache.commons.lang.StringUtils; import org.slf4j.Logger; import org.slf4j.LoggerFactory; @@ -85,36 +82,4 @@ public Map getDicSecondList(DSSDictionaryRequestVO dictionaryRequ retMap.put("mapList",mapList); return retMap; } - - - /** - * 获取空间默认部门 - * @return - */ - @Override - public List getDefaultDepartmentVOList(){ - QueryWrapper dictionaryQueryWrapper = new QueryWrapper<>(); - dictionaryQueryWrapper.eq("workspace_id", 0); - dictionaryQueryWrapper.eq("dic_key", DSSDictionaryConstant.W_WORKSPACE_DEPARTMENT); - List dictionarieList = dssDictionaryMapper.selectList(dictionaryQueryWrapper); - List retList = new ArrayList<>(); - if(CollectionUtils.isEmpty(dictionarieList)){ - return retList; - } - String dicValue = dictionarieList.get(0).getDicValue(); - if(StringUtils.isBlank(dicValue)){ - return retList; - } - String[] tempStrArr = dicValue.split(";"); - DepartmentVO departmentVO = null; - for(int i=0;i getRealComponents(List subMenus, String workspaceId, String username) { -// Set dssMenuSet = new HashSet<>(subMenus); -// subMenus.forEach(subMenu -> { -// if (subMenu.isComponent()){ -// int applicationId = subMenu.getApplicationId(); -// List roles = dssWorkspaceUserMapper.getRoleInWorkspace(Integer.parseInt(workspaceId),username); -// int priv = 0; -// for (Integer role : roles) { -// Integer rolePriv = dssWorkspaceMenuMapper.getOneCompoentRolePriv(Integer.parseInt(workspaceId), role, applicationId); -// if(rolePriv != null){ -// priv += rolePriv; -// } -// } -// if (priv == 0){ -// dssMenuSet.remove(subMenu); -// } -// } -// }); -// return new ArrayList<>(dssMenuSet); -// } -//} diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/impl/DSSWorkspacePrivServiceImpl.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/impl/DSSWorkspacePrivServiceImpl.java index 18476226d3..0e21f8a999 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/impl/DSSWorkspacePrivServiceImpl.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/impl/DSSWorkspacePrivServiceImpl.java @@ -50,15 +50,15 @@ public void updateRoleMenuPriv(int workspaceId, int menuId, String updater, List @Override @Transactional(rollbackFor = Exception.class) - public void updateRoleComponentPriv(int workspaceId, int componentId, String username, List> pairs) { + public void updateRoleComponentPriv(int workspaceId, int appconnId, String username, List> pairs) { pairs.forEach(pair -> { int roleId = pair.getKey(); int priv = pair.getValue() ? 1 : 0; - int count = dssWorkspacePrivMapper.queryCntOfRCP(workspaceId, componentId, roleId); + int count = dssWorkspacePrivMapper.queryCntOfRCP(workspaceId, appconnId, roleId); if (count >= 1) { - dssWorkspacePrivMapper.updateRoleComponentPriv(workspaceId, componentId, roleId, priv); + dssWorkspacePrivMapper.updateRoleComponentPriv(workspaceId, appconnId, roleId, priv); } else { - dssWorkspacePrivMapper.insertRolComponentPriv(workspaceId, componentId, roleId, priv, username); + dssWorkspacePrivMapper.insertRolComponentPriv(workspaceId, appconnId, roleId, priv, username); } }); } diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/impl/DSSWorkspaceRoleCheckServiceImpl.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/impl/DSSWorkspaceRoleCheckServiceImpl.java new file mode 100644 index 0000000000..415718ad6d --- /dev/null +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/impl/DSSWorkspaceRoleCheckServiceImpl.java @@ -0,0 +1,28 @@ +package com.webank.wedatasphere.dss.framework.workspace.service.impl; + +import com.webank.wedatasphere.dss.framework.workspace.dao.DSSWorkspaceMapper; +import com.webank.wedatasphere.dss.framework.workspace.service.DSSWorkspaceRoleCheckService; +import com.webank.wedatasphere.dss.framework.workspace.service.DSSWorkspaceService; +import org.apache.commons.lang.StringUtils; +import org.springframework.beans.factory.annotation.Autowired; + +import java.util.List; + +public class DSSWorkspaceRoleCheckServiceImpl implements DSSWorkspaceRoleCheckService { + + @Autowired + private DSSWorkspaceMapper dssWorkspaceMapper; + + @Autowired + DSSWorkspaceService dssWorkspaceService; + + @Override + public boolean checkRolesOperation(int workspaceId, String loginUser, String username, List roles) { + // 获取工作空间创建者 + String createBy = dssWorkspaceMapper.getWorkspace(workspaceId).getCreateBy(); + return StringUtils.equals(loginUser, createBy) ? + (!StringUtils.equals(createBy, username) || roles.contains(1)) : + (dssWorkspaceService.isAdminUser((long) workspaceId, loginUser) && ((!dssWorkspaceService.isAdminUser((long) workspaceId, username) && !roles.contains(1)) + || (StringUtils.equals(loginUser, username) && roles.contains(1)))); + } +} diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/impl/DSSWorkspaceRoleServiceImpl.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/impl/DSSWorkspaceRoleServiceImpl.java index 80ce4dea8c..6e464ad764 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/impl/DSSWorkspaceRoleServiceImpl.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/impl/DSSWorkspaceRoleServiceImpl.java @@ -46,7 +46,7 @@ public class DSSWorkspaceRoleServiceImpl implements DSSWorkspaceRoleService { @Override @Transactional(rollbackFor = Exception.class) - public void addWorkspaceRole(String roleName, int workspaceId, List menuIds, List componentIds, String username) { + public DSSWorkspaceRole addWorkspaceRole(String roleName, int workspaceId, List menuIds, List componentIds, String username) { DSSWorkspaceRole dssRole = new DSSWorkspaceRole(); dssRole.setWorkspaceId(workspaceId); dssRole.setFrontName(roleName); @@ -71,6 +71,7 @@ public void addWorkspaceRole(String roleName, int workspaceId, List men dssWorkspaceRoleMapper.updateRoleComponent(dssRole.getId(), workspaceId, allComponentIds, username, 0); } workspaceDBHelper.retrieveFromDB(); + return dssRole; } @Override @@ -95,7 +96,10 @@ public Integer getWorkspaceIdByUser(String username) { @Override public int getApiPriv(String username, Integer workspaceId, String roleName, String appName) { - int roleId = dssWorkspaceRoleMapper.getRoleId(roleName, -1); + Integer roleId = dssWorkspaceRoleMapper.getRoleId(roleName, -1); + if (roleId == null) { + return -1; + } DSSApplicationBean applicationBean = workspaceDBHelper.getAppConn(appName); if (applicationBean == null) { return -1; diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/impl/DSSWorkspaceServiceImpl.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/impl/DSSWorkspaceServiceImpl.java index 31a441d5b4..113be06783 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/impl/DSSWorkspaceServiceImpl.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/impl/DSSWorkspaceServiceImpl.java @@ -51,6 +51,7 @@ import org.apache.commons.lang.ArrayUtils; import org.apache.commons.lang.StringUtils; import org.apache.linkis.common.exception.ErrorException; +import org.apache.linkis.protocol.util.ImmutablePair; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.beans.factory.annotation.Autowired; @@ -62,7 +63,6 @@ import static com.webank.wedatasphere.dss.framework.workspace.util.DSSWorkspaceConstant.DEFAULT_DEMO_WORKSPACE_NAME; -//@Service public class DSSWorkspaceServiceImpl implements DSSWorkspaceService { private static final Logger LOGGER = LoggerFactory.getLogger(DSSWorkspaceServiceImpl.class); @@ -73,8 +73,6 @@ public class DSSWorkspaceServiceImpl implements DSSWorkspaceService { @Autowired private DSSWorkspaceMenuMapper dssWorkspaceMenuMapper; @Autowired - private DSSWorkspaceInfoMapper dssWorkspaceInfoMapper; - @Autowired private WorkspaceDBHelper workspaceDBHelper; @Autowired private DSSWorkspaceService dssWorkspaceService; @@ -122,7 +120,7 @@ public int createWorkspace(String workspaceName, String tags, String userName, } Long userId = dssWorkspaceUserMapper.getUserID(userName); dssWorkspaceUserMapper.setUserRoleInWorkspace(dssWorkspace.getId(), - workspaceDBHelper.getRoleIdByName(CommonRoleEnum.ADMIN.getName()), userName, userName, userId); + workspaceDBHelper.getRoleIdByName(CommonRoleEnum.ADMIN.getName()), userName, userName, userId, userName); dssMenuRoleMapper.insertBatch(workspaceDBHelper.generateDefaultWorkspaceMenuRole(dssWorkspace.getId(), userName)); dssWorkspaceHomepageMapper.insertBatch(workspaceDBHelper.generateDefaultWorkspaceHomepage(dssWorkspace.getId(), userName)); dssComponentRoleMapper.insertBatch(workspaceDBHelper.generateDefaultWorkspaceComponentPrivs(dssWorkspace.getId(), userName)); @@ -151,7 +149,7 @@ public void addWorkspaceUser(List roleIds, Workspace workspace, String } //保存 - 保存用户角色关系 dss_workspace_user_role for (Integer roleId : roleIds) { - dssWorkspaceUserMapper.setUserRoleInWorkspace((int) workspace.getWorkspaceId(), roleId, userName, creator, userId == null ? null : Long.parseLong(userId)); + dssWorkspaceUserMapper.insertUserRoleInWorkspace((int) workspace.getWorkspaceId(), roleId, new Date(), userName, creator, userId == null ? null : Long.parseLong(userId), creator); } } @@ -177,32 +175,29 @@ public List getWorkspaces(String userName) { @Override public DSSWorkspaceHomePageVO getWorkspaceHomePage(String userName, String moduleName) throws DSSErrorException { - //根据用户名 拿到用户ID - //根据用户id 和工作空间id 拿到 角色id - //根据role id 和工作空间id 拿到 重定向的 url - List tempWorkspaceIds = dssWorkspaceUserMapper.getWorkspaceIds(userName); -// if (tempWorkspaceIds == null || tempWorkspaceIds.isEmpty()) { -// throw new DSSErrorException(30020, "该账号尚未加入工作空间,请联系管理员分配工作空间及用户角色"); -// } - List workspaceIds = new ArrayList<>(); - tempWorkspaceIds.stream(). - map(dssWorkspaceInfoMapper::getWorkspaceNameById). - filter(name -> !DEFAULT_DEMO_WORKSPACE_NAME.getValue().equals(name)). - map(dssWorkspaceInfoMapper::getWorkspaceIdByName). - forEach(workspaceIds::add); + List dssWorkspaces = dssWorkspaceMapper.getWorkspaces(userName); + List workspaceIds = dssWorkspaces.stream().filter(l -> !DEFAULT_DEMO_WORKSPACE_NAME.getValue().equals(l.getName())) + .map(DSSWorkspace::getId).collect(Collectors.toList()); DSSWorkspaceHomePageVO dssWorkspaceHomePageVO = new DSSWorkspaceHomePageVO(); if (workspaceIds.size() == 0) { Long userId = dssWorkspaceUserMapper.getUserID(userName); - int workspaceId = dssWorkspaceInfoMapper.getWorkspaceIdByName(DSSWorkspaceConstant.DEFAULT_WORKSPACE_NAME.getValue()); + int workspaceId = dssWorkspaceMapper.getWorkspaceIdByName(DSSWorkspaceConstant.DEFAULT_WORKSPACE_NAME.getValue()); dssWorkspaceUserMapper.setUserRoleInWorkspace(workspaceId, workspaceDBHelper.getRoleIdByName(CommonRoleEnum.ANALYSER.getName()), - userName, "system", userId); - String homepageUrl = dssWorkspaceUserMapper.getHomepageUrl(workspaceId, workspaceDBHelper.getRoleIdByName(CommonRoleEnum.ANALYSER.getName())); + userName, "system", userId, "system"); + Integer workspace0xId = dssWorkspaceMapper.getWorkspaceIdByName(DSSWorkspaceConstant.DEFAULT_0XWORKSPACE_NAME.getValue()); + if (workspace0xId != null) { + dssWorkspaceUserMapper.setUserRoleInWorkspace(workspace0xId, workspaceDBHelper.getRoleIdByName(CommonRoleEnum.ANALYSER.getName()), + userName, "system", userId, "system"); + } + //todo 初始化做的各项事情改为listener模式 + //为新用户自动加入部门关联的工作空间 + joinWorkspaceForNewUser(userName, userId); + + //若路径没有workspaceId会出现页面没有首页、管理台 + String homepageUrl = "/home" + "?workspaceId=" + workspaceId; if (ApplicationConf.HOMEPAGE_MODULE_NAME.getValue().equalsIgnoreCase(moduleName)) { homepageUrl = ApplicationConf.HOMEPAGE_URL.getValue() + workspaceIds.get(0); } - if (StringUtils.isEmpty(homepageUrl)) { - homepageUrl = "/home" + "?workspaceId=" + workspaceId; - } dssWorkspaceHomePageVO.setHomePageUrl(homepageUrl); dssWorkspaceHomePageVO.setWorkspaceId(workspaceId); dssWorkspaceHomePageVO.setRoleName(CommonRoleEnum.ANALYSER.getName()); @@ -226,7 +221,6 @@ public DSSWorkspaceHomePageVO getWorkspaceHomePage(String userName, String modul dssWorkspaceHomePageVO.setWorkspaceId(workspaceIds.get(0)); dssWorkspaceHomePageVO.setRoleName(workspaceDBHelper.getRoleNameById(minRoleId)); } else { - //排除掉默认的默认工作空间bdapWorkspace String homepageUrl = "/workspaceHome?workspaceId=" + workspaceIds.get(0); if (ApplicationConf.HOMEPAGE_MODULE_NAME.getValue().equalsIgnoreCase(moduleName)) { homepageUrl = ApplicationConf.HOMEPAGE_URL.getValue() + workspaceIds.get(0); @@ -240,25 +234,22 @@ public DSSWorkspaceHomePageVO getWorkspaceHomePage(String userName, String modul @Override public List getWorkspaceUsers(String workspaceId, String department, String username, String roleName, int pageNow, int pageSize, List total) { - int roleId = -1; - if (StringUtils.isNotEmpty(roleName)) { - roleId = workspaceDBHelper.getRoleIdByName(roleName); - } + String roleId = StringUtils.isBlank(roleName) ? null : String.valueOf(workspaceDBHelper.getRoleIdByName(roleName)); PageHelper.startPage(pageNow, pageSize); - List workspaceUsers = new ArrayList<>(); - try { - workspaceUsers = dssWorkspaceUserMapper.getWorkspaceUsers(workspaceId, username); - } finally { - //PageHelper.clearPage(); - } + List workspaceUsers = dssWorkspaceUserMapper.getWorkspaceUsers(workspaceId, username, roleId); PageInfo pageInfo = new PageInfo<>(workspaceUsers); total.add(pageInfo.getTotal()); - List dssWorkspaceUserVOs = new ArrayList<>(); - for (DSSWorkspaceUser workspaceUser : workspaceUsers) { - List roles = dssWorkspaceUserMapper.getRoleInWorkspace(Integer.parseInt(workspaceId), workspaceUser.getUsername()); - dssWorkspaceUserVOs.add(changeToUserVO(workspaceUser, roles)); - } - return dssWorkspaceUserVOs; + return workspaceUsers.stream().map( + workspaceUser -> changeToUserVO(workspaceUser, + Arrays.stream(workspaceUser.getRoleIds().split(",")).map(Integer::valueOf).collect(Collectors.toList()))) + .collect(Collectors.toList()); + } + + @Override + public List getWorkspaceUsers(String workspaceId) { + return + dssWorkspaceUserMapper.getWorkspaceUsers(workspaceId, null, null).stream() + .map(DSSWorkspaceUser::getUsername).collect(Collectors.toList()); } private DSSWorkspaceUserVO changeToUserVO(DSSWorkspaceUser dssWorkspaceUser, List roles) { @@ -281,6 +272,8 @@ private DSSWorkspaceUserVO changeToUserVO(DSSWorkspaceUser dssWorkspaceUser, Lis vo.setRoles(roles); vo.setCreator(dssWorkspaceUser.getCreator()); vo.setJoinTime(dssWorkspaceUser.getJoinTime()); + vo.setUpdateTime(dssWorkspaceUser.getUpdateTime()); + vo.setUpdateUser(dssWorkspaceUser.getUpdateUser()); return vo; } @@ -401,7 +394,7 @@ public DSSWorkspacePrivVO getWorkspaceMenuPrivs(String workspaceId) { @Override public DSSWorkspaceOverviewVO getOverview(String username, int workspaceId, boolean isEnglish) { DSSWorkspaceOverviewVO dssWorkspaceOverviewVO = new DSSWorkspaceOverviewVO(); - DSSWorkspace dssWorkspace = dssWorkspaceInfoMapper.getWorkspace(workspaceId); + DSSWorkspace dssWorkspace = dssWorkspaceMapper.getWorkspace(workspaceId); dssWorkspaceOverviewVO.setTitle(dssWorkspace.getName()); dssWorkspaceOverviewVO.setDescription(dssWorkspace.getDescription()); return dssWorkspaceOverviewVO; @@ -427,8 +420,8 @@ public DSSWorkspaceHomepageSettingVO getWorkspaceHomepageSettings(int workspaceI } @Override - public String getWorkspaceName(String workspaceId) { - return dssWorkspaceInfoMapper.getWorkspaceNameById(Integer.parseInt(workspaceId)); + public String getWorkspaceName(Long workspaceId) { + return dssWorkspaceMapper.getWorkspaceNameById(workspaceId); } @Override @@ -443,19 +436,28 @@ public boolean checkAdminByWorkspace(String username, int workspaceId) { } @Override - public List getDepartments() { + public List getAllDepartmentWithOffices() { List allDepartments = staffInfoGetter.getAllDepartments(); - List departmentVOs = new ArrayList<>(); - int count = 1; - for (String department : allDepartments) { - DepartmentVO departmentVO = new DepartmentVO(); - departmentVO.setFrontName(department); - departmentVO.setName(department); - departmentVO.setId(count); - departmentVOs.add(departmentVO); - count++; + return allDepartments; + } + + @Override + public void associateDepartments(Long workspaceId, String departments, String roles, String user) throws DSSErrorException { + List userRoles = dssWorkspaceUserMapper.getRoleInWorkspace(workspaceId.intValue(), user); + //管理员鉴权 + if (userRoles.stream().noneMatch(l -> l == workspaceDBHelper.getRoleIdByName(CommonRoleEnum.ADMIN.getName()))) { + throw new DSSErrorException(80000, "无权限操作"); } - return departmentVOs; + if (dssWorkspaceMapper.getAssociateDepartmentsByWorkspaceId(workspaceId) != null) { + dssWorkspaceMapper.updateDepartmentsForWorkspace(workspaceId, departments, roles, user); + } else { + dssWorkspaceMapper.addDepartmentsForWorkspace(workspaceId, departments, roles, user); + } + } + + @Override + public DSSWorkspaceAssociateDepartments getAssociateDepartmentsInfo(Long workspaceId) { + return dssWorkspaceMapper.getAssociateDepartmentsByWorkspaceId(workspaceId); } @Override @@ -481,11 +483,11 @@ public List getWorkspaceUsersByRole(int workspaceId, String private DSSWorkspace getWorkspace(Supplier workspaceSupplier, String username) throws DSSErrorException { DSSWorkspace dssWorkSpace = workspaceSupplier.get(); if (dssWorkSpace == null) { - throw new DSSErrorException(30022, "workspace is not exists."); + throw new DSSFrameworkWarnException(30022, "workspace is not exists."); } List users = dssWorkspaceUserMapper.getAllWorkspaceUsers(dssWorkSpace.getId()); if (!users.contains(username)) { - throw new DSSErrorException(30021, "You have no permission to access this workspace " + dssWorkSpace.getName()); + throw new DSSFrameworkWarnException(30021, "You have no permission to access this workspace " + dssWorkSpace.getName()); } String originDepartId = dssWorkSpace.getDepartment(); if (StringUtils.isNotBlank(originDepartId)) { @@ -560,7 +562,7 @@ private List getMenuAppInstances(List menuVos, AppConn appConn = AppConnManager.getAppConnManager().getAppConn(menuAppconn.getName()); List instanceList = new ArrayList<>(); SSOUrlBuilderOperation operation; - if(appConn instanceof OnlySSOAppConn) { + if (appConn instanceof OnlySSOAppConn) { operation = ((OnlySSOAppConn) appConn).getOrCreateSSOStandard().getSSOBuilderService().createSSOUrlBuilderOperation(); SSOHelper.setSSOUrlBuilderOperation(operation, workspace); operation.setAppName(appConn.getAppDesc().getAppName()); @@ -636,9 +638,9 @@ public Long addFavorite(String username, Long workspaceId, Long menuApplicationI } @Override - public Long deleteFavorite(String username, Long applicationId, Long workspaceId, String type) { - workspaceMapper.deleteFavorite(username, applicationId, workspaceId, type); - return applicationId; + public Long deleteFavorite(String username, Long appconnId, Long workspaceId, String type) { + workspaceMapper.deleteFavorite(username, appconnId, workspaceId, type); + return appconnId; } @@ -665,4 +667,25 @@ public boolean isAdminUser(Long workspaceId, String username) { org.apache.commons.lang3.ArrayUtils.contains(AdminConf.SUPER_ADMIN_LIST, username)) || username.equals(workspace.getCreateBy()); } + + private void joinWorkspaceForNewUser(String userName, Long userId) { + String userOrgName = staffInfoGetter.getFullOrgNameByUsername(userName); + List workspaceAssociateDepartments = dssWorkspaceMapper.getWorkspaceAssociateDepartments(); + Set> needToAdd = new HashSet<>(); + for (DSSWorkspaceAssociateDepartments item : workspaceAssociateDepartments) { + String departments = item.getDepartments(); + if (StringUtils.isNotBlank(departments) && StringUtils.isNotBlank(item.getRoleIds())) { + Arrays.stream(departments.split(",")).forEach(org -> { + if (org.equals(userOrgName)) { + needToAdd.add(new ImmutablePair<>(item.getWorkspaceId(), item.getRoleIds())); + } + }); + } + } + needToAdd.forEach(pair -> { + Arrays.stream(pair.getValue().split(",")).forEach(roleId -> { + dssWorkspaceUserMapper.setUserRoleInWorkspace(pair.getKey().intValue(), Integer.parseInt(roleId), userName, "system", userId, "system"); + }); + }); + } } diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/impl/DSSWorkspaceUserServiceImpl.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/impl/DSSWorkspaceUserServiceImpl.java index fdbf934008..7a8f04d716 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/impl/DSSWorkspaceUserServiceImpl.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/impl/DSSWorkspaceUserServiceImpl.java @@ -16,11 +16,13 @@ package com.webank.wedatasphere.dss.framework.workspace.service.impl; +import com.webank.wedatasphere.dss.framework.workspace.bean.DSSWorkspaceUser; import com.webank.wedatasphere.dss.framework.workspace.bean.StaffInfo; import com.webank.wedatasphere.dss.framework.workspace.bean.vo.StaffInfoVO; import com.webank.wedatasphere.dss.framework.workspace.dao.DSSWorkspaceUserMapper; import com.webank.wedatasphere.dss.framework.workspace.service.DSSWorkspaceUserService; import com.webank.wedatasphere.dss.framework.workspace.service.StaffInfoGetter; +import com.webank.wedatasphere.dss.framework.workspace.util.WorkspaceDBHelper; import com.webank.wedatasphere.dss.framework.workspace.util.WorkspaceServerConstant; import org.apache.commons.lang.StringUtils; import org.slf4j.Logger; @@ -29,7 +31,7 @@ import org.springframework.stereotype.Service; import org.springframework.transaction.annotation.Transactional; -import java.util.List; +import java.util.*; import java.util.stream.Collectors; @@ -43,12 +45,17 @@ public class DSSWorkspaceUserServiceImpl implements DSSWorkspaceUserService { @Autowired private StaffInfoGetter staffInfoGetter; + @Autowired + private WorkspaceDBHelper workspaceDBHelper; + @Override @Transactional(rollbackFor = Throwable.class) public void updateWorkspaceUser(List roles, int workspaceId, String userName, String creator) { + //获取用户创建时间 + DSSWorkspaceUser workspaceUsers = dssWorkspaceUserMapper.getWorkspaceUsers(String.valueOf(workspaceId), userName, null).stream().findFirst().get(); dssWorkspaceUserMapper.removeAllRolesForUser(userName, workspaceId); roles.forEach(role ->{ - dssWorkspaceUserMapper.setUserRoleInWorkspace(workspaceId, role, userName, creator, 0L); + dssWorkspaceUserMapper.insertUserRoleInWorkspace(workspaceId, role, workspaceUsers.getJoinTime(), userName, workspaceUsers.getCreator(), 0L, creator); }); } @@ -110,4 +117,31 @@ public List getWorkspaceReleaseUsers(int workspaceId) { public Long getCountByUsername(String username,int workspaceId){ return dssWorkspaceUserMapper.getCountByUsername(username,workspaceId); } + + @Override + public List> getUserRoleByUserName(String userName) { + List workspaceRoles = dssWorkspaceUserMapper.getWorkspaceRoleByUsername(userName); + List> list = new ArrayList<>(); + workspaceRoles.forEach(workspaceRole -> { + Map map = new HashMap<>(); + map.put("workspaceId", workspaceRole.getWorkspaceId()); + map.put("roleId", workspaceRole.getRoleIds()); + map.put("roleName", workspaceDBHelper.getRoleFrontName(Integer.parseInt(workspaceRole.getRoleIds()))); + list.add(map); + }); + return list; + } + + @Override + @Transactional(rollbackFor = Exception.class) + public boolean clearUserByUserName(String userName) { + if(staffInfoGetter.getAllUsers().stream().anyMatch(staffInfo -> staffInfo.getEnglishName().equals(userName))) { + dssWorkspaceUserMapper.deleteUserByUserName(userName); + dssWorkspaceUserMapper.deleteUserRolesByUserName(userName); + dssWorkspaceUserMapper.deleteProxyUserByUserName(userName); + return true; + }else{ + return false; + } + } } diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/impl/DefaultStaffInfoGetter.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/impl/DefaultStaffInfoGetter.java index da752f3996..6f2cd2d33f 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/impl/DefaultStaffInfoGetter.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/impl/DefaultStaffInfoGetter.java @@ -41,7 +41,7 @@ public String getFullOrgNameByUsername(String username) { @Override public List getAllDepartments() { - List allDepartments = Arrays.asList("WeDataSphere","DataSP", "linkis"); + List allDepartments = Arrays.asList("WeDataSphere-linkisGroup"); return allDepartments; } } diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/impl/NoticeServiceImpl.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/impl/NoticeServiceImpl.java new file mode 100644 index 0000000000..9262a453dc --- /dev/null +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/service/impl/NoticeServiceImpl.java @@ -0,0 +1,27 @@ +package com.webank.wedatasphere.dss.framework.workspace.service.impl; + +import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; +import com.webank.wedatasphere.dss.framework.workspace.bean.NoticeContent; +import com.webank.wedatasphere.dss.framework.workspace.dao.NoticeMapper; +import com.webank.wedatasphere.dss.framework.workspace.service.NoticeService; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Service; + +import java.util.Date; +import java.util.List; + +@Service +public class NoticeServiceImpl implements NoticeService { + + @Autowired + private NoticeMapper noticeMapper; + + @Override + public List getNoticeContent() { + QueryWrapper queryWrapper = new QueryWrapper<>(); + Date now = new Date(); + queryWrapper.gt("end_time", now); + queryWrapper.orderByDesc("id"); + return noticeMapper.selectList(queryWrapper); + } +} diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/util/DSSWorkspaceConstant.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/util/DSSWorkspaceConstant.java index fd1e269afe..62f81053d9 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/util/DSSWorkspaceConstant.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/util/DSSWorkspaceConstant.java @@ -25,6 +25,9 @@ public interface DSSWorkspaceConstant { CommonVars DEFAULT_WORKSPACE_NAME = CommonVars.apply("wds.dss.workspace.default.name", "bdapWorkspace"); + CommonVars DEFAULT_0XWORKSPACE_NAME + = CommonVars.apply("wds.dss.workspace.0x.default.name", "bdapWorkspace_0X"); + CommonVars DEFAULT_DEMO_WORKSPACE_NAME = CommonVars.apply("wds.dss.default.demo.workspace", "WDS_DSS_DEMO"); String WORKSPACE_MANAGEMENT_NAME = CommonVars.apply("wds.dss.workspace.management.name", "工作空间管理").getValue(); diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/util/RestfulUtils.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/util/RestfulUtils.java deleted file mode 100644 index d5c97c15c1..0000000000 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/util/RestfulUtils.java +++ /dev/null @@ -1,45 +0,0 @@ - /* - * - * * Copyright 2019 WeBank - * * - * * 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. - * - */ - - package com.webank.wedatasphere.dss.framework.workspace.util; - - @Deprecated - public class RestfulUtils { - -// public static Response dealError(String reason){ -// Message message = Message.error(reason); -// -// return Message.response(message); -// } -// -// public static Response dealOk(String msg){ -// Message message = Message.ok(msg); -// return Message.messageToResponse(message); -// } -// -// -// -// @SafeVarargs -// public static Response dealOk(String msg, Pair... data){ -// Message message = Message.ok(msg); -// Arrays.stream(data).forEach(p -> message.data(p.getKey(), p.getValue())); -// return Message.messageToResponse(message); -// } - - - } diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/util/RestulHelper.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/util/RestulHelper.java deleted file mode 100644 index 286e3f9a6f..0000000000 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/util/RestulHelper.java +++ /dev/null @@ -1,45 +0,0 @@ -/* - * Copyright 2019 WeBank - * 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. - * - */ - -package com.webank.wedatasphere.dss.framework.workspace.util; - - -import com.webank.wedatasphere.dss.framework.workspace.exception.DSSWorkspaceLoginFailException; -import org.apache.linkis.common.exception.ErrorException; -import org.apache.linkis.server.security.SecurityFilter; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.stereotype.Component; - -import javax.servlet.http.HttpServletRequest; - -@Deprecated -@Component -public class RestulHelper { - - @Autowired - private WorkspaceDBHelper workspaceDBHelper; - - public String getLoginUser(HttpServletRequest request)throws ErrorException { - try{ - return SecurityFilter.getLoginUsername(request); - }catch(Exception e){ - throw new DSSWorkspaceLoginFailException(80013, "You are not logged in"); - } - } - - - -} diff --git a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/util/ApplicationUtils.java b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/util/WorkspaceUtils.java similarity index 71% rename from dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/util/ApplicationUtils.java rename to dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/util/WorkspaceUtils.java index 2ab47b5495..7b8184c7ba 100644 --- a/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/util/ApplicationUtils.java +++ b/dss-framework/dss-framework-workspace-server/src/main/java/com/webank/wedatasphere/dss/framework/workspace/util/WorkspaceUtils.java @@ -17,16 +17,19 @@ package com.webank.wedatasphere.dss.framework.workspace.util; import com.webank.wedatasphere.dss.common.exception.DSSErrorException; +import com.webank.wedatasphere.dss.standard.app.sso.Workspace; +import com.webank.wedatasphere.dss.standard.sso.utils.SSOHelper; import org.slf4j.Logger; import org.slf4j.LoggerFactory; +import javax.servlet.http.HttpServletRequest; import java.io.UnsupportedEncodingException; import java.net.URLEncoder; -public class ApplicationUtils { +public class WorkspaceUtils { - private static final Logger LOGGER = LoggerFactory.getLogger(ApplicationUtils.class); + private static final Logger LOGGER = LoggerFactory.getLogger(WorkspaceUtils.class); private static final String REDIRECT_FORMAT = "%s?redirect=%s&dssurl=${dssurl}&cookies=${cookies}"; @@ -47,4 +50,11 @@ public static void main(String[] args) throws DSSErrorException { System.out.println(redirectUrlFormat("http://127.0..0.1:8090/qualitis/api/v1/redirect","http://127.0..0.1:8090/#/projects/list?id={projectId}&flow=true")); } + public static void validateWorkspace(long workspaceId, HttpServletRequest httpServletRequest) throws DSSErrorException { + Workspace workspace = SSOHelper.getWorkspace(httpServletRequest); + if (workspace.getWorkspaceId() != workspaceId) { + throw new DSSErrorException(80001, "请求参数的workspaceId和cookie中的workspaceId不一致,请切换至正确的workspace再操作。"); + } + } + } diff --git a/dss-framework/framework-plugins/dss-framework-migrate-server/pom.xml b/dss-framework/framework-plugins/dss-framework-migrate-server/pom.xml index 365c610aa2..23c4100d44 100644 --- a/dss-framework/framework-plugins/dss-framework-migrate-server/pom.xml +++ b/dss-framework/framework-plugins/dss-framework-migrate-server/pom.xml @@ -5,13 +5,20 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 + ../../../pom.xml 4.0.0 dss-framework-migrate-server + + com.webank.wedatasphere.dss + dss-sender-service + ${dss.version} + provided + com.webank.wedatasphere.dss dss-framework-project-server diff --git a/dss-framework/framework-plugins/dss-framework-migrate-server/src/main/java/com/webank/wedatasphere/dss/migrate/conf/MigrateConf.java b/dss-framework/framework-plugins/dss-framework-migrate-server/src/main/java/com/webank/wedatasphere/dss/migrate/conf/MigrateConf.java deleted file mode 100644 index 2a46b8f419..0000000000 --- a/dss-framework/framework-plugins/dss-framework-migrate-server/src/main/java/com/webank/wedatasphere/dss/migrate/conf/MigrateConf.java +++ /dev/null @@ -1,12 +0,0 @@ -package com.webank.wedatasphere.dss.migrate.conf; - -import org.apache.linkis.common.conf.CommonVars; - -public class MigrateConf { - - public static final String ORC_SERVER_NAME = - CommonVars.apply("wds.dss.framework.migrate.orc.name", "dss-framework-orchestrator-server-dev").getValue(); - - public static final String WORKFLOW_SERVER_NAME = - CommonVars.apply("wds.dss.workflow.server.name", "dss-workflow-server-dev").getValue(); -} diff --git a/dss-framework/framework-plugins/dss-framework-migrate-server/src/main/java/com/webank/wedatasphere/dss/migrate/restful/DSSMigrateRestful.java b/dss-framework/framework-plugins/dss-framework-migrate-server/src/main/java/com/webank/wedatasphere/dss/migrate/restful/DSSMigrateRestful.java index 3531c55bf7..4cdfd1c636 100644 --- a/dss-framework/framework-plugins/dss-framework-migrate-server/src/main/java/com/webank/wedatasphere/dss/migrate/restful/DSSMigrateRestful.java +++ b/dss-framework/framework-plugins/dss-framework-migrate-server/src/main/java/com/webank/wedatasphere/dss/migrate/restful/DSSMigrateRestful.java @@ -1,6 +1,7 @@ package com.webank.wedatasphere.dss.migrate.restful; import com.google.common.collect.Lists; +import com.webank.wedatasphere.dss.common.entity.BmlResource; import com.webank.wedatasphere.dss.common.entity.IOType; import com.webank.wedatasphere.dss.common.entity.Resource; import com.webank.wedatasphere.dss.common.exception.DSSErrorException; @@ -14,12 +15,11 @@ import com.webank.wedatasphere.dss.framework.project.entity.request.ProjectCreateRequest; import com.webank.wedatasphere.dss.framework.project.entity.vo.DSSProjectVo; import com.webank.wedatasphere.dss.framework.project.exception.DSSProjectErrorException; -import com.webank.wedatasphere.dss.framework.project.server.service.BMLService; +import com.webank.wedatasphere.dss.common.service.BMLService; import com.webank.wedatasphere.dss.framework.project.service.DSSFrameworkProjectService; import com.webank.wedatasphere.dss.framework.project.service.DSSProjectService; import com.webank.wedatasphere.dss.framework.workspace.bean.DSSWorkspace; import com.webank.wedatasphere.dss.framework.workspace.service.DSSWorkspaceService; -import com.webank.wedatasphere.dss.migrate.conf.MigrateConf; import com.webank.wedatasphere.dss.migrate.exception.MigrateErrorException; import com.webank.wedatasphere.dss.migrate.service.MetaService; import com.webank.wedatasphere.dss.migrate.service.MigrateService; @@ -27,6 +27,7 @@ import com.webank.wedatasphere.dss.orchestrator.common.entity.DSSOrchestratorVersion; import com.webank.wedatasphere.dss.orchestrator.common.entity.OrchestratorVo; import com.webank.wedatasphere.dss.orchestrator.common.protocol.*; +import com.webank.wedatasphere.dss.sender.service.DSSSenderServiceFactory; import com.webank.wedatasphere.dss.standard.app.sso.Workspace; import com.webank.wedatasphere.dss.standard.app.sso.builder.SSOUrlBuilderOperation; import com.webank.wedatasphere.dss.standard.app.sso.builder.impl.SSOUrlBuilderOperationImpl; @@ -49,6 +50,7 @@ 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.web.bind.annotation.*; import org.springframework.web.multipart.MultipartFile; @@ -56,6 +58,7 @@ import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.*; +import java.nio.charset.StandardCharsets; import java.util.*; @RequestMapping(path = "/dss/framework/release", produces = {"application/json"}) @@ -81,13 +84,13 @@ public class DSSMigrateRestful { @Autowired private DSSFrameworkProjectService dssFrameworkProjectService; @Autowired + @Qualifier("projectBmlService") BMLService bmlService; @Autowired private MetaService metaService; - // todo only dev开发中心 - private Sender orchestratorSender = Sender.getSender(MigrateConf.ORC_SERVER_NAME); - private Sender workflowSender = Sender.getSender(MigrateConf.WORKFLOW_SERVER_NAME); + private Sender orchestratorSender = DSSSenderServiceFactory.getOrCreateServiceInstance().getOrcSender(); + private Sender workflowSender = DSSSenderServiceFactory.getOrCreateServiceInstance().getWorkflowSender(); @PostMapping("/importOldDSSProject") @@ -102,14 +105,13 @@ public Message importOldDSSProject(HttpServletRequest req, //4.上传到bml //5.通过resourceId 和 version 导入到 dev的 orchestrator-server String userName = SecurityFilter.getLoginUsername(req); -// List files = form.getFields("file"); if (files == null || files.size() <= 0) { LOG.error("files are null, can not continue"); return Message.error("no files to import"); } //只取第一个文件 MultipartFile p = files.get(0); - String fileName = new String(p.getOriginalFilename().getBytes("ISO8859-1"), "UTF-8"); + String fileName = new String(Objects.requireNonNull(p.getOriginalFilename()).getBytes("ISO8859-1"), StandardCharsets.UTF_8); InputStream is = null; OutputStream os = null; try { @@ -121,8 +123,6 @@ public Message importOldDSSProject(HttpServletRequest req, is = p.getInputStream(); os = IoUtils.generateExportOutputStream(inputPath); IOUtils.copy(is, os); -// Workspace workspace = new Workspace(); - Cookie[] cookies = req.getCookies(); Workspace workspace = getWorkspace(req); migrateService.migrate(userName, inputPath, workspace); } catch (Exception e) { @@ -183,7 +183,6 @@ public static Workspace getWorkspaceForOldVersion(HttpServletRequest request) { } ssoUrlBuilderOperation.setDSSUrl(gateWayUrl); ssoUrlBuilderOperation.setWorkspace(workspace.getWorkspaceName()); -// workspace.setSSOUrlBuilderOperation(ssoUrlBuilderOperation); return workspace; } @@ -209,17 +208,13 @@ public Message importWorkFlow(HttpServletRequest req, @RequestParam(value = "dssLabels", required = false) String dssLabels, @RequestParam(name = "file") List files) throws Exception { String userName = SecurityFilter.getLoginUsername(req); -// List files = form.getFields("file"); if (files == null || files.size() <= 0) { LOG.error("files are null, can not continue"); return Message.error("no files to import"); } //只取第一个文件 -// FormDataBodyPart p = files.get(0); MultipartFile p = files.get(0); -// FormDataContentDisposition fileDetail = p.getFormDataContentDisposition(); -// String fileName = new String(fileDetail.getFileName().getBytes("ISO8859-1"), "UTF-8"); - String fileName = new String(p.getOriginalFilename().getBytes("ISO8859-1"), "UTF-8"); + String fileName = new String(Objects.requireNonNull(p.getOriginalFilename()).getBytes("ISO8859-1"), StandardCharsets.UTF_8); InputStream is = null; OutputStream os = null; Message responseMsg = Message.ok(); @@ -269,7 +264,7 @@ public Message importWorkFlow(HttpServletRequest req, // mkdir(orcPath); orchestratorInfo.setProjectId(projectVo.getId()); orchestratorInfo.setName(flowName); - String oldUUID = migrateService.queryOrcUUIDByName(new Long(dssWorkspace.getId()), projectVo.getId(), flowName); + String oldUUID = migrateService.queryOrcUUIDByName((long) dssWorkspace.getId(), projectVo.getId(), flowName); if (null != oldUUID) { orchestratorInfo.setUUID(oldUUID); } else { @@ -286,9 +281,9 @@ public Message importWorkFlow(HttpServletRequest req, InputStream inputStream = new FileInputStream(orcZipPath); long importOrcId = 0L; try { - Map uploadMap = bmlService.upload(userName, inputStream, "default_orc.zip", projectVo.getName()); - String resourceId = uploadMap.get("resourceId").toString(); - String version = uploadMap.get("version").toString(); + BmlResource bmlResource = bmlService.upload(userName, inputStream, "default_orc.zip", projectVo.getName()); + String resourceId = bmlResource.getResourceId(); + String version = bmlResource.getVersion(); //不能走release的importservice接口 因为dev标签没有import操作 importOrcId = migrateService.importOrcToOrchestrator(resourceId, version, projectVo, userName, "dev", workspace, orchestratorInfo); } finally { @@ -378,16 +373,19 @@ public void exportOrcSqlFile(HttpServletRequest req, ResponseExportOrchestrator exportResponse = null; OrchestratorVo orchestratorVo; if (orcVersionId != null) { - orchestratorVo = (OrchestratorVo) orchestratorSender.ask(new RequestQueryByIdOrchestrator(orchestratorId, orcVersionId)); + orchestratorVo = RpcAskUtils.processAskException(orchestratorSender.ask(new RequestQueryByIdOrchestrator(orchestratorId, orcVersionId)), + OrchestratorVo.class, RequestQueryByIdOrchestrator.class); } else { - orchestratorVo = (OrchestratorVo) orchestratorSender.ask(new RequestQueryByIdOrchestrator(orchestratorId, null)); + orchestratorVo = RpcAskUtils.processAskException(orchestratorSender.ask(new RequestQueryByIdOrchestrator(orchestratorId, null)), + OrchestratorVo.class, RequestQueryByIdOrchestrator.class); } orcVersionId = orchestratorVo.getDssOrchestratorVersion().getId(); LOG.info("export orchestrator orchestratorId " + orchestratorId + ",orcVersionId:" + orcVersionId); try { RequestExportOrchestrator requestExportOrchestrator = new RequestExportOrchestrator( userName, orchestratorId, orcVersionId, projectName, dssLabelList, addOrcVersion, workspace); - exportResponse = (ResponseExportOrchestrator) orchestratorSender.ask(requestExportOrchestrator); + exportResponse = RpcAskUtils.processAskException(orchestratorSender.ask(requestExportOrchestrator), + ResponseExportOrchestrator.class, RequestExportOrchestrator.class); } catch (Exception e) { LOG.error("export orchestrator failed for ", e); throw new DSSErrorException(100789, "export orchestrator failed for " + e.getMessage()); @@ -544,19 +542,17 @@ private DSSOrchestratorVersion getLatestOrchestratorVersion(Long projectId, Long requestOrchestratorVersion.setProjectId(projectId); requestOrchestratorVersion.setOrchestratorId(orcId); requestOrchestratorVersion.setUsername(userName); - ResponseOrchetratorVersion orchetratorVersion = (ResponseOrchetratorVersion) orchestratorSender.ask(requestOrchestratorVersion); - DSSOrchestratorVersion orchestratorLatestVersion = orchetratorVersion.getOrchestratorVersions().stream() - .filter((v) -> v.getValidFlag() == 1).sorted(new Comparator() { - @Override - public int compare(DSSOrchestratorVersion o1, DSSOrchestratorVersion o2) { - // 注意是逆序 - if (o1.getVersion().compareToIgnoreCase(o2.getVersion()) < 0) { - return 1; - } else if (o1.getVersion().compareToIgnoreCase(o2.getVersion()) == 0) { - return 0; - } else { - return -1; - } + ResponseOrchetratorVersion orchestratorVersion = RpcAskUtils.processAskException(orchestratorSender.ask(requestOrchestratorVersion), + ResponseOrchetratorVersion.class, RequestOrchestratorVersion.class); + DSSOrchestratorVersion orchestratorLatestVersion = orchestratorVersion.getOrchestratorVersions().stream() + .filter((v) -> v.getValidFlag() == 1).sorted((o1, o2) -> { + // 注意是逆序 + if (o1.getVersion().compareToIgnoreCase(o2.getVersion()) < 0) { + return 1; + } else if (o1.getVersion().compareToIgnoreCase(o2.getVersion()) == 0) { + return 0; + } else { + return -1; } }).findFirst().get(); return orchestratorLatestVersion; @@ -564,7 +560,8 @@ public int compare(DSSOrchestratorVersion o1, DSSOrchestratorVersion o2) { private String getLatestFlowBmlVersion(String username, long flowId) { RequestQueryWorkFlow requestQueryWorkFlow = new RequestQueryWorkFlow(username, flowId); - ResponseQueryWorkflow responseQueryWorkflow = (ResponseQueryWorkflow) workflowSender.ask(requestQueryWorkFlow); + ResponseQueryWorkflow responseQueryWorkflow = RpcAskUtils.processAskException(workflowSender.ask(requestQueryWorkFlow), + ResponseQueryWorkflow.class, RequestQueryWorkFlow.class); if (null != responseQueryWorkflow && null != responseQueryWorkflow.getDssFlow()) { return responseQueryWorkflow.getDssFlow().getBmlVersion(); } else { @@ -591,7 +588,7 @@ public Message exportAllFlowInProject(HttpServletRequest req, // 1,获取项目下所有的工作流 // 获取工作空间id,没有就报错 List workspaces = dssWorkspaceService.getWorkspaces(username); - long workspaceId = 0l; + long workspaceId = 0L; DSSWorkspace dssWorkspace = workspaces.stream().filter((w) -> w.getName().equals(workspaceName)).findFirst().get(); if (null == dssWorkspace) { LOG.error("Cannot find workspace {}, please create it first.", workspaceName); @@ -620,7 +617,8 @@ public Message exportAllFlowInProject(HttpServletRequest req, if (StringUtils.isBlank(requestOrchestratorInfos.getOrchestratorMode())) { requestOrchestratorInfos.setOrchestratorMode(DEFAULT_PROJECT_ORCHESTRATOR_MODE); } - ResponseOrchestratorInfos responseOrchestratorInfos = (ResponseOrchestratorInfos) orchestratorSender.ask(requestOrchestratorInfos); + ResponseOrchestratorInfos responseOrchestratorInfos = RpcAskUtils.processAskException(orchestratorSender.ask(requestOrchestratorInfos), + ResponseOrchestratorInfos.class, RequestOrchestratorInfos.class); int count = 0; if (CollectionUtils.isNotEmpty(responseOrchestratorInfos.getOrchestratorInfos())) { for (DSSOrchestratorInfo orchestratorInfo : responseOrchestratorInfos.getOrchestratorInfos()) { @@ -632,7 +630,8 @@ public Message exportAllFlowInProject(HttpServletRequest req, requestOrchestratorVersion.setProjectId(orchestratorInfo.getProjectId()); ResponseOrchetratorVersion responseOrchetratorVersion = null; try { - responseOrchetratorVersion = (ResponseOrchetratorVersion) orchestratorSender.ask(requestOrchestratorVersion); + responseOrchetratorVersion = RpcAskUtils.processAskException(orchestratorSender.ask(requestOrchestratorVersion), + ResponseOrchetratorVersion.class, RequestOrchestratorVersion.class); } catch (Exception e) { DSSExceptionUtils.dealErrorException(60015, "Ask orchestrotor version failed " + BDPJettyServerHelper.gson().toJson(requestOrchestratorVersion), e, DSSErrorException.class); @@ -644,7 +643,7 @@ public int compare(Long o1, Long o2) { // 注意,正序 if (o1 < o2) { return -1; - } else if (o1 == o2) { + } else if (o1.equals(o2)) { return 0; } else { return 1; @@ -672,7 +671,7 @@ private void exportFlow(String username, Workspace workspace, String projectName projectName, labels, addOrcVersion, workspace); ResponseExportOrchestrator exportResponse = null; try { - exportResponse = (ResponseExportOrchestrator) orchestratorSender.ask(exportRequest); + exportResponse = RpcAskUtils.processAskException(orchestratorSender.ask(exportRequest), ResponseExportOrchestrator.class, RequestExportOrchestrator.class); } catch (Exception e) { DSSExceptionUtils.dealErrorException(60015, "export orchestrator ref failed " + BDPJettyServerHelper.gson().toJson(exportRequest), e, DSSErrorException.class); diff --git a/dss-framework/framework-plugins/dss-framework-migrate-server/src/main/java/com/webank/wedatasphere/dss/migrate/service/impl/MetaReader.java b/dss-framework/framework-plugins/dss-framework-migrate-server/src/main/java/com/webank/wedatasphere/dss/migrate/service/impl/MetaReader.java index aee2d75636..5e3842525e 100644 --- a/dss-framework/framework-plugins/dss-framework-migrate-server/src/main/java/com/webank/wedatasphere/dss/migrate/service/impl/MetaReader.java +++ b/dss-framework/framework-plugins/dss-framework-migrate-server/src/main/java/com/webank/wedatasphere/dss/migrate/service/impl/MetaReader.java @@ -66,13 +66,18 @@ private void readT() { body.stream().map(DSSExceptionUtils.map(this::lineToT)).forEach(datas::add); } - private T lineToT(List list) throws IllegalAccessException, InstantiationException, NoSuchFieldException, ParseException { + private T lineToT(List list) throws IllegalAccessException, InstantiationException, ParseException { T t = tClass.newInstance(); for (int i = 0; i < list.size(); i++) { try{ String valueStr = list.get(i); if ("null".equalsIgnoreCase(valueStr)) continue; - Field declaredField = tClass.getDeclaredField(fields.get(i)); + Field declaredField; + try { + declaredField = tClass.getDeclaredField(fields.get(i)); + }catch ( java.lang.NoSuchFieldException e){ + continue; + } declaredField.setAccessible(true); Object value = null; String type = declaredField.getType().getSimpleName(); diff --git a/dss-framework/framework-plugins/dss-framework-migrate-server/src/main/java/com/webank/wedatasphere/dss/migrate/service/impl/MigrateServiceImpl.java b/dss-framework/framework-plugins/dss-framework-migrate-server/src/main/java/com/webank/wedatasphere/dss/migrate/service/impl/MigrateServiceImpl.java index 74510be918..73ed4e4832 100644 --- a/dss-framework/framework-plugins/dss-framework-migrate-server/src/main/java/com/webank/wedatasphere/dss/migrate/service/impl/MigrateServiceImpl.java +++ b/dss-framework/framework-plugins/dss-framework-migrate-server/src/main/java/com/webank/wedatasphere/dss/migrate/service/impl/MigrateServiceImpl.java @@ -1,28 +1,26 @@ package com.webank.wedatasphere.dss.migrate.service.impl; import com.google.common.collect.Lists; +import com.webank.wedatasphere.dss.common.entity.BmlResource; import com.webank.wedatasphere.dss.common.entity.IOType; import com.webank.wedatasphere.dss.common.exception.DSSErrorException; import com.webank.wedatasphere.dss.common.label.DSSLabel; import com.webank.wedatasphere.dss.common.label.EnvDSSLabel; import com.webank.wedatasphere.dss.common.protocol.ResponseImportOrchestrator; -import com.webank.wedatasphere.dss.common.utils.DSSCommonUtils; -import com.webank.wedatasphere.dss.common.utils.DSSExceptionUtils; -import com.webank.wedatasphere.dss.common.utils.IoUtils; -import com.webank.wedatasphere.dss.common.utils.ZipHelper; +import com.webank.wedatasphere.dss.common.utils.*; import com.webank.wedatasphere.dss.framework.project.entity.DSSProjectDO; import com.webank.wedatasphere.dss.framework.project.entity.request.ProjectCreateRequest; import com.webank.wedatasphere.dss.framework.project.entity.vo.DSSProjectVo; -import com.webank.wedatasphere.dss.framework.project.server.service.BMLService; +import com.webank.wedatasphere.dss.common.service.BMLService; import com.webank.wedatasphere.dss.framework.project.service.DSSFrameworkProjectService; import com.webank.wedatasphere.dss.framework.project.service.DSSProjectService; -import com.webank.wedatasphere.dss.migrate.conf.MigrateConf; import com.webank.wedatasphere.dss.migrate.exception.MigrateErrorException; import com.webank.wedatasphere.dss.migrate.service.MetaService; import com.webank.wedatasphere.dss.migrate.service.MigrateService; import com.webank.wedatasphere.dss.orchestrator.common.entity.DSSOrchestratorInfo; import com.webank.wedatasphere.dss.orchestrator.common.entity.OrchestratorVo; import com.webank.wedatasphere.dss.orchestrator.common.protocol.*; +import com.webank.wedatasphere.dss.sender.service.DSSSenderServiceFactory; import com.webank.wedatasphere.dss.standard.app.sso.Workspace; import com.webank.wedatasphere.dss.workflow.common.entity.DSSFlow; import com.webank.wedatasphere.dss.workflow.common.entity.DSSFlowRelation; @@ -34,6 +32,7 @@ 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.stereotype.Service; import java.io.*; @@ -58,9 +57,10 @@ public class MigrateServiceImpl implements MigrateService { DSSFrameworkProjectService dssFrameworkProjectService; @Autowired + @Qualifier("projectBmlService") BMLService bmlService; - private Sender orchestratorSender = Sender.getSender(MigrateConf.ORC_SERVER_NAME); + private Sender orchestratorSender = DSSSenderServiceFactory.getOrCreateServiceInstance().getOrcSender(); @Override public void migrate(String userName, String inputZipPath, Workspace workspace) throws Exception { @@ -80,9 +80,9 @@ public void migrate(String userName, String inputZipPath, Workspace workspace) t DSSProjectDO dssProject = metaService.readProject(inputPath); DSSProjectVo finalProject; DSSProjectDO dbProject = dssProjectService.getProjectByName(dssProject.getName()); - if(!dssProject.getUsername().equalsIgnoreCase(userName)){ + if (!dssProject.getUsername().equalsIgnoreCase(userName)) { LOG.error("fatal error, project owner is {} ,but export user is {}", dssProject.getUsername(), userName); - throw new MigrateErrorException(40013, "project has been exported by others,not owner "+dssProject.getUsername()); + throw new MigrateErrorException(40013, "project has been exported by others,not owner " + dssProject.getUsername()); } if (dbProject == null) { //判断如果没有该工程,则开始调用接口来进行工程创建 @@ -119,7 +119,7 @@ public void migrate(String userName, String inputZipPath, Workspace workspace) t dssProjectService.modifyOldProject(dssProject, dbProject); } List dssFlows = metaService.readFlow(inputPath); - dssFlows.stream().forEach(dssFlow -> dssFlow.setProjectID(finalProject.getId())); + dssFlows.stream().forEach(dssFlow -> dssFlow.setProjectId(finalProject.getId())); List dssFlowRelations = metaService.readFlowRelation(inputPath); List rootFlows = dssFlows.stream().filter(DSSFlow::getRootFlow).collect(Collectors.toList()); @@ -173,9 +173,9 @@ public void migrate(String userName, String inputZipPath, Workspace workspace) t String orcZipPath = ZipHelper.zip(orcPath); InputStream inputStream = new FileInputStream(orcZipPath); try { - Map uploadMap = bmlService.upload(userName, inputStream, "default_orc.zip", finalProject.getName()); - String resourceId = uploadMap.get("resourceId").toString(); - String version = uploadMap.get("version").toString(); + BmlResource bmlResource = bmlService.upload(userName, inputStream, "default_orc.zip", finalProject.getName()); + String resourceId = bmlResource.getResourceId(); + String version = bmlResource.getVersion(); //不能走release的importservice接口 因为dev标签没有import操作 importOrcToOrchestrator(resourceId, version, finalProject, userName, "dev", workspace, orchestratorInfo); } finally { @@ -229,7 +229,8 @@ public long importOrcToOrchestrator(String resourceId, String version, DSSProjec RequestImportOrchestrator requestImportOrchestrator = new RequestImportOrchestrator(username, project.getName(), project.getId(), resourceId, version, dssOrchestratorInfo.getName(), dssLabels, workspace); - ResponseImportOrchestrator responseImportOrchestrator = (ResponseImportOrchestrator) this.orchestratorSender.ask(requestImportOrchestrator); + ResponseImportOrchestrator responseImportOrchestrator = RpcAskUtils.processAskException(this.orchestratorSender.ask(requestImportOrchestrator), + ResponseImportOrchestrator.class, RequestImportOrchestrator.class); return responseImportOrchestrator.orcId(); } @@ -259,8 +260,8 @@ public DSSOrchestratorInfo buildOrchestratorInfo(DSSFlow dssFlow, DSSProjectVo d @Override public String queryOrcUUIDByName(Long workspaceId, Long projectId, String orcName) throws DSSErrorException { String uuid = null; - ResponseOrchestratorInfos responseOrchestratorInfos = (ResponseOrchestratorInfos) orchestratorSender - .ask(new RequestOrchestratorInfos(null, projectId, orcName, null)); + ResponseOrchestratorInfos responseOrchestratorInfos = RpcAskUtils.processAskException(orchestratorSender.ask(new RequestOrchestratorInfos(null, projectId, orcName, null)), + ResponseOrchestratorInfos.class, RequestOrchestratorInfos.class); if (CollectionUtils.isNotEmpty(responseOrchestratorInfos.getOrchestratorInfos())) { uuid = responseOrchestratorInfos.getOrchestratorInfos().get(0).getUUID(); } diff --git a/dss-framework/framework-plugins/dss-framework-orchestrator-publish/pom.xml b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/pom.xml index d04fad0f84..648919cd42 100644 --- a/dss-framework/framework-plugins/dss-framework-orchestrator-publish/pom.xml +++ b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/pom.xml @@ -21,7 +21,8 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 + ../../../pom.xml 4.0.0 @@ -57,7 +58,7 @@ org.springframework spring-tx - ${spring.version} + ${spring-framework.version} provided @@ -81,7 +82,7 @@ org.springframework spring-context - ${spring.version} + ${spring-framework.version} provided diff --git a/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/ConversionDSSOrchestratorPlugin.java b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/ConversionDSSOrchestratorPlugin.java index 390b28e90c..307b8dd009 100644 --- a/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/ConversionDSSOrchestratorPlugin.java +++ b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/ConversionDSSOrchestratorPlugin.java @@ -27,10 +27,21 @@ public interface ConversionDSSOrchestratorPlugin extends DSSOrchestratorPlugin { - + /** + * 编排转化(转为具体调度形式,比如转为schedulis调度工作流) + * 这里只是个rpc的client简单封装。 + * @param userName 发布人 + * @param project 编排所属项目 + * @param workspace 所属工作空间 + * @param orchestrationIdMap 要发布的编排,key为编排的appId,value为编排的refOrchestrationId + * @param dssLabels 标签 + * @param approvalId 审批单号,如果不需要,可以传null + * @return + */ ResponseOperateOrchestrator convert(String userName, DSSProject project, Workspace workspace, Map orchestrationIdMap, - List dssLabels); + List dssLabels, + String approvalId); } diff --git a/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/ExportDSSOrchestratorPlugin.java b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/ExportDSSOrchestratorPlugin.java index d1d1efb34b..d2130b45f2 100644 --- a/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/ExportDSSOrchestratorPlugin.java +++ b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/ExportDSSOrchestratorPlugin.java @@ -20,23 +20,25 @@ import com.webank.wedatasphere.dss.common.label.DSSLabel; import com.webank.wedatasphere.dss.orchestrator.common.entity.DSSOrchestratorInfo; import com.webank.wedatasphere.dss.orchestrator.core.plugin.DSSOrchestratorPlugin; +import com.webank.wedatasphere.dss.orchestrator.publish.entity.OrchestratorExportResult; import com.webank.wedatasphere.dss.standard.app.sso.Workspace; import java.util.List; -import java.util.Map; public interface ExportDSSOrchestratorPlugin extends DSSOrchestratorPlugin { /** - * 导出Orchestrator基本信息和工作流基本信息 + * 导出Orchestrator基本信息和工作流,并放到BML中。 + * 注意:导入导出接口只适合于不同环境下,先从A环境导出工作流,再导入到B环境的情况。 + * 不适合用于在同一环境下的复制。同一环境的复制操作需使用copyOperation。 + * @param addOrcVersion 导出之后,是否要升级一个版本 */ - Map exportOrchestrator(String userName, - Long orchestratorId, - Long orcVersionId, - String projectName, - List dssLabels, - boolean addOrcVersion, Workspace workspace) throws DSSErrorException; - + OrchestratorExportResult exportOrchestrator(String userName, + Long orchestratorId, + Long orcVersionId, + String projectName, + List dssLabels, + boolean addOrcVersion, Workspace workspace) throws DSSErrorException; Long orchestratorVersionIncrease(Long orcId, String userName, String comment, diff --git a/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/ImportDSSOrchestratorPlugin.java b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/ImportDSSOrchestratorPlugin.java index 8332d0d2ea..a724632860 100644 --- a/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/ImportDSSOrchestratorPlugin.java +++ b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/ImportDSSOrchestratorPlugin.java @@ -18,6 +18,7 @@ import com.webank.wedatasphere.dss.common.exception.DSSErrorException; import com.webank.wedatasphere.dss.common.label.DSSLabel; +import com.webank.wedatasphere.dss.orchestrator.common.entity.DSSOrchestratorVersion; import com.webank.wedatasphere.dss.orchestrator.common.protocol.RequestImportOrchestrator; import com.webank.wedatasphere.dss.orchestrator.core.plugin.DSSOrchestratorPlugin; import com.webank.wedatasphere.dss.standard.app.sso.Workspace; @@ -30,13 +31,22 @@ public interface ImportDSSOrchestratorPlugin extends DSSOrchestratorPlugin { /** * 导入Orchestrator + * 注意:导入导出接口只适合于不同环境下,先从A环境导出工作流,再导入到B环境的情况。 + * 不适合用于在同一环境下的复制。同一环境的复制操作需使用copyOperation。 * @param requestImportOrchestrator * @return * @throws DSSErrorException * @throws IOException * @throws ExternalOperationFailedException */ - Long importOrchestrator(RequestImportOrchestrator requestImportOrchestrator) throws Exception; + DSSOrchestratorVersion importOrchestrator(RequestImportOrchestrator requestImportOrchestrator) throws Exception; + /** + * 导入复制的编排 + * @param requestImportOrchestrator + * @return + * @throws Exception + */ + Long importCopyOrchestrator(RequestImportOrchestrator requestImportOrchestrator, String suffix) throws Exception; } diff --git a/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/conf/DSSOrchestratorConf.java b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/conf/DSSOrchestratorConf.java index 5740b6e9b7..5afb26a7f4 100644 --- a/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/conf/DSSOrchestratorConf.java +++ b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/conf/DSSOrchestratorConf.java @@ -21,4 +21,8 @@ public class DSSOrchestratorConf { public static final CommonVars DSS_EXPORT_ENV = CommonVars.apply("wds.dss.server.export.env", "dev"); + /** + * 导入之后的工作流直接生效,而不需要发布之后。表明工作流的导入和发布是分开的,要发布的话,还需要手动触发 + */ + public static final CommonVars DSS_IMPORT_VALID_IMMEDIATELY = CommonVars.apply("wds.dss.server.import.valid.immediately", false); } diff --git a/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/entity/OrchestratorExportResult.java b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/entity/OrchestratorExportResult.java new file mode 100644 index 0000000000..aaa2577d27 --- /dev/null +++ b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/entity/OrchestratorExportResult.java @@ -0,0 +1,43 @@ +package com.webank.wedatasphere.dss.orchestrator.publish.entity; + +import com.webank.wedatasphere.dss.common.entity.BmlResource; + +/** + * 工作流导出结果 + * Author: xlinliu + * Date: 2022/8/22 + */ +public class OrchestratorExportResult { + /** + * 导出工作流的bml文件 + */ + private BmlResource bmlResource; + /** + * 工作流的版本 + */ + private String orcVersionId; + + public OrchestratorExportResult() { + } + + public OrchestratorExportResult(BmlResource bmlResource, String orcVersionId) { + this.bmlResource = bmlResource; + this.orcVersionId = orcVersionId; + } + + public BmlResource getBmlResource() { + return bmlResource; + } + + public void setBmlResource(BmlResource bmlResource) { + this.bmlResource = bmlResource; + } + + public String getOrcVersionId() { + return orcVersionId; + } + + public void setOrcVersionId(String orcVersionId) { + this.orcVersionId = orcVersionId; + } +} diff --git a/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/impl/ConversionDSSOrchestratorPluginImpl.java b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/impl/ConversionDSSOrchestratorPluginImpl.java index d6a36ac5c9..6c6be62a7a 100644 --- a/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/impl/ConversionDSSOrchestratorPluginImpl.java +++ b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/impl/ConversionDSSOrchestratorPluginImpl.java @@ -18,6 +18,7 @@ import com.webank.wedatasphere.dss.common.entity.project.DSSProject; import com.webank.wedatasphere.dss.common.label.DSSLabel; +import com.webank.wedatasphere.dss.common.utils.RpcAskUtils; import com.webank.wedatasphere.dss.orchestrator.common.protocol.RequestConvertOrchestrations; import com.webank.wedatasphere.dss.orchestrator.common.protocol.ResponseOperateOrchestrator; import com.webank.wedatasphere.dss.orchestrator.core.plugin.AbstractDSSOrchestratorPlugin; @@ -37,15 +38,17 @@ public ResponseOperateOrchestrator convert(String userName, DSSProject project, Workspace workspace, Map orchestrationIdMap, - List dssLabels) { + List dssLabels, + String approvalId) { //1、发布DSS编排,如DSS工作流 RequestConvertOrchestrations requestConvertOrchestrator = new RequestConvertOrchestrations(); requestConvertOrchestrator.setOrchestrationIdMap(orchestrationIdMap); requestConvertOrchestrator.setProject(project); requestConvertOrchestrator.setWorkspace(workspace); + requestConvertOrchestrator.setApprovalId(approvalId); requestConvertOrchestrator.setDSSLabels(dssLabels); requestConvertOrchestrator.setUserName(userName); Sender sender = DSSSenderServiceFactory.getOrCreateServiceInstance().getWorkflowSender(dssLabels); - return (ResponseOperateOrchestrator) sender.ask(requestConvertOrchestrator); + return RpcAskUtils.processAskException(sender.ask(requestConvertOrchestrator), ResponseOperateOrchestrator.class, RequestConvertOrchestrations.class); } } diff --git a/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/impl/ExportDSSOrchestratorPluginImpl.java b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/impl/ExportDSSOrchestratorPluginImpl.java index 1977b10272..76414479c9 100644 --- a/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/impl/ExportDSSOrchestratorPluginImpl.java +++ b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/impl/ExportDSSOrchestratorPluginImpl.java @@ -16,6 +16,7 @@ package com.webank.wedatasphere.dss.orchestrator.publish.impl; +import com.webank.wedatasphere.dss.common.entity.BmlResource; import com.webank.wedatasphere.dss.common.entity.IOType; import com.webank.wedatasphere.dss.common.exception.DSSErrorException; import com.webank.wedatasphere.dss.common.label.DSSLabel; @@ -29,13 +30,15 @@ import com.webank.wedatasphere.dss.orchestrator.core.DSSOrchestrator; import com.webank.wedatasphere.dss.orchestrator.core.exception.DSSOrchestratorErrorException; import com.webank.wedatasphere.dss.orchestrator.core.plugin.AbstractDSSOrchestratorPlugin; -import com.webank.wedatasphere.dss.orchestrator.core.service.BMLService; +import com.webank.wedatasphere.dss.common.service.BMLService; import com.webank.wedatasphere.dss.orchestrator.core.utils.OrchestratorUtils; import com.webank.wedatasphere.dss.orchestrator.db.dao.OrchestratorMapper; import com.webank.wedatasphere.dss.orchestrator.loader.OrchestratorManager; import com.webank.wedatasphere.dss.orchestrator.publish.ExportDSSOrchestratorPlugin; +import com.webank.wedatasphere.dss.orchestrator.publish.entity.OrchestratorExportResult; import com.webank.wedatasphere.dss.orchestrator.publish.io.export.MetaExportService; import com.webank.wedatasphere.dss.orchestrator.publish.utils.OrchestrationDevelopmentOperationUtils; +import com.webank.wedatasphere.dss.standard.app.development.operation.DevelopmentOperation; import com.webank.wedatasphere.dss.standard.app.development.operation.RefCopyOperation; import com.webank.wedatasphere.dss.standard.app.development.operation.RefExportOperation; import com.webank.wedatasphere.dss.standard.app.development.ref.*; @@ -45,8 +48,8 @@ import com.webank.wedatasphere.dss.standard.app.sso.Workspace; import org.apache.commons.lang.StringUtils; import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.beans.factory.annotation.Qualifier; import org.springframework.stereotype.Component; -import org.springframework.transaction.annotation.Transactional; import java.io.IOException; import java.io.InputStream; @@ -55,8 +58,10 @@ import java.util.Date; import java.util.List; import java.util.Map; +import java.util.function.BiFunction; +import java.util.function.Consumer; -import static com.webank.wedatasphere.dss.common.utils.ZipHelper.zipExportProject; +import static com.webank.wedatasphere.dss.common.utils.ZipHelper.zip; @Component @@ -65,10 +70,12 @@ public class ExportDSSOrchestratorPluginImpl extends AbstractDSSOrchestratorPlug static final String DEFAULT_ORC_NAME = "default_orc"; @Autowired + @Qualifier("orchestratorBmlService") private BMLService bmlService; @Autowired private OrchestratorMapper orchestratorMapper; @Autowired + @Qualifier("orcMetaExportService") private MetaExportService metaExportService; @Autowired private ContextService contextService; @@ -76,9 +83,8 @@ public class ExportDSSOrchestratorPluginImpl extends AbstractDSSOrchestratorPlug private OrchestratorManager orchestratorManager; @Override - @Transactional(rollbackFor = Exception.class) - public Map exportOrchestrator(String userName, Long orchestratorId, Long orcVersionId, String projectName, - List dssLabels, boolean addOrcVersion, Workspace workspace) throws DSSErrorException { + public OrchestratorExportResult exportOrchestrator(String userName, Long orchestratorId, Long orcVersionId, String projectName, + List dssLabels, boolean addOrcVersion, Workspace workspace) throws DSSErrorException { //1、导出info信息 if (orcVersionId == null || orcVersionId < 0){ LOGGER.info("orchestratorVersionId is {}.", orcVersionId); @@ -102,41 +108,51 @@ public Map exportOrchestrator(String userName, Long orchestrator } LOGGER.info("{} 开始导出Orchestrator: {} 版本ID为: {}.", userName, dssOrchestratorInfo.getName(), orcVersionId); - //2、导出第三方应用信息,如工作流、Visualis、Qualities + //2、导出第三方应用信息,如工作流、Visualis、Qualitis DSSOrchestrator dssOrchestrator = orchestratorManager.getOrCreateOrchestrator(userName, workspace.getWorkspaceName(), dssOrchestratorInfo.getType(), dssLabels); - ExportResponseRef responseRef = OrchestrationDevelopmentOperationUtils.tryOrchestrationOperation(dssOrchestratorInfo, dssOrchestrator, userName, - workspace, dssLabels, DevelopmentIntegrationStandard::getRefExportService, + //定义操作结果处理器 + BiFunction responseRefConsumer = (developmentOperation, developmentRequestRef) -> { + RefJobContentRequestRef requestRef = (RefJobContentRequestRef) developmentRequestRef; + requestRef.setRefJobContent(MapUtils.newCommonMap(OrchestratorRefConstant.ORCHESTRATION_ID_KEY, dssOrchestratorVersion.getAppId())); + return ((RefExportOperation) developmentOperation).exportRef(requestRef); + }; + //定义项目相关的处理器,处着编排的RequestRef的项目相关信息 + Consumer projectRefRequestRefConsumer = projectRefRequestRef -> projectRefRequestRef.setProjectName(projectName).setRefProjectId(dssOrchestratorVersion.getProjectId()); + + ExportResponseRef responseRef = OrchestrationDevelopmentOperationUtils.tryOrchestrationOperation( + dssOrchestratorInfo, + dssOrchestrator, + userName, + workspace, + dssLabels, + //指明DevelopmentService是RefExportService + DevelopmentIntegrationStandard::getRefExportService, + //指明operation是ExportOperation developmentService -> ((RefExportService) developmentService).getRefExportOperation(), null, - projectRefRequestRef -> projectRefRequestRef.setProjectName(projectName).setRefProjectId(dssOrchestratorVersion.getProjectId()), - (developmentOperation, developmentRequestRef) -> { - RefJobContentRequestRef requestRef = (RefJobContentRequestRef) developmentRequestRef; - requestRef.setRefJobContent(MapUtils.newCommonMap(OrchestratorRefConstant.ORCHESTRATION_ID_KEY, dssOrchestratorVersion.getAppId())); - return ((RefExportOperation) developmentOperation).exportRef(requestRef); - }, "export"); + projectRefRequestRefConsumer, + responseRefConsumer, + "export"); String resourceId = (String) responseRef.getResourceMap().get(ImportRequestRef.RESOURCE_ID_KEY); String version = (String) responseRef.getResourceMap().get(ImportRequestRef.RESOURCE_VERSION_KEY); bmlService.downloadToLocalPath(userName, resourceId, version, orcExportSaveBasePath + "orc_flow.zip"); //打包导出工程 - String exportPath = zipExportProject(orcExportSaveBasePath); + String exportPath = zip(orcExportSaveBasePath); //3、打包新的zip包上传BML InputStream inputStream = bmlService.readLocalResourceFile(userName, exportPath); - Map resultMap = bmlService.upload(userName, inputStream, + BmlResource uploadResult = bmlService.upload(userName, inputStream, dssOrchestratorInfo.getName() + ".OrcExport", projectName); //4、判断导出后是否改变Orc的版本 if (addOrcVersion) { - Long orcIncreaseVersionId = orchestratorVersionIncrease(dssOrchestratorInfo.getId(), + orcVersionId = orchestratorVersionIncrease(dssOrchestratorInfo.getId(), userName, dssOrchestratorInfo.getComment(), workspace, dssOrchestratorInfo, projectName, dssLabels); - resultMap.put("orcVersionId", orcIncreaseVersionId); - } else { - resultMap.put("orcVersionId", orcVersionId); } - return resultMap; + return new OrchestratorExportResult(uploadResult,String.valueOf(orcVersionId)); //4、返回BML存储信息 } else { throw new DSSErrorException(90038, "该Orchestrator的版本号不存在,请检查版本号是否正确."); @@ -166,14 +182,12 @@ public Long orchestratorVersionIncrease(Long orcId, //更新老版本的comment DSSOrchestratorVersion updateCommentVersion = new DSSOrchestratorVersion(); updateCommentVersion.setId(oldOrcVersionId); - String realComment = comment != null ? comment : "release comment"; + String realComment = StringUtils.isNotBlank(comment) ? comment : "release comment"; updateCommentVersion.setComment(realComment); if(StringUtils.isNotBlank(userName)){ updateCommentVersion.setUpdater(userName); dssOrchestratorVersion.setUpdater(userName); } - updateCommentVersion.setUpdateTime(new Date()); - orchestratorMapper.updateOrchestratorVersion(updateCommentVersion); //要求AppConn对应第三方应用拷贝一个新的app出来关联,如工作流,需要新建一个新的工作流进行关联。 //1、生成上下文ContextId @@ -198,6 +212,7 @@ public Long orchestratorVersionIncrease(Long orcId, dssOrchestratorVersion.setContent((String) responseRef.getRefJobContent().get(OrchestratorRefConstant.ORCHESTRATION_CONTENT_KEY)); //update appConn node contextId dssOrchestratorVersion.setFormatContextId(contextId); + orchestratorMapper.updateOrchestratorVersion(updateCommentVersion); orchestratorMapper.addOrchestratorVersion(dssOrchestratorVersion); return dssOrchestratorVersion.getId(); } diff --git a/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/impl/ImportDSSOrchestratorPluginImpl.java b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/impl/ImportDSSOrchestratorPluginImpl.java index ded19df024..b712f6281f 100644 --- a/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/impl/ImportDSSOrchestratorPluginImpl.java +++ b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/impl/ImportDSSOrchestratorPluginImpl.java @@ -16,47 +16,49 @@ package com.webank.wedatasphere.dss.orchestrator.publish.impl; +import com.webank.wedatasphere.dss.common.entity.BmlResource; +import com.webank.wedatasphere.dss.common.entity.node.DSSNodeDefault; import com.webank.wedatasphere.dss.common.exception.DSSErrorException; import com.webank.wedatasphere.dss.common.label.DSSLabel; import com.webank.wedatasphere.dss.common.label.DSSLabelUtil; -import com.webank.wedatasphere.dss.common.utils.DSSExceptionUtils; -import com.webank.wedatasphere.dss.common.utils.IoUtils; -import com.webank.wedatasphere.dss.common.utils.MapUtils; -import com.webank.wedatasphere.dss.common.utils.ZipHelper; +import com.webank.wedatasphere.dss.common.utils.*; import com.webank.wedatasphere.dss.contextservice.service.ContextService; import com.webank.wedatasphere.dss.orchestrator.common.entity.DSSOrchestratorInfo; import com.webank.wedatasphere.dss.orchestrator.common.entity.DSSOrchestratorVersion; import com.webank.wedatasphere.dss.orchestrator.common.protocol.RequestImportOrchestrator; -import com.webank.wedatasphere.dss.orchestrator.common.protocol.RequestProjectImportOrchestrator; import com.webank.wedatasphere.dss.orchestrator.common.ref.OrchestratorRefConstant; import com.webank.wedatasphere.dss.orchestrator.core.DSSOrchestrator; import com.webank.wedatasphere.dss.orchestrator.core.plugin.AbstractDSSOrchestratorPlugin; -import com.webank.wedatasphere.dss.orchestrator.core.service.BMLService; +import com.webank.wedatasphere.dss.common.service.BMLService; import com.webank.wedatasphere.dss.orchestrator.core.utils.OrchestratorUtils; import com.webank.wedatasphere.dss.orchestrator.db.dao.OrchestratorMapper; import com.webank.wedatasphere.dss.orchestrator.loader.OrchestratorManager; import com.webank.wedatasphere.dss.orchestrator.publish.ImportDSSOrchestratorPlugin; +import com.webank.wedatasphere.dss.orchestrator.publish.conf.DSSOrchestratorConf; +import com.webank.wedatasphere.dss.orchestrator.publish.io.export.MetaExportService; import com.webank.wedatasphere.dss.orchestrator.publish.io.input.MetaInputService; import com.webank.wedatasphere.dss.orchestrator.publish.utils.OrchestrationDevelopmentOperationUtils; -import com.webank.wedatasphere.dss.sender.service.DSSSenderServiceFactory; import com.webank.wedatasphere.dss.standard.app.development.operation.RefImportOperation; import com.webank.wedatasphere.dss.standard.app.development.ref.ImportRequestRef; import com.webank.wedatasphere.dss.standard.app.development.ref.RefJobContentResponseRef; import com.webank.wedatasphere.dss.standard.app.development.service.RefImportService; import com.webank.wedatasphere.dss.standard.app.development.standard.DevelopmentIntegrationStandard; import com.webank.wedatasphere.dss.standard.app.sso.Workspace; +import com.webank.wedatasphere.dss.workflow.common.entity.DSSFlow; +import com.webank.wedatasphere.dss.workflow.common.entity.DSSFlowRelation; +import org.apache.commons.io.FileUtils; import org.apache.commons.lang.StringUtils; -import org.springframework.beans.BeanUtils; +import org.apache.linkis.server.BDPJettyServerHelper; import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.beans.factory.annotation.Qualifier; import org.springframework.stereotype.Component; -import org.springframework.transaction.annotation.Transactional; -import java.io.File; -import java.io.InputStream; +import java.io.*; import java.util.Date; import java.util.List; import java.util.Map; import java.util.UUID; +import java.util.stream.Collectors; import static com.webank.wedatasphere.dss.orchestrator.publish.impl.ExportDSSOrchestratorPluginImpl.DEFAULT_ORC_NAME; @@ -65,10 +67,15 @@ public class ImportDSSOrchestratorPluginImpl extends AbstractDSSOrchestratorPlugin implements ImportDSSOrchestratorPlugin { @Autowired + @Qualifier("orcMetaInputService") private MetaInputService metaInputService; @Autowired + @Qualifier("orcMetaExportService") + private MetaExportService metaExportService; + @Autowired private OrchestratorMapper orchestratorMapper; @Autowired + @Qualifier("orchestratorBmlService") private BMLService bmlService; @Autowired private ContextService contextService; @@ -76,8 +83,7 @@ public class ImportDSSOrchestratorPluginImpl extends AbstractDSSOrchestratorPlug private OrchestratorManager orchestratorManager; @Override - @Transactional(rollbackFor = Exception.class) - public Long importOrchestrator(RequestImportOrchestrator requestImportOrchestrator) throws Exception { + public DSSOrchestratorVersion importOrchestrator(RequestImportOrchestrator requestImportOrchestrator) throws Exception { String userName = requestImportOrchestrator.getUserName(); String projectName = requestImportOrchestrator.getProjectName(); Long projectId = requestImportOrchestrator.getProjectId(); @@ -94,7 +100,7 @@ public Long importOrchestrator(RequestImportOrchestrator requestImportOrchestrat //2、导入Info信息(导入冲突处理) List dssOrchestratorInfos = metaInputService.importOrchestrator(inputPath); DSSOrchestratorInfo importDssOrchestratorInfo = dssOrchestratorInfos.get(0); - + importDssOrchestratorInfo.setProjectId(projectId); //复制工程,直接使用新的UUID和复制后的工程ID if (requestImportOrchestrator.getCopyProjectId() != null && StringUtils.isNotBlank(requestImportOrchestrator.getCopyProjectName())) { @@ -109,16 +115,14 @@ public Long importOrchestrator(RequestImportOrchestrator requestImportOrchestrat //add 和update都需要更新成当前环境id信息,放到新的版本记录中 //todo 跨环境必须保证工程ID一样,或者需要更新导入包中的所有工程ID为当前最新ID,不一致的话关系到上下文、第三方工程的映射问题 if (null != existFlag) { - //判断是否存在相同名称的编排名称 + //判断是否存在相同名称的编排 if (StringUtils.isNotBlank(uuid) && !uuid.equals(importDssOrchestratorInfo.getUUID())) { DSSExceptionUtils .dealErrorException(61002, "The same orchestration name already exists", DSSErrorException.class); } importDssOrchestratorInfo.setId(existFlag.getId()); - //如果Orchestrator已经导入过,目前只更新版本信息,并更新基础信息name,其它信息不修改。 - orchestratorMapper.updateOrchestrator(importDssOrchestratorInfo); } else { - //判断是否存在相同名称的编排名称 + //判断是否存在相同名称的编排 if (StringUtils.isNotBlank(uuid)) { DSSExceptionUtils .dealErrorException(61002, "The same orchestration name already exists", DSSErrorException.class); @@ -137,14 +141,13 @@ public Long importOrchestrator(RequestImportOrchestrator requestImportOrchestrat if (StringUtils.isEmpty(importDssOrchestratorInfo.getOrchestratorWay())) { importDssOrchestratorInfo.setOrchestratorWay(",pom_work_flow_DAG,"); } - orchestratorMapper.addOrchestrator(importDssOrchestratorInfo); } String flowZipPath = inputPath + File.separator + "orc_flow.zip"; //3、上传工作流zip包到bml InputStream inputStream = bmlService.readLocalResourceFile(userName, flowZipPath); - Map resultMap = bmlService.upload(userName, inputStream, importDssOrchestratorInfo.getName() + "_orc_flow.zip", projectName); - String orcResourceId = resultMap.get("resourceId").toString(); - String orcBmlVersion = resultMap.get("version").toString(); + BmlResource resultMap = bmlService.upload(userName, inputStream, importDssOrchestratorInfo.getName() + "_orc_flow.zip", projectName); + String orcResourceId = resultMap.getResourceId(); + String orcBmlVersion = resultMap.getVersion(); //4、导入版本Version信息 DSSOrchestratorVersion dssOrchestratorVersion = new DSSOrchestratorVersion(); @@ -155,8 +158,9 @@ public Long importOrchestrator(RequestImportOrchestrator requestImportOrchestrat dssOrchestratorVersion.setProjectId(projectId); dssOrchestratorVersion.setSource("Orchestrator create"); dssOrchestratorVersion.setUpdater(userName); - //生产导入:默认是为无效,开发环境为有效 - dssOrchestratorVersion.setValidFlag(DSSLabelUtil.isDevEnv(dssLabels) ? 1 : 0); + //生产导入:默认是为无效,除非开启直接有效(在生成环境独立部署的时候,导入和发布是分开的);开发环境为有效 + int valid = DSSOrchestratorConf.DSS_IMPORT_VALID_IMMEDIATELY.getValue() || DSSLabelUtil.isDevEnv(dssLabels) ? 1 : 0; + dssOrchestratorVersion.setValidFlag(valid); String oldVersion = orchestratorMapper.getLatestVersion(importDssOrchestratorInfo.getId(), 1); if (StringUtils.isNotEmpty(oldVersion)) { @@ -170,10 +174,7 @@ public Long importOrchestrator(RequestImportOrchestrator requestImportOrchestrat dssOrchestratorVersion.setFormatContextId(contextId); LOGGER.info("Create a new ContextId for import: {} ", contextId); - dssOrchestratorVersion.setUpdateTime(new Date()); - orchestratorMapper.addOrchestratorVersion(dssOrchestratorVersion); - - //6、导出第三方应用信息,如工作流、Visualis、Qualities + //6、导入第三方应用信息,如工作流、Visualis、Qualities DSSOrchestrator dssOrchestrator = orchestratorManager.getOrCreateOrchestrator(userName, workspace.getWorkspaceName(), importDssOrchestratorInfo.getType(), dssLabels); Long finalProjectId = projectId; @@ -191,24 +192,194 @@ public Long importOrchestrator(RequestImportOrchestrator requestImportOrchestrat }, "import"); long orchestrationId = (Long) responseRef.getRefJobContent().get(OrchestratorRefConstant.ORCHESTRATION_ID_KEY); String orchestrationContent = (String) responseRef.getRefJobContent().get(OrchestratorRefConstant.ORCHESTRATION_CONTENT_KEY); + if(null != existFlag){ + //如果Orchestrator已经导入过,目前只更新版本信息,并更新基础信息name,其它信息不修改。 + orchestratorMapper.updateOrchestrator(importDssOrchestratorInfo); + }else{ + orchestratorMapper.addOrchestrator(importDssOrchestratorInfo); + } //更新返回內容 + dssOrchestratorVersion.setUpdateTime(new Date()); + dssOrchestratorVersion.setAppId(orchestrationId); + dssOrchestratorVersion.setContent(orchestrationContent); + dssOrchestratorVersion.setOrchestratorId(importDssOrchestratorInfo.getId()); + orchestratorMapper.addOrchestratorVersion(dssOrchestratorVersion); + LOGGER.info("import orchestrator success,orcId:{},appId:{}",importDssOrchestratorInfo.getId(),orchestrationId); + + return dssOrchestratorVersion; + } + + @Override + public Long importCopyOrchestrator(RequestImportOrchestrator requestImportOrchestrator, String nodeSuffix) throws Exception { + String userName = requestImportOrchestrator.getUserName(); + String projectName = requestImportOrchestrator.getProjectName(); + String resourceId = requestImportOrchestrator.getResourceId(); + String version = requestImportOrchestrator.getBmlVersion(); + List dssLabels = requestImportOrchestrator.getDssLabels(); + Workspace workspace = requestImportOrchestrator.getWorkspace(); + + String targetProjectName = requestImportOrchestrator.getCopyProjectName(); + Long targetProjectId = requestImportOrchestrator.getCopyProjectId(); + String targetOrchestratorName = requestImportOrchestrator.getOrchestratorName(); + + //1、下载BML的Orchestrator的导入包 + String inputZipPath = IoUtils.generateIOPath(userName, targetProjectName, DEFAULT_ORC_NAME + ".zip"); + bmlService.downloadToLocalPath(userName, resourceId, version, inputZipPath); + String inputPath = ZipHelper.unzip(inputZipPath); + + if (StringUtils.isBlank(nodeSuffix)){ + nodeSuffix = "copy"; + } + String flowZipPath = inputPath + File.separator + "orc_flow.zip"; + ZipHelper.unzip(flowZipPath); + String sourceProjectDir = inputPath + File.separator + projectName; + //修改flow的json文件 + List dssFlows = metaInputService.inputFlow(sourceProjectDir); + for (DSSFlow dssFlow: dssFlows) { + LOGGER.info("Start to modify dssFlow {} json.", dssFlow.getName()); + String flowInputPath = sourceProjectDir + File.separator + dssFlow.getName(); + String flowJsonPath = flowInputPath + File.separator + dssFlow.getName() + ".json"; + // 修改原有的json内容 + String flowJson = bmlService.readLocalTextFile(userName, flowJsonPath); + Map flowJsonObject = BDPJettyServerHelper.jacksonJson().readValue(flowJson, Map.class); + List workflowNodes = DSSCommonUtils.getWorkFlowNodes(flowJson); + String finalNodeSuffix = nodeSuffix; + List targetWorkflowNodes = workflowNodes.stream().peek(s -> { + String name = s.getName(); + s.setName(name + "_" + finalNodeSuffix); + }).collect(Collectors.toList()); + flowJsonObject.replace("nodes", targetWorkflowNodes); + String updatedJson = DSSCommonUtils.COMMON_GSON.toJson(flowJsonObject); + //修改json文件保存路径 + try (BufferedWriter bufferedWriter = new BufferedWriter(new FileWriter(flowJsonPath))) { + bufferedWriter.write(updatedJson); + bufferedWriter.flush(); + } + if (dssFlow.getRootFlow()) { + LOGGER.info("Modify root dssFlow {} json to dssFlow {} json.", dssFlow.getName(), targetOrchestratorName); + if (!targetOrchestratorName.equals(dssFlow.getName())){ + FileUtils.moveFile(new File(flowJsonPath), new File(flowInputPath + File.separator + targetOrchestratorName + ".json")); + } + FileUtils.moveDirectory(new File(flowInputPath), new File(sourceProjectDir + File.separator + targetOrchestratorName)); + } else { + LOGGER.info("Modify dssFlow {} json to dssFlow {} json.", dssFlow.getName(), dssFlow.getName() + "_" + nodeSuffix); + FileUtils.moveFile(new File(flowJsonPath), new File(flowInputPath + File.separator + dssFlow.getName() + "_" + nodeSuffix + ".json")); + FileUtils.moveDirectory(new File(flowInputPath), new File(sourceProjectDir + File.separator + dssFlow.getName() + "_" + nodeSuffix)); + } + } + //修改meta.txt并保存 + LOGGER.info("Modify dssFlows meta.txt"); + modifyFlowMeta(dssFlows, targetOrchestratorName, targetProjectId, nodeSuffix); + + List dssFlowRelations = metaInputService.inputFlowRelation(sourceProjectDir); + metaExportService.exportFlowBaseInfo(dssFlows, dssFlowRelations, sourceProjectDir); + + //2、导入Info信息(导入冲突处理) + List dssOrchestratorInfos = metaInputService.importOrchestrator(inputPath); + DSSOrchestratorInfo importDssOrchestratorInfo = dssOrchestratorInfos.get(0); + //修改orchestrator信息为target信息 + importDssOrchestratorInfo.setProjectId(targetProjectId); + importDssOrchestratorInfo.setUUID(UUID.randomUUID().toString()); + importDssOrchestratorInfo.setId(null); + importDssOrchestratorInfo.setCreator(userName); + importDssOrchestratorInfo.setCreateTime(new Date()); + importDssOrchestratorInfo.setName(targetOrchestratorName); + + //0.x工作流导入兼容 + if (importDssOrchestratorInfo.getWorkspaceId() == null) { + importDssOrchestratorInfo.setWorkspaceId(workspace.getWorkspaceId()); + } + if (StringUtils.isEmpty(importDssOrchestratorInfo.getOrchestratorMode())) { + importDssOrchestratorInfo.setOrchestratorMode("pom_work_flow"); + } + if (StringUtils.isEmpty(importDssOrchestratorInfo.getOrchestratorWay())) { + importDssOrchestratorInfo.setOrchestratorWay(",pom_work_flow_DAG,"); + } + + // rename orc_flow.zip from source projectName to target projectName. + FileUtils.delete(new File(flowZipPath)); + if (!projectName.equals(targetProjectName)) { + FileUtils.moveDirectory(new File(inputPath + File.separator + projectName), new File(inputPath + File.separator + targetProjectName)); + } + ZipHelper.zip(inputPath + File.separator + targetProjectName); + FileUtils.moveFile(new File(inputPath + File.separator + targetProjectName + ".zip"), new File(flowZipPath)); + + + //3、上传工作流zip包到bml + InputStream inputStream = bmlService.readLocalResourceFile(userName, flowZipPath); + BmlResource resultMap = bmlService.upload(userName, inputStream, importDssOrchestratorInfo.getName() + "_orc_flow.zip", targetProjectName); + + String orcResourceId = resultMap.getResourceId(); + String orcBmlVersion = resultMap.getVersion(); + + //4、导入版本Version信息 + DSSOrchestratorVersion dssOrchestratorVersion = new DSSOrchestratorVersion(); + dssOrchestratorVersion.setAppId(null); + dssOrchestratorVersion.setComment("orchestrator copy"); + dssOrchestratorVersion.setOrchestratorId(importDssOrchestratorInfo.getId()); + dssOrchestratorVersion.setContent(""); + dssOrchestratorVersion.setProjectId(targetProjectId); + dssOrchestratorVersion.setSource("Orchestrator create"); + dssOrchestratorVersion.setUpdater(userName); + + //生产导入:默认是为无效,开发环境为有效 + int valid = DSSOrchestratorConf.DSS_IMPORT_VALID_IMMEDIATELY.getValue() || DSSLabelUtil.isDevEnv(dssLabels) ? 1 : 0; + dssOrchestratorVersion.setValidFlag(valid); + + String oldVersion = orchestratorMapper.getLatestVersion(importDssOrchestratorInfo.getId(), 1); + if (StringUtils.isNotEmpty(oldVersion)) { + dssOrchestratorVersion.setVersion(oldVersion); + } else { + dssOrchestratorVersion.setVersion(OrchestratorUtils.generateNewVersion()); + } + + //5、生成上下文ContextId,所有信息都要转换成target信息 + String contextId = contextService.createContextID(workspace.getWorkspaceName(), targetProjectName, importDssOrchestratorInfo.getName(), dssOrchestratorVersion.getVersion(), userName); + dssOrchestratorVersion.setFormatContextId(contextId); + LOGGER.info("Create a new ContextId {} for copy a new orchestrator.", contextId); + + //6、导入第三方应用信息,如工作流、Visualis、Qualities + DSSOrchestrator dssOrchestrator = orchestratorManager.getOrCreateOrchestrator(userName, + workspace.getWorkspaceName(), importDssOrchestratorInfo.getType(), dssLabels); + RefJobContentResponseRef responseRef = OrchestrationDevelopmentOperationUtils.tryOrchestrationOperation(importDssOrchestratorInfo, + dssOrchestrator, userName, workspace, dssLabels, + DevelopmentIntegrationStandard::getRefImportService, + developmentService -> ((RefImportService) developmentService).getRefImportOperation(), + dssContextRequestRef -> dssContextRequestRef.setContextId(contextId), + projectRefRequestRef -> projectRefRequestRef.setRefProjectId(targetProjectId).setProjectName(targetProjectName), + (developmentOperation, developmentRequestRef) -> { + ImportRequestRef requestRef = (ImportRequestRef) developmentRequestRef; + requestRef.setResourceMap(MapUtils.newCommonMap(ImportRequestRef.RESOURCE_ID_KEY, orcResourceId, ImportRequestRef.RESOURCE_VERSION_KEY, orcBmlVersion)); + requestRef.setNewVersion(dssOrchestratorVersion.getVersion()); + return ((RefImportOperation) developmentOperation).importRef(requestRef); + }, "import"); + long orchestrationId = (Long) responseRef.getRefJobContent().get(OrchestratorRefConstant.ORCHESTRATION_ID_KEY); + String orchestrationContent = (String) responseRef.getRefJobContent().get(OrchestratorRefConstant.ORCHESTRATION_CONTENT_KEY); + + orchestratorMapper.addOrchestrator(importDssOrchestratorInfo); + //此处需要获取ID后再写入 + dssOrchestratorVersion.setOrchestratorId(importDssOrchestratorInfo.getId()); + dssOrchestratorVersion.setUpdateTime(new Date()); dssOrchestratorVersion.setAppId(orchestrationId); dssOrchestratorVersion.setContent(orchestrationContent); - orchestratorMapper.updateOrchestratorVersion(dssOrchestratorVersion); -// synProjectOrchestrator(importDssOrchestratorInfo, dssOrchestratorVersion, dssLabels); + orchestratorMapper.addOrchestratorVersion(dssOrchestratorVersion); + return dssOrchestratorVersion.getOrchestratorId(); } - public void synProjectOrchestrator(DSSOrchestratorInfo importDssOrchestratorInfo, DSSOrchestratorVersion dssOrchestratorVersion, List dssLabels) { - //Is dev environment - if (DSSLabelUtil.isDevEnv(dssLabels)) { - RequestProjectImportOrchestrator projectImportOrchestrator = new RequestProjectImportOrchestrator(); - BeanUtils.copyProperties(importDssOrchestratorInfo, projectImportOrchestrator); - projectImportOrchestrator.setVersionId(dssOrchestratorVersion.getId()); - //保存工程级别的编排模式 - DSSSenderServiceFactory.getOrCreateServiceInstance().getProjectServerSender() - .ask(projectImportOrchestrator); + private void modifyFlowMeta(List dssFlows, String targetOrchestratorName, Long targetProjectId, String flowNodeSuffix) { + for (DSSFlow dssFlow : dssFlows) { + dssFlow.setCreateTime(new Date()); + dssFlow.setProjectId(targetProjectId); + if (dssFlow.getChildren() != null) { + modifyFlowMeta((List) dssFlow.getChildren(), null, targetProjectId, flowNodeSuffix); + } + if (dssFlow.getRootFlow()) { + dssFlow.setName(targetOrchestratorName); + } else { + dssFlow.setName(dssFlow.getName() + "_" + flowNodeSuffix); + } } } } diff --git a/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/io/export/MetaExportService.java b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/io/export/MetaExportService.java index c35c817eb4..01afcd2b8a 100644 --- a/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/io/export/MetaExportService.java +++ b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/io/export/MetaExportService.java @@ -18,8 +18,11 @@ import com.webank.wedatasphere.dss.orchestrator.common.entity.DSSOrchestratorInfo; +import com.webank.wedatasphere.dss.workflow.common.entity.DSSFlow; +import com.webank.wedatasphere.dss.workflow.common.entity.DSSFlowRelation; import java.io.IOException; +import java.util.List; public interface MetaExportService { @@ -33,5 +36,8 @@ public interface MetaExportService { */ void export(DSSOrchestratorInfo dssOrchestratorInfo, String savePath) throws IOException; + void exportFlowBaseInfo(List allDSSFlows, List allFlowRelations, String savePath) throws IOException; + + } diff --git a/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/io/export/impl/MetaExportServiceImpl.java b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/io/export/impl/MetaExportServiceImpl.java index 929148fafb..fc6d20153d 100644 --- a/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/io/export/impl/MetaExportServiceImpl.java +++ b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/io/export/impl/MetaExportServiceImpl.java @@ -21,13 +21,16 @@ import com.webank.wedatasphere.dss.orchestrator.common.entity.DSSOrchestratorInfo; import com.webank.wedatasphere.dss.orchestrator.publish.io.export.MetaExportService; import com.webank.wedatasphere.dss.orchestrator.publish.io.export.MetaWriter; +import com.webank.wedatasphere.dss.workflow.common.entity.DSSFlow; +import com.webank.wedatasphere.dss.workflow.common.entity.DSSFlowRelation; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.stereotype.Service; import java.io.*; +import java.util.List; -@Service +@Service("orcMetaExportService") public class MetaExportServiceImpl implements MetaExportService { private Logger logger = LoggerFactory.getLogger(this.getClass()); @@ -46,6 +49,30 @@ public void export(DSSOrchestratorInfo dssOrchestratorInfo, String savePath) thr } } + @Override + public void exportFlowBaseInfo(List allDSSFlows, List allFlowRelations, String savePath) throws IOException { + + try ( + OutputStream outputStream = generateOutputStream(savePath) + ) { + exportFlowBaseInfo(allDSSFlows, outputStream); + exportFlowRelation(allFlowRelations, outputStream); + } + + } + + private void exportFlowBaseInfo(List DSSFlows, OutputStream outputStream) throws IOException { + + MetaWriter.of("dss_flow", DSSFlow.class).data(DSSFlows).write(outputStream); + + } + + private void exportFlowRelation(List flowRelations, OutputStream outputStream) throws IOException { + + MetaWriter.of("dss_workflow_relation", DSSFlowRelation.class).data(flowRelations).write(outputStream); + + } + private OutputStream generateOutputStream(String basePath) throws IOException { return IoUtils.generateExportOutputStream(basePath + File.separator + fileName); diff --git a/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/io/input/MetaInputService.java b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/io/input/MetaInputService.java index 230ad50b76..a672c6c4f4 100644 --- a/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/io/input/MetaInputService.java +++ b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/io/input/MetaInputService.java @@ -21,6 +21,8 @@ import com.webank.wedatasphere.dss.orchestrator.common.entity.DSSOrchestratorInfo; +import com.webank.wedatasphere.dss.workflow.common.entity.DSSFlow; +import com.webank.wedatasphere.dss.workflow.common.entity.DSSFlowRelation; import java.io.IOException; import java.util.List; @@ -31,4 +33,8 @@ public interface MetaInputService { List importOrchestrator(String basePath) throws IOException; + List inputFlow(String basePath) throws IOException; + + List inputFlowRelation(String basePath) throws IOException; + } diff --git a/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/io/input/MetaReader.java b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/io/input/MetaReader.java index 7b08541063..b24cef36f3 100644 --- a/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/io/input/MetaReader.java +++ b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/io/input/MetaReader.java @@ -81,12 +81,17 @@ private void readT() { body.stream().map(DSSExceptionUtils.map(this::lineToT)).forEach(datas::add); } - private T lineToT(List list) throws IllegalAccessException, InstantiationException, NoSuchFieldException, ParseException { + private T lineToT(List list) throws IllegalAccessException, InstantiationException, ParseException { T t = tClass.newInstance(); for (int i = 0; i < list.size(); i++) { String valueStr = list.get(i); if ("null".equalsIgnoreCase(valueStr)) continue; - Field declaredField = tClass.getDeclaredField(fields.get(i)); + Field declaredField; + try { + declaredField = tClass.getDeclaredField(fields.get(i)); + }catch ( java.lang.NoSuchFieldException e){ + continue; + } declaredField.setAccessible(true); Object value = null; String type = declaredField.getType().getSimpleName(); diff --git a/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/io/input/impl/MetaInputServiceImpl.java b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/io/input/impl/MetaInputServiceImpl.java index 52b9babbcc..2691959d89 100644 --- a/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/io/input/impl/MetaInputServiceImpl.java +++ b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/io/input/impl/MetaInputServiceImpl.java @@ -22,6 +22,8 @@ import com.webank.wedatasphere.dss.orchestrator.publish.io.input.MetaInputService; import com.webank.wedatasphere.dss.orchestrator.publish.io.input.MetaReader; +import com.webank.wedatasphere.dss.workflow.common.entity.DSSFlow; +import com.webank.wedatasphere.dss.workflow.common.entity.DSSFlowRelation; import org.springframework.stereotype.Service; import java.io.File; @@ -31,7 +33,7 @@ import java.util.List; -@Service +@Service("orcMetaInputService") public class MetaInputServiceImpl implements MetaInputService { // TODO: 2020/3/13 防止表结构发生改变的version 字段的添加 @@ -45,6 +47,19 @@ public List importOrchestrator(String basePath) throws IOEx } } + @Override + public List inputFlow(String basePath) throws IOException { + try (InputStream inputStream = generateInputstream(basePath)) { + return MetaReader.of("dss_flow", DSSFlow.class).read(inputStream); + } + } + + @Override + public List inputFlowRelation(String basePath) throws IOException { + try (InputStream inputStream = generateInputstream(basePath)) { + return MetaReader.of("dss_workflow_relation", DSSFlowRelation.class).read(inputStream); + } + } /** * 获取inputStream diff --git a/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/job/CommonUpdateConvertJobStatus.java b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/job/CommonUpdateConvertJobStatus.java new file mode 100644 index 0000000000..8780e725bd --- /dev/null +++ b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/job/CommonUpdateConvertJobStatus.java @@ -0,0 +1,43 @@ +package com.webank.wedatasphere.dss.orchestrator.publish.job; + +import com.webank.wedatasphere.dss.common.protocol.JobStatus; +import com.webank.wedatasphere.dss.orchestrator.common.entity.OrchestratorPublishJob; +import com.webank.wedatasphere.dss.orchestrator.db.dao.OrchestratorJobMapper; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.stereotype.Component; + +import java.util.Date; + +@Component +public class CommonUpdateConvertJobStatus { + + private static final Logger LOGGER = LoggerFactory.getLogger(CommonUpdateConvertJobStatus.class); + + @Autowired + private OrchestratorJobMapper orchestratorJobMapper; + + public void updateConvertJobStatus(OrchestratorPublishJob orchestratorPublishJob) { + LOGGER.info("Update convert orchestrator job status to {}", orchestratorPublishJob.getStatus()); + orchestratorPublishJob.setUpdateTime(new Date(System.currentTimeMillis())); + orchestratorJobMapper.updatePublishJob(orchestratorPublishJob); + } + + public void toRunningStatus(OrchestratorPublishJob orchestratorPublishJob) { + orchestratorPublishJob.setStatus(JobStatus.Running.getStatus()); + updateConvertJobStatus(orchestratorPublishJob); + } + + public void toSuccessStatus(OrchestratorPublishJob orchestratorPublishJob) { + orchestratorPublishJob.setStatus(JobStatus.Success.getStatus()); + updateConvertJobStatus(orchestratorPublishJob); + } + + public void toFailedStatus(OrchestratorPublishJob orchestratorPublishJob, String errorMsg) { + orchestratorPublishJob.setStatus(JobStatus.Failed.getStatus()); + orchestratorPublishJob.setErrorMsg(errorMsg); + updateConvertJobStatus(orchestratorPublishJob); + } + +} diff --git a/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/job/OrchestratorConversionJob.java b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/job/OrchestratorConversionJob.java index 3648942746..a7ebf544f9 100644 --- a/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/job/OrchestratorConversionJob.java +++ b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/job/OrchestratorConversionJob.java @@ -19,6 +19,8 @@ import com.webank.wedatasphere.dss.common.entity.project.DSSProject; import com.webank.wedatasphere.dss.common.exception.DSSErrorException; import com.webank.wedatasphere.dss.common.protocol.project.ProjectInfoRequest; +import com.webank.wedatasphere.dss.common.utils.RpcAskUtils; +import com.webank.wedatasphere.dss.orchestrator.common.entity.OrchestratorPublishJob; import com.webank.wedatasphere.dss.orchestrator.common.protocol.ResponseOperateOrchestrator; import com.webank.wedatasphere.dss.orchestrator.core.plugin.DSSOrchestratorPlugin; import com.webank.wedatasphere.dss.orchestrator.publish.ConversionDSSOrchestratorPlugin; @@ -42,6 +44,17 @@ public final class OrchestratorConversionJob implements Runnable { private ConversionJobEntity conversionJobEntity; private List conversionDSSOrchestratorPlugins; private Consumer consumer; + private CommonUpdateConvertJobStatus commonUpdateConvertJobStatus; + + private OrchestratorPublishJob orchestratorPublishJob; + + public OrchestratorPublishJob getOrchestratorPublishJob() { + return orchestratorPublishJob; + } + + public void setOrchestratorPublishJob(OrchestratorPublishJob orchestratorPublishJob) { + this.orchestratorPublishJob = orchestratorPublishJob; + } public String getId() { return id; @@ -67,6 +80,14 @@ public void afterConversion(Consumer consumer) { this.consumer = consumer; } + public CommonUpdateConvertJobStatus getCommonUpdateConvertJobStatus() { + return commonUpdateConvertJobStatus; + } + + public void setCommonUpdateConvertJobStatus(CommonUpdateConvertJobStatus commonUpdateConvertJobStatus) { + this.commonUpdateConvertJobStatus = commonUpdateConvertJobStatus; + } + @Override public void run() { //1.从编排中心导出一次工作流,进行一次版本升级 @@ -74,6 +95,7 @@ public void run() { LOGGER.info("Job {} begin to convert project {} for user {} to scheduler, the orchestrationIds is {}.", id, conversionJobEntity.getProject().getId(), conversionJobEntity.getUserName(), conversionJobEntity.getOrchestrationIdMap().keySet()); conversionJobEntity.setResponse(ResponseOperateOrchestrator.running()); + this.commonUpdateConvertJobStatus.toRunningStatus(this.orchestratorPublishJob); ConversionDSSOrchestratorPlugin conversionDSSOrchestratorPlugin = null; for (DSSOrchestratorPlugin plugin: conversionDSSOrchestratorPlugins) { if(plugin instanceof ConversionDSSOrchestratorPlugin) { @@ -83,24 +105,27 @@ public void run() { ProjectInfoRequest projectInfoRequest = new ProjectInfoRequest(); projectInfoRequest.setProjectId(conversionJobEntity.getProject().getId()); try{ - DSSProject project = (DSSProject) DSSSenderServiceFactory.getOrCreateServiceInstance().getProjectServerSender() - .ask(projectInfoRequest); + DSSProject project = RpcAskUtils.processAskException(DSSSenderServiceFactory.getOrCreateServiceInstance().getProjectServerSender() + .ask(projectInfoRequest), DSSProject.class, ProjectInfoRequest.class); conversionJobEntity.setProject(project); Workspace workspace = conversionJobEntity.getWorkspace(); ResponseOperateOrchestrator response = conversionDSSOrchestratorPlugin.convert(conversionJobEntity.getUserName(), project, workspace, - conversionJobEntity.getOrchestrationIdMap(), conversionJobEntity.getLabels()); + conversionJobEntity.getOrchestrationIdMap(), conversionJobEntity.getLabels(),null); if(response.isFailed()) { String msg = response.getMessage() == null ? "Unknown reason, please ask admin for help!" : response.getMessage(); throw new DSSErrorException(50000, msg); } //3.如果都没有报错,那么默认任务应该是成功的,那么则将所有的状态进行置为完成 consumer.accept(response); + LOGGER.info("{} completed with status {}.", getId(), response.getJobStatus()); conversionJobEntity.setResponse(response); + this.commonUpdateConvertJobStatus.toSuccessStatus(this.orchestratorPublishJob); } catch (final Exception t){ LOGGER.error("Job {} convert for project {} failed.", id, conversionJobEntity.getProject().getId(), t); ResponseOperateOrchestrator response = ResponseOperateOrchestrator.failed(ExceptionUtils.getRootCauseMessage(t)); conversionJobEntity.setResponse(response); consumer.accept(response); + this.commonUpdateConvertJobStatus.toFailedStatus(this.orchestratorPublishJob, response.getMessage()); } LOGGER.info("Job {} convert project {} for user {} to Orchestrator {}, costs {}.", id, conversionJobEntity.getProject().getId(), conversionJobEntity.getUserName(), conversionJobEntity.getResponse().getJobStatus(), ByteTimeUtils.msDurationToString(conversionJobEntity.getUpdateTime().getTime() - conversionJobEntity.getCreateTime().getTime())); diff --git a/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/utils/OrchestrationDevelopmentOperationUtils.java b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/utils/OrchestrationDevelopmentOperationUtils.java index f9e3f32209..7136c48f7d 100644 --- a/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/utils/OrchestrationDevelopmentOperationUtils.java +++ b/dss-framework/framework-plugins/dss-framework-orchestrator-publish/src/main/java/com/webank/wedatasphere/dss/orchestrator/publish/utils/OrchestrationDevelopmentOperationUtils.java @@ -23,12 +23,29 @@ import java.util.function.Function; /** + * 编排相关operation执行工具 * @author enjoyyin * @date 2022-03-12 * @since 0.5.0 */ public class OrchestrationDevelopmentOperationUtils { - + /** + * 执行编排相关的operation + * @param dssOrchestratorInfo 编排信息 + * @param dssOrchestrator 编排系统资源,作为操作的执行者。 + * @param userName 用户名 + * @param workspace 工作空间 + * @param dssLabels label,比如环境label + * @param getDevelopmentService DevelopmentService提供器,比如可以提供一个WorkflowExportService用于最终提供工作流导出服务 + * @param getDevelopmentOperation Operation提供器,比如可以提供一个WorkflowRefExportOperation用于工作流导出操作 + * @param contextRequestRefConsumer requestRef的context处理器,用以对DSSContextRequestRef的requestRef做处理 + * @param projectRefRequestRefConsumer requestRef的project处理器,用以对ProjectRefRequestRef的requestRef做处理 + * @param responseRefConsumer 上游提供的operation执行器,用以operation执行以及对responseRef的处理 + * @param operationName 执行的操作名 + * @return 得到的操作结果ResponseRef + * @param 操作入参RequestRef具体类型 + * @param 操作结果ResponseRef的具体类型 + */ public static V tryOrchestrationOperation(DSSOrchestratorInfo dssOrchestratorInfo, DSSOrchestrator dssOrchestrator, String userName, @@ -40,10 +57,24 @@ public static V tryOrch Consumer projectRefRequestRefConsumer, BiFunction responseRefConsumer, String operationName) throws DSSErrorException { + + //第一步,获取编排实现的APPInstance(比如工作流服务)和对应的DevelopmentIntegrationStandard ImmutablePair standMap = OrchestratorLoaderUtils.getOrchestratorDevelopmentStandard(dssOrchestrator, dssLabels); - return DevelopmentOperationUtils.tryDevelopmentRequestRefOperation(() -> getDevelopmentService.apply(standMap.getValue(), standMap.getKey()), - getDevelopmentOperation, contextRequestRefConsumer, projectRefRequestRefConsumer, + //第二步,执行这个APPInstance上实现的DevelopmentIntegrationStandard。 + //至于要执行DevelopmentIntegrationStandard里的什么operation,就需要根据外部传入的getDevelopmentService和getDevelopmentOperation决定 + return DevelopmentOperationUtils.tryDevelopmentRequestRefOperation( + //目的是让下游得到DevelopmentService。 + // 根据不同的AppInstance,getDevelopmentService可以返回不同是实现类,比如是一个WorkflowExportService。 + // 因此下游就会得到一个具体的DevelopmentService,比如一个WorkflowExportService,这样下游便可以执行工作流导出了。 + () -> getDevelopmentService.apply(standMap.getValue(), standMap.getKey()), + //告诉下游如何从DevelopmentService得到operation,比如从从RefExportOperation::getRefExportOperation,得到WorkflowRefExportOperation + getDevelopmentOperation, + //给下游一个context处理器,对DSSContextRequestRef类型的requestRef做处理 + contextRequestRefConsumer, + //给下游一个project处理器,对ProjectRefRequestRef类型的requestRef做处理 + projectRefRequestRefConsumer, + //给下游一个operation的执行器,执行operation(最终也是通过上游提供的responseRefConsumer去执行operation) (developmentOperation, developmentRequestRef) -> { developmentRequestRef.setWorkspace(workspace).setUserName(userName).setDSSLabels(dssLabels).setType(dssOrchestratorInfo.getType()); return responseRefConsumer.apply(developmentOperation, (K) developmentRequestRef); diff --git a/dss-framework/pom.xml b/dss-framework/pom.xml index 902ad17ba3..d6676d01f4 100644 --- a/dss-framework/pom.xml +++ b/dss-framework/pom.xml @@ -21,7 +21,8 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 + ../pom.xml 4.0.0 @@ -37,6 +38,7 @@ dss-framework-project-server dss-framework-admin-service framework-plugins/dss-framework-migrate-server + dss-framework-proxy-user-service \ No newline at end of file diff --git a/dss-orchestrator/dss-orchestrator-common/pom.xml b/dss-orchestrator/dss-orchestrator-common/pom.xml index 4e31995b57..533719303a 100644 --- a/dss-orchestrator/dss-orchestrator-common/pom.xml +++ b/dss-orchestrator/dss-orchestrator-common/pom.xml @@ -21,7 +21,8 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 + ../../pom.xml 4.0.0 diff --git a/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/entity/DSSOrchestratorCopyInfo.java b/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/entity/DSSOrchestratorCopyInfo.java new file mode 100644 index 0000000000..a6866b873d --- /dev/null +++ b/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/entity/DSSOrchestratorCopyInfo.java @@ -0,0 +1,257 @@ +package com.webank.wedatasphere.dss.orchestrator.common.entity; + +import java.util.Date; +import java.util.List; + +public class DSSOrchestratorCopyInfo { + + private String id; + + private String username; + + private String type; + + private Long workspaceId; + + private Long sourceOrchestratorId; + + private String sourceOrchestratorName; + + private String targetOrchestratorName; + + private String sourceProjectName; + + private String targetProjectName; + + /** + * 目标工作流节点后缀 + */ + private String workflowNodeSuffix; + + private String microserverName; + + private String exceptionInfo; + + /** + * 复制任务最终状态,0失败,1成功 + */ + private Integer status; + + private String instanceName; + + /** + * 当前编排是否在被复制,1有,0没有 + */ + private Integer isCopying; + + /** + * 复制成功的节点 + */ + private List successNode; + + /** + * 复制开始时间 + */ + private Date startTime; + + /** + * 复制结束时间 + */ + private Date endTime; + + public DSSOrchestratorCopyInfo() { + } + + public DSSOrchestratorCopyInfo(String id) { + this.id = id; + } + + public DSSOrchestratorCopyInfo(String id, String username, String type, Long workspaceId, Long sourceOrchestratorId, + String sourceOrchestratorName, String targetOrchestratorName, + String sourceProjectName, String targetProjectName, String workflowNodeSuffix, + String microserverName, Integer isCopying, Date startTime, Date endTime, String instanceName) { + this.id = id; + this.username = username; + this.type = type; + this.workspaceId = workspaceId; + this.sourceOrchestratorId = sourceOrchestratorId; + this.sourceOrchestratorName = sourceOrchestratorName; + this.targetOrchestratorName = targetOrchestratorName; + this.sourceProjectName = sourceProjectName; + this.targetProjectName = targetProjectName; + this.workflowNodeSuffix = workflowNodeSuffix; + this.microserverName = microserverName; + this.isCopying = isCopying; + this.startTime = startTime; + this.endTime = endTime; + this.instanceName = instanceName; + } + + public String getId() { + return id; + } + + public void setId(String id) { + this.id = id; + } + + public String getUsername() { + return username; + } + + public void setUsername(String username) { + this.username = username; + } + + public String getType() { + return type; + } + + public void setType(String type) { + this.type = type; + } + + public Long getWorkspaceId() { + return workspaceId; + } + + public void setWorkspaceId(Long workspaceId) { + this.workspaceId = workspaceId; + } + + public Long getSourceOrchestratorId() { + return sourceOrchestratorId; + } + + public void setSourceOrchestratorId(Long sourceOrchestratorId) { + this.sourceOrchestratorId = sourceOrchestratorId; + } + + public String getSourceOrchestratorName() { + return sourceOrchestratorName; + } + + public void setSourceOrchestratorName(String sourceOrchestratorName) { + this.sourceOrchestratorName = sourceOrchestratorName; + } + + public String getTargetOrchestratorName() { + return targetOrchestratorName; + } + + public void setTargetOrchestratorName(String targetOrchestratorName) { + this.targetOrchestratorName = targetOrchestratorName; + } + + public String getSourceProjectName() { + return sourceProjectName; + } + + public void setSourceProjectName(String sourceProjectName) { + this.sourceProjectName = sourceProjectName; + } + + public String getTargetProjectName() { + return targetProjectName; + } + + public void setTargetProjectName(String targetProjectName) { + this.targetProjectName = targetProjectName; + } + + public String getWorkflowNodeSuffix() { + return workflowNodeSuffix; + } + + public void setWorkflowNodeSuffix(String workflowNodeSuffix) { + this.workflowNodeSuffix = workflowNodeSuffix; + } + + public String getMicroserverName() { + return microserverName; + } + + public void setMicroserverName(String microserverName) { + this.microserverName = microserverName; + } + + public String getExceptionInfo() { + return exceptionInfo; + } + + public void setExceptionInfo(String exceptionInfo) { + this.exceptionInfo = exceptionInfo; + } + + public Integer getStatus() { + return status; + } + + public void setStatus(Integer status) { + this.status = status; + } + + public String getInstanceName() { + return instanceName; + } + + public void setInstanceName(String instanceName) { + this.instanceName = instanceName; + } + + public Integer getIsCopying() { + return isCopying; + } + + public void setIsCopying(Integer isCopying) { + this.isCopying = isCopying; + } + + public String getSuccessNode() { + return successNode == null ? null : String.join(",", successNode); + } + + public void setSuccessNode(List successNode) { + this.successNode = successNode; + } + + public Date getStartTime() { + return startTime; + } + + public void setStartTime(Date startTime) { + this.startTime = startTime; + } + + public Date getEndTime() { + return endTime; + } + + public void setEndTime(Date endTime) { + this.endTime = endTime; + } + + @Override + public String toString() { + return "DSSOrchestratorCopyInfo{" + + "id='" + id + '\'' + + ", username='" + username + '\'' + + ", type='" + type + '\'' + + ", workspaceId=" + workspaceId + + ", sourceOrchestratorId=" + sourceOrchestratorId + + ", sourceOrchestratorName='" + sourceOrchestratorName + '\'' + + ", targetOrchestratorName='" + targetOrchestratorName + '\'' + + ", sourceProjectName='" + sourceProjectName + '\'' + + ", targetProjectName='" + targetProjectName + '\'' + + ", workflowNodeSuffix='" + workflowNodeSuffix + '\'' + + ", microserverName='" + microserverName + '\'' + + ", exceptionInfo='" + exceptionInfo + '\'' + + ", status='" + status + '\'' + + ", instanceName='" + instanceName + '\'' + + ", isCopying=" + isCopying + + ", successNode=" + successNode + + ", startTime=" + startTime + + ", endTime=" + endTime + + '}'; + } +} diff --git a/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/entity/DSSOrchestratorInfo.java b/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/entity/DSSOrchestratorInfo.java index 97233b66e7..2035ae9124 100644 --- a/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/entity/DSSOrchestratorInfo.java +++ b/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/entity/DSSOrchestratorInfo.java @@ -19,40 +19,71 @@ import java.util.Date; import java.util.List; +/** + * 一个具体的编排实例 + */ public class DSSOrchestratorInfo implements DSSOrchestration { private Long id; - + /** + * 编排名 + */ private String name; - + /** + * 编排具体的实现类型,比如工作流workflow + */ private String type; - + /** + * 编排描述 + */ private String desc; - + /** + * 创建者 + */ private String creator; - + /** + * 创建时间 + */ private Date createTime; - + /** + * 用途标签,作为描述信息 + */ private String uses; - + /** + * 可以支持的实现的appconn节点 + */ private String appConnName; - + /** + * 所属工程 + */ private Long projectId; - + /** + * 工作流的唯一id,不同环境里,同一个工作流保持一致,用来判断是否是同一个工作流。 + */ private String uuid; - + /** + * 实现的二级类型,比如workflow_DAG + */ private String secondaryType; - + /** + * 所属工作空间 + */ private Long workspaceId; private String orchestratorMode; private String orchestratorWay; - + /** + * 编排的重要程度,调度会考察这个重要程度 + */ private String orchestratorLevel; - + /** + * 更新人 + */ private String updateUser; - + /** + * 更新时间 + */ private Date updateTime; public DSSOrchestratorInfo() { diff --git a/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/entity/DSSOrchestratorVersion.java b/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/entity/DSSOrchestratorVersion.java index a27b26d402..1175731d95 100644 --- a/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/entity/DSSOrchestratorVersion.java +++ b/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/entity/DSSOrchestratorVersion.java @@ -28,17 +28,49 @@ public class DSSOrchestratorVersion { private Long id; + /** + * 编排id + */ private Long orchestratorId; + /** + * 对应的实现的id,比如workflow的id。每个版本{@link #getVersion()}对应的实现可以不一样。 + */ private Long appId; + /** + * 所属项目id + */ private Long projectId; + /** + * 创建的来源。比如新建、回滚。 用于记录信息description用,一般不消费。 + */ private String source; + /** + * 编排的版本哈 + */ private String version; + /** + * 备注 + */ private String comment; + /** + * 更新时间 + */ private Date updateTime; + /** + * 更新人 + */ private String updater; + /** + * 废弃字段。 具体实现的内容已经放到实现模块了。比如工作流在工作流模块,通过关联获取 + */ private String content; + /** + * 这个编排版本所对应的contextId + */ private String contextId; - //有效标示 0:无效;1:有效,默认是有效 + /** + * 有效标示 0:无效;1:有效,默认是有效 + */ private Integer validFlag; diff --git a/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/entity/OrchestratorPublishJob.java b/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/entity/OrchestratorPublishJob.java new file mode 100644 index 0000000000..d852b45b7d --- /dev/null +++ b/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/entity/OrchestratorPublishJob.java @@ -0,0 +1,81 @@ +package com.webank.wedatasphere.dss.orchestrator.common.entity; + +import java.util.Date; + +public class OrchestratorPublishJob { + + private Long id; + private String jobId; + private String conversionJobJson; + private Date createTime; + private Date updateTime; + private String instanceName; + private String status; + private String errorMsg; + + public String getInstanceName() { + return instanceName; + } + + public void setInstanceName(String instanceName) { + this.instanceName = instanceName; + } + + public String getStatus() { + return status; + } + + public void setStatus(String status) { + this.status = status; + } + + public Long getId() { + return id; + } + + public void setId(Long id) { + this.id = id; + } + + public String getJobId() { + return jobId; + } + + public OrchestratorPublishJob setJobId(String jobId) { + this.jobId = jobId; + return this; + } + + public String getConversionJobJson() { + return conversionJobJson; + } + + public OrchestratorPublishJob setConversionJobJson(String conversionJobJson) { + this.conversionJobJson = conversionJobJson; + return this; + } + + public Date getCreateTime() { + return createTime; + } + + public void setCreateTime(Date createTime) { + this.createTime = createTime; + } + + public Date getUpdateTime() { + return updateTime; + } + + public void setUpdateTime(Date updateTime) { + this.updateTime = updateTime; + } + + public String getErrorMsg() { + return errorMsg; + } + + public void setErrorMsg(String errorMsg) { + this.errorMsg = errorMsg; + } +} diff --git a/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/protocol/RequestConvertOrchestrations.java b/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/protocol/RequestConvertOrchestrations.java index 50ad403c0a..13627901ec 100644 --- a/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/protocol/RequestConvertOrchestrations.java +++ b/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/protocol/RequestConvertOrchestrations.java @@ -32,6 +32,7 @@ public class RequestConvertOrchestrations { private Project project; private DSSWorkspace workspace; private List dssLabels; + private String approvalId; public Map getOrchestrationIdMap() { return orchestrationIdMap; @@ -72,4 +73,12 @@ public List getDSSLabels() { public void setDSSLabels(List dssLabels) { this.dssLabels = dssLabels; } + + public String getApprovalId() { + return approvalId; + } + + public void setApprovalId(String approvalId) { + this.approvalId = approvalId; + } } diff --git a/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/protocol/RequestFrameworkConvertOrchestration.java b/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/protocol/RequestFrameworkConvertOrchestration.java index 310c283c9a..19e0c2d57e 100644 --- a/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/protocol/RequestFrameworkConvertOrchestration.java +++ b/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/protocol/RequestFrameworkConvertOrchestration.java @@ -20,14 +20,36 @@ import java.util.List; import java.util.Map; - +/** + * 编排转化为调度工作流(发布)的请求类。 + * 可以发布单个工作流,多个编排,也可以是发布整个工程。 + */ public class RequestFrameworkConvertOrchestration { private String userName; private DSSWorkspace workspace; + /** + * 可以指定一个编排实现的id,比如工作流的id,来发布这个编排 + */ private Long orcAppId; + /** + * 也可以直接指定要发布的编排列表 + */ private List orcIds; + /** + * 工程id。如果工程id不为空,则表示要发布这个工程下所有未发布过以及已发布过 + */ + private Long projectId; private Map labels; + /** + * 关联的审批单号 + */ + private String approveId; + + /** + * 是否要发布工程下所有的编排。 + * 如果为true,那么不仅仅会发布{@link #orcIds }或者{@link #orcAppId}指定的编排,还会发布对应工程下其他所有的编排 + */ private boolean convertAllOrcs; private String comment; @@ -62,6 +84,21 @@ public List getOrcIds() { public void setOrcIds(List orcIds) { this.orcIds = orcIds; } + public String getApproveId() { + return approveId; + } + + public void setApproveId(String approveId) { + this.approveId = approveId; + } + + public Long getProjectId() { + return projectId; + } + + public void setProjectId(Long projectId) { + this.projectId = projectId; + } public Map getLabels() { return labels; diff --git a/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/protocol/RequestImportOrchestrator.java b/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/protocol/RequestImportOrchestrator.java index 0784ee16e7..9a64d3eaf5 100644 --- a/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/protocol/RequestImportOrchestrator.java +++ b/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/protocol/RequestImportOrchestrator.java @@ -54,6 +54,27 @@ public RequestImportOrchestrator(String userName, this.orchestratorName = orchestratorName; } + public RequestImportOrchestrator(String userName, + String projectName, + Long projectId, + String resourceId, + String bmlVersion, + String orchestratorName, + List dssLabels, + Workspace workspace, + Long copyProjectId, + String copyProjectName) { + this.userName = userName; + this.projectName = projectName; + this.projectId = projectId; + this.resourceId = resourceId; + this.bmlVersion = bmlVersion; + this.orchestratorName = orchestratorName; + this.dssLabels = dssLabels; + this.workspace = workspace; + this.copyProjectId = copyProjectId; + this.copyProjectName = copyProjectName; + } public String getUserName() { return userName; diff --git a/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/protocol/ResponseOperateOrchestrator.java b/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/protocol/ResponseOperateOrchestrator.java index 6cc64d2305..b483c683ff 100644 --- a/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/protocol/ResponseOperateOrchestrator.java +++ b/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/protocol/ResponseOperateOrchestrator.java @@ -71,12 +71,6 @@ public static ResponseOperateOrchestrator success(String message) { return response; } - public static ResponseOperateOrchestrator failed() { - ResponseOperateOrchestrator response = new ResponseOperateOrchestrator(); - response.setJobStatus(JobStatus.Failed); - return response; - } - public static ResponseOperateOrchestrator failed(String message) { ResponseOperateOrchestrator response = new ResponseOperateOrchestrator(); response.setJobStatus(JobStatus.Failed); diff --git a/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/ref/OrchestratorRefConstant.java b/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/ref/OrchestratorRefConstant.java index 1b1f7df8f5..d6470db0f8 100644 --- a/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/ref/OrchestratorRefConstant.java +++ b/dss-orchestrator/dss-orchestrator-common/src/main/java/com/webank/wedatasphere/dss/orchestrator/common/ref/OrchestratorRefConstant.java @@ -30,4 +30,7 @@ public interface OrchestratorRefConstant { String ORCHESTRATION_USES = "orchestrationUses"; String ORCHESTRATION_SCHEDULER_APP_CONN = "schedulerAppConnName"; + //workflow node suffix input by the user during workflow replication. + String ORCHESTRATION_NODE_SUFFIX = "nodeSuffix"; + } diff --git a/dss-orchestrator/dss-orchestrator-conversion-standard/pom.xml b/dss-orchestrator/dss-orchestrator-conversion-standard/pom.xml index 012de0c443..d00dd765c4 100644 --- a/dss-orchestrator/dss-orchestrator-conversion-standard/pom.xml +++ b/dss-orchestrator/dss-orchestrator-conversion-standard/pom.xml @@ -21,7 +21,8 @@ dss-orchestrator com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 + ../pom.xml 4.0.0 diff --git a/dss-orchestrator/dss-orchestrator-conversion-standard/src/main/java/com/webank/wedatasphere/dss/orchestrator/converter/standard/operation/ConversionOperation.java b/dss-orchestrator/dss-orchestrator-conversion-standard/src/main/java/com/webank/wedatasphere/dss/orchestrator/converter/standard/operation/ConversionOperation.java index 5d3ebbbd23..386189c3c0 100644 --- a/dss-orchestrator/dss-orchestrator-conversion-standard/src/main/java/com/webank/wedatasphere/dss/orchestrator/converter/standard/operation/ConversionOperation.java +++ b/dss-orchestrator/dss-orchestrator-conversion-standard/src/main/java/com/webank/wedatasphere/dss/orchestrator/converter/standard/operation/ConversionOperation.java @@ -26,6 +26,11 @@ public interface ConversionOperation implements ProjectToRelConversionRequestRef {} diff --git a/dss-orchestrator/dss-orchestrator-core/pom.xml b/dss-orchestrator/dss-orchestrator-core/pom.xml index 64cf710808..22ed8a0fa6 100644 --- a/dss-orchestrator/dss-orchestrator-core/pom.xml +++ b/dss-orchestrator/dss-orchestrator-core/pom.xml @@ -21,7 +21,8 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 + ../../pom.xml 4.0.0 diff --git a/dss-orchestrator/dss-orchestrator-core/src/main/java/com/webank/wedatasphere/dss/orchestrator/core/DSSOrchestrator.java b/dss-orchestrator/dss-orchestrator-core/src/main/java/com/webank/wedatasphere/dss/orchestrator/core/DSSOrchestrator.java index 2060913a47..d323658466 100644 --- a/dss-orchestrator/dss-orchestrator-core/src/main/java/com/webank/wedatasphere/dss/orchestrator/core/DSSOrchestrator.java +++ b/dss-orchestrator/dss-orchestrator-core/src/main/java/com/webank/wedatasphere/dss/orchestrator/core/DSSOrchestrator.java @@ -22,7 +22,10 @@ import java.util.List; - +/** + * 一个编排抽象,代表了一类编排。包括了编排的类型、关联的appconn、支持的工具条类型等信息。 + * 注意,编排并不是工作流,因此不会包含任何的节点内容 + */ public interface DSSOrchestrator { /** diff --git a/dss-orchestrator/dss-orchestrator-core/src/main/java/com/webank/wedatasphere/dss/orchestrator/core/type/DSSOrchestratorRelationManager.java b/dss-orchestrator/dss-orchestrator-core/src/main/java/com/webank/wedatasphere/dss/orchestrator/core/type/DSSOrchestratorRelationManager.java index 32e796e544..dcacaa83cc 100644 --- a/dss-orchestrator/dss-orchestrator-core/src/main/java/com/webank/wedatasphere/dss/orchestrator/core/type/DSSOrchestratorRelationManager.java +++ b/dss-orchestrator/dss-orchestrator-core/src/main/java/com/webank/wedatasphere/dss/orchestrator/core/type/DSSOrchestratorRelationManager.java @@ -29,8 +29,8 @@ public class DSSOrchestratorRelationManager { int index = 1; while(true) { String key = "wds.dss.orchestrator.new." + index + "th"; - if(BDPConfiguration.contains(key)) { - String relation = BDPConfiguration.get(key); + if(BDPConfiguration.contains(key,false)) { + String relation = BDPConfiguration.get(key,false); OrchestratorJsonRelation jsonRelation = new OrchestratorJsonRelation(relation); jsonRelation.init(); LOGGER.info("loaded DSSOrchestratorRelation is {}.", relation); diff --git a/dss-orchestrator/dss-orchestrator-core/src/main/java/com/webank/wedatasphere/dss/orchestrator/core/utils/OrchestratorUtils.java b/dss-orchestrator/dss-orchestrator-core/src/main/java/com/webank/wedatasphere/dss/orchestrator/core/utils/OrchestratorUtils.java index c239e52360..cc1caa8d5a 100644 --- a/dss-orchestrator/dss-orchestrator-core/src/main/java/com/webank/wedatasphere/dss/orchestrator/core/utils/OrchestratorUtils.java +++ b/dss-orchestrator/dss-orchestrator-core/src/main/java/com/webank/wedatasphere/dss/orchestrator/core/utils/OrchestratorUtils.java @@ -33,6 +33,10 @@ public static String generateNewVersion() { return "v000001"; } + public static String generateNewCopyVersion(String suffix) { + return suffix + "_v000001"; + } + /** * 注意: flow版本更新需要同步更新ContextID * @@ -40,7 +44,17 @@ public static String generateNewVersion() { * @return */ public static String increaseVersion(String oldVersion) { - int num = Integer.parseInt(oldVersion.substring(1)) + 1; + if (oldVersion.length() <= 7) { + return newNormalVersion(oldVersion); + } else { + int i = oldVersion.lastIndexOf("_"); + String newVer = newNormalVersion(oldVersion.substring(i + 1)); + return oldVersion.substring(0, i + 1) + newVer; + } + } + + static String newNormalVersion(String oldVersion) { + long num = Long.parseLong(oldVersion.substring(1)) + 1; String tmp = "00000" + num; return "v" + tmp.substring(tmp.length() - 6); } @@ -53,10 +67,10 @@ public static String getModeStr(List strList) { return strList.stream().map(String::trim).filter((s) -> StringUtils.isNotBlank(s)).distinct().collect(Collectors.joining(MODE_SPLIT, MODE_SPLIT, MODE_SPLIT)); } - public static List convertList(String str){ - if(StringUtils.isEmpty(str)){ + public static List convertList(String str) { + if (StringUtils.isEmpty(str)) { return new ArrayList<>(); } - return Arrays.stream(str.split(MODE_SPLIT)).map(String::trim).filter((s)-> StringUtils.isNotBlank(s)).distinct().collect(Collectors.toList()); + return Arrays.stream(str.split(MODE_SPLIT)).map(String::trim).filter((s) -> StringUtils.isNotBlank(s)).distinct().collect(Collectors.toList()); } } diff --git a/dss-orchestrator/dss-orchestrator-core/src/main/scala/com/webank/wedatasphere/dss/orchestrator/core/service/BMLService.scala b/dss-orchestrator/dss-orchestrator-core/src/main/scala/com/webank/wedatasphere/dss/orchestrator/core/service/BMLService.scala deleted file mode 100644 index 133115a180..0000000000 --- a/dss-orchestrator/dss-orchestrator-core/src/main/scala/com/webank/wedatasphere/dss/orchestrator/core/service/BMLService.scala +++ /dev/null @@ -1,183 +0,0 @@ -/* - * Copyright 2019 WeBank - * 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. - * - */ - -package com.webank.wedatasphere.dss.orchestrator.core.service - -import java.io.{ByteArrayInputStream, InputStream} -import java.util -import java.util.UUID - -import com.webank.wedatasphere.dss.common.exception.DSSErrorException -import com.webank.wedatasphere.dss.common.utils.IoUtils -import org.apache.linkis.bml.client.{BmlClient, BmlClientFactory} -import org.apache.linkis.bml.protocol.{BmlDownloadResponse, BmlUpdateResponse, BmlUploadResponse} -import org.apache.linkis.common.utils.{JavaLog, Utils} -import org.apache.commons.io.IOUtils -import org.springframework.stereotype.Component - -import scala.collection.JavaConversions._ - - -@Component -class BMLService extends JavaLog{ - - def upload(userName: String, content: String, fileName: String, projectName:String): util.Map[String, Object] = { - val inputStream = new ByteArrayInputStream(content.getBytes("utf-8")) - val client: BmlClient = createBMLClient(userName) - val resource: BmlUploadResponse = client.uploadShareResource(userName, projectName, fileName, inputStream) - if (!resource.isSuccess) throw new DSSErrorException(911113, "上传失败") - val map = new util.HashMap[String, Object] - map += "resourceId" -> resource.resourceId - map += "version" -> resource.version - } - - def upload(userName: String, inputStream: InputStream, fileName: String, projectName:String): util.Map[String, Object] = { - val client: BmlClient = createBMLClient(userName) - val resource: BmlUploadResponse = client.uploadShareResource(userName, projectName, fileName, inputStream) - if (!resource.isSuccess) throw new DSSErrorException(911113, "上传失败") - val map = new util.HashMap[String, Object] - map += "resourceId" -> resource.resourceId - map += "version" -> resource.version - } - - def update(userName: String, resourceId: String, inputStream: InputStream): util.Map[String, Object] = { - val client: BmlClient = createBMLClient(userName) - val resource: BmlUpdateResponse = client.updateShareResource(userName, resourceId, "", inputStream) - if (!resource.isSuccess) throw new DSSErrorException(911114, "更新失败") - val map = new util.HashMap[String, Object] - map += "resourceId" -> resource.resourceId - map += "version" -> resource.version - } - - def update(userName: String, resourceId: String, content: String): util.Map[String, Object] = { - val inputStream = new ByteArrayInputStream(content.getBytes("utf-8")) - val client: BmlClient = createBMLClient(userName) - val resource: BmlUpdateResponse = client.updateShareResource(userName, resourceId, UUID.randomUUID().toString+".json", inputStream) - if (!resource.isSuccess) throw new DSSErrorException(911114, "更新失败") - val map = new util.HashMap[String, Object] - map += "resourceId" -> resource.resourceId - map += "version" -> resource.version - } - - def query(userName: String, resourceId: String, version: String): util.Map[String, Object] = { - val client: BmlClient = createBMLClient(userName) - var resource: BmlDownloadResponse = null - if (version == null) { - resource = client.downloadShareResource(userName, resourceId) - } else { - resource = client.downloadShareResource(userName, resourceId, version) - } - if (!resource.isSuccess) throw new DSSErrorException(911115, "下载失败") - val map = new util.HashMap[String, Object] - map += "path" -> resource.fullFilePath - map += "string" -> inputstremToString(resource.inputStream) - } - - def download(userName: String, resourceId: String, version: String): util.Map[String, Object] = { - val client: BmlClient = createBMLClient(userName) - var resource: BmlDownloadResponse = null - if (version == null) { - resource = client.downloadShareResource(userName, resourceId) - } else { - resource = client.downloadShareResource(userName, resourceId, version) - } - if (!resource.isSuccess) throw new DSSErrorException(911115, "下载失败") - val map = new util.HashMap[String, Object] - map += "path" -> resource.fullFilePath - map += "is" -> resource.inputStream - } - - def downloadToLocalPath(userName: String, resourceId: String, version: String, path: String): String = { - val result = download(userName,resourceId,version) - val is = result.get("is").asInstanceOf[InputStream] - val os = IoUtils.generateExportOutputStream(path) - Utils.tryFinally(IOUtils.copy(is,os)){ - IOUtils.closeQuietly(os) - IOUtils.closeQuietly(is) - } - path - } - - def downloadAndGetFlowJson(userName: String, resourceId: String, version: String, path: String): String = { - downloadToLocalPath(userName,resourceId,version,path) - //因为下载到指定目录后返回的resource的Stream为null,只能从文件重新读取。 - val is = IoUtils.generateInputInputStream(path) - Utils.tryFinally(inputstremToString(is))(IOUtils.closeQuietly(is)) - } - - def readLocalResourceFile(userName: String,readPath: String): InputStream ={ - IoUtils.generateInputInputStream(readPath) - } - - def readLocalFlowJsonFile(userName: String,readPath: String): String ={ - var inputStream:InputStream = null - var flowJson:String = null - Utils.tryFinally{ - inputStream = IoUtils.generateInputInputStream(readPath) - flowJson = inputstremToString(inputStream) - }{ - IOUtils.closeQuietly(inputStream) - } - flowJson - } - - private def inputstremToString(inputStream: InputStream): String = { - scala.io.Source.fromInputStream(inputStream).mkString - } - - private def createBMLClient(userName: String): BmlClient = { - if (userName == null) - BmlClientFactory.createBmlClient() - else - BmlClientFactory.createBmlClient(userName) - } - - def createBmlProject(username:String, projectName:String, editUsers:util.List[String], - accessUsers:util.List[String] ): Unit ={ - val client = createBMLClient(username) - val response = client.createBmlProject(username, projectName, accessUsers, editUsers) - if (response.isSuccess){ - logger.info(s"for user $username create bml project $projectName success") - }else{ - logger.error(s"for user $username create bml project $projectName failed") - } - } - - def attachResourceAndProject(username:String, projectName:String, resourceId:String):Unit = { - val client = createBMLClient(username) - val response = client.attachResourceAndProject(projectName, resourceId) - if (response.isSuccess){ - logger.info(s"attach $username and $projectName success") - }else{ - logger.error(s"attach $username and $projectName failed") - } - } - - - def updateProjectPriv(projectName:String, username:String, editUsers:util.List[String], - accessUsers:util.List[String]): Unit ={ - val client = createBMLClient(username) - val response = client.updateProjectPriv(username, projectName, editUsers, accessUsers) - if (response.isSuccess){ - logger.info(s"attach $username and $projectName success") - }else{ - logger.error(s"attach $username and $projectName failed") - } - } - - - -} diff --git a/dss-orchestrator/dss-orchestrator-db/pom.xml b/dss-orchestrator/dss-orchestrator-db/pom.xml index f0d23a2faf..f165f38b16 100644 --- a/dss-orchestrator/dss-orchestrator-db/pom.xml +++ b/dss-orchestrator/dss-orchestrator-db/pom.xml @@ -21,7 +21,8 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 + ../../pom.xml 4.0.0 diff --git a/dss-orchestrator/dss-orchestrator-db/src/main/java/com/webank/wedatasphere/dss/orchestrator/db/dao/OrchestratorCopyJobMapper.java b/dss-orchestrator/dss-orchestrator-db/src/main/java/com/webank/wedatasphere/dss/orchestrator/db/dao/OrchestratorCopyJobMapper.java new file mode 100644 index 0000000000..52a5b01ecd --- /dev/null +++ b/dss-orchestrator/dss-orchestrator-db/src/main/java/com/webank/wedatasphere/dss/orchestrator/db/dao/OrchestratorCopyJobMapper.java @@ -0,0 +1,30 @@ +package com.webank.wedatasphere.dss.orchestrator.db.dao; + + +import com.webank.wedatasphere.dss.orchestrator.common.entity.DSSOrchestratorCopyInfo; +import org.apache.ibatis.annotations.Mapper; +import org.apache.ibatis.annotations.Param; + +import java.util.List; + + +@Mapper +public interface OrchestratorCopyJobMapper { + + List getOrchestratorCopyInfoList(Long orchestratorId); + + String getOrchestratorCopyStatus(Long sourceOrchestratorId); + + void insertOrchestratorCopyInfo(DSSOrchestratorCopyInfo dssOrchestratorCopyInfo); + + void updateCopyStatus(DSSOrchestratorCopyInfo dssOrchestratorCopyInfo); + + void updateErrorMsgById(DSSOrchestratorCopyInfo dssOrchestratorCopyInfo); + + void batchUpdateCopyJob(@Param("failedJobs") List failedJobs); + + DSSOrchestratorCopyInfo getOrchestratorCopyInfoById(String id); + + List getRunningJob(); + +} diff --git a/dss-orchestrator/dss-orchestrator-db/src/main/java/com/webank/wedatasphere/dss/orchestrator/db/dao/OrchestratorJobMapper.java b/dss-orchestrator/dss-orchestrator-db/src/main/java/com/webank/wedatasphere/dss/orchestrator/db/dao/OrchestratorJobMapper.java index 306c253be1..cbdb4e3a81 100644 --- a/dss-orchestrator/dss-orchestrator-db/src/main/java/com/webank/wedatasphere/dss/orchestrator/db/dao/OrchestratorJobMapper.java +++ b/dss-orchestrator/dss-orchestrator-db/src/main/java/com/webank/wedatasphere/dss/orchestrator/db/dao/OrchestratorJobMapper.java @@ -1,19 +1,22 @@ package com.webank.wedatasphere.dss.orchestrator.db.dao; -import com.webank.wedatasphere.dss.orchestrator.db.entity.OrchestratorPublishJob; +import com.webank.wedatasphere.dss.orchestrator.common.entity.OrchestratorPublishJob; import org.apache.ibatis.annotations.Mapper; +import org.apache.ibatis.annotations.Param; + +import java.util.List; @Mapper public interface OrchestratorJobMapper { - OrchestratorPublishJob getPublishJobById(long id); + OrchestratorPublishJob getPublishJobByJobId(@Param("jobId") String jobId); - OrchestratorPublishJob getPublishJobByJobId(String jobId); + List getPublishJobByJobStatuses(@Param("statuses") List statuses); long insertPublishJob(OrchestratorPublishJob job); void updatePublishJob(OrchestratorPublishJob job); - void deletePublishJob(OrchestratorPublishJob job); + void batchUpdatePublishJob(@Param("failedJobs") List failedJobs); } diff --git a/dss-orchestrator/dss-orchestrator-db/src/main/java/com/webank/wedatasphere/dss/orchestrator/db/dao/OrchestratorMapper.java b/dss-orchestrator/dss-orchestrator-db/src/main/java/com/webank/wedatasphere/dss/orchestrator/db/dao/OrchestratorMapper.java index b2490d1b7b..b95f7df7c1 100644 --- a/dss-orchestrator/dss-orchestrator-db/src/main/java/com/webank/wedatasphere/dss/orchestrator/db/dao/OrchestratorMapper.java +++ b/dss-orchestrator/dss-orchestrator-db/src/main/java/com/webank/wedatasphere/dss/orchestrator/db/dao/OrchestratorMapper.java @@ -19,7 +19,6 @@ import com.webank.wedatasphere.dss.orchestrator.common.entity.*; import org.apache.ibatis.annotations.*; -import java.util.Date; import java.util.List; import java.util.Map; @@ -101,8 +100,8 @@ DSSOrchestratorVersion getOrcVersionByIdAndOrcVersionId(@Param("orchestratorId") @Select("select uuid from dss_orchestrator_info where project_id = #{projectId} and name = #{name}") String getOrcNameByParam(@Param("projectId") Long projectId, @Param("name") String name); - @Select("select id from `dss_orchestrator_info` where `project_id` = #{projectId} and `is_published` = 1") - List getAllOrcIdsByProjectId(@Param("projectId") Long projectId); + @Select("select id from `dss_orchestrator_info` where `project_id` = #{projectId} and `is_published` = #{isPublished}") + List getAllOrcIdsByProjectId(@Param("projectId") Long projectId, @Param("isPublished")Integer isPublished); @Select("select max(`app_id`) from `dss_orchestrator_version_info` where `orchestrator_id` = #{orchestratorId} and `version` = #{version}") Long getAppIdByVersion(@Param("orchestratorId") Long orchestratorId, @Param("version") String version); @@ -123,4 +122,8 @@ DSSOrchestratorVersion getOrcVersionByIdAndOrcVersionId(@Param("orchestratorId") List queryOrchestratorInfos(@Param("params") Map params); DSSOrchestratorRefOrchestration getRefOrchestrationId(@Param("orchestratorId") Long orchestratorId); + + List getHistoryOrcVersion(@Param("remainVersion") int remainVersion); + + void batchUpdateOrcInfo(@Param("list") List historyOrcVersion); } diff --git a/dss-orchestrator/dss-orchestrator-db/src/main/java/com/webank/wedatasphere/dss/orchestrator/db/dao/impl/OrchestratorCopyJobMapper.xml b/dss-orchestrator/dss-orchestrator-db/src/main/java/com/webank/wedatasphere/dss/orchestrator/db/dao/impl/OrchestratorCopyJobMapper.xml new file mode 100644 index 0000000000..ebda6f8f14 --- /dev/null +++ b/dss-orchestrator/dss-orchestrator-db/src/main/java/com/webank/wedatasphere/dss/orchestrator/db/dao/impl/OrchestratorCopyJobMapper.xml @@ -0,0 +1,102 @@ + + + + + + + + + + `id`,`username`,`type`,`source_orchestrator_id`,`source_orchestrator_name`,`target_orchestrator_name`,`source_project_name`, + `target_project_name`,`workspace_id`,`workflow_node_suffix`,`microserver_name`,`exception_info`,`status`,`is_copying`,`success_node`,`start_time`,`end_time`,`instance_name` + + + + INSERT INTO `dss_orchestrator_copy_info` () + VALUES + (#{id}, #{username}, #{type}, #{sourceOrchestratorId}, #{sourceOrchestratorName}, #{targetOrchestratorName}, #{sourceProjectName}, + #{targetProjectName}, #{workspaceId}, #{workflowNodeSuffix}, #{microserverName}, #{exceptionInfo}, #{status}, #{isCopying}, #{successNode}, #{startTime}, #{endTime}, #{instanceName}) + + + + UPDATE dss_orchestrator_copy_info + + `is_copying`=#{isCopying}, + `end_time`=#{endTime}, + `success_node`=#{successNode}, + `status`=#{status} + + WHERE id = #{id} + + + + + UPDATE dss_orchestrator_copy_info + + `is_copying`=#{isCopying}, + `end_time`=#{endTime}, + `success_node`=#{successNode}, + `status`=#{status}, + `exception_info`=#{exceptionInfo} + + WHERE id = #{id} + + + + UPDATE dss_orchestrator_copy_info set status = + + when id = #{item.id} then #{item.status} + + is_copying = + + when id = #{item.id} then #{item.isCopying} + + exception_info = + + when id = #{item.id} then #{item.exceptionInfo} + + end_time = + + when id = #{item.id} then #{item.endTime} + + WHERE id in + + #{item.id} + + + + + + + + + + + + \ No newline at end of file diff --git a/dss-orchestrator/dss-orchestrator-db/src/main/java/com/webank/wedatasphere/dss/orchestrator/db/dao/impl/OrchestratorJobMapper.xml b/dss-orchestrator/dss-orchestrator-db/src/main/java/com/webank/wedatasphere/dss/orchestrator/db/dao/impl/OrchestratorJobMapper.xml index 7fcf5d70ae..8049f82755 100644 --- a/dss-orchestrator/dss-orchestrator-db/src/main/java/com/webank/wedatasphere/dss/orchestrator/db/dao/impl/OrchestratorJobMapper.xml +++ b/dss-orchestrator/dss-orchestrator-db/src/main/java/com/webank/wedatasphere/dss/orchestrator/db/dao/impl/OrchestratorJobMapper.xml @@ -21,48 +21,68 @@ - id, job_id, conversion_job_json, created_time, updated_time + id, job_id, conversion_job_json, create_time, update_time, instance_name, status, error_msg - - + + + parameterType="com.webank.wedatasphere.dss.orchestrator.common.entity.OrchestratorPublishJob"> INSERT INTO dss_orchestrator_job_info () VALUES - (#{id},#{jobId},#{conversionJobJson},#{createdTime}) + (#{id},#{jobId},#{conversionJobJson},#{createTime},#{updateTime},#{instanceName},#{status},#{errorMsg}) - + UPDATE dss_orchestrator_job_info - job_id =#{jobId}, - conversion_job_json =#{conversionJobJson}, - created_time =#{createdTime}, - updated_time =#{updatedTime}, + update_time =#{updateTime}, + status = #{status}, + error_msg = #{errorMsg}, WHERE id = #{id} - - DELETE - FROM - dss_orchestrator_job_info - WHERE id = #{id} - + + UPDATE dss_orchestrator_job_info set status = + + when id = #{item.id} then #{item.status} + + update_time = + + when id = #{item.id} then #{item.updateTime} + + error_msg = + + when id = #{item.id} then #{item.errorMsg} + + WHERE id in + + #{item.id} + + - + - - + + + + + diff --git a/dss-orchestrator/dss-orchestrator-db/src/main/java/com/webank/wedatasphere/dss/orchestrator/db/dao/impl/orchestratorMapper.xml b/dss-orchestrator/dss-orchestrator-db/src/main/java/com/webank/wedatasphere/dss/orchestrator/db/dao/impl/orchestratorMapper.xml index edc7f455e2..77e8669fd6 100644 --- a/dss-orchestrator/dss-orchestrator-db/src/main/java/com/webank/wedatasphere/dss/orchestrator/db/dao/impl/orchestratorMapper.xml +++ b/dss-orchestrator/dss-orchestrator-db/src/main/java/com/webank/wedatasphere/dss/orchestrator/db/dao/impl/orchestratorMapper.xml @@ -155,7 +155,7 @@ @@ -239,4 +239,39 @@ from dss_orchestrator_ref_orchestration_relation where orchestrator_id = #{orchestratorId} + + + + + + update dss_orchestrator_version_info set context_id ='' where id in + + #{item.id} + + \ No newline at end of file diff --git a/dss-orchestrator/dss-orchestrator-db/src/main/java/com/webank/wedatasphere/dss/orchestrator/db/entity/OrchestratorPublishJob.java b/dss-orchestrator/dss-orchestrator-db/src/main/java/com/webank/wedatasphere/dss/orchestrator/db/entity/OrchestratorPublishJob.java deleted file mode 100644 index 854eebbfa2..0000000000 --- a/dss-orchestrator/dss-orchestrator-db/src/main/java/com/webank/wedatasphere/dss/orchestrator/db/entity/OrchestratorPublishJob.java +++ /dev/null @@ -1,56 +0,0 @@ -package com.webank.wedatasphere.dss.orchestrator.db.entity; - -import java.util.Date; - -public class OrchestratorPublishJob { - - private Long id; - private String jobId; - private String conversionJobJson; - private Date createdTime; - private Date updatedTime; - - public Long getId() { - return id; - } - - public void setId(Long id) { - this.id = id; - } - - public String getJobId() { - return jobId; - } - - public OrchestratorPublishJob setJobId(String jobId) { - this.jobId = jobId; - return this; - } - - public String getConversionJobJson() { - return conversionJobJson; - } - - public OrchestratorPublishJob setConversionJobJson(String conversionJobJson) { - this.conversionJobJson = conversionJobJson; - return this; - } - - public Date getCreatedTime() { - return createdTime; - } - - public OrchestratorPublishJob setCreatedTime(Date createdTime) { - this.createdTime = createdTime; - return this; - } - - public Date getUpdatedTime() { - return updatedTime; - } - - public OrchestratorPublishJob setUpdatedTime(Date updatedTime) { - this.updatedTime = updatedTime; - return this; - } -} diff --git a/dss-orchestrator/dss-orchestrator-loader/pom.xml b/dss-orchestrator/dss-orchestrator-loader/pom.xml index 3082f7c46b..2cc19553d6 100644 --- a/dss-orchestrator/dss-orchestrator-loader/pom.xml +++ b/dss-orchestrator/dss-orchestrator-loader/pom.xml @@ -21,7 +21,8 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 + ../../pom.xml 4.0.0 @@ -58,7 +59,7 @@ org.springframework spring-context - ${spring.version} + ${spring-framework.version} provided diff --git a/dss-orchestrator/dss-orchestrator-loader/src/main/java/com/webank/wedatasphere/dss/orchestrator/loader/DefaultLinkedAppConnResolver.java b/dss-orchestrator/dss-orchestrator-loader/src/main/java/com/webank/wedatasphere/dss/orchestrator/loader/DefaultLinkedAppConnResolver.java index 9db2b1255b..fd2f7fcbcb 100644 --- a/dss-orchestrator/dss-orchestrator-loader/src/main/java/com/webank/wedatasphere/dss/orchestrator/loader/DefaultLinkedAppConnResolver.java +++ b/dss-orchestrator/dss-orchestrator-loader/src/main/java/com/webank/wedatasphere/dss/orchestrator/loader/DefaultLinkedAppConnResolver.java @@ -28,6 +28,9 @@ @Component class DefaultLinkedAppConnResolver implements LinkedAppConnResolver { + /** + * 当前实现是获取所有系统中注册过的appconn,并没有根据条件做过滤 + */ @Override public List resolveAppConnByUser(String userName, String workspaceName, String typeName) { //todo 后面可以使用数据库表来定义用户可以加载的AppConn. diff --git a/dss-orchestrator/dss-orchestrator-loader/src/main/java/com/webank/wedatasphere/dss/orchestrator/loader/DefaultOrchestratorLoader.java b/dss-orchestrator/dss-orchestrator-loader/src/main/java/com/webank/wedatasphere/dss/orchestrator/loader/DefaultOrchestratorLoader.java index 6255c680a9..e16adf6881 100644 --- a/dss-orchestrator/dss-orchestrator-loader/src/main/java/com/webank/wedatasphere/dss/orchestrator/loader/DefaultOrchestratorLoader.java +++ b/dss-orchestrator/dss-orchestrator-loader/src/main/java/com/webank/wedatasphere/dss/orchestrator/loader/DefaultOrchestratorLoader.java @@ -66,7 +66,7 @@ protected DSSOrchestratorContext createOrchestratorContext() { if (appConnList.stream().anyMatch(t -> t instanceof SchedulerAppConn)) { List schedulerAppConns = AppConnManager.getAppConnManager().listAppConns(SchedulerAppConn.class); SchedulerAppConn schedulerAppConn; - if(StringUtils.isBlank(relation.getBindingSchedulerAppConnName())) { + if (StringUtils.isBlank(relation.getBindingSchedulerAppConnName())) { schedulerAppConn = schedulerAppConns.get(0); } else { schedulerAppConn = schedulerAppConns.stream().filter(appConn1 -> appConn1.getAppDesc().getAppName().equals(relation.getBindingSchedulerAppConnName())) diff --git a/dss-orchestrator/dss-orchestrator-loader/src/main/java/com/webank/wedatasphere/dss/orchestrator/loader/LinkedAppConnResolver.java b/dss-orchestrator/dss-orchestrator-loader/src/main/java/com/webank/wedatasphere/dss/orchestrator/loader/LinkedAppConnResolver.java index 66bc4c748b..af71d8c791 100644 --- a/dss-orchestrator/dss-orchestrator-loader/src/main/java/com/webank/wedatasphere/dss/orchestrator/loader/LinkedAppConnResolver.java +++ b/dss-orchestrator/dss-orchestrator-loader/src/main/java/com/webank/wedatasphere/dss/orchestrator/loader/LinkedAppConnResolver.java @@ -22,5 +22,12 @@ import java.util.List; public interface LinkedAppConnResolver { + /** + * 根据用户、工作空间、appconn的类型,获取所有注册过的符合条件的appconn + * @param userName 用户名 + * @param workspaceName 工作空间名 + * @param typeName appconn类型 + * @return 注册过的符合条件的appconn + */ List resolveAppConnByUser(String userName, String workspaceName, String typeName); } diff --git a/dss-orchestrator/dss-orchestrator-loader/src/main/java/com/webank/wedatasphere/dss/orchestrator/loader/OrchestratorManager.java b/dss-orchestrator/dss-orchestrator-loader/src/main/java/com/webank/wedatasphere/dss/orchestrator/loader/OrchestratorManager.java index 7a301cf0cb..aa4cc1c94a 100644 --- a/dss-orchestrator/dss-orchestrator-loader/src/main/java/com/webank/wedatasphere/dss/orchestrator/loader/OrchestratorManager.java +++ b/dss-orchestrator/dss-orchestrator-loader/src/main/java/com/webank/wedatasphere/dss/orchestrator/loader/OrchestratorManager.java @@ -16,16 +16,23 @@ package com.webank.wedatasphere.dss.orchestrator.loader; +import com.webank.wedatasphere.dss.appconn.core.AppConn; +import com.webank.wedatasphere.dss.appconn.manager.AppConnManager; +import com.webank.wedatasphere.dss.appconn.manager.impl.AbstractAppConnManager; +import com.webank.wedatasphere.dss.appconn.manager.service.AppConnRefreshListener; import com.webank.wedatasphere.dss.common.label.DSSLabel; import com.webank.wedatasphere.dss.orchestrator.core.DSSOrchestrator; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Component; +import javax.annotation.PostConstruct; import java.util.List; import java.util.Map; import java.util.concurrent.ConcurrentHashMap; - +/** + * 编排管理器 + */ @Component public class OrchestratorManager { @@ -34,16 +41,39 @@ public class OrchestratorManager { @Autowired private DefaultOrchestratorLoader defaultOrchestratorLoader; + private volatile static boolean registered = false; + + private AppConnRefreshListener cacheRemoveListener = appconnName -> cacheDssOrchestrator.forEach((key, value) -> { + if (value.getAppConn().getAppDesc().getAppName().equals(appconnName) || + value.getSchedulerAppConn().getAppDesc().getAppName().equals(appconnName)) { + cacheDssOrchestrator.remove(key); + } + }); + + /** + * 获取一个编排,可以从缓存获取,或者重新load一个 + * + * @param userName 用户名 + * @param workspaceName 工作空间名 + * @param typeName 编排的实现类型 + * @param dssLabels 环境标签 + * @return 编排 + */ public DSSOrchestrator getOrCreateOrchestrator(String userName, String workspaceName, String typeName, List dssLabels) { + if (!registered) { + ((AbstractAppConnManager) AppConnManager.getAppConnManager()).getAppConnRefreshThread() + .registerRefreshListener(cacheRemoveListener); + registered = true; + } String findKey = getCacheKey(userName, workspaceName, typeName); DSSOrchestrator dssOrchestrator = cacheDssOrchestrator.get(findKey); if (null == dssOrchestrator) { - synchronized (cacheDssOrchestrator) { + synchronized (cacheDssOrchestrator) { dssOrchestrator = cacheDssOrchestrator.get(findKey); - if(null == dssOrchestrator) { + if (null == dssOrchestrator) { dssOrchestrator = defaultOrchestratorLoader.loadOrchestrator(userName, workspaceName, typeName, dssLabels); cacheDssOrchestrator.put(findKey, dssOrchestrator); } @@ -55,4 +85,6 @@ public DSSOrchestrator getOrCreateOrchestrator(String userName, protected String getCacheKey(String userName, String workspaceName, String typeName) { return userName + "_" + workspaceName + "_" + typeName; } + + } diff --git a/dss-orchestrator/dss-orchestrator-loader/src/main/java/com/webank/wedatasphere/dss/orchestrator/loader/utils/OrchestratorLoaderUtils.java b/dss-orchestrator/dss-orchestrator-loader/src/main/java/com/webank/wedatasphere/dss/orchestrator/loader/utils/OrchestratorLoaderUtils.java index 34b67831a5..9eb74444ee 100644 --- a/dss-orchestrator/dss-orchestrator-loader/src/main/java/com/webank/wedatasphere/dss/orchestrator/loader/utils/OrchestratorLoaderUtils.java +++ b/dss-orchestrator/dss-orchestrator-loader/src/main/java/com/webank/wedatasphere/dss/orchestrator/loader/utils/OrchestratorLoaderUtils.java @@ -45,6 +45,14 @@ public void init(){ orchestratorLoaderUtils = this; } + /** + * 根据标签和编排资源配置,解析出合适的AppInstance,并得到编排对应实现(比如工作流实现)的DevelopmentIntegrationStandard。 + * 以pair的方式返回AppInstance和的DevelopmentIntegrationStandard + * @param dssOrchestrator 编排资源 + * @param dssLabels 标签 + * @return AppInstance和的DevelopmentIntegrationStandard 对 + * @throws NoSuchAppInstanceException 如果编排不存在实现实例 + */ public static ImmutablePair getOrchestratorDevelopmentStandard( DSSOrchestrator dssOrchestrator, List dssLabels) throws NoSuchAppInstanceException { AppConn orchestratorAppConn = dssOrchestrator.getAppConn(); diff --git a/dss-orchestrator/orchestrators/dss-workflow/dss-flow-execution-server/pom.xml b/dss-orchestrator/orchestrators/dss-workflow/dss-flow-execution-server/pom.xml index 0ae5a3f9ed..6dfbe302b7 100644 --- a/dss-orchestrator/orchestrators/dss-workflow/dss-flow-execution-server/pom.xml +++ b/dss-orchestrator/orchestrators/dss-workflow/dss-flow-execution-server/pom.xml @@ -21,13 +21,19 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 + ../../../../pom.xml 4.0.0 dss-flow-execution-server + + org.apache.linkis + linkis-jobhistory + ${linkis.version} + org.apache.linkis linkis-entrance @@ -55,6 +61,37 @@ + + org.apache.linkis + linkis-scheduler + ${linkis.version} + + + linkis-common + org.apache.linkis + + + spring-tx + org.springframework + + + commons-text + org.apache.commons + + + json4s-jackson_2.11 + org.json4s + + + linkis-label-common + org.apache.linkis + + + commons-beanutils + commons-beanutils + + + com.webank.wedatasphere.dss @@ -76,6 +113,16 @@ + + spring-tx + org.springframework + ${spring-framework.version} + + + spring-context-support + org.springframework + ${spring-framework.version} + org.apache.linkis linkis-rpc @@ -94,6 +141,10 @@ org.glassfish.hk2.external bean-validator + + spring-aop + org.springframework + @@ -145,8 +196,6 @@ provided - - diff --git a/dss-orchestrator/orchestrators/dss-workflow/dss-flow-execution-server/src/main/java/com/webank/wedatasphere/dss/flow/execution/entrance/conf/FlowEntranceSpringConfiguration.java b/dss-orchestrator/orchestrators/dss-workflow/dss-flow-execution-server/src/main/java/com/webank/wedatasphere/dss/flow/execution/entrance/conf/FlowEntranceSpringConfiguration.java new file mode 100644 index 0000000000..73b13bf11f --- /dev/null +++ b/dss-orchestrator/orchestrators/dss-workflow/dss-flow-execution-server/src/main/java/com/webank/wedatasphere/dss/flow/execution/entrance/conf/FlowEntranceSpringConfiguration.java @@ -0,0 +1,32 @@ +package com.webank.wedatasphere.dss.flow.execution.entrance.conf; + +/** + * Author: xlinliu + * Date: 2023/1/16 + */ + +import org.apache.linkis.entrance.interceptor.EntranceInterceptor; +import org.apache.linkis.entrance.interceptor.impl.CommentInterceptor; +import org.apache.linkis.entrance.interceptor.impl.LabelCheckInterceptor; +import org.apache.linkis.entrance.interceptor.impl.LogPathCreateInterceptor; +import org.apache.linkis.entrance.interceptor.impl.StorePathEntranceInterceptor; +import org.apache.linkis.entrance.constant.ServiceNameConsts; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; + +@Configuration +public class FlowEntranceSpringConfiguration { + private final Logger logger = LoggerFactory.getLogger(getClass()); + @Bean(name=ServiceNameConsts.ENTRANCE_INTERCEPTOR) + public EntranceInterceptor[] entranceInterceptors() { + logger.info("dss workflow entrance load entranceInterceptors"); + return new EntranceInterceptor[] { + new LabelCheckInterceptor(), + new LogPathCreateInterceptor(), + new StorePathEntranceInterceptor(), + new CommentInterceptor(), + }; + } +} \ No newline at end of file diff --git a/dss-orchestrator/orchestrators/dss-workflow/dss-flow-execution-server/src/main/java/com/webank/wedatasphere/dss/flow/execution/entrance/dao/TaskMapper.java b/dss-orchestrator/orchestrators/dss-workflow/dss-flow-execution-server/src/main/java/com/webank/wedatasphere/dss/flow/execution/entrance/dao/TaskMapper.java index af84a34cec..5306a55446 100644 --- a/dss-orchestrator/orchestrators/dss-workflow/dss-flow-execution-server/src/main/java/com/webank/wedatasphere/dss/flow/execution/entrance/dao/TaskMapper.java +++ b/dss-orchestrator/orchestrators/dss-workflow/dss-flow-execution-server/src/main/java/com/webank/wedatasphere/dss/flow/execution/entrance/dao/TaskMapper.java @@ -30,6 +30,8 @@ public interface TaskMapper { void updateTask(WorkflowQueryTask queryTask); + void batchUpdateTasks(@Param("failedJobs") List failedJobs); + List search(@Param("taskID") Long taskID, @Param("umUser") String username, @Param("status") List status, @Param("startDate") Date startDate, @Param("endDate") Date endDate, @Param("executeApplicationName") String executeApplicationName, @Param("instance") String instance, @Param("execId") String execId); diff --git a/dss-orchestrator/orchestrators/dss-workflow/dss-flow-execution-server/src/main/java/com/webank/wedatasphere/dss/flow/execution/entrance/dao/impl/taskMapper.xml b/dss-orchestrator/orchestrators/dss-workflow/dss-flow-execution-server/src/main/java/com/webank/wedatasphere/dss/flow/execution/entrance/dao/impl/taskMapper.xml index d43e9763f0..b9c9ae1cea 100644 --- a/dss-orchestrator/orchestrators/dss-workflow/dss-flow-execution-server/src/main/java/com/webank/wedatasphere/dss/flow/execution/entrance/dao/impl/taskMapper.xml +++ b/dss-orchestrator/orchestrators/dss-workflow/dss-flow-execution-server/src/main/java/com/webank/wedatasphere/dss/flow/execution/entrance/dao/impl/taskMapper.xml @@ -112,6 +112,28 @@ WHERE id =#{taskID} + + UPDATE dss_workflow_task set status = + + when id = #{item.taskID} then #{item.status} + + err_desc = + + when id = #{item.taskID} then #{item.errDesc} + + updated_time = + + when id = #{item.taskID} then #{item.updatedTime} + + WHERE id in + + #{item.taskID} + + + - INSERT INTO dss_workflow () VALUES - (#{id},#{name},#{state},#{source},#{description},#{createTime},#{creator},#{isRootFlow},#{rank},#{projectID},#{hasSaved},#{uses},#{bmlVersion},#{resourceId},#{linkedAppConnNames},#{dssLabels}) + (#{id},#{name},#{state},#{source},#{description},#{createTime},#{creator},#{isRootFlow},#{rank},#{projectId},#{hasSaved},#{uses},#{bmlVersion},#{resourceId},#{linkedAppConnNames},#{dssLabels}) - - INSERT INTO dss_workflow_relation (flow_id,parent_flow_id) VALUES @@ -59,8 +56,8 @@ SELECT fr.flow_id FROM `dss_workflow_relation` fr WHERE fr.parent_flow_id = #{parentFlowID} - + select * from dss_workflow where id in (SELECT flow_id FROM `dss_workflow_relation` WHERE parent_flow_id = #{parentFlowID}) @@ -86,20 +83,14 @@ flow_id = #{flowID} - - + + + + + + + + + \ No newline at end of file diff --git a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/entity/InputRelation.java b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/entity/InputRelation.java deleted file mode 100644 index ff82ae8858..0000000000 --- a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/entity/InputRelation.java +++ /dev/null @@ -1,78 +0,0 @@ -/* - * Copyright 2019 WeBank - * 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. - * - */ - -package com.webank.wedatasphere.dss.workflow.entity; - - -import com.webank.wedatasphere.dss.common.entity.IOEnv; - -public class InputRelation { - - private Long id; - private String type; - private IOEnv sourceEnv; - private Long sourceID; - private IOEnv targetEnv; - private Long targetID; - - public Long getId() { - return id; - } - - public void setId(Long id) { - this.id = id; - } - - public String getType() { - return type; - } - - public void setType(String type) { - this.type = type; - } - - public IOEnv getSourceEnv() { - return sourceEnv; - } - - public void setSourceEnv(IOEnv sourceEnv) { - this.sourceEnv = sourceEnv; - } - - public Long getSourceID() { - return sourceID; - } - - public void setSourceID(Long sourceID) { - this.sourceID = sourceID; - } - - public IOEnv getTargetEnv() { - return targetEnv; - } - - public void setTargetEnv(IOEnv targetEnv) { - this.targetEnv = targetEnv; - } - - public Long getTargetID() { - return targetID; - } - - public void setTargetID(Long targetID) { - this.targetID = targetID; - } -} diff --git a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/entity/vo/FlowInfoVo.java b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/entity/vo/FlowInfoVo.java new file mode 100644 index 0000000000..dda3a33c5a --- /dev/null +++ b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/entity/vo/FlowInfoVo.java @@ -0,0 +1,32 @@ +package com.webank.wedatasphere.dss.workflow.entity.vo; + +import com.webank.wedatasphere.dss.workflow.common.entity.DSSFlow; +import com.webank.wedatasphere.dss.workflow.common.entity.DSSFlowRelation; + +import java.util.List; + +public class FlowInfoVo { + + List subFlowList; + + Long parentFlowId; + + public List getSubFlowList() { + return subFlowList; + } + + public void setSubFlowList(List subFlowList) { + this.subFlowList = subFlowList; + } + + public FlowInfoVo() { + } + + public Long getParentFlowId() { + return parentFlowId; + } + + public void setParentFlowId(Long parentFlowId) { + this.parentFlowId = parentFlowId; + } +} diff --git a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/io/export/WorkFlowExportService.java b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/io/export/WorkFlowExportService.java index 15b3233020..55eef00641 100644 --- a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/io/export/WorkFlowExportService.java +++ b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/io/export/WorkFlowExportService.java @@ -22,10 +22,37 @@ import java.util.List; public interface WorkFlowExportService { - + /** + * 将一个工作流导出成压缩包,存放到本地磁盘,并返回压缩包的路径 + * @param dssProjectId 所属工作空间id + * @param projectName 所属项目名称 + * @param rootFlowId 工作流根节点id + * @param userName 用户名 + * @param workspace 所属工作空间 + * @param dssLabels label列表 + * @return 导出的工作流压缩包的文件地址 + * @throws Exception + */ String exportFlowInfo(Long dssProjectId, String projectName, long rootFlowId, String userName, Workspace workspace, List dssLabels) throws Exception; + /** + * 导出工作流中的各种资源,放到projectSavePath中。 + * 工作流中的资源包括工作流资源和节点资源 + * @param projectSavePath 保存资源的目录 + * @param flowJson 工作流元信息 + * @param flowName 工作流明 + * @param dssLabels label列表 + * @throws Exception + */ void exportFlowResources(String userName, Long projectId, String projectName, String projectSavePath, String flowJson, String flowName, Workspace workspace,List dssLabels) throws Exception; + /** + * 从bml中取出工作流的元数据信息 + * @param userName 执行用户 + * @param resourceId 工作流元数据的resourceId + * @param version 工作流元数据信息bml文件的版本 + * @param savePath 取出后的元数据信息存放目录 + * @return 工作流的元数据信息 + */ String downloadFlowJsonFromBml(String userName, String resourceId, String version, String savePath); } diff --git a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/io/export/impl/NodeExportServiceImpl.java b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/io/export/impl/NodeExportServiceImpl.java index a0ada025fe..0e4aca71e5 100644 --- a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/io/export/impl/NodeExportServiceImpl.java +++ b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/io/export/impl/NodeExportServiceImpl.java @@ -25,13 +25,14 @@ import com.webank.wedatasphere.dss.workflow.dao.NodeInfoMapper; import com.webank.wedatasphere.dss.workflow.entity.CommonAppConnNode; import com.webank.wedatasphere.dss.workflow.io.export.NodeExportService; -import com.webank.wedatasphere.dss.workflow.service.BMLService; +import com.webank.wedatasphere.dss.common.service.BMLService; import com.webank.wedatasphere.dss.workflow.service.WorkflowNodeService; import org.apache.commons.io.FileUtils; import org.apache.commons.io.IOUtils; 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.stereotype.Service; import java.io.Closeable; @@ -47,6 +48,7 @@ public class NodeExportServiceImpl implements NodeExportService { private final Logger LOGGER = LoggerFactory.getLogger(this.getClass()); @Autowired + @Qualifier("workflowBmlService") private BMLService bmlService; @Autowired private NodeInfoMapper nodeInfoMapper; diff --git a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/io/export/impl/WorkFlowExportServiceImpl.java b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/io/export/impl/WorkFlowExportServiceImpl.java index 7e2eaa3b2b..7417a682fc 100644 --- a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/io/export/impl/WorkFlowExportServiceImpl.java +++ b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/io/export/impl/WorkFlowExportServiceImpl.java @@ -20,10 +20,10 @@ import com.webank.wedatasphere.dss.common.entity.IOType; import com.webank.wedatasphere.dss.common.entity.Resource; import com.webank.wedatasphere.dss.common.entity.node.DSSEdge; -import com.webank.wedatasphere.dss.common.entity.node.DSSEdgeDefault; import com.webank.wedatasphere.dss.common.entity.node.DSSNode; import com.webank.wedatasphere.dss.common.entity.node.Node; import com.webank.wedatasphere.dss.common.exception.DSSErrorException; +import com.webank.wedatasphere.dss.common.exception.DSSRuntimeException; import com.webank.wedatasphere.dss.common.label.DSSLabel; import com.webank.wedatasphere.dss.common.utils.IoUtils; import com.webank.wedatasphere.dss.common.utils.ZipHelper; @@ -38,23 +38,21 @@ import com.webank.wedatasphere.dss.workflow.io.export.MetaExportService; import com.webank.wedatasphere.dss.workflow.io.export.NodeExportService; import com.webank.wedatasphere.dss.workflow.io.export.WorkFlowExportService; -import com.webank.wedatasphere.dss.workflow.service.BMLService; +import com.webank.wedatasphere.dss.common.service.BMLService; import com.webank.wedatasphere.dss.workflow.service.DSSFlowService; import org.apache.commons.collections.CollectionUtils; -import org.apache.commons.collections.MapUtils; import org.apache.commons.lang.StringUtils; 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.stereotype.Service; import java.io.File; import java.util.*; -import java.util.concurrent.ConcurrentSkipListSet; import java.util.concurrent.CountDownLatch; import java.util.concurrent.FutureTask; import java.util.concurrent.TimeUnit; -import java.util.concurrent.atomic.AtomicInteger; import java.util.stream.Collectors; import static com.webank.wedatasphere.dss.workflow.constant.DSSWorkFlowConstant.NODE_EXPORT_IMPORT_TIMEOUT_MINUTES; @@ -67,6 +65,7 @@ public class WorkFlowExportServiceImpl implements WorkFlowExportService { private Logger logger = LoggerFactory.getLogger(this.getClass()); @Autowired + @Qualifier("workflowBmlService") private BMLService bmlService; @Autowired @@ -130,7 +129,7 @@ public String exportFlowInfo(Long dssProjectId, String projectName, long rootFlo throw new DSSErrorException(90037, "该工程没有可以导出的工作流,请检查工作流是否都为空"); } //打包导出工程 - return ZipHelper.zipExportProject(flowExportSaveBasePath); + return ZipHelper.zip(flowExportSaveBasePath); } @@ -183,6 +182,11 @@ public void exportFlowResources(String userName, Long projectId, String projectN for (DSSNode node : nodes) { nodeExportService.downloadNodeResourceToLocal(userName, node, workFlowResourceSavePath); NodeInfo nodeInfo = nodeInfoMapper.getWorkflowNodeByType(node.getNodeType()); + if(nodeInfo==null){ + String msg = String.format("%s note type not exist,please check appconn install successfully", node.getNodeType()); + logger.error(msg); + throw new DSSRuntimeException(msg); + } if (Boolean.TRUE.equals(nodeInfo.getSupportJump()) && nodeInfo.getJumpType() == 1) { logger.info("node.getJobContent() is :{}", node.getJobContent()); nodeExportService.downloadAppConnResourceToLocal(userName, projectId, projectName, node, appConnResourceSavePath, workspace, dssLabels); @@ -249,6 +253,11 @@ public void exportFlowResources_for_multi_thread(String userName, Long projectId try { nodeExportService.downloadNodeResourceToLocal(userName, node, workFlowResourceSavePath); NodeInfo nodeInfo = nodeInfoMapper.getWorkflowNodeByType(node.getNodeType()); + if(nodeInfo==null){ + String msg = String.format("%s note type not exist,please check appconn install successfully", node.getNodeType()); + logger.error(msg); + throw new DSSRuntimeException(msg); + } if (Boolean.TRUE.equals(nodeInfo.getSupportJump()) && nodeInfo.getJumpType() == 1) { logger.info("node.getJobContent() is :{}", node.getJobContent()); nodeExportService.downloadAppConnResourceToLocal(userName, projectId, projectName, node, appConnResourceSavePath, workspace, dssLabels); @@ -289,7 +298,7 @@ public void exportFlowResources_for_multi_thread(String userName, Long projectId @Override public String downloadFlowJsonFromBml(String userName, String resourceId, String version, String savePath) { - return bmlService.downloadAndGetFlowJson(userName, resourceId, version, savePath); + return bmlService.downloadAndGetText(userName, resourceId, version, savePath); } private String downloadFlowResourceFromBml(String userName, Resource resource, String savePath) { diff --git a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/io/input/MetaReader.java b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/io/input/MetaReader.java index 3d14832bf0..eea43170cd 100644 --- a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/io/input/MetaReader.java +++ b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/io/input/MetaReader.java @@ -81,12 +81,17 @@ private void readT() { body.stream().map(DSSExceptionUtils.map(this::lineToT)).forEach(datas::add); } - private T lineToT(List list) throws IllegalAccessException, InstantiationException, NoSuchFieldException, ParseException { + private T lineToT(List list) throws IllegalAccessException, InstantiationException, ParseException { T t = tClass.newInstance(); for (int i = 0; i < list.size(); i++) { String valueStr = list.get(i); if ("null".equalsIgnoreCase(valueStr)) continue; - Field declaredField = tClass.getDeclaredField(fields.get(i)); + Field declaredField; + try { + declaredField = tClass.getDeclaredField(fields.get(i)); + }catch ( java.lang.NoSuchFieldException e){ + continue; + } declaredField.setAccessible(true); Object value = null; String type = declaredField.getType().getSimpleName(); diff --git a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/io/input/impl/NodeInputServiceImpl.java b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/io/input/impl/NodeInputServiceImpl.java index 906dbb713f..df2869718b 100644 --- a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/io/input/impl/NodeInputServiceImpl.java +++ b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/io/input/impl/NodeInputServiceImpl.java @@ -17,6 +17,8 @@ package com.webank.wedatasphere.dss.workflow.io.input.impl; +import com.google.common.collect.ImmutableMap; +import com.webank.wedatasphere.dss.common.entity.BmlResource; import com.webank.wedatasphere.dss.common.entity.Resource; import com.webank.wedatasphere.dss.common.exception.DSSErrorException; import com.webank.wedatasphere.dss.common.label.DSSLabel; @@ -30,26 +32,25 @@ import com.webank.wedatasphere.dss.workflow.common.parser.NodeParser; import com.webank.wedatasphere.dss.workflow.entity.CommonAppConnNode; import com.webank.wedatasphere.dss.workflow.io.input.NodeInputService; -import com.webank.wedatasphere.dss.workflow.service.BMLService; +import com.webank.wedatasphere.dss.common.service.BMLService; import com.webank.wedatasphere.dss.workflow.service.WorkflowNodeService; import org.apache.linkis.server.BDPJettyServerHelper; 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.stereotype.Service; import java.io.File; import java.io.IOException; import java.io.InputStream; -import java.util.LinkedHashMap; -import java.util.List; -import java.util.Map; -import java.util.UUID; +import java.util.*; import java.util.function.Supplier; @Service public class NodeInputServiceImpl implements NodeInputService { @Autowired + @Qualifier("workflowBmlService") private BMLService bmlService; @Autowired @@ -70,10 +71,10 @@ public String uploadResourceToBml(String userName, String nodeJson, String input resources.forEach(resource -> { if (resource.getVersion() != null && resource.getFileName() != null && resource.getResourceId() != null) { InputStream resourceInputStream = readResource(userName, resource, inputResourcePath); - Map bmlReturnMap = bmlService.upload(userName, + BmlResource bmlReturnMap = bmlService.upload(userName, resourceInputStream, UUID.randomUUID().toString() + ".json", projectName); - resource.setResourceId(bmlReturnMap.get("resourceId").toString()); - resource.setVersion(bmlReturnMap.get("version").toString()); + resource.setResourceId(bmlReturnMap.getResourceId()); + resource.setVersion(bmlReturnMap.getVersion()); } else { logger.warn("Illegal resource information"); logger.warn("username:{},fileName:{},version:{},resourceId:{}", userName, resource.getFileName(), resource.getVersion(), resource.getResourceId()); @@ -110,7 +111,7 @@ public String uploadAppConnResource(String userName, String projectName, DSSFlow appConnNode.setNodeType(nodeType); appConnNode.setJobContent(nodeContent); appConnNode.setFlowId(dssFlow.getId()); - appConnNode.setProjectId(dssFlow.getProjectID()); + appConnNode.setProjectId(dssFlow.getProjectId()); appConnNode.setWorkspace(workspace); appConnNode.setContextId(flowContextId); @@ -119,8 +120,14 @@ public String uploadAppConnResource(String userName, String projectName, DSSFlow File file = new File(nodeResourcePath); if (file.exists()) { InputStream resourceInputStream = bmlService.readLocalResourceFile(userName, nodeResourcePath); - Supplier> bmlResourceMap = () -> bmlService.upload(userName, resourceInputStream, UUID.randomUUID().toString() + ".json", - projectName); + Supplier> bmlResourceMap = () -> { + BmlResource resource = bmlService.upload(userName, resourceInputStream, UUID.randomUUID().toString() + ".json", + projectName); + return ImmutableMap.of( + "resourceId", resource.getResourceId(), + "version", resource.getVersion() + ); + }; Supplier> streamResourceMap = () -> MapUtils.newCommonMap(ImportRequestRef.INPUT_STREAM_KEY, resourceInputStream); try { nodeExportContent = nodeService.importNode(userName, appConnNode, bmlResourceMap, streamResourceMap, orcVersion); @@ -142,7 +149,7 @@ public String uploadAppConnResource(String userName, String projectName, DSSFlow return BDPJettyServerHelper.jacksonJson().writeValueAsString(nodeJsonMap); } } else { - logger.error("appConn node resource file does not exists. nodeId: {}" + nodeId); + logger.warn("appConn node resource file does not exists. nodeId: {}" + nodeId); } return nodeJson; diff --git a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/io/input/impl/WorkFlowInputServiceImpl.java b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/io/input/impl/WorkFlowInputServiceImpl.java index 21bce5d2a3..ecf74338c7 100644 --- a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/io/input/impl/WorkFlowInputServiceImpl.java +++ b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/io/input/impl/WorkFlowInputServiceImpl.java @@ -16,6 +16,10 @@ package com.webank.wedatasphere.dss.workflow.io.input.impl; +import com.google.gson.JsonArray; +import com.google.gson.JsonElement; +import com.google.gson.JsonParser; +import com.webank.wedatasphere.dss.common.entity.BmlResource; import com.webank.wedatasphere.dss.common.entity.Resource; import com.webank.wedatasphere.dss.common.exception.DSSErrorException; import com.webank.wedatasphere.dss.common.label.DSSLabel; @@ -31,33 +35,35 @@ import com.webank.wedatasphere.dss.workflow.dao.FlowMapper; import com.webank.wedatasphere.dss.workflow.io.input.NodeInputService; import com.webank.wedatasphere.dss.workflow.io.input.WorkFlowInputService; -import com.webank.wedatasphere.dss.workflow.io.scheduler.NodeImportJob; -import com.webank.wedatasphere.dss.workflow.service.BMLService; +import com.webank.wedatasphere.dss.common.service.BMLService; import org.apache.commons.collections.CollectionUtils; +import org.apache.commons.lang.StringUtils; import org.apache.linkis.cs.common.utils.CSCommonUtils; import org.apache.linkis.server.BDPJettyServerHelper; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.beans.BeanUtils; import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.beans.factory.annotation.Qualifier; import org.springframework.stereotype.Service; import java.io.File; import java.io.IOException; import java.io.InputStream; +import java.nio.file.Files; +import java.nio.file.Path; +import java.nio.file.Paths; import java.util.*; -import java.util.concurrent.CountDownLatch; -import java.util.concurrent.TimeUnit; -import java.util.concurrent.atomic.AtomicInteger; +import java.util.function.Function; import java.util.stream.Collectors; -import static com.webank.wedatasphere.dss.workflow.scheduler.DssJobThreadPool.nodeExportThreadPool; @Service public class WorkFlowInputServiceImpl implements WorkFlowInputService { private Logger logger = LoggerFactory.getLogger(this.getClass()); @Autowired + @Qualifier("workflowBmlService") private BMLService bmlService; @Autowired @@ -86,19 +92,30 @@ public void inputWorkFlow(String userName, //todo 不同服务共享导入工程文件包,可采用共享存储 String flowInputPath = inputProjectPath + File.separator + dssFlow.getName(); String flowJsonPath = flowInputPath + File.separator + dssFlow.getName() + ".json"; - String flowJson = bmlService.readLocalFlowJsonFile(userName, flowJsonPath); - //如果包含subflow,需要一同导入subflow内容,并更新parrentflow的json内容 + String flowJson = bmlService.readLocalTextFile(userName, flowJsonPath); + //生成新的节点key。 + Set nodeKeys=findFlowNodeKeys(flowJson); + Map oldNewNodeKeyMap = nodeKeys.stream().collect(Collectors.toMap(Function.identity(), v -> UUID.randomUUID().toString())); + String updateFlowJson=flowJson; + for (Map.Entry entry : oldNewNodeKeyMap.entrySet()) { + String oldKey=entry.getKey(); + String newKey = entry.getValue(); + updateFlowJson = updateFlowJson.replaceAll(oldKey, newKey); + } + //对于第三方节点,resource名也要更新,否则在寻找resource的时候会找不到。 + renameAppConnResource(flowInputPath,oldNewNodeKeyMap); // TODO: 2020/7/31 优化update方法里面的saveContent - String updateFlowJson = updateFlowContextIdAndVersion(userName, + updateFlowJson = updateFlowContextIdAndVersion(userName, workspace.getWorkspaceName(), projectName, - flowJson, + updateFlowJson, dssFlow, parentFlowId, contextId, orcVersion); updateFlowJson = inputWorkFlowNodes(userName, projectName, updateFlowJson, dssFlow, flowInputPath, workspace, orcVersion, dssLabels); + //如果包含subflow,需要一同导入subflow内容,并更新parentflow的json内容 List subFlows = dssFlow.getChildren(); if (subFlows != null) { for (DSSFlow subFlow : subFlows) { @@ -117,7 +134,41 @@ public void inputWorkFlow(String userName, } + /** + * 找到所有的 + * @param flowJson + * @return + */ + private Set findFlowNodeKeys(String flowJson){ + Set nodeKeys = new HashSet<>(); + JsonArray jsonElements; + if(StringUtils.isBlank(flowJson)||(jsonElements=new JsonParser().parse(flowJson).getAsJsonObject().getAsJsonArray("nodes")) + ==null){ + return nodeKeys; + } + for (JsonElement node : jsonElements) { + String nodeKey= node.getAsJsonObject().get("key").getAsString(); + nodeKeys.add(nodeKey); + } + return nodeKeys; + } + /** + * 修改解压包中appconn资源的文件名 + */ + private void renameAppConnResource(String flowInputPath, Map oldNewName){ + String appConnResourceSavePath = flowInputPath + File.separator + "appconn-resource"; + oldNewName.forEach((oldKey,newKey)->{ + Path oldNodeResourcePath =Paths.get( appConnResourceSavePath ,oldKey + ".appconnre"); + Path newNodeResourcePath =Paths.get( appConnResourceSavePath ,newKey + ".appconnre"); + try { + Files.move(oldNodeResourcePath,newNodeResourcePath); + } catch (IOException e) { + logger.error("move appconn resource file failed,path:{}",oldNodeResourcePath); + //do nothing + } + }); + } private String updateFlowContextIdAndVersion(String userName, String workspaceName, String projectName, @@ -155,7 +206,7 @@ private String inputWorkFlowNodes(String userName, String projectName, String fl List> nodeJsonListRes = new ArrayList<>(); if (nodeJsonList.size() > 0) { for (String nodeJson : nodeJsonList) { - // TODO: 2020/3/20 暂时注视掉appconn相关 + // TODO: 2020/3/20 暂时注释掉appconn相关 String updateNodeJson = nodeInputService.uploadResourceToBml(userName, nodeJson, workFlowResourceSavePath, projectName); updateNodeJson = nodeInputService.uploadAppConnResource(userName, projectName, dssFlow, updateNodeJson, updateContextId, appConnResourceSavePath, @@ -173,18 +224,18 @@ private String inputWorkFlowNodes(String userName, String projectName, String fl if ("workflow.subflow".equals(nodeType) && CollectionUtils.isNotEmpty(subflows)) { String subFlowName = nodeJsonMap.get("title").toString(); logger.info("subflows:{}", subflows); - List DSSFlowList = subflows.stream().filter(subflow -> + List dssFlowList = subflows.stream().filter(subflow -> subflow.getName().equals(subFlowName) ).collect(Collectors.toList()); - if (DSSFlowList.size() == 1) { - updateNodeJson = nodeInputService.updateNodeSubflowID(updateNodeJson, DSSFlowList.get(0).getId()); + if (dssFlowList.size() == 1) { + updateNodeJson = nodeInputService.updateNodeSubflowID(updateNodeJson, dssFlowList.get(0).getId()); nodeJsonMap = BDPJettyServerHelper.jacksonJson().readValue(updateNodeJson, Map.class); nodeJsonListRes.add(nodeJsonMap); - } else if (DSSFlowList.size() > 1) { + } else if (dssFlowList.size() > 1) { logger.error("工程内存在重复的子工作流节点名称,导入失败" + subFlowName); throw new DSSErrorException(90077, "工程内存在重复的子工作流节点名称,导入失败" + subFlowName); } else { - logger.error("工程内存在重复的子工作流节点名称,导入失败" + subFlowName); + logger.error("工程内未能找到子工作流节点,导入失败" + subFlowName); throw new DSSErrorException(90078, "工程内未能找到子工作流节点,导入失败" + subFlowName); } } else { @@ -197,62 +248,6 @@ private String inputWorkFlowNodes(String userName, String projectName, String fl } - - private String inputWorkFlowNodes_for_multi_thread(String userName, String projectName, String flowJson, - DSSFlow dssFlow, String flowPath, Workspace workspace, - String orcVersion, List dssLabels) throws DSSErrorException, IOException { - List nodeJsonList = workFlowParser.getWorkFlowNodesJson(flowJson); - if (nodeJsonList == null) { - throw new DSSErrorException(90073, "工作流内没有工作流节点,导入失败 " + dssFlow.getName()); - } - String updateContextId = workFlowParser.getValueWithKey(flowJson, CSCommonUtils.CONTEXT_ID_STR); - if (nodeJsonList.size() == 0) { - return flowJson; - } - List subflows = (List) dssFlow.getChildren(); - String workFlowResourceSavePath = flowPath + File.separator + "resource" + File.separator; - String appConnResourceSavePath = flowPath + File.separator + "appconn-resource"; -// List> nodeJsonListRes = new ArrayList<>(); - List> nodeJsonListRes = Collections.synchronizedList(new ArrayList<>()); - CountDownLatch cdl = new CountDownLatch(nodeJsonList.size()); - AtomicInteger failedCount = new AtomicInteger(0); - if (nodeJsonList.size() > 0) { - for (String nodeJson : nodeJsonList) { - NodeImportJob.ImportJobEntity jobEntity = new NodeImportJob.ImportJobEntity(); - jobEntity.setDssFlow(dssFlow); - jobEntity.setNodeJson(nodeJson); - jobEntity.setUserName(userName); - jobEntity.setProjectName(projectName); - jobEntity.setWorkFlowResourceSavePath(workFlowResourceSavePath); - jobEntity.setAppConnResourceSavePath(appConnResourceSavePath); - jobEntity.setDssLabels(dssLabels); - jobEntity.setWorkspace(workspace); - jobEntity.setUpdateContextId(updateContextId); - jobEntity.setSubflows(subflows); - jobEntity.setOrcVersion(orcVersion); - NodeImportJob nodeImportJob = new NodeImportJob(); - nodeImportJob.setNodeInputService(nodeInputService); - nodeImportJob.setJobEntity(jobEntity); - nodeImportJob.setFailedCount(failedCount); - nodeImportJob.setCountDownLatch(cdl); - nodeImportJob.setNodeJsonListRes(nodeJsonListRes); - nodeExportThreadPool.submit(nodeImportJob); - } - } - // 用户需要等待所有节点导入完成 - boolean success = false; - try { - success = cdl.await(30, TimeUnit.MINUTES); - } catch (InterruptedException e) { - logger.error("failed to import node for workflow:{}", dssFlow.getName(), e); - throw new DSSErrorException(90071, "导入节点超时!"); - } - if (failedCount.get() > 0) { - throw new DSSErrorException(90074, "有节点导入失败,请重试!"); - } - return workFlowParser.updateFlowJsonWithKey(flowJson, "nodes", nodeJsonListRes); - } - private String uploadFlowResourceToBml(String userName, String flowJson, String flowResourcePath, String projectName) throws IOException { List resourceList = workFlowParser.getWorkFlowResources(flowJson); @@ -260,9 +255,9 @@ private String uploadFlowResourceToBml(String userName, String flowJson, String if (resourceList != null) { resourceList.forEach(resource -> { InputStream resourceInputStream = readFlowResource(userName, resource, flowResourcePath); - Map bmlReturnMap = bmlService.upload(userName, resourceInputStream, UUID.randomUUID().toString() + ".json", projectName); - resource.setResourceId(bmlReturnMap.get("resourceId").toString()); - resource.setVersion(bmlReturnMap.get("version").toString()); + BmlResource bmlReturnMap = bmlService.upload(userName, resourceInputStream, UUID.randomUUID().toString() + ".json", projectName); + resource.setResourceId(bmlReturnMap.getResourceId()); + resource.setVersion(bmlReturnMap.getVersion()); }); if (resourceList.size() == 0) { return flowJson; @@ -322,7 +317,7 @@ public DSSFlow setSubFlow(DSSFlow dssFlow, List dssFlows, DSSFlow cyFlow = new DSSFlow(); BeanUtils.copyProperties(dssFlow, cyFlow, "children", "flowVersions"); //封装flow信息 - cyFlow.setProjectID(projectId); + cyFlow.setProjectId(projectId); cyFlow.setCreator(username); cyFlow.setCreateTime(new Date()); cyFlow.setId(null); diff --git a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/io/scheduler/NodeCopyJob.java b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/io/scheduler/NodeCopyJob.java deleted file mode 100644 index 1b1a6d2d7c..0000000000 --- a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/io/scheduler/NodeCopyJob.java +++ /dev/null @@ -1,269 +0,0 @@ -package com.webank.wedatasphere.dss.workflow.io.scheduler; - -import com.google.common.reflect.TypeToken; -import com.google.gson.Gson; -import com.google.gson.JsonObject; -import com.google.gson.JsonParser; -import com.webank.wedatasphere.dss.common.entity.node.DSSNode; -import com.webank.wedatasphere.dss.common.entity.node.DSSNodeDefault; -import com.webank.wedatasphere.dss.common.exception.DSSErrorException; -import com.webank.wedatasphere.dss.common.label.DSSLabel; -import com.webank.wedatasphere.dss.common.utils.IoUtils; -import com.webank.wedatasphere.dss.standard.app.sso.Workspace; -import com.webank.wedatasphere.dss.workflow.common.entity.DSSFlow; -import com.webank.wedatasphere.dss.workflow.dao.NodeInfoMapper; -import com.webank.wedatasphere.dss.workflow.entity.CommonAppConnNode; -import com.webank.wedatasphere.dss.workflow.entity.NodeInfo; -import com.webank.wedatasphere.dss.workflow.io.export.NodeExportService; -import com.webank.wedatasphere.dss.workflow.io.input.NodeInputService; -import com.webank.wedatasphere.dss.workflow.scheduler.DssJob; -import com.webank.wedatasphere.dss.workflow.service.WorkflowNodeService; -import org.apache.linkis.server.BDPJettyServerHelper; - -import java.io.File; -import java.io.IOException; -import java.util.List; -import java.util.Map; -import java.util.concurrent.CountDownLatch; -import java.util.concurrent.atomic.AtomicInteger; -import java.util.stream.Collectors; - -public class NodeCopyJob extends DssJob { - - private NodeInputService nodeInputService; - - private NodeExportService nodeExportService; - - private NodeInfoMapper nodeInfoMapper; - - private WorkflowNodeService workflowNodeService; - - private JobEntity jobEntity; - - private List> nodeJsonListRes; - - private AtomicInteger failedCount; - - private CountDownLatch countDownLatch; - - @Override - public void run() { - try { - //重新上传一份jar文件到bml - String updateNodeJson = inputNodeFiles(jobEntity.userName, jobEntity.projectName, jobEntity.nodeJson); - - Map nodeJsonMap = BDPJettyServerHelper.jacksonJson().readValue(updateNodeJson, Map.class); - //更新subflowID - String nodeType = nodeJsonMap.get("jobType").toString(); - NodeInfo nodeInfo = nodeInfoMapper.getWorkflowNodeByType(nodeType); - if ("workflow.subflow".equals(nodeType)) { - String subFlowName = nodeJsonMap.get("title").toString(); - List dssFlowList = jobEntity.getSubflows().stream().filter(subflow -> - subflow.getName().equals(subFlowName) - ).collect(Collectors.toList()); - if (dssFlowList.size() == 1) { - updateNodeJson = nodeInputService.updateNodeSubflowID(updateNodeJson, dssFlowList.get(0).getId()); - nodeJsonMap = BDPJettyServerHelper.jacksonJson().readValue(updateNodeJson, Map.class); - nodeJsonListRes.add(nodeJsonMap); - } else if (dssFlowList.size() > 1) { - logger.error("工程内存在重复的子工作流节点名称,导入失败" + subFlowName); - throw new DSSErrorException(90077, "工程内存在重复的子工作流节点名称,导入失败" + subFlowName); - } else { - logger.error("工程内存在重复的子工作流节点名称,导入失败" + subFlowName); - throw new DSSErrorException(90078, "工程内未能找到子工作流节点,导入失败" + subFlowName); - } -// } else if (nodeJsonMap.get("jobContent") != null && !((Map) nodeJsonMap.get("jobContent")).containsKey("script")) { - } else if (Boolean.TRUE.equals(nodeInfo.getSupportJump()) && nodeInfo.getJumpType() == 1) { - logger.info("nodeJsonMap.jobContent is:{}", nodeJsonMap.get("jobContent")); - CommonAppConnNode newNode = new CommonAppConnNode(); - CommonAppConnNode oldNode = new CommonAppConnNode(); - oldNode.setJobContent((Map) nodeJsonMap.get("jobContent")); - oldNode.setContextId(jobEntity.getUpdateContextId()); - oldNode.setNodeType(nodeType); - oldNode.setName((String) nodeJsonMap.get("title")); - oldNode.setFlowId(jobEntity.getDssFlow().getId()); - oldNode.setWorkspace(jobEntity.getWorkspace()); - oldNode.setDssLabels(jobEntity.getDssLabels()); - oldNode.setFlowName(jobEntity.getDssFlow().getName()); - oldNode.setProjectId(jobEntity.getDssFlow().getProjectID()); - newNode.setName(oldNode.getName()); - Map jobContent = workflowNodeService.copyNode(jobEntity.userName, newNode, oldNode, jobEntity.orcVersion); - nodeJsonMap.put("jobContent", jobContent); - nodeJsonListRes.add(nodeJsonMap); - } else { - nodeJsonListRes.add(nodeJsonMap); - } - } catch (Exception e) { - failedCount.getAndAdd(1); - logger.error("failed to copy node:", e); - } finally { - countDownLatch.countDown(); - } - } - - //由于每一个节点可能含有jar文件,这个功能不能直接复制使用,因为删掉新版本节点会直接删掉旧版本的node中的jar文件 - //所以重新上传一份jar文件到bml - private String inputNodeFiles(String userName, String projectName, String nodeJson) throws IOException { - String flowPath = IoUtils.generateIOPath(userName, projectName, ""); - String workFlowResourceSavePath = flowPath + File.separator + "resource" + File.separator; - Gson gson = new Gson(); - JsonParser parser = new JsonParser(); - JsonObject jsonObject = parser.parse(nodeJson).getAsJsonObject(); - DSSNode node = gson.fromJson(jsonObject, new TypeToken() { - }.getType()); - //先导出来 - nodeExportService.downloadNodeResourceToLocal(userName, node, workFlowResourceSavePath); - //后导入到bml - String updateNodeJson = nodeInputService.uploadResourceToBml(userName, nodeJson, workFlowResourceSavePath, projectName); - return updateNodeJson; - } - - public NodeInputService getNodeInputService() { - return nodeInputService; - } - - public void setNodeInputService(NodeInputService nodeInputService) { - this.nodeInputService = nodeInputService; - } - - public JobEntity getJobEntity() { - return jobEntity; - } - - public void setJobEntity(JobEntity jobEntity) { - this.jobEntity = jobEntity; - } - - public List> getNodeJsonListRes() { - return nodeJsonListRes; - } - - public void setNodeJsonListRes(List> nodeJsonListRes) { - this.nodeJsonListRes = nodeJsonListRes; - } - - public CountDownLatch getCountDownLatch() { - return countDownLatch; - } - - public void setCountDownLatch(CountDownLatch countDownLatch) { - this.countDownLatch = countDownLatch; - } - - public AtomicInteger getFailedCount() { - return failedCount; - } - - public void setFailedCount(AtomicInteger failedCount) { - this.failedCount = failedCount; - } - - public NodeExportService getNodeExportService() { - return nodeExportService; - } - - public void setNodeExportService(NodeExportService nodeExportService) { - this.nodeExportService = nodeExportService; - } - - public NodeInfoMapper getNodeInfoMapper() { - return nodeInfoMapper; - } - - public void setNodeInfoMapper(NodeInfoMapper nodeInfoMapper) { - this.nodeInfoMapper = nodeInfoMapper; - } - - public WorkflowNodeService getWorkflowNodeService() { - return workflowNodeService; - } - - public void setWorkflowNodeService(WorkflowNodeService workflowNodeService) { - this.workflowNodeService = workflowNodeService; - } - - - public static class JobEntity { - private String userName; - private String projectName; - private DSSFlow dssFlow; - private List subflows; - private Workspace workspace; - private String orcVersion; - private List dssLabels; - private String nodeJson; - private String updateContextId; - - public String getUpdateContextId() { - return updateContextId; - } - - public void setUpdateContextId(String updateContextId) { - this.updateContextId = updateContextId; - } - - public String getUserName() { - return userName; - } - - public void setUserName(String userName) { - this.userName = userName; - } - - public String getProjectName() { - return projectName; - } - - public void setProjectName(String projectName) { - this.projectName = projectName; - } - - public DSSFlow getDssFlow() { - return dssFlow; - } - - public void setDssFlow(DSSFlow dssFlow) { - this.dssFlow = dssFlow; - } - - public Workspace getWorkspace() { - return workspace; - } - - public void setWorkspace(Workspace workspace) { - this.workspace = workspace; - } - - public String getOrcVersion() { - return orcVersion; - } - - public void setOrcVersion(String orcVersion) { - this.orcVersion = orcVersion; - } - - public List getDssLabels() { - return dssLabels; - } - - public void setDssLabels(List dssLabels) { - this.dssLabels = dssLabels; - } - - public String getNodeJson() { - return nodeJson; - } - - public void setNodeJson(String nodeJson) { - this.nodeJson = nodeJson; - } - - public List getSubflows() { - return subflows; - } - - public void setSubflows(List subflows) { - this.subflows = subflows; - } - } -} diff --git a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/io/scheduler/NodeExportJob.java b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/io/scheduler/NodeExportJob.java deleted file mode 100644 index 6534d29f0e..0000000000 --- a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/io/scheduler/NodeExportJob.java +++ /dev/null @@ -1,4 +0,0 @@ -package com.webank.wedatasphere.dss.workflow.io.scheduler; - -public class NodeExportJob { -} diff --git a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/io/scheduler/NodeImportJob.java b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/io/scheduler/NodeImportJob.java deleted file mode 100644 index 7524db3015..0000000000 --- a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/io/scheduler/NodeImportJob.java +++ /dev/null @@ -1,220 +0,0 @@ -package com.webank.wedatasphere.dss.workflow.io.scheduler; - -import com.webank.wedatasphere.dss.common.exception.DSSErrorException; -import com.webank.wedatasphere.dss.common.label.DSSLabel; -import com.webank.wedatasphere.dss.standard.app.sso.Workspace; -import com.webank.wedatasphere.dss.workflow.common.entity.DSSFlow; -import com.webank.wedatasphere.dss.workflow.io.input.NodeInputService; -import com.webank.wedatasphere.dss.workflow.scheduler.DssJob; -import org.apache.commons.collections.CollectionUtils; -import org.apache.linkis.server.BDPJettyServerHelper; - -import java.util.List; -import java.util.Map; -import java.util.concurrent.CountDownLatch; -import java.util.concurrent.atomic.AtomicInteger; -import java.util.stream.Collectors; - -public class NodeImportJob extends DssJob { - - private NodeInputService nodeInputService; - - private ImportJobEntity jobEntity; - - private List> nodeJsonListRes; - - private AtomicInteger failedCount; - - private CountDownLatch countDownLatch; - - @Override - public void run() { - try { - // TODO: 2020/3/20 暂时注视掉appconn相关 - String updateNodeJson = nodeInputService.uploadResourceToBml(jobEntity.getUserName(), jobEntity.getNodeJson(), - jobEntity.getWorkFlowResourceSavePath(), jobEntity.getProjectName()); - updateNodeJson = nodeInputService.uploadAppConnResource(jobEntity.getUserName(), jobEntity.getProjectName(), - jobEntity.getDssFlow(), updateNodeJson, jobEntity.getUpdateContextId(), jobEntity.getAppConnResourceSavePath(), - jobEntity.getWorkspace(), jobEntity.getOrcVersion(), jobEntity.getDssLabels()); - //兼容0.x的key修改 - if (updateNodeJson.contains("wds.linkis.yarnqueue")) { - updateNodeJson = updateNodeJson.replace("wds.linkis.yarnqueue", "wds.linkis.rm.yarnqueue"); - } - Map nodeJsonMap = BDPJettyServerHelper.jacksonJson().readValue(updateNodeJson, Map.class); - //更新subflowID - String nodeType = nodeJsonMap.get("jobType").toString(); - if (nodeType.contains("appjoint")) { - nodeJsonMap.replace("jobType", nodeType.replace("appjoint", "appconn")); - } - if ("workflow.subflow".equals(nodeType) && CollectionUtils.isNotEmpty(jobEntity.getSubflows())) { - String subFlowName = nodeJsonMap.get("title").toString(); - logger.info("subflows:{}", jobEntity.getSubflows()); - List dssFlowList = jobEntity.getSubflows().stream().filter(subflow -> subflow.getName().equals(subFlowName)).collect(Collectors.toList()); - if (dssFlowList.size() == 1) { - updateNodeJson = nodeInputService.updateNodeSubflowID(updateNodeJson, dssFlowList.get(0).getId()); - nodeJsonMap = BDPJettyServerHelper.jacksonJson().readValue(updateNodeJson, Map.class); - nodeJsonListRes.add(nodeJsonMap); - } else if (dssFlowList.size() > 1) { - logger.error("工程内存在重复的子工作流节点名称,导入失败" + subFlowName); - throw new DSSErrorException(90077, "工程内存在重复的子工作流节点名称,导入失败" + subFlowName); - } else { - logger.error("工程内未能找到子工作流节点,导入失败" + subFlowName); - throw new DSSErrorException(90078, "工程内未能找到子工作流节点,导入失败" + subFlowName); - } - } else { - nodeJsonListRes.add(nodeJsonMap); - } -// countDownLatch.countDown(); - } catch (Exception e) { - //todo 失败重试 - failedCount.getAndAdd(1); - logger.error("failed to import node:", e); - }finally { - countDownLatch.countDown(); - } - - } - - public NodeInputService getNodeInputService() { - return nodeInputService; - } - - public void setNodeInputService(NodeInputService nodeInputService) { - this.nodeInputService = nodeInputService; - } - - public ImportJobEntity getJobEntity() { - return jobEntity; - } - - public void setJobEntity(ImportJobEntity jobEntity) { - this.jobEntity = jobEntity; - } - - public List> getNodeJsonListRes() { - return nodeJsonListRes; - } - - public void setNodeJsonListRes(List> nodeJsonListRes) { - this.nodeJsonListRes = nodeJsonListRes; - } - - public CountDownLatch getCountDownLatch() { - return countDownLatch; - } - - public void setCountDownLatch(CountDownLatch countDownLatch) { - this.countDownLatch = countDownLatch; - } - - public AtomicInteger getFailedCount() { - return failedCount; - } - - public void setFailedCount(AtomicInteger failedCount) { - this.failedCount = failedCount; - } - - - public static class ImportJobEntity { - private String userName; - private String projectName; - private DSSFlow dssFlow; - private List subflows; - private String workFlowResourceSavePath; - private String appConnResourceSavePath; - private Workspace workspace; - private String orcVersion; - private List dssLabels; - private String nodeJson; - private String updateContextId; - - public String getUpdateContextId() { - return updateContextId; - } - - public void setUpdateContextId(String updateContextId) { - this.updateContextId = updateContextId; - } - - public String getUserName() { - return userName; - } - - public void setUserName(String userName) { - this.userName = userName; - } - - public String getProjectName() { - return projectName; - } - - public void setProjectName(String projectName) { - this.projectName = projectName; - } - - public DSSFlow getDssFlow() { - return dssFlow; - } - - public void setDssFlow(DSSFlow dssFlow) { - this.dssFlow = dssFlow; - } - - public Workspace getWorkspace() { - return workspace; - } - - public void setWorkspace(Workspace workspace) { - this.workspace = workspace; - } - - public String getOrcVersion() { - return orcVersion; - } - - public void setOrcVersion(String orcVersion) { - this.orcVersion = orcVersion; - } - - public List getDssLabels() { - return dssLabels; - } - - public void setDssLabels(List dssLabels) { - this.dssLabels = dssLabels; - } - - public String getNodeJson() { - return nodeJson; - } - - public void setNodeJson(String nodeJson) { - this.nodeJson = nodeJson; - } - - public String getWorkFlowResourceSavePath() { - return workFlowResourceSavePath; - } - - public void setWorkFlowResourceSavePath(String workFlowResourceSavePath) { - this.workFlowResourceSavePath = workFlowResourceSavePath; - } - - public String getAppConnResourceSavePath() { - return appConnResourceSavePath; - } - - public void setAppConnResourceSavePath(String appConnResourceSavePath) { - this.appConnResourceSavePath = appConnResourceSavePath; - } - - public List getSubflows() { - return subflows; - } - - public void setSubflows(List subflows) { - this.subflows = subflows; - } - } -} diff --git a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/lock/DSSFlowEditLockManager.java b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/lock/DSSFlowEditLockManager.java index 2789d936f3..2bb3298aa3 100644 --- a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/lock/DSSFlowEditLockManager.java +++ b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/lock/DSSFlowEditLockManager.java @@ -161,7 +161,7 @@ private static String generateLock(Long flowID, String username, String owner) t unLockEvents.offer(unLockEvent); return lockContent; } catch (DuplicateKeyException e) { - LOGGER.warn("acquire lock failed", e); + LOGGER.warn("acquire lock failed", e.getMessage()); DSSFlowEditLock personalFlowEditLock = lockMapper.getPersonalFlowEditLock(flowID, null); String userName = Optional.ofNullable(personalFlowEditLock).map(DSSFlowEditLock::getUsername).orElse(null); throw new DSSErrorException(DSSWorkFlowConstant.EDIT_LOCK_ERROR_CODE, "用户" + userName + "已锁定编辑"); diff --git a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/restful/ContextServiceRestful.java b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/restful/ContextServiceRestful.java index b4e322a875..a73fb81515 100644 --- a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/restful/ContextServiceRestful.java +++ b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/restful/ContextServiceRestful.java @@ -22,6 +22,8 @@ import com.webank.wedatasphere.dss.workflow.entity.request.TablesRequest; import org.apache.linkis.server.Message; import org.apache.linkis.server.security.SecurityFilter; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.web.bind.annotation.RequestBody; import org.springframework.web.bind.annotation.RequestMapping; @@ -34,6 +36,8 @@ @RestController public class ContextServiceRestful { + private final Logger logger = LoggerFactory.getLogger(getClass()); + @Autowired private CSTableService csTableService; @@ -42,6 +46,7 @@ public Message tables(HttpServletRequest req, @RequestBody TablesRequest tablesR String userName = SecurityFilter.getLoginUsername(req); String contextIDStr = tablesRequest.getContextID(); String nodeName = tablesRequest.getNodeName(); + logger.info("Begin to get cs tables, contextId:{}, nodeName:{}", contextIDStr, nodeName); return Message.ok().data("tables", csTableService.queryTables("default", contextIDStr, nodeName)); } @@ -50,6 +55,7 @@ public Message queryTableMeta(HttpServletRequest req,@RequestBody QueryTableMeta String userName = SecurityFilter.getLoginUsername(req); String contextIDStr = queryTableMetaRequest.getContextID(); String contextKeyStr = queryTableMetaRequest.getContextKey(); + logger.info("Begin to get cs columns, contextId:{}, contextKey:{}", contextIDStr, contextKeyStr); return Message.ok().data("columns", csTableService.queryTableMeta("default", contextIDStr, contextKeyStr)); } diff --git a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/restful/FlowRestfulApi.java b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/restful/FlowRestfulApi.java index 6ce40ccfbe..2e8fa292c0 100644 --- a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/restful/FlowRestfulApi.java +++ b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/restful/FlowRestfulApi.java @@ -17,10 +17,16 @@ package com.webank.wedatasphere.dss.workflow.restful; import com.fasterxml.jackson.core.JsonProcessingException; +import com.google.gson.Gson; +import com.google.gson.JsonObject; import com.webank.wedatasphere.dss.appconn.manager.utils.AppConnManagerUtils; +import com.webank.wedatasphere.dss.common.auditlog.OperateTypeEnum; +import com.webank.wedatasphere.dss.common.auditlog.TargetTypeEnum; import com.webank.wedatasphere.dss.common.exception.DSSErrorException; import com.webank.wedatasphere.dss.common.label.DSSLabel; import com.webank.wedatasphere.dss.common.label.EnvDSSLabel; +import com.webank.wedatasphere.dss.common.utils.AuditLogUtils; +import com.webank.wedatasphere.dss.common.utils.DSSExceptionUtils; import com.webank.wedatasphere.dss.contextservice.service.ContextService; import com.webank.wedatasphere.dss.contextservice.service.impl.ContextServiceImpl; import com.webank.wedatasphere.dss.orchestrator.common.protocol.ResponseConvertOrchestrator; @@ -29,18 +35,26 @@ import com.webank.wedatasphere.dss.workflow.WorkFlowManager; import com.webank.wedatasphere.dss.workflow.common.entity.DSSFlow; import com.webank.wedatasphere.dss.workflow.constant.DSSWorkFlowConstant; +import com.webank.wedatasphere.dss.workflow.dao.LockMapper; +import com.webank.wedatasphere.dss.workflow.entity.DSSFlowEditLock; import com.webank.wedatasphere.dss.workflow.entity.request.*; import com.webank.wedatasphere.dss.workflow.entity.vo.ExtraToolBarsVO; -import com.webank.wedatasphere.dss.workflow.exception.DSSWorkflowErrorException; import com.webank.wedatasphere.dss.workflow.lock.DSSFlowEditLockManager; +import com.webank.wedatasphere.dss.common.service.BMLService; import com.webank.wedatasphere.dss.workflow.service.DSSFlowService; import com.webank.wedatasphere.dss.workflow.service.PublishService; +import org.apache.commons.collections.CollectionUtils; import org.apache.commons.lang.StringUtils; +import org.apache.linkis.common.exception.ErrorException; +import org.apache.linkis.cs.client.utils.SerializeHelper; +import org.apache.linkis.cs.common.entity.source.LinkisHAWorkFlowContextID; +import org.apache.linkis.cs.common.utils.CSCommonUtils; import org.apache.linkis.server.Message; import org.apache.linkis.server.security.SecurityFilter; 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.web.bind.annotation.*; import javax.annotation.PostConstruct; @@ -50,6 +64,7 @@ import java.io.IOException; import java.util.*; + @RestController @RequestMapping(path = "/dss/workflow", produces = {"application/json"}) public class FlowRestfulApi { @@ -61,9 +76,14 @@ public class FlowRestfulApi { @Autowired private PublishService publishService; @Autowired - private DSSFlowService dssFlowService; + @Qualifier("workflowBmlService") + private BMLService bmlService; @Autowired private WorkFlowManager workFlowManager; + @Autowired + private LockMapper lockMapper; + @Autowired + private HttpServletRequest httpServletRequest; @PostConstruct public void init() { @@ -73,74 +93,89 @@ public void init() { /** * 添加subflow节点 * - * @param req + * @param addFlowRequest * @return * @throws DSSErrorException * @throws JsonProcessingException */ @RequestMapping(value = "addFlow", method = RequestMethod.POST) - public Message addFlow(HttpServletRequest req, @RequestBody AddFlowRequest addFlowRequest) throws DSSErrorException, JsonProcessingException { + public Message addFlow( @RequestBody AddFlowRequest addFlowRequest) throws ErrorException, JsonProcessingException { //如果是子工作流,那么分类应该是和父类一起的? - String userName = SecurityFilter.getLoginUsername(req); + String userName = SecurityFilter.getLoginUsername(httpServletRequest); + Workspace workspace = SSOHelper.getWorkspace(httpServletRequest); // TODO: 2019/5/23 flowName工程名下唯一校验 String name = addFlowRequest.getName(); + Long parentFlowID = addFlowRequest.getParentFlowID(); + // 判断parentFlowID中是否已存在名称为name的subflow + if (flowService.checkExistSameSubflow(parentFlowID, name)){ + return Message.error("子工作流名不能重复"); + } String workspaceName = addFlowRequest.getWorkspaceName(); String projectName = addFlowRequest.getProjectName(); String version = addFlowRequest.getVersion(); + //subflow中的subflow节点的version需要从父工作流获取 + DSSFlow parentFlow = flowService.getFlow(parentFlowID); + JsonObject jsonObject = new Gson().fromJson(parentFlow.getFlowJson(), JsonObject.class); + //schedulerAppconnName从parentFlow的json中获取 + String schedulerAppconnName = jsonObject.get(DSSWorkFlowConstant.SCHEDULER_APP_CONN_NAME) == null + ? null : jsonObject.get(DSSWorkFlowConstant.SCHEDULER_APP_CONN_NAME).getAsString(); + if (StringUtils.isBlank(version)) { + LinkisHAWorkFlowContextID contextID = (LinkisHAWorkFlowContextID) SerializeHelper + .deserializeContextID(jsonObject.get(CSCommonUtils.CONTEXT_ID_STR).getAsString()); + LOGGER.info("contextID version from parent flow is: {}", contextID.getVersion()); + version = contextID.getVersion(); + } String description = addFlowRequest.getDescription(); - Long parentFlowID = addFlowRequest.getParentFlowID(); String uses = addFlowRequest.getUses(); List dssLabelList = new ArrayList<>(); dssLabelList.add(new EnvDSSLabel(addFlowRequest.getLabels().getRoute())); String contextId = contextService.createContextID(workspaceName, projectName, name, version, userName); + AuditLogUtils.printLog(userName, String.valueOf(workspace.getWorkspaceId()), workspaceName, TargetTypeEnum.CONTEXTID, + contextId, name, OperateTypeEnum.CREATE, addFlowRequest); + //projectId在后面逻辑中填充 DSSFlow dssFlow = workFlowManager.createWorkflow(userName, null, name, contextId, description, parentFlowID, - uses, new ArrayList<>(), dssLabelList, null, null); - - // TODO: 2019/5/16 空值校验,重复名校验 + uses, new ArrayList<>(), dssLabelList, version, schedulerAppconnName); + AuditLogUtils.printLog(userName, workspace.getWorkspaceId(), workspaceName, TargetTypeEnum.WORKFLOW, + dssFlow.getId(), name, OperateTypeEnum.CREATE, addFlowRequest); + LOGGER.info("User {} end to add flow, name:{}, projectName:{}", userName, name, projectName); return Message.ok().data("flow", dssFlow); } @RequestMapping(value = "publishWorkflow", method = RequestMethod.POST) - public Message publishWorkflow(HttpServletRequest request, @RequestBody PublishWorkflowRequest publishWorkflowRequest) throws Exception { + public Message publishWorkflow(@RequestBody PublishWorkflowRequest publishWorkflowRequest) throws Exception { Long workflowId = publishWorkflowRequest.getWorkflowId(); Map labels = new HashMap<>(); labels.put(EnvDSSLabel.DSS_ENV_LABEL_KEY, publishWorkflowRequest.getLabels().getRoute()); String comment = publishWorkflowRequest.getComment(); - Workspace workspace = SSOHelper.getWorkspace(request); - String publishUser = SecurityFilter.getLoginUsername(request); - Message message; - try { - String taskId = publishService.submitPublish(publishUser, workflowId, labels, workspace, comment); - LOGGER.info("submit publish task ok ,taskId is {}.", taskId); - if (DSSWorkFlowConstant.PUBLISHING_ERROR_CODE.equals(taskId)) { - message = Message.error("发布工程已经含有工作流,正在发布中,请稍后再试"); - } else if (StringUtils.isNotEmpty(taskId)) { - message = Message.ok("生成工作流发布任务成功").data("releaseTaskId", taskId); - } else { - LOGGER.error("taskId {} is error.", taskId); - message = Message.error("发布工作流失败"); - } - } catch (DSSWorkflowErrorException e) { - throw e; - } catch (final Throwable t) { - LOGGER.error("failed to submit publish task for workflow id {}.", workflowId, t); - message = Message.error("发布工作流失败"); - } - return message; + Workspace workspace = SSOHelper.getWorkspace(httpServletRequest); + String publishUser = SecurityFilter.getLoginUsername(httpServletRequest); + return DSSExceptionUtils.getMessage(() -> publishService.submitPublish(publishUser, workflowId, labels, workspace, comment), + taskId -> { + if (DSSWorkFlowConstant.PUBLISHING_ERROR_CODE.equals(taskId)) { + return Message.error("发布工程已经含有工作流,正在发布中,请稍后再试"); + } else if (StringUtils.isNotEmpty(taskId)) { + //发布正常,说明dssflow一定存在,所以不需要判空。 + DSSFlow dssFlow = flowService.getFlowByID(workflowId); + AuditLogUtils.printLog(publishUser,workspace.getWorkspaceId(), workspace.getWorkspaceName(), + TargetTypeEnum.WORKFLOW,workflowId, dssFlow.getName(),OperateTypeEnum.PUBLISH, publishWorkflowRequest); + return Message.ok("生成工作流发布任务成功").data("releaseTaskId", taskId); + } else { + LOGGER.error("taskId {} is error.", taskId); + return Message.error("发布工作流失败"); + }}, + String.format("用户 %s 发布工作流 %s 失败.", publishUser, workflowId)); } /** * 获取发布任务状态 * - * @param request * @param releaseTaskId * @return */ @RequestMapping(value = "getReleaseStatus", method = RequestMethod.GET) - public Message getReleaseStatus(HttpServletRequest request, - @NotNull(message = "查询的发布id不能为空") @RequestParam(required = false, name = "releaseTaskId") Long releaseTaskId) { - String username = SecurityFilter.getLoginUsername(request); + public Message getReleaseStatus(@NotNull(message = "查询的发布id不能为空") @RequestParam(required = false, name = "releaseTaskId") Long releaseTaskId) { + String username = SecurityFilter.getLoginUsername(httpServletRequest); Message message; try { ResponseConvertOrchestrator response = publishService.getStatus(username, releaseTaskId.toString()); @@ -171,18 +206,24 @@ public Message getReleaseStatus(HttpServletRequest request, /** * 更新工作流的基本信息,不包括更新Json,BML版本等 * - * @param req * @param updateFlowBaseInfoRequest * @return * @throws DSSErrorException */ @RequestMapping(value = "updateFlowBaseInfo", method = RequestMethod.POST) - public Message updateFlowBaseInfo(HttpServletRequest req, @RequestBody UpdateFlowBaseInfoRequest updateFlowBaseInfoRequest) throws DSSErrorException { + public Message updateFlowBaseInfo(@RequestBody UpdateFlowBaseInfoRequest updateFlowBaseInfoRequest) throws DSSErrorException { + String username = SecurityFilter.getLoginUsername(httpServletRequest); + Workspace workspace = SSOHelper.getWorkspace(httpServletRequest); Long flowID = updateFlowBaseInfoRequest.getId(); String name = updateFlowBaseInfoRequest.getName(); String description = updateFlowBaseInfoRequest.getDescription(); String uses = updateFlowBaseInfoRequest.getUses(); + Long parentFlowID = flowService.getParentFlowID(flowID); + DSSFlow flowByID = flowService.getFlowByID(flowID); + if (flowService.checkExistSameFlow(parentFlowID, name, flowByID.getName())){ + return Message.error("子工作流名不能重复"); + } // TODO: 2019/6/13 projectVersionID的更新校验 //这里可以不做事务 DSSFlow dssFlow = new DSSFlow(); @@ -191,37 +232,47 @@ public Message updateFlowBaseInfo(HttpServletRequest req, @RequestBody UpdateFlo dssFlow.setDescription(description); dssFlow.setUses(uses); flowService.updateFlowBaseInfo(dssFlow); + AuditLogUtils.printLog( username,workspace.getWorkspaceId(), workspace.getWorkspaceName(), + TargetTypeEnum.WORKFLOW,flowID,name,OperateTypeEnum.UPDATE,updateFlowBaseInfoRequest); return Message.ok(); } /** * 读取工作流的Json数据,提供给前端渲染 * - * @param req * @param flowID * @return * @throws DSSErrorException */ @RequestMapping(value = "get", method = RequestMethod.GET) - public Message get(HttpServletRequest req, - @RequestParam(required = false, name = "flowId") Long flowID, + public Message get(@RequestParam(required = false, name = "flowId") Long flowID, @RequestParam(required = false, name = "isNotHaveLock") Boolean isNotHaveLock) throws DSSErrorException { - String username = SecurityFilter.getLoginUsername(req); - DSSFlow dssFlow = flowService.getFlow(flowID); + String username = SecurityFilter.getLoginUsername(httpServletRequest); + LOGGER.info("User {} start to open workflow {}", username, flowID); + DSSFlow dssFlow; + try { + dssFlow = flowService.getFlow(flowID); + } catch (NullPointerException e) { + return Message.error("The workflow is not exists, please check to delete. (打开了不存在的工作流,请确保是否已删除.)"); + } if (isNotHaveLock != null && isNotHaveLock) { return Message.ok().data("flow", dssFlow); } - Cookie[] cookies = req.getCookies(); + Cookie[] cookies = httpServletRequest.getCookies(); String ticketId = Arrays.stream(cookies).filter(cookie -> DSSWorkFlowConstant.BDP_USER_TICKET_ID.equals(cookie.getName())).findFirst().map(Cookie::getValue).get(); if (dssFlow != null) { // 尝试获取工作流编辑锁 try { - String flowEditLock = DSSFlowEditLockManager.tryAcquireLock(dssFlow, username, ticketId); - dssFlow.setFlowEditLock(flowEditLock); + //只有父工作流才有锁,子工作流复用父工作流的锁 + if(dssFlow.getRootFlow()) { + String flowEditLock = DSSFlowEditLockManager.tryAcquireLock(dssFlow, username, ticketId); + dssFlow.setFlowEditLock(flowEditLock); + } } catch (DSSErrorException e) { if (DSSWorkFlowConstant.EDIT_LOCK_ERROR_CODE == e.getErrCode()) { - return Message.error(e.getDesc()); + DSSFlowEditLock flowEditLock = lockMapper.getFlowEditLockByID(flowID); + return Message.error(e.getDesc()).data("editLockInfo", flowEditLock); } throw e; } @@ -230,57 +281,75 @@ public Message get(HttpServletRequest req, } @RequestMapping(value = "deleteFlow", method = RequestMethod.POST) - public Message deleteFlow(HttpServletRequest req, @RequestBody DeleteFlowRequest deleteFlowRequest) throws DSSErrorException { + public Message deleteFlow(@RequestBody DeleteFlowRequest deleteFlowRequest) throws DSSErrorException { + String username = SecurityFilter.getLoginUsername(httpServletRequest); + Workspace workspace = SSOHelper.getWorkspace(httpServletRequest); Long flowID = deleteFlowRequest.getId(); - boolean sure = deleteFlowRequest.getSure() != null && deleteFlowRequest.getSure().booleanValue(); + boolean sure = deleteFlowRequest.getSure() != null && deleteFlowRequest.getSure(); // TODO: 2019/6/13 projectVersionID的更新校验 //state为true代表曾经发布过 - if (flowService.getFlowByID(flowID).getState() && !sure) { + DSSFlow dssFlow = flowService.getFlowByID(flowID); + if (dssFlow != null && dssFlow.getState() && !sure) { return Message.ok().data("warmMsg", "该工作流曾经发布过,删除将会将该工作流的所有版本都删除,是否继续?"); } flowService.batchDeleteFlow(Arrays.asList(flowID)); + AuditLogUtils.printLog(username, workspace.getWorkspaceId(), workspace.getWorkspaceName(), TargetTypeEnum.WORKFLOW, + flowID, dssFlow == null ? null : dssFlow.getName(), OperateTypeEnum.DELETE, deleteFlowRequest); return Message.ok(); } /** * 工作流保存接口,如工作流Json内容有变化,会更新工作流的Json内容 * - * @param req * @param saveFlowRequest * @return * @throws DSSErrorException * @throws IOException */ @RequestMapping(value = "saveFlow", method = RequestMethod.POST) -// @ProjectPrivChecker - public Message saveFlow(HttpServletRequest req, @RequestBody SaveFlowRequest saveFlowRequest) throws DSSErrorException, IOException { + public Message saveFlow( @RequestBody SaveFlowRequest saveFlowRequest) throws IOException { + String username = SecurityFilter.getLoginUsername(httpServletRequest); + Long flowID = saveFlowRequest.getId(); String jsonFlow = saveFlowRequest.getJson(); + Workspace workspace = SSOHelper.getWorkspace(httpServletRequest); + DSSFlow dssFlow = flowService.getFlowByID(flowID); String workspaceName = saveFlowRequest.getWorkspaceName(); String projectName = saveFlowRequest.getProjectName(); + // 判断工作流中是否存在命名相同的节点 + if (flowService.checkIsExistSameFlow(jsonFlow)) { + return Message.error("It exists same flow.(存在相同的节点)"); + } + // 判断工作流中是否有子工作流未被保存 + List unSaveNodes = flowService.checkIsSave(flowID, jsonFlow); + if (CollectionUtils.isNotEmpty(unSaveNodes)) { + return Message.error("工作流中存在子工作流未被保存,请先保存子工作流:" + unSaveNodes); + } - Boolean isNotHaveLock = saveFlowRequest.getNotHaveLock(); - String userName = SecurityFilter.getLoginUsername(req); + String userName = SecurityFilter.getLoginUsername(httpServletRequest); String version; - synchronized (DSSWorkFlowConstant.saveFlowLock.intern(flowID)) { - if (isNotHaveLock != null && isNotHaveLock.booleanValue()) { - version = flowService.saveFlow(flowID, jsonFlow, null, userName, workspaceName, projectName); - return Message.ok().data("flowVersion", version).data("flowEditLock", null); - } - version = flowService.saveFlow(flowID, jsonFlow, null, userName, workspaceName, projectName); + //若工作流已经被其他用户抢锁,则当前用户不能再保存成功 + String ticketId = Arrays.stream(httpServletRequest.getCookies()).filter(cookie -> DSSWorkFlowConstant.BDP_USER_TICKET_ID.equals(cookie.getName())) + .findFirst().map(Cookie::getValue).get(); + DSSFlowEditLock flowEditLock = lockMapper.getFlowEditLockByID(flowID); + if (flowEditLock != null && !flowEditLock.getOwner().equals(ticketId)) { + return Message.error("当前工作流被用户" + flowEditLock.getUsername() + "已锁定编辑,您编辑的内容不能再被保存。如有疑问,请与" + flowEditLock.getUsername() + "确认"); } + version = flowService.saveFlow(flowID, jsonFlow, null, userName, workspaceName, projectName); + AuditLogUtils.printLog( username,workspace.getWorkspaceId(), workspaceName,TargetTypeEnum.WORKFLOW, + flowID,dssFlow.getName(),OperateTypeEnum.UPDATE,saveFlowRequest); return Message.ok().data("flowVersion", version); } + /** * 工作流编辑锁更新接口 * - * @param req request * @param flowEditLock 老的编辑锁 * @return 新的编辑锁 */ @RequestMapping(value = "/updateFlowEditLock", method = RequestMethod.GET) - public Message updateFlowEditLock(HttpServletRequest req, @RequestParam(required = false, name = "flowEditLock") String flowEditLock) throws DSSErrorException { + public Message updateFlowEditLock(@RequestParam(required = false, name = "flowEditLock") String flowEditLock) throws DSSErrorException { if (StringUtils.isBlank(flowEditLock)) { throw new DSSErrorException(60067, "update flowEditLock failed,because flowEditLock is empty"); } @@ -288,9 +357,8 @@ public Message updateFlowEditLock(HttpServletRequest req, @RequestParam(required } @RequestMapping(value = "/getExtraToolBars", method = RequestMethod.POST) - public Message getExtraToolBars(HttpServletRequest req, @RequestBody GetExtraToolBarsRequest getExtraToolBarsRequest) throws DSSErrorException { - String userName = SecurityFilter.getLoginUsername(req); - Workspace workspace = SSOHelper.getWorkspace(req); + public Message getExtraToolBars( @RequestBody GetExtraToolBarsRequest getExtraToolBarsRequest) throws DSSErrorException { + Workspace workspace = SSOHelper.getWorkspace(httpServletRequest); List barsVOList = flowService.getExtraToolBars(workspace.getWorkspaceId(), getExtraToolBarsRequest.getProjectId()); return Message.ok().data("extraBars", barsVOList); } diff --git a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/restful/NodeRestfulApi.java b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/restful/NodeRestfulApi.java index e634be52cb..695fd75e9a 100644 --- a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/restful/NodeRestfulApi.java +++ b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/restful/NodeRestfulApi.java @@ -21,6 +21,7 @@ import com.webank.wedatasphere.dss.common.exception.DSSErrorException; import com.webank.wedatasphere.dss.common.exception.DSSRuntimeException; import com.webank.wedatasphere.dss.common.label.EnvDSSLabel; +import com.webank.wedatasphere.dss.common.utils.DSSExceptionUtils; import com.webank.wedatasphere.dss.standard.app.development.utils.DSSJobContentConstant; import com.webank.wedatasphere.dss.standard.app.sso.Workspace; import com.webank.wedatasphere.dss.standard.common.exception.operation.ExternalOperationFailedException; @@ -55,10 +56,7 @@ import javax.servlet.http.HttpServletRequest; import java.io.File; import java.io.IOException; -import java.util.ArrayList; -import java.util.Collections; -import java.util.List; -import java.util.Map; +import java.util.*; import java.util.function.Function; import java.util.stream.Collectors; @@ -87,7 +85,7 @@ public Message listNodeType(HttpServletRequest req) { try { return transfer(n, req); } catch (IOException e) { - logger.error("listNodeType get icons failed.", e); + logger.error("ListNodeType get AppConn {} icons failed.", n.getAppConnName(), e); throw new DSSRuntimeException(81200, e.getMessage(), e); } }).collect(Collectors.toList())); @@ -129,13 +127,19 @@ private NodeInfoVO transfer(NodeInfo nodeInfo, HttpServletRequest req) throws IO Function descriptionSupplier = internationalization(req, NodeUi::getDescriptionEn, NodeUi::getDescription); Function labelNameSupplier = internationalization(req, NodeUi::getLableNameEn, NodeUi::getLableName); ArrayList nodeUiVOS = new ArrayList<>(); + Set keySet = new HashSet<>(nodeInfo.getNodeUis().size()); for (NodeUi nodeUi : nodeInfo.getNodeUis()) { + //避免重复的ui key,因为第三方组件可能会重复配置。 + if(keySet.contains(nodeUi.getKey())){ + continue; + } NodeUiVO nodeUiVO = new NodeUiVO(); BeanUtils.copyProperties(nodeUi, nodeUiVO); nodeUiVO.setDesc(descriptionSupplier.apply(nodeUi)); nodeUiVO.setLableName(labelNameSupplier.apply(nodeUi)); nodeUiVO.setNodeUiValidateVOS(nodeUi.getNodeUiValidates().stream().map(v -> transfer(v, req)).sorted(NodeUiValidateVO::compareTo).collect(Collectors.toList())); nodeUiVOS.add(nodeUiVO); + keySet.add(nodeUi.getKey()); } nodeUiVOS.sort(NodeUiVO::compareTo); nodeInfoVO.setNodeUiVOS(nodeUiVOS); @@ -146,9 +150,9 @@ private String getIcon(NodeInfo nodeInfo) throws IOException { String appConnHomePath = AppConnManager.getAppConnManager().getAppConnHomePath(nodeInfo.getAppConnName()); File iconPath = new File(appConnHomePath, nodeInfo.getIconPath()); if(!iconPath.exists()) { - throw new IOException("Get icon failed. Caused by: " + iconPath + " not exists."); + throw new IOException("Get icon failed. Caused by: AppConn " + nodeInfo.getAppConnName() + "'s " + iconPath + " not exists."); } else if(!iconPath.isFile()) { - throw new IOException("Get icon failed. Caused by: " + iconPath + " is not a file."); + throw new IOException("Get icon failed. Caused by: AppConn " + nodeInfo.getAppConnName() + "'s " + iconPath + " is not a file."); } return FileUtils.readFileToString(iconPath); } @@ -163,8 +167,7 @@ public Message createExternalNode(HttpServletRequest req, @RequestBody CreateExt Map params = createExternalNodeRequest.getParams(); String nodeId = createExternalNodeRequest.getNodeID(); - logger.info("User {} try to create a {} node for workflow {} in projectId {}, params is {}.", - userName, nodeType, flowId, projectId, params); + logger.info("User {} try to create a {} node for workflow {}, params is {}.", userName, nodeType, flowId, params); CommonAppConnNode node = new CommonAppConnNode(); node.setNodeType(nodeType); node.setFlowId(flowId); @@ -195,7 +198,7 @@ public Message createExternalNode(HttpServletRequest req, @RequestBody CreateExt dssNodes = workFlowParser.getWorkFlowNodes(flowContent).stream() .filter(dssNode -> ArrayUtils.contains(upStreams, dssNode.getId())).collect(Collectors.toList()); if(dssNodes.isEmpty() && upStreams.length > 0) { - return Message.error("create node failed! Caused by: the banding up-stream nodes are not exists(绑定的上游节点不存在)."); + return Message.error("Create node failed! Caused by: the banding up-stream nodes are not exists(绑定的上游节点不存在)."); } params.put(DSSJobContentConstant.UP_STREAM_KEY, dssNodes); } @@ -271,6 +274,7 @@ public Message batchDeleteAppConnNode(HttpServletRequest req, @RequestBody Batch node.setNodeType(nodeType); node.setJobContent(params); node.setFlowId(flowId); + node.setName(json.get("name").toString()); workflowNodeService.deleteNode(userName, node); }); @@ -278,7 +282,7 @@ public Message batchDeleteAppConnNode(HttpServletRequest req, @RequestBody Batch } @RequestMapping(value = "/getAppConnNodeUrl",method = RequestMethod.POST) - public Message getAppConnNodeUrl(HttpServletRequest req, @RequestBody AppConnNodeUrlRequest appConnNodeUrlRequest) throws IllegalAccessException, ExternalOperationFailedException, InstantiationException { + public Message getAppConnNodeUrl(HttpServletRequest req, @RequestBody AppConnNodeUrlRequest appConnNodeUrlRequest) { String userName = SecurityFilter.getLoginUsername(req); Workspace workspace = SSOHelper.getWorkspace(req); Long projectId = appConnNodeUrlRequest.getProjectID(); diff --git a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/service/DSSFlowService.java b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/service/DSSFlowService.java index 1907728d74..a001f038ac 100644 --- a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/service/DSSFlowService.java +++ b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/service/DSSFlowService.java @@ -21,6 +21,7 @@ import com.webank.wedatasphere.dss.standard.app.sso.Workspace; import com.webank.wedatasphere.dss.workflow.common.entity.DSSFlow; import com.webank.wedatasphere.dss.workflow.entity.vo.ExtraToolBarsVO; +import org.apache.linkis.common.exception.ErrorException; import java.io.IOException; import java.util.List; @@ -36,12 +37,14 @@ public interface DSSFlowService { /** * 通过flowID获取最新版本的dwsFlow,版本信息在latestVersion + * * @return */ DSSFlow getFlow(Long flowId); - void updateFlowBaseInfo(DSSFlow dssFlow) throws DSSErrorException; + List getSubFlowContextIdsByFlowIds(List flowIdList) throws ErrorException; + void updateFlowBaseInfo(DSSFlow dssFlow) throws DSSErrorException; void batchDeleteFlow(List flowIdlist); @@ -51,13 +54,22 @@ String saveFlow(Long flowId, String userName, String workspaceName, String projectName - ) throws DSSErrorException; + ) throws IOException; DSSFlow copyRootFlow(Long rootFlowId, String userName, Workspace workspace, String projectName, String version, String contextIdStr, - String description, List dssLabels) throws DSSErrorException, IOException; + String description, List dssLabels,String nodeSuffix, + String newFlowName, Long newProjectId) throws DSSErrorException, IOException; Long getParentFlowID(Long id); List getExtraToolBars(long workspaceId, long projectId); + + boolean checkExistSameSubflow(Long parentFlowID, String name); + + boolean checkExistSameFlow(Long parentFlowID, String name, String existName); + + boolean checkIsExistSameFlow(String jsonFlow); + + List checkIsSave(Long parentFlowID, String jsonFlow); } diff --git a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/service/WorkflowNodeService.java b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/service/WorkflowNodeService.java index 89e87f00ee..2baaeb6b39 100644 --- a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/service/WorkflowNodeService.java +++ b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/service/WorkflowNodeService.java @@ -16,6 +16,7 @@ package com.webank.wedatasphere.dss.workflow.service; +import com.webank.wedatasphere.dss.common.entity.BmlResource; import com.webank.wedatasphere.dss.common.exception.DSSErrorException; import com.webank.wedatasphere.dss.standard.app.development.ref.ExportResponseRef; import com.webank.wedatasphere.dss.standard.common.exception.operation.ExternalOperationFailedException; diff --git a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/service/impl/DSSFlowServiceImpl.java b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/service/impl/DSSFlowServiceImpl.java index 16d4c7d8f7..50939f8858 100644 --- a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/service/impl/DSSFlowServiceImpl.java +++ b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/service/impl/DSSFlowServiceImpl.java @@ -21,10 +21,14 @@ import com.google.gson.Gson; import com.google.gson.JsonObject; import com.google.gson.JsonParser; +import com.webank.wedatasphere.dss.common.entity.BmlResource; import com.webank.wedatasphere.dss.common.entity.Resource; +import com.webank.wedatasphere.dss.common.entity.node.DSSEdge; import com.webank.wedatasphere.dss.common.entity.node.DSSNode; import com.webank.wedatasphere.dss.common.entity.node.DSSNodeDefault; +import com.webank.wedatasphere.dss.common.entity.node.Node; import com.webank.wedatasphere.dss.common.exception.DSSErrorException; +import com.webank.wedatasphere.dss.common.exception.DSSRuntimeException; import com.webank.wedatasphere.dss.common.label.DSSLabel; import com.webank.wedatasphere.dss.common.utils.DSSCommonUtils; import com.webank.wedatasphere.dss.common.utils.IoUtils; @@ -35,8 +39,13 @@ import com.webank.wedatasphere.dss.standard.app.sso.Workspace; import com.webank.wedatasphere.dss.workflow.common.entity.DSSFlow; import com.webank.wedatasphere.dss.workflow.common.entity.DSSFlowRelation; +import com.webank.wedatasphere.dss.workflow.common.parser.NodeParser; import com.webank.wedatasphere.dss.workflow.common.parser.WorkFlowParser; import com.webank.wedatasphere.dss.workflow.constant.DSSWorkFlowConstant; +import com.webank.wedatasphere.dss.workflow.core.WorkflowFactory; +import com.webank.wedatasphere.dss.workflow.core.entity.Workflow; +import com.webank.wedatasphere.dss.workflow.core.entity.WorkflowWithContextImpl; +import com.webank.wedatasphere.dss.workflow.core.json2flow.JsonToFlowParser; import com.webank.wedatasphere.dss.workflow.dao.FlowMapper; import com.webank.wedatasphere.dss.workflow.dao.NodeInfoMapper; import com.webank.wedatasphere.dss.workflow.entity.CommonAppConnNode; @@ -45,18 +54,22 @@ import com.webank.wedatasphere.dss.workflow.io.export.NodeExportService; import com.webank.wedatasphere.dss.workflow.io.input.NodeInputService; import com.webank.wedatasphere.dss.workflow.lock.Lock; -import com.webank.wedatasphere.dss.workflow.service.BMLService; +import com.webank.wedatasphere.dss.common.service.BMLService; import com.webank.wedatasphere.dss.workflow.service.DSSFlowService; import com.webank.wedatasphere.dss.workflow.service.WorkflowNodeService; import org.apache.commons.collections.CollectionUtils; import org.apache.commons.lang.StringUtils; +import org.apache.commons.lang.exception.ExceptionUtils; import org.apache.linkis.common.conf.CommonVars; +import org.apache.linkis.common.exception.ErrorException; +import org.apache.linkis.cs.client.utils.SerializeHelper; import org.apache.linkis.cs.common.utils.CSCommonUtils; import org.apache.linkis.server.BDPJettyServerHelper; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.beans.BeanUtils; import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.beans.factory.annotation.Qualifier; import org.springframework.dao.DuplicateKeyException; import org.springframework.stereotype.Service; import org.springframework.transaction.annotation.Transactional; @@ -67,7 +80,7 @@ import java.util.*; import java.util.stream.Collectors; -import static com.webank.wedatasphere.dss.workflow.constant.DSSWorkFlowConstant.SCHEDULER_APP_CONN_NAME; +import static com.webank.wedatasphere.dss.workflow.constant.DSSWorkFlowConstant.*; @Service public class DSSFlowServiceImpl implements DSSFlowService { @@ -82,6 +95,9 @@ public class DSSFlowServiceImpl implements DSSFlowService { @Autowired private WorkFlowParser workFlowParser; @Autowired + private NodeParser nodeParser; + @Autowired + @Qualifier("workflowBmlService") private BMLService bmlService; @Autowired private NodeExportService nodeExportService; @@ -165,25 +181,55 @@ public DSSFlow addFlow(DSSFlow dssFlow, @Lock @Transactional(rollbackFor = DSSErrorException.class) @Override - public DSSFlow addSubFlow(DSSFlow DSSFlow, Long parentFlowID, String contextIDStr, String orcVersion, String schedulerAppConn) throws DSSErrorException { + public DSSFlow addSubFlow(DSSFlow dssFlow, Long parentFlowID, String contextIDStr, String orcVersion, String schedulerAppConn) throws DSSErrorException { DSSFlow parentFlow = flowMapper.selectFlowByID(parentFlowID); - DSSFlow.setProjectID(parentFlow.getProjectID()); - DSSFlow subFlow = addFlow(DSSFlow, contextIDStr, orcVersion, schedulerAppConn); + dssFlow.setProjectId(parentFlow.getProjectId()); + DSSFlow subFlow = addFlow(dssFlow, contextIDStr, orcVersion, schedulerAppConn); //数据库中插入关联信息 flowMapper.insertFlowRelation(subFlow.getId(), parentFlowID); return subFlow; } @Override - public DSSFlow getFlow(Long flowID) { - DSSFlow DSSFlow = getFlowByID(flowID); + public DSSFlow getFlow(Long flowID) throws NullPointerException { + DSSFlow dssFlow = getFlowByID(flowID); //todo update - String userName = DSSFlow.getCreator(); - Map query = bmlService.query(userName, DSSFlow.getResourceId(), DSSFlow.getBmlVersion()); - DSSFlow.setFlowJson(query.get("string").toString()); - return DSSFlow; + String userName = dssFlow.getCreator(); + Map query = bmlService.query(userName, dssFlow.getResourceId(), dssFlow.getBmlVersion()); + dssFlow.setFlowJson(query.get("string").toString()); + return dssFlow; + } + + @Override + public List getSubFlowContextIdsByFlowIds(List flowIdList) throws ErrorException { + ArrayList contextIdList = new ArrayList<>(); + // 查出所有子工作流的上下文ID + for (Long flowId : flowIdList) { + generateSubFlowContextIdByFlowID(flowId, contextIdList); + } + return contextIdList; + } + + private void generateSubFlowContextIdByFlowID(Long flowId, ArrayList contextIdList) throws ErrorException { + List subFlowIDs = flowMapper.selectSubFlowIDByParentFlowID(flowId); + if (subFlowIDs.size() > 0) { + // 获取子工作流 contextId + JsonToFlowParser jsonToFlowParser = WorkflowFactory.INSTANCE.getJsonToFlowParser(); + for (Long subFlowID : subFlowIDs) { + DSSFlow dssFlow = getFlow(subFlowID); + Workflow workflow = jsonToFlowParser.parse(dssFlow); + String contextIDStr = ((WorkflowWithContextImpl) workflow).getContextID(); + if (StringUtils.isNotBlank(contextIDStr)) { + // 获取需要清理的contextId + contextIdList.add(SerializeHelper.deserializeContextID(contextIDStr).getContextId()); + } + //获取下一层的subflow的上下文Id + generateSubFlowContextIdByFlowID(subFlowID, contextIdList); + } + } } + @Lock @Transactional(rollbackFor = DSSErrorException.class) @Override @@ -196,6 +242,7 @@ public void updateFlowBaseInfo(DSSFlow DSSFlow) throws DSSErrorException { } } + @Lock @Transactional(rollbackFor = DSSErrorException.class) @Override @@ -203,15 +250,13 @@ public void batchDeleteFlow(List flowIDlist) { flowIDlist.forEach(this::deleteFlow); } - /*@Lock*/ - @Transactional(rollbackFor = {DSSErrorException.class}) @Override public String saveFlow(Long flowID, String jsonFlow, String comment, String userName, String workspaceName, - String projectName) { + String projectName) throws IOException{ DSSFlow dssFlow = flowMapper.selectFlowByID(flowID); String creator = dssFlow.getCreator(); @@ -225,6 +270,8 @@ public String saveFlow(Long flowID, } else { logger.info("saveFlow is change"); } + checkSubflowDependencies(userName, flowID, jsonFlow); + String resourceId = dssFlow.getResourceId(); Long parentFlowID = flowMapper.getParentFlowID(flowID); // 这里不要检查ContextID具体版本等,只要存在就不创建 2020-0423 @@ -238,6 +285,7 @@ public String saveFlow(Long flowID, dssFlow.setDescription(comment); dssFlow.setResourceId(bmlReturnMap.get("resourceId").toString()); dssFlow.setBmlVersion(bmlReturnMap.get("version").toString()); + updateMetrics(dssFlow, jsonFlow); //todo 数据库增加版本更新 flowMapper.updateFlowInputInfo(dssFlow); @@ -245,12 +293,60 @@ public String saveFlow(Long flowID, contextService.checkAndSaveContext(jsonFlow, String.valueOf(parentFlowID)); } catch (DSSErrorException e) { logger.error("Failed to saveContext: ", e); + throw new DSSRuntimeException(e.getErrCode(),"保存ContextId失败,您可以尝试重新发布工作流!原因:" + ExceptionUtils.getRootCauseMessage(e),e); } String version = bmlReturnMap.get("version").toString(); return version; } + /** + * 当数据库的父子工作流依赖关系和json中不一致时,该方法会删除数据库中的脏数据 + * + * @param user + * @param flowId + * @param flowJson + * @throws IOException + */ + private void checkSubflowDependencies(String user, long flowId, String flowJson) throws IOException { + List nodeJsonList = workFlowParser.getWorkFlowNodesJson(flowJson); + if (CollectionUtils.isEmpty(nodeJsonList)) { + return; + } + List subflowInfos = flowMapper.getSubflowInfoByParentId(flowId); + List subflowTitleList = new ArrayList<>(); + for (String nodeJson : nodeJsonList) { + Map nodeMap = BDPJettyServerHelper.jacksonJson().readValue(nodeJson, Map.class); + String nodeType = nodeMap.get(JOBTYPE_KEY).toString(); + if ("workflow.subflow".equals(nodeType)) { + String title = nodeMap.get(TITLE_KEY).toString(); + subflowTitleList.add(title); + } + } + subflowInfos.forEach(subflow -> { + if (subflowTitleList.stream().noneMatch(t -> t.equals(subflow.getName()))) { + //查看子工作流有无内容 + String subflowJson = bmlService.query(user, subflow.getResourceId(), subflow.getBmlVersion()).get("string").toString(); + if (CollectionUtils.isEmpty(workFlowParser.getWorkFlowNodesJson(subflowJson))) { + flowMapper.deleteFlowRelation(subflow.getId()); + } else { + logger.warn("子工作流内容不为空,不予删除和父工作流的依赖关系。"); + } + } + }); + } + + private void updateMetrics(DSSFlow dssFlow, String flowJson) { + Map metricsMap = new HashMap<>(); + List nodes = workFlowParser.getWorkFlowNodes(flowJson); + metricsMap.put("totalNodes", nodes == null ? 0 : nodes.size()); + if (nodes != null) { + Map mapCnt = nodes.stream().collect(Collectors.groupingBy(DSSNode::getNodeType, Collectors.counting())); + metricsMap.putAll(mapCnt); + } + dssFlow.setMetrics(new Gson().toJson(metricsMap)); + } + public boolean isEqualTwoJson(String oldJsonNode, String newJsonNode) { Gson gson = new Gson(); JsonParser parser = new JsonParser(); @@ -269,7 +365,7 @@ public boolean isEqualTwoJson(String oldJsonNode, String newJsonNode) { public String getFlowJson(String userName, String projectName, DSSFlow dssFlow) { String flowExportSaveBasePath = IoUtils.generateIOPath(userName, projectName, ""); String savePath = flowExportSaveBasePath + File.separator + dssFlow.getName() + File.separator + dssFlow.getName() + ".json"; - String flowJson = bmlService.downloadAndGetFlowJson(userName, dssFlow.getResourceId(), dssFlow.getBmlVersion(), savePath); + String flowJson = bmlService.downloadAndGetText(userName, dssFlow.getResourceId(), dssFlow.getBmlVersion(), savePath); return flowJson; } @@ -306,30 +402,96 @@ private void deleteDWSDB(Long flowID) { //第一期没有工作流的发布,所以不需要删除dws工作流的发布表 } - + /** + * 支持跨项目拷贝工作流 + * + * @param rootFlowId 拷贝基于的工作流id + * @param userName + * @param workspace + * @param projectName 目标工程名 + * @param version 新的编排版本 + * @param contextIdStr 新的contextId + * @param description + * @param dssLabels + * @param nodeSuffix 节点后缀 + * @param newFlowName 新的工作流名称 + * @param newProjectId 目标工程id + * @return + * @throws DSSErrorException + * @throws IOException + */ @Override public DSSFlow copyRootFlow(Long rootFlowId, String userName, Workspace workspace, String projectName, String version, String contextIdStr, - String description, List dssLabels) throws DSSErrorException, IOException { + String description, List dssLabels, String nodeSuffix, + String newFlowName, Long newProjectId) throws DSSErrorException, IOException { DSSFlow dssFlow = flowMapper.selectFlowByID(rootFlowId); - if(dssFlow == null) { - throw new DSSErrorException(50030, "Workflow " + rootFlowId + " is not exists[工作流不存在,请检查是否已被删除]."); - } - logger.info("User {} try to copy workflow {} in project {} with newVersion {} and contextId {}.", userName, - dssFlow.getName(), projectName, version, contextIdStr); - DSSFlow rootFlowWithSubFlows = copyFlowAndSetSubFlowInDB(dssFlow, userName, description); + DSSFlow rootFlowWithSubFlows = copyFlowAndSetSubFlowInDB(dssFlow, userName, description, nodeSuffix, newFlowName, newProjectId); updateFlowJson(userName, projectName, rootFlowWithSubFlows, version, null, - contextIdStr, workspace, dssLabels); + contextIdStr, workspace, dssLabels, nodeSuffix); return flowMapper.selectFlowByID(rootFlowWithSubFlows.getId()); } - private DSSFlow copyFlowAndSetSubFlowInDB(DSSFlow dssFlow, - String userName, String description) { + + @Override + public boolean checkExistSameSubflow(Long parentFlowID, String name) { + List subflowName = flowMapper.getSubflowName(parentFlowID); + return subflowName.stream().anyMatch(s -> s.equals(name)); + } + + @Override + public boolean checkExistSameFlow(Long parentFlowID, String name, String existName) { + List subflowName = flowMapper.getSubflowName(parentFlowID); + if (name.equals(existName)) { + return false; + } + return subflowName.stream().anyMatch(s -> s.equals(name)); + } + + @Override + public List checkIsSave(Long flowID, String jsonFlow) { + List subflowName = flowMapper.getSubflowName(flowID); + List workFlowNodes = workFlowParser.getWorkFlowNodes(jsonFlow); + return workFlowNodes.stream().filter(t -> Objects.equals("workflow.subflow", t.getNodeType())) + .map(DSSNode::getName) + .filter(name -> !subflowName.contains(name)) + .collect(Collectors.toList()); + } + + @Override + public boolean checkIsExistSameFlow(String jsonFlow) { + List workFlowNodes = workFlowParser.getWorkFlowNodes(jsonFlow); + long distinctSize = workFlowNodes.stream().map(Node::getName).distinct().count(); + return distinctSize < workFlowNodes.size(); + } + + /** + * @param dssFlow + * @param userName + * @param description + * @param newFlowName 拷贝时新的工作流名 + * @param newProjectId 拷贝到新的工程id + * @param subFlowNameSuffix 工作流拷贝时需要为所有子工作流节点名添加后缀 + * @return + */ + private DSSFlow copyFlowAndSetSubFlowInDB(DSSFlow dssFlow, String userName, String description, + String subFlowNameSuffix, String newFlowName, Long newProjectId) { DSSFlow cyFlow = new DSSFlow(); BeanUtils.copyProperties(dssFlow, cyFlow, "children", "flowVersions"); //封装flow信息 cyFlow.setCreator(userName); cyFlow.setCreateTime(new Date()); + //工作流复制时用户输入的的目标工作流名 + if (StringUtils.isNotBlank(newFlowName)) { + cyFlow.setName(newFlowName); + } + //跨项目复制时,newProjectId为目标工程id + if (newProjectId != null) { + cyFlow.setProjectId(newProjectId); + } + if (StringUtils.isNotBlank(subFlowNameSuffix) && !cyFlow.getRootFlow()) { + cyFlow.setName(cyFlow.getName() + "_" + subFlowNameSuffix); + } if (StringUtils.isNotBlank(description)) { cyFlow.setDescription(description); } @@ -341,7 +503,7 @@ private DSSFlow copyFlowAndSetSubFlowInDB(DSSFlow dssFlow, if (dssFlow.getChildren() == null) { dssFlow.setChildren(new ArrayList()); } - DSSFlow copySubFlow = copyFlowAndSetSubFlowInDB(subDSSFlow, userName, description); + DSSFlow copySubFlow = copyFlowAndSetSubFlowInDB(subDSSFlow, userName, description, subFlowNameSuffix, null, newProjectId); persistenceFlowRelation(copySubFlow.getId(), cyFlow.getId()); cyFlow.addChildren(copySubFlow); } @@ -350,11 +512,19 @@ private DSSFlow copyFlowAndSetSubFlowInDB(DSSFlow dssFlow, private void updateFlowJson(String userName, String projectName, DSSFlow rootFlow, String version, Long parentFlowId, String contextIdStr, - Workspace workspace, List dssLabels) throws DSSErrorException, IOException { - String flowJson = bmlService.readFlowJsonFromBML(userName, rootFlow.getResourceId(), rootFlow.getBmlVersion()); + Workspace workspace, List dssLabels, String nodeSuffix) throws DSSErrorException, IOException { + String flowJson = bmlService.readTextFromBML(userName, rootFlow.getResourceId(), rootFlow.getBmlVersion()); //如果包含subflow,需要一同导入subflow内容,并更新parrentflow的json内容 // TODO: 2020/7/31 优化update方法里面的saveContent + //copy subflow need new contextID + if (!rootFlow.getRootFlow()) { + contextIdStr = contextService.checkAndInitContext(flowJson, parentFlowId.toString(), workspace.getWorkspaceName(), projectName, rootFlow.getName(), version, userName); + logger.info("update subflow contextID :" + contextIdStr + ",flow name is " + rootFlow.getName()); + } String updateFlowJson = updateFlowContextIdAndVersion(flowJson, contextIdStr, version); + if (StringUtils.isNotBlank(nodeSuffix)) { + updateFlowJson = addFLowNodeSuffix(updateFlowJson, nodeSuffix); + } //重新上传工作流资源 updateFlowJson = uploadFlowResourceToBml(userName, updateFlowJson, projectName, rootFlow); //上传节点的资源或调用appconn的copyRef @@ -364,7 +534,7 @@ private void updateFlowJson(String userName, String projectName, DSSFlow rootFlo if (subFlows != null) { for (DSSFlow subflow : subFlows) { updateFlowJson(userName, projectName, subflow, version, rootFlow.getId(), - contextIdStr, workspace, dssLabels); + contextIdStr, workspace, dssLabels, nodeSuffix); } } @@ -374,6 +544,51 @@ private void updateFlowJson(String userName, String projectName, DSSFlow rootFlo contextService.checkAndSaveContext(updateFlowJson, String.valueOf(parentFlowId)); } + private String addFLowNodeSuffix(String flowJson, String nodeSuffix) throws IOException { + List nodeJsonList = workFlowParser.getWorkFlowNodesJson(flowJson); + List edgeList = workFlowParser.getWorkFlowEdges(flowJson); + if (CollectionUtils.isEmpty(nodeJsonList)) { + return flowJson; + } + List> nodeList = new ArrayList<>(); + for (String nodeJson : nodeJsonList) { + Map nodeJsonMap = BDPJettyServerHelper.jacksonJson().readValue(nodeJson, Map.class); + nodeJsonMap.replace(TITLE_KEY, nodeJsonMap.get(TITLE_KEY) + "_" + nodeSuffix); + List resourceList = nodeParser.getNodeResource(nodeJson); + if (CollectionUtils.isNotEmpty(resourceList)) { + String oldKey = (String) nodeJsonMap.get("key"); + final String newKey = UUID.randomUUID().toString(); + //需要替换resource的fileName + resourceList.forEach(resource -> { + resource.setFileName(resource.getFileName().replace(oldKey, newKey)); + }); + nodeJsonMap.put("resources", resourceList); + Map jobContent = (Map) nodeJsonMap.get("jobContent"); + if (MapUtils.isNotEmpty(jobContent)) { + jobContent.forEach((k, v) -> { + jobContent.put(k, ((String) v).replace(oldKey, newKey)); + }); + nodeJsonMap.put("jobContent", jobContent); + } + nodeJsonMap.put("id", newKey); + nodeJsonMap.put("key", newKey); + if (CollectionUtils.isNotEmpty(edgeList)) { + edgeList.forEach(e -> { + if (e.getSource().equals(oldKey)) { + e.setSource(newKey); + } + if (e.getTarget().equals(oldKey)) { + e.setTarget(newKey); + } + }); + } + } + nodeList.add(nodeJsonMap); + } + flowJson = workFlowParser.updateFlowJsonWithKey(flowJson, "edges", edgeList); + return workFlowParser.updateFlowJsonWithKey(flowJson, "nodes", nodeList); + } + //上传工作流资源 public String uploadFlowResourceToBml(String userName, String flowJson, String projectName, DSSFlow dssFlow) throws IOException { List resourceList = workFlowParser.getWorkFlowResources(flowJson); @@ -389,9 +604,9 @@ public String uploadFlowResourceToBml(String userName, String flowJson, String p bmlService.downloadToLocalPath(userName, resource.getResourceId(), resource.getVersion(), flowResourcePath); //重新上传resource InputStream resourceInputStream = bmlService.readLocalResourceFile(userName, flowResourcePath); - Map bmlReturnMap = bmlService.upload(userName, resourceInputStream, UUID.randomUUID().toString() + ".json", projectName); - resource.setResourceId(bmlReturnMap.get("resourceId").toString()); - resource.setVersion(bmlReturnMap.get("version").toString()); + BmlResource bmlReturnMap = bmlService.upload(userName, resourceInputStream, UUID.randomUUID().toString() + ".json", projectName); + resource.setResourceId(bmlReturnMap.getResourceId()); + resource.setVersion(bmlReturnMap.getVersion()); }); //更新flowJson的resources return workFlowParser.updateFlowJsonWithKey(flowJson, "resources", resourceList); @@ -443,10 +658,10 @@ private String updateWorkFlowNodeJson(String userName, String projectName, Map nodeJsonMap = BDPJettyServerHelper.jacksonJson().readValue(updateNodeJson, Map.class); //更新subflowID - String nodeType = nodeJsonMap.get("jobType").toString(); + String nodeType = nodeJsonMap.get(JOBTYPE_KEY).toString(); NodeInfo nodeInfo = nodeInfoMapper.getWorkflowNodeByType(nodeType); if ("workflow.subflow".equals(nodeType)) { - String subFlowName = nodeJsonMap.get("title").toString(); + String subFlowName = nodeJsonMap.get(TITLE_KEY).toString(); List dssFlowList = subflows.stream().filter(subflow -> subflow.getName().equals(subFlowName) ).collect(Collectors.toList()); @@ -462,6 +677,10 @@ private String updateWorkFlowNodeJson(String userName, String projectName, throw new DSSErrorException(90078, "工程内未能找到子工作流节点,导入失败" + subFlowName); } // } else if (nodeJsonMap.get("jobContent") != null && !((Map) nodeJsonMap.get("jobContent")).containsKey("script")) { + }else if(nodeInfo==null){ + String msg = String.format("%s note type not exist,please check appconn install successfully", nodeType); + logger.error(msg); + throw new DSSRuntimeException(msg); } else if (Boolean.TRUE.equals(nodeInfo.getSupportJump()) && nodeInfo.getJumpType() == 1) { logger.info("nodeJsonMap.jobContent is:{}", nodeJsonMap.get("jobContent")); CommonAppConnNode newNode = new CommonAppConnNode(); @@ -469,12 +688,12 @@ private String updateWorkFlowNodeJson(String userName, String projectName, oldNode.setJobContent((Map) nodeJsonMap.get("jobContent")); oldNode.setContextId(updateContextId); oldNode.setNodeType(nodeType); - oldNode.setName((String) nodeJsonMap.get("title")); + oldNode.setName((String) nodeJsonMap.get(TITLE_KEY)); oldNode.setFlowId(dssFlow.getId()); oldNode.setWorkspace(workspace); oldNode.setDssLabels(dssLabels); oldNode.setFlowName(dssFlow.getName()); - oldNode.setProjectId(dssFlow.getProjectID()); + oldNode.setProjectId(dssFlow.getProjectId()); oldNode.setProjectName(projectName); newNode.setName(oldNode.getName()); Map jobContent = workflowNodeService.copyNode(userName, newNode, oldNode, version); diff --git a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/service/impl/PublishServiceImpl.java b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/service/impl/PublishServiceImpl.java index 1600aad41b..92bdc280b5 100644 --- a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/service/impl/PublishServiceImpl.java +++ b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/service/impl/PublishServiceImpl.java @@ -18,8 +18,11 @@ import com.webank.wedatasphere.dss.appconn.manager.AppConnManager; import com.webank.wedatasphere.dss.appconn.scheduler.SchedulerAppConn; +import com.webank.wedatasphere.dss.common.entity.project.DSSProject; import com.webank.wedatasphere.dss.common.exception.DSSErrorException; +import com.webank.wedatasphere.dss.common.protocol.project.ProjectInfoRequest; import com.webank.wedatasphere.dss.common.utils.DSSExceptionUtils; +import com.webank.wedatasphere.dss.common.utils.RpcAskUtils; import com.webank.wedatasphere.dss.orchestrator.common.protocol.RequestFrameworkConvertOrchestration; import com.webank.wedatasphere.dss.orchestrator.common.protocol.RequestFrameworkConvertOrchestrationStatus; import com.webank.wedatasphere.dss.orchestrator.common.protocol.ResponseConvertOrchestrator; @@ -64,34 +67,34 @@ protected Sender getOrchestratorSender() { @Override public String submitPublish(String convertUser, Long workflowId, - Map dssLabel, Workspace workspace, String comment) throws Exception { + Map dssLabel, Workspace workspace, String comment) throws Exception { LOGGER.info("User {} begins to convert workflow {}.", convertUser, workflowId); //1 获取对应的orcId 和 orcVersionId //2.进行提交 - RequestFrameworkConvertOrchestration requestFrameworkConvertOrchestration = new RequestFrameworkConvertOrchestration(); - requestFrameworkConvertOrchestration.setComment(comment); - requestFrameworkConvertOrchestration.setOrcAppId(workflowId); - requestFrameworkConvertOrchestration.setUserName(convertUser); - requestFrameworkConvertOrchestration.setWorkspace(workspace); DSSFlow dssFlow = null; try { dssFlow = dssFlowService.getFlow(workflowId); - if(dssFlow == null) { + if (dssFlow == null) { DSSExceptionUtils.dealErrorException(63325, "workflow " + workflowId + " is not exists.", DSSErrorException.class); return null; } + ProjectInfoRequest projectInfoRequest = new ProjectInfoRequest(); + projectInfoRequest.setProjectId(dssFlow.getProjectId()); + DSSProject dssProject = (DSSProject) DSSSenderServiceFactory.getOrCreateServiceInstance().getProjectServerSender().ask(projectInfoRequest); + if (dssProject.getWorkspaceId() != workspace.getWorkspaceId()) { + DSSExceptionUtils.dealErrorException(63335, "工作流所在工作空间和cookie中不一致,请刷新页面后,再次发布!", DSSErrorException.class); + } String schedulerAppConnName = workFlowParser.getValueWithKey(dssFlow.getFlowJson(), DSSWorkFlowConstant.SCHEDULER_APP_CONN_NAME); - if(StringUtils.isBlank(schedulerAppConnName)) { + if (StringUtils.isBlank(schedulerAppConnName)) { // 向下兼容老版本 schedulerAppConnName = DEFAULT_SCHEDULER_APP_CONN.getValue(); } SchedulerAppConn schedulerAppConn = (SchedulerAppConn) AppConnManager.getAppConnManager().getAppConn(schedulerAppConnName); // 只是为了获取是否需要发布所有Orc,这里直接拿第一个AppInstance即可。 AppInstance appInstance = schedulerAppConn.getAppDesc().getAppInstances().get(0); - requestFrameworkConvertOrchestration.setConvertAllOrcs(schedulerAppConn.getOrCreateConversionStandard().getDSSToRelConversionService(appInstance).isConvertAllOrcs()); - requestFrameworkConvertOrchestration.setLabels(dssLabel); - ResponseConvertOrchestrator response = (ResponseConvertOrchestrator) getOrchestratorSender().ask(requestFrameworkConvertOrchestration); - if(response.getResponse().isFailed()) { + ResponseConvertOrchestrator response = requestConvertOrchestration(comment, workflowId, convertUser, + workspace, schedulerAppConn, dssLabel, appInstance); + if (response.getResponse().isFailed()) { throw new DSSErrorException(50311, response.getResponse().getMessage()); } return response.getId(); @@ -105,22 +108,53 @@ public String submitPublish(String convertUser, Long workflowId, return null; } + /** + * 可覆写该方法自定义实现request对象 + * + * @return + */ + protected ResponseConvertOrchestrator requestConvertOrchestration(String comment, Long workflowId, String convertUser, Workspace workspace, + SchedulerAppConn schedulerAppConn, Map dssLabel, AppInstance appInstance) { + RequestFrameworkConvertOrchestration requestFrameworkConvertOrchestration = new RequestFrameworkConvertOrchestration(); + requestFrameworkConvertOrchestration.setComment(comment); + requestFrameworkConvertOrchestration.setOrcAppId(workflowId); + requestFrameworkConvertOrchestration.setUserName(convertUser); + requestFrameworkConvertOrchestration.setWorkspace(workspace); + requestFrameworkConvertOrchestration.setConvertAllOrcs(schedulerAppConn.getOrCreateConversionStandard().getDSSToRelConversionService(appInstance).isConvertAllOrcs()); + requestFrameworkConvertOrchestration.setLabels(dssLabel); + ResponseConvertOrchestrator response = RpcAskUtils.processAskException(getOrchestratorSender().ask(requestFrameworkConvertOrchestration), + ResponseConvertOrchestrator.class, RequestFrameworkConvertOrchestration.class); + return response; + } + + @Override public ResponseConvertOrchestrator getStatus(String username, String taskId) { - if (LOGGER.isDebugEnabled()){ + if (LOGGER.isDebugEnabled()) { LOGGER.debug("{} is asking status of {}.", username, taskId); } - ResponseConvertOrchestrator response =new ResponseConvertOrchestrator(); + ResponseConvertOrchestrator response = new ResponseConvertOrchestrator(); //通过rpc的方式去获取到最新status try { - RequestFrameworkConvertOrchestrationStatus req = new RequestFrameworkConvertOrchestrationStatus(taskId); - response = (ResponseConvertOrchestrator) getOrchestratorSender().ask(req); + response = requestConvertOrchestrationStatus(taskId); LOGGER.info("user {} gets status of {}, status is {},msg is {}", username, taskId, response.getResponse().getJobStatus(), response.getResponse().getMessage()); - }catch (Exception t){ + } catch (Exception t) { LOGGER.error("failed to getStatus {} ", taskId, t); } return response; } + /** + * 可覆写该方法自定义实现request对象 + * + * @return + */ + protected ResponseConvertOrchestrator requestConvertOrchestrationStatus(String taskId){ + RequestFrameworkConvertOrchestrationStatus req = new RequestFrameworkConvertOrchestrationStatus(); + req.setId(taskId); + return RpcAskUtils.processAskException(getOrchestratorSender().ask(req), ResponseConvertOrchestrator.class, + RequestFrameworkConvertOrchestrationStatus.class); + } + } diff --git a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/service/impl/WorkflowNodeServiceImpl.java b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/service/impl/WorkflowNodeServiceImpl.java index 602ea350a0..c06bf3a4b5 100644 --- a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/service/impl/WorkflowNodeServiceImpl.java +++ b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/java/com/webank/wedatasphere/dss/workflow/service/impl/WorkflowNodeServiceImpl.java @@ -16,14 +16,19 @@ package com.webank.wedatasphere.dss.workflow.service.impl; +import com.google.gson.Gson; +import com.google.gson.JsonObject; import com.webank.wedatasphere.dss.appconn.core.AppConn; import com.webank.wedatasphere.dss.appconn.core.ext.OnlyDevelopmentAppConn; import com.webank.wedatasphere.dss.appconn.core.ext.OnlySSOAppConn; import com.webank.wedatasphere.dss.appconn.manager.AppConnManager; +import com.webank.wedatasphere.dss.common.entity.BmlResource; import com.webank.wedatasphere.dss.common.exception.DSSErrorException; +import com.webank.wedatasphere.dss.common.exception.DSSRuntimeException; import com.webank.wedatasphere.dss.common.label.DSSLabel; import com.webank.wedatasphere.dss.common.protocol.project.ProjectRelationRequest; import com.webank.wedatasphere.dss.common.protocol.project.ProjectRelationResponse; +import com.webank.wedatasphere.dss.common.utils.RpcAskUtils; import com.webank.wedatasphere.dss.sender.service.DSSSenderServiceFactory; import com.webank.wedatasphere.dss.standard.app.development.operation.*; import com.webank.wedatasphere.dss.standard.app.development.ref.*; @@ -50,14 +55,19 @@ import com.webank.wedatasphere.dss.workflow.service.DSSFlowService; import com.webank.wedatasphere.dss.workflow.service.WorkflowNodeService; import org.apache.commons.lang.StringUtils; +import org.apache.linkis.common.exception.ErrorException; +import org.apache.linkis.cs.client.utils.SerializeHelper; +import org.apache.linkis.cs.common.entity.source.LinkisHAWorkFlowContextID; +import org.apache.linkis.cs.common.utils.CSCommonUtils; import org.apache.linkis.rpc.Sender; +import org.apache.linkis.server.BDPJettyServerHelper; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Service; import java.io.IOException; -import java.util.HashMap; -import java.util.List; -import java.util.Map; +import java.util.*; import java.util.function.BiConsumer; import java.util.function.BiFunction; import java.util.function.Function; @@ -77,6 +87,8 @@ public class WorkflowNodeServiceImpl implements WorkflowNodeService { private Sender projectSender = DSSSenderServiceFactory.getOrCreateServiceInstance().getProjectServerSender(); + private static final Logger logger = LoggerFactory.getLogger(WorkflowNodeServiceImpl.class); + @Override public List listNodeGroups() { //cache @@ -97,7 +109,11 @@ private T getDevelopmentService(AppConn appConn, try { appInstance = appConn.getAppDesc().getAppInstancesByLabels(dssLabels).get(0); } catch (NoSuchAppInstanceException e) { - throw new ExternalOperationFailedException(50020, "cannot find the appInstance with label " + dssLabels.get(0).getStringValue()); + if (dssLabels.get(0).getStringValue() == null){ + throw new ExternalOperationFailedException(50020, "未正确退出生产中心或流失生产中,请刷新页面后重试!"); + } + throw new ExternalOperationFailedException(50020, "Cannot find the appInstance with label " + dssLabels.get(0).getStringValue() + + ". (在" + dssLabels.get(0).getStringValue() + "中心找不到" + appConn.getAppDesc().getAppName() + "实例)"); } return getDevelopmentService.apply(developmentIntegrationStandard, appInstance); } @@ -107,7 +123,7 @@ public Map createNode(String userName, CommonAppConnNode node) t RefJobContentResponseRef responseRef = tryNodeOperation(userName, node, this::getRefCRUDService, developmentService -> ((RefCRUDService) developmentService).getRefCreationOperation(), (developmentOperation, developmentRequestRef) -> - ((RefCreationOperation) developmentOperation).createRef((DSSJobContentRequestRef) developmentRequestRef) + ((RefCreationOperation) developmentOperation).createRef((DSSJobContentRequestRef) developmentRequestRef) , (developmentRequestRef, refJobContentResponseRef) -> { if (developmentRequestRef instanceof ProjectRefRequestRef) { Long projectRefId = ((ProjectRefRequestRef) developmentRequestRef).getRefProjectId(); @@ -124,7 +140,17 @@ private V tryNodeOperat BiConsumer responseRefConsumer, String operation) { NodeInfo nodeInfo = nodeInfoMapper.getWorkflowNodeByType(node.getNodeType()); + if(nodeInfo==null){ + String msg = String.format("%s note type not exist,please check appconn install successfully", node.getNodeType()); + logger.error(msg); + throw new DSSRuntimeException(msg); + } AppConn appConn = AppConnManager.getAppConnManager().getAppConn(nodeInfo.getAppConnName()); + if(appConn==null){ + String msg = String.format("%s appconn not exist,please check appconn install successfully", node.getNodeType()); + logger.error(msg); + throw new DSSRuntimeException(msg); + } String name; if (StringUtils.isBlank(node.getName())) { name = node.getJobContent().get(DSSWorkFlowConstant.TITLE_KEY).toString(); @@ -133,7 +159,7 @@ private V tryNodeOperat } if(node.getProjectId() == null || node.getProjectId() <= 0) { DSSFlow dssFlow = dssFlowService.getFlow(node.getFlowId()); - node.setProjectId(dssFlow.getProjectID()); + node.setProjectId(dssFlow.getProjectId()); } return DevelopmentOperationUtils.tryDevelopmentOperation(() -> developmentServiceFunction.apply(appConn, node.getDssLabels()), developmentOperationFunction, @@ -143,7 +169,7 @@ private V tryNodeOperat try { orcVersion = getOrcVersion(node); } catch (Exception e) { - throw new ExternalOperationFailedException(50205, "get workflow version failed.", e); + throw new ExternalOperationFailedException(50205, "Get workflow version failed." + e.getMessage(), e); } if(node.getParams() != null) { dssJobContentRequestRef.getDSSJobContent().putAll(node.getParams()); @@ -154,6 +180,15 @@ private V tryNodeOperat }, refJobContentRequestRef -> { refJobContentRequestRef.setRefJobContent(node.getJobContent()); + if ("query".equals(operation)) { + try { + Map runtimeParams = getRuntimeParams(node); + refJobContentRequestRef.getRefJobContent().put("runtime",runtimeParams); + } catch (IOException e) { + throw new ExternalOperationFailedException(50205, "Get workflow runtimeParams failed." + e.getMessage(), e); + } + } + if (refJobContentRequestRef instanceof QueryJumpUrlRequestRef) { ((QueryJumpUrlRequestRef) refJobContentRequestRef).setSSOUrlBuilderOperation(getSSOUrlBuilderOperation(appConn, node.getWorkspace())); } @@ -172,12 +207,31 @@ private V tryNodeOperat (developmentOperation, developmentRequestRef) -> { developmentRequestRef.setDSSLabels(node.getDssLabels()).setUserName(userName).setWorkspace(node.getWorkspace()).setName(name).setType(node.getNodeType()); return requestRefOperationFunction.apply(developmentOperation, (K) developmentRequestRef); - }, responseRefConsumer, operation + " workflow node " + name); + }, responseRefConsumer, appConn.getAppDesc().getAppName() + " try to " + operation + " workflow node " + name); + } + + @SuppressWarnings("all") + private Map getRuntimeParams(CommonAppConnNode commonAppConnNode) throws IOException{ + if (commonAppConnNode.getFlowId() == null) { + return new LinkedHashMap(); + } + DSSFlow dssFlow = dssFlowService.getFlow(commonAppConnNode.getFlowId()); + Map flowJsonObject = BDPJettyServerHelper.jacksonJson().readValue(dssFlow.getFlowJson(), Map.class); + final ArrayList nodes = (ArrayList) flowJsonObject.get("nodes"); + final Optional first = nodes.stream().filter(map -> commonAppConnNode.getJobContent().get("title").toString().equals(map.get("title"))).findFirst(); + if (!first.isPresent()) return new LinkedHashMap(); + final LinkedHashMap node = first.get(); + final LinkedHashMap params = (LinkedHashMap) node.get("params"); + if (params == null) return new LinkedHashMap(); + final LinkedHashMap configuration = (LinkedHashMap) params.get("configuration"); + if (configuration == null) return new LinkedHashMap(); + final LinkedHashMap runtime = (LinkedHashMap) configuration.get("runtime"); + return runtime == null ? new LinkedHashMap() : runtime; } private Long parseProjectId(Long dssProjectId, String appConnName, List dssLabels) { ProjectRelationRequest projectRelationRequest = new ProjectRelationRequest(dssProjectId, appConnName, dssLabels); - ProjectRelationResponse projectRelationResponse = (ProjectRelationResponse) projectSender.ask(projectRelationRequest); + ProjectRelationResponse projectRelationResponse = RpcAskUtils.processAskException(projectSender.ask(projectRelationRequest), ProjectRelationResponse.class, ProjectRelationRequest.class); return projectRelationResponse.getAppInstanceProjectId(); } @@ -228,12 +282,13 @@ public List listNodes(String userName, CommonAppConnNode no @Override public ExportResponseRef exportNode(String userName, CommonAppConnNode node) { - return tryNodeOperation(userName, node, + return tryNodeOperation(userName, + node, (appConn, dssLabels) -> getDevelopmentService(appConn, dssLabels, DevelopmentIntegrationStandard::getRefExportService), developmentService -> ((RefExportService) developmentService).getRefExportOperation(), - (developmentOperation, developmentRequestRef) -> - ((RefExportOperation) developmentOperation).exportRef((RefJobContentRequestRef) developmentRequestRef) - , null, "export"); + (developmentOperation, developmentRequestRef) -> ((RefExportOperation) developmentOperation).exportRef((RefJobContentRequestRef) developmentRequestRef), + null, + "export"); } @Override @@ -277,10 +332,22 @@ private String getOrcVersion(CommonAppConnNode node) throws IOException { throw new NullPointerException("The flowId is null, please ask admin for help!"); } DSSFlow dssFlow = dssFlowService.getFlow(node.getFlowId()); - if(StringUtils.isBlank(node.getFlowName())) { + if (StringUtils.isBlank(node.getFlowName())) { node.setFlowName(dssFlow.getName()); } - return workFlowParser.getValueWithKey(dssFlow.getFlowJson(), DSSJobContentConstant.ORC_VERSION_KEY); + String version = workFlowParser.getValueWithKey(dssFlow.getFlowJson(), DSSJobContentConstant.ORC_VERSION_KEY); + //兼容老的flowJson外层没有orcVersion字段的情况,需要从contextId中获取 + if (version == null) { + try { + JsonObject jsonObject = new Gson().fromJson(dssFlow.getFlowJson(), JsonObject.class); + LinkisHAWorkFlowContextID contextID = (LinkisHAWorkFlowContextID) SerializeHelper + .deserializeContextID(jsonObject.get(CSCommonUtils.CONTEXT_ID_STR).getAsString()); + version = contextID.getVersion(); + } catch (ErrorException e) { + logger.error("Invalid contextID, please contact with administrator: ", e); + } + } + return version; } private SSOUrlBuilderOperation getSSOUrlBuilderOperation(AppConn appConn, Workspace workspace) { diff --git a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/scala/com/webank/wedatasphere/dss/workflow/receiver/DSSWorkflowChooser.scala b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/scala/com/webank/wedatasphere/dss/workflow/receiver/DSSWorkflowChooser.scala index e81bf6832b..7501faf90c 100644 --- a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/scala/com/webank/wedatasphere/dss/workflow/receiver/DSSWorkflowChooser.scala +++ b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/scala/com/webank/wedatasphere/dss/workflow/receiver/DSSWorkflowChooser.scala @@ -19,7 +19,7 @@ package com.webank.wedatasphere.dss.workflow.receiver import com.webank.wedatasphere.dss.common.protocol.{RequestDeleteWorkflow, RequestExportWorkflow, RequestQueryWorkFlow, RequestUpdateWorkflow} import com.webank.wedatasphere.dss.orchestrator.common.protocol._ import com.webank.wedatasphere.dss.workflow.WorkFlowManager -import com.webank.wedatasphere.dss.workflow.common.protocol.{RequestCopyWorkflow, RequestCreateWorkflow, RequestImportWorkflow} +import com.webank.wedatasphere.dss.workflow.common.protocol.{RequestCopyWorkflow, RequestCreateWorkflow, RequestDeleteBmlSource, RequestImportWorkflow, RequestSubFlowContextIds, RequestUnlockWorkflow} import org.apache.linkis.rpc.{RPCMessageEvent, Receiver, ReceiverChooser} import javax.annotation.PostConstruct @@ -46,7 +46,11 @@ class DSSWorkflowChooser extends ReceiverChooser { case _: RequestImportWorkflow => receiver case _: RequestCopyWorkflow => receiver case _: RequestQueryWorkFlow => receiver + case _: RequestUnlockWorkflow => receiver case _: RequestConvertOrchestrations => receiver + case _: RequestSubFlowContextIds => receiver + case _: RequestDeleteBmlSource => receiver + case _ => None } } \ No newline at end of file diff --git a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/scala/com/webank/wedatasphere/dss/workflow/receiver/DSSWorkflowReceiver.scala b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/scala/com/webank/wedatasphere/dss/workflow/receiver/DSSWorkflowReceiver.scala index 7fa001d2c4..c189ebb3cd 100644 --- a/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/scala/com/webank/wedatasphere/dss/workflow/receiver/DSSWorkflowReceiver.scala +++ b/dss-orchestrator/orchestrators/dss-workflow/dss-workflow-server/src/main/scala/com/webank/wedatasphere/dss/workflow/receiver/DSSWorkflowReceiver.scala @@ -16,8 +16,7 @@ package com.webank.wedatasphere.dss.workflow.receiver -import java.util - +import com.webank.wedatasphere.dss.common.entity.BmlResource import com.webank.wedatasphere.dss.common.exception.DSSErrorException import com.webank.wedatasphere.dss.common.protocol._ import com.webank.wedatasphere.dss.common.utils.DSSCommonUtils @@ -56,15 +55,18 @@ class DSSWorkflowReceiver(workflowManager: WorkFlowManager) extends Receiver { workflowManager.deleteWorkflow(reqDeleteFlow.userName, reqDeleteFlow.flowID) new ResponseDeleteWorkflow(JobStatus.Success) + case reqUnlockWorkflow: RequestUnlockWorkflow => + workflowManager.unlockWorkflow(reqUnlockWorkflow.getUsername, reqUnlockWorkflow.getFlowId, reqUnlockWorkflow.getConfirmDelete) + case reqExportFlow: RequestExportWorkflow => - val dssExportFlowResource: util.Map[String, AnyRef] = workflowManager.exportWorkflow( + val dssExportFlowResource: BmlResource = workflowManager.exportWorkflow( reqExportFlow.userName, reqExportFlow.flowID, reqExportFlow.projectId, reqExportFlow.projectName, DSSCommonUtils.COMMON_GSON.fromJson(reqExportFlow.workspaceStr, classOf[Workspace]), reqExportFlow.dssLabelList) - ResponseExportWorkflow(dssExportFlowResource.get("resourceId").toString, dssExportFlowResource.get("version").toString, + ResponseExportWorkflow(dssExportFlowResource.getResourceId, dssExportFlowResource.getVersion, reqExportFlow.flowID) case requestImportWorkflow: RequestImportWorkflow => @@ -91,7 +93,10 @@ class DSSWorkflowReceiver(workflowManager: WorkFlowManager) extends Receiver { requestCopyWorkflow.getContextIdStr, requestCopyWorkflow.getOrcVersion, requestCopyWorkflow.getDescription, - requestCopyWorkflow.getDssLabels) + requestCopyWorkflow.getDssLabels, + requestCopyWorkflow.getNodeSuffix, + requestCopyWorkflow.getNewFlowName, + requestCopyWorkflow.getTargetProjectId) new ResponseCopyWorkflow(copyFlow) case requestQueryWorkFlow: RequestQueryWorkFlow => @@ -101,6 +106,8 @@ class DSSWorkflowReceiver(workflowManager: WorkFlowManager) extends Receiver { case requestConvertOrchestrator: RequestConvertOrchestrations => workflowManager.convertWorkflow(requestConvertOrchestrator) + case requestWorkflowIdList : RequestSubFlowContextIds => + workflowManager.getSubFlowContextIdsByFlowIds(requestWorkflowIdList) case _ => throw new DSSErrorException(90000, "Not support protocol " + message) } diff --git a/dss-orchestrator/orchestrators/dss-workflow/pom.xml b/dss-orchestrator/orchestrators/dss-workflow/pom.xml index 8fab7b78ea..ce45e45357 100644 --- a/dss-orchestrator/orchestrators/dss-workflow/pom.xml +++ b/dss-orchestrator/orchestrators/dss-workflow/pom.xml @@ -21,7 +21,8 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 + ../../../pom.xml 4.0.0 diff --git a/dss-orchestrator/pom.xml b/dss-orchestrator/pom.xml index 599f996be9..baff41b185 100644 --- a/dss-orchestrator/pom.xml +++ b/dss-orchestrator/pom.xml @@ -21,7 +21,8 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 + ../pom.xml 4.0.0 diff --git a/dss-server/pom.xml b/dss-server/pom.xml new file mode 100644 index 0000000000..fdb4bd22aa --- /dev/null +++ b/dss-server/pom.xml @@ -0,0 +1,152 @@ + + + + + + dss + com.webank.wedatasphere.dss + 1.1.2 + ../pom.xml + + 4.0.0 + + dss-server + + + + org.apache.linkis + linkis-module + ${linkis.version} + provided + + + org.springframework.cloud + spring-cloud-netflix + + + spring-cloud-starter-netflix-eureka-client + org.springframework.cloud + + + + + com.webank.wedatasphere.dss + dss-framework-project-server + ${dss.version} + + + com.webank.wedatasphere.dss + dss-framework-orchestrator-server + ${dss.version} + + + com.webank.wedatasphere.dss + dss-workflow-server + ${dss.version} + + + com.webank.wedatasphere.dss + dss-flow-execution-server + ${dss.version} + + + com.webank.wedatasphere.dss + dss-framework-migrate-server + ${dss.version} + + + com.webank.wedatasphere.dss + dss-scriptis-server + ${dss.version} + + + + + + + + org.apache.maven.plugins + maven-deploy-plugin + + + + net.alchim31.maven + scala-maven-plugin + + + org.apache.maven.plugins + maven-jar-plugin + + + org.apache.maven.plugins + maven-assembly-plugin + 2.3 + false + + + make-assembly + package + + single + + + + src/main/assembly/distribution.xml + + + + + + false + out + false + false + + src/main/assembly/distribution.xml + + + + + org.apache.maven.plugins + maven-compiler-plugin + + 8 + 8 + + + + + + src/main/java + + **/*.xml + + + + + + + + + + + + + + \ No newline at end of file diff --git a/dss-server/src/main/assembly/distribution.xml b/dss-server/src/main/assembly/distribution.xml new file mode 100644 index 0000000000..7ed92bb63a --- /dev/null +++ b/dss-server/src/main/assembly/distribution.xml @@ -0,0 +1,43 @@ + + + + dss-server + + dir + + true + dss-server + + + + + + lib + true + true + false + true + true + + + + + + diff --git a/dss-server/src/main/scala/com/webank/wedatasphere/dss/DSSServerApplication.scala b/dss-server/src/main/scala/com/webank/wedatasphere/dss/DSSServerApplication.scala new file mode 100644 index 0000000000..2bb9f8750e --- /dev/null +++ b/dss-server/src/main/scala/com/webank/wedatasphere/dss/DSSServerApplication.scala @@ -0,0 +1,35 @@ +package com.webank.wedatasphere.dss + +import com.webank.wedatasphere.dss.common.utils.DSSMainHelper +import org.apache.linkis.DataWorkCloudApplication +import org.apache.linkis.common.conf.DSSConfiguration +import org.apache.linkis.common.utils.{Logging, Utils} + +import java.io.File + +object DSSServerApplication extends Logging { + + val userName: String = System.getProperty("user.name") + val hostName: String = Utils.getComputerName + + val legacyServerConfigs: Array[String] = Array("dss-framework-project-server.properties", "dss-framework-orchestrator-server.properties", + "dss-workflow-server.properties", "dss-flow-execution-server.properties") + + def main(args: Array[String]): Unit = { + val serviceName = System.getProperty("serviceName") //ProjectConf.SERVICE_NAME.getValue + DSSMainHelper.formatPropertyFiles(serviceName) + // 若dss-server配置不存在,则加载先前配置 + val serverConfFileURL = getClass.getClassLoader.getResource(serviceName + ".properties") + if (serverConfFileURL == null || !new File(serverConfFileURL.getPath).exists) { + logger.info(s"$serviceName.properties is not exists, now try to load legacy configs.") + DSSConfiguration.addLegacyConfiguration(legacyServerConfigs) + DSSConfiguration.setSpringApplicationName("dss-server-dev") + } + val allArgs = args ++ DSSMainHelper.getExtraSpringOptions + System.setProperty("hostName", hostName) + System.setProperty("userName", userName) + info(s"Ready to start $serviceName with args: ${allArgs.toList}.") + println(s"Test Ready to start $serviceName with args: ${allArgs.toList}.") + DataWorkCloudApplication.main(allArgs) + } +} diff --git a/dss-standard/development-standard/development-process-standard-execution/pom.xml b/dss-standard/development-standard/development-process-standard-execution/pom.xml index ed02aa03b4..87e3220185 100644 --- a/dss-standard/development-standard/development-process-standard-execution/pom.xml +++ b/dss-standard/development-standard/development-process-standard-execution/pom.xml @@ -21,7 +21,7 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 ../../../pom.xml 4.0.0 @@ -66,10 +66,6 @@ UTF-8 - - net.alchim31.maven - scala-maven-plugin - diff --git a/dss-standard/development-standard/development-process-standard/pom.xml b/dss-standard/development-standard/development-process-standard/pom.xml index d1c61661ca..b4efe63966 100644 --- a/dss-standard/development-standard/development-process-standard/pom.xml +++ b/dss-standard/development-standard/development-process-standard/pom.xml @@ -21,7 +21,7 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 ../../../pom.xml 4.0.0 diff --git a/dss-standard/development-standard/development-process-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/development/ref/CopyRequestRef.java b/dss-standard/development-standard/development-process-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/development/ref/CopyRequestRef.java index 96c3b590d2..7539d0bedd 100644 --- a/dss-standard/development-standard/development-process-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/development/ref/CopyRequestRef.java +++ b/dss-standard/development-standard/development-process-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/development/ref/CopyRequestRef.java @@ -17,6 +17,12 @@ package com.webank.wedatasphere.dss.standard.app.development.ref; +/** + * 若AppConn实现了二级规范,则在实现CopyRequestRef的同时必须要实现ProjectRequestRef。 + * 原因是dss调用节点复制接口时,传的refProjectId可能是节点源工程id(同一个项目内拷贝),也可能是另一个工程Id(跨项目拷贝)。 + * + * @param + */ public interface CopyRequestRef> extends RefJobContentRequestRef { @@ -29,6 +35,12 @@ default R setNewVersion(String version) { * The new version comes from DSS Orchestrator framework. * When the orchestrator, such as DSSWorkflow, added a new version, we hope all nodes of this workflow, * can also update a new version. + *

+ * 特别注意: + * 此版本号格式不一定是v000001的格式,在工作流复制操作时会加上前缀或者后缀(suffix_v000001的形式),三方节点最好将其当做一个普通字符串处理。 + * 三方节点新增节点版本时,需要将之前版本号去掉后缀,然后用节点名前缀拼接新的版本号。 + * 比如节点名:widget_1001_v000001 在新增版本号(v000002)后需变为:widget_1001_v000002,而不是widget_1001_v000001_v000002 + * * @return the new version of the orchestrator, such as DSSWorkflow. */ default String getNewVersion() { diff --git a/dss-standard/development-standard/development-process-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/development/ref/ImportRequestRef.java b/dss-standard/development-standard/development-process-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/development/ref/ImportRequestRef.java index f2c1b464d0..a3e2b329ed 100644 --- a/dss-standard/development-standard/development-process-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/development/ref/ImportRequestRef.java +++ b/dss-standard/development-standard/development-process-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/development/ref/ImportRequestRef.java @@ -17,6 +17,8 @@ package com.webank.wedatasphere.dss.standard.app.development.ref; +import com.webank.wedatasphere.dss.common.entity.BmlResource; + import java.util.Map; public interface ImportRequestRef> diff --git a/dss-standard/development-standard/development-process-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/development/ref/ProjectRefRequestRef.java b/dss-standard/development-standard/development-process-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/development/ref/ProjectRefRequestRef.java index 1086338a86..36328d8cc5 100644 --- a/dss-standard/development-standard/development-process-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/development/ref/ProjectRefRequestRef.java +++ b/dss-standard/development-standard/development-process-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/development/ref/ProjectRefRequestRef.java @@ -14,6 +14,11 @@ default R setRefProjectId(Long refProjectId) { /** * 返回的是第三方 AppConn 的工程 Id,由第三方 AppConn {@code ProjectCreationOperation} 的实现类返回的 refProjectId。 + * 特别注意:当dss调用三方appconn的节点拷贝接口时,若工作流是在同一个项目内拷贝,则refProjectId代表的是节点源工程Id; + * 若工作流是跨工程拷贝,则refProjectId代表的是目标工程id。 + * 三方接入系统在实现节点拷贝操作时,需要在自己系统内先根据节点名获取源工程Id,判断和dss传入的refProjectId是否一致。 + * 若一致,则代表dss端发起的请求是工程内拷贝;若不一致,则代表dss端发起的请求是跨工程拷贝。三方系统在实现节点拷贝操作时需要根据此两种情况做不同实现。 + * * @return 返回的是第三方 AppConn 的工程Id */ default Long getRefProjectId() { diff --git a/dss-standard/development-standard/development-process-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/development/utils/DevelopmentOperationUtils.java b/dss-standard/development-standard/development-process-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/development/utils/DevelopmentOperationUtils.java index 4537337bf0..93ead04e9b 100644 --- a/dss-standard/development-standard/development-process-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/development/utils/DevelopmentOperationUtils.java +++ b/dss-standard/development-standard/development-process-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/development/utils/DevelopmentOperationUtils.java @@ -5,14 +5,19 @@ import com.webank.wedatasphere.dss.standard.app.development.ref.*; import com.webank.wedatasphere.dss.standard.app.development.service.DevelopmentService; import com.webank.wedatasphere.dss.standard.common.entity.ref.ResponseRef; +import com.webank.wedatasphere.dss.standard.common.exception.operation.ExternalOperationFailedException; import com.webank.wedatasphere.dss.standard.common.exception.operation.ExternalOperationWarnException; import com.webank.wedatasphere.dss.standard.common.utils.RequestRefUtils; +import org.apache.commons.lang3.StringUtils; +import org.apache.commons.lang3.exception.ExceptionUtils; +import org.apache.linkis.common.exception.WarnException; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import java.util.function.*; /** + * operation工具 * @author enjoyyin * @date 2022-03-12 * @since 0.5.0 @@ -21,6 +26,24 @@ public class DevelopmentOperationUtils { private static final Logger LOGGER = LoggerFactory.getLogger(DevelopmentOperationUtils.class); + /** + * 执行operation。 + * 先通过getDevelopmentService和getDevelopmentOperation获取operation,在根据operation中的requestRef不同实现以及不同的 + * RequestRefConsumer先对requestRef做处理,最后执行operation得到 ResponseRef结果后返回 + * + * @param getDevelopmentService DevelopmentService获取器 + * @param getDevelopmentOperation operation获取器 + * @param jobContentRequestRefConsumer 对RequestRef的jobContent处理器。如果处理器不为空,且requestRef是DSSJobContentRequestRef的话,会本执行此处理器 + * @param refJobContentRequestRefConsumer 对RequestRef的refJobContent处理器。如果处理不为空,且requestRef是RefJobContentRequestRef的话,会执行此处理器 + * @param contextRequestRefConsumer 对RequestRef的context处理器。如果处理不为空,且requestRef是DSSContextRequestReff的话,会执行此处理器 + * @param projectRefRequestRefConsumer 对RequestRef的project处理器。如果处理不为空,且requestRef是ProjectRefRequestRef的话,会执行此处理器 + * @param requestRefOperationFunction operation的执行器,由外部传入 + * @param responseRefConsumer + * @param errorMsg + * @return + * @param + * @param + */ public static V tryDevelopmentOperation(Supplier getDevelopmentService, Function getDevelopmentOperation, Consumer jobContentRequestRefConsumer, @@ -53,7 +76,21 @@ public static V tryDeve if(projectRefRequestRefConsumer != null && requestRef instanceof ProjectRefRequestRef) { projectRefRequestRefConsumer.accept((ProjectRefRequestRef) requestRef); } - V responseRef = requestRefOperationFunction.apply(operation, requestRef); + V responseRef; + try { + responseRef = requestRefOperationFunction.apply(operation, requestRef); + } catch (WarnException e) { + String error; + if(StringUtils.isBlank(e.getDesc())) { + error = String.format("%s failed, no detail error returned by this AppConn, please ask admin for help.", errorMsg); + } else { + error = String.format("%s failed. Caused by: %s.", errorMsg, e.getDesc()); + } + throw new ExternalOperationFailedException(50010, error, e); + } catch (RuntimeException e) { + String error = String.format("%s failed. Caused by: %s.", errorMsg, ExceptionUtils.getRootCauseMessage(e)); + throw new ExternalOperationFailedException(50010, error, e); + } if(responseRef.isFailed()) { LOGGER.error("{} failed. Caused by: {}.", errorMsg, responseRef.getErrorMsg()); DSSExceptionUtils.dealWarnException(61123, diff --git a/dss-standard/dss-standard-common/pom.xml b/dss-standard/dss-standard-common/pom.xml index f82d4f624a..31c6cf8c80 100644 --- a/dss-standard/dss-standard-common/pom.xml +++ b/dss-standard/dss-standard-common/pom.xml @@ -21,7 +21,7 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 ../../pom.xml 4.0.0 diff --git a/dss-standard/dss-standard-common/src/main/java/com/webank/wedatasphere/dss/standard/common/desc/AppDescImpl.java b/dss-standard/dss-standard-common/src/main/java/com/webank/wedatasphere/dss/standard/common/desc/AppDescImpl.java index 5bbb43786d..0bf7b738f0 100644 --- a/dss-standard/dss-standard-common/src/main/java/com/webank/wedatasphere/dss/standard/common/desc/AppDescImpl.java +++ b/dss-standard/dss-standard-common/src/main/java/com/webank/wedatasphere/dss/standard/common/desc/AppDescImpl.java @@ -75,7 +75,7 @@ public List getAppInstancesByLabels(List labels) throws N } // if all the labels is different form the target, then return null. if(maxSimilarity <= 0) { - LOG.error("{} has no such AppInstance machs the labels: {}.", appName, labels); + LOG.error("{} has no such AppInstance match the labels: {}.", appName, labels); throw new NoSuchAppInstanceException(60002, "No such AppInstance machs the labels."); } return Collections.singletonList(targetAppInstance); diff --git a/dss-standard/dss-standard-common/src/main/java/com/webank/wedatasphere/dss/standard/common/desc/AppInstance.java b/dss-standard/dss-standard-common/src/main/java/com/webank/wedatasphere/dss/standard/common/desc/AppInstance.java index 1bfa014050..d2cea9c150 100644 --- a/dss-standard/dss-standard-common/src/main/java/com/webank/wedatasphere/dss/standard/common/desc/AppInstance.java +++ b/dss-standard/dss-standard-common/src/main/java/com/webank/wedatasphere/dss/standard/common/desc/AppInstance.java @@ -20,7 +20,9 @@ import java.util.List; import java.util.Map; - +/** + * AppConn的一个实例 + */ public interface AppInstance { Long getId(); diff --git a/dss-standard/dss-standard-common/src/main/java/com/webank/wedatasphere/dss/standard/common/desc/AppInstanceImpl.java b/dss-standard/dss-standard-common/src/main/java/com/webank/wedatasphere/dss/standard/common/desc/AppInstanceImpl.java index dfcbe422f8..0335c5ed19 100644 --- a/dss-standard/dss-standard-common/src/main/java/com/webank/wedatasphere/dss/standard/common/desc/AppInstanceImpl.java +++ b/dss-standard/dss-standard-common/src/main/java/com/webank/wedatasphere/dss/standard/common/desc/AppInstanceImpl.java @@ -92,4 +92,15 @@ public boolean equals(Object o) { public int hashCode() { return Objects.hashCode(id); } + + @Override + public String toString() { + return "AppInstanceImpl{" + + "id=" + id + + ", baseUrl='" + baseUrl + '\'' + + ", homepageUri='" + homepageUri + '\'' + + ", config=" + config + + ", labels=" + labels + + '}'; + } } diff --git a/dss-standard/dss-standard-common/src/main/java/com/webank/wedatasphere/dss/standard/common/utils/AppStandardClassUtils.java b/dss-standard/dss-standard-common/src/main/java/com/webank/wedatasphere/dss/standard/common/utils/AppStandardClassUtils.java index 5778e5b28c..d660e22d6c 100644 --- a/dss-standard/dss-standard-common/src/main/java/com/webank/wedatasphere/dss/standard/common/utils/AppStandardClassUtils.java +++ b/dss-standard/dss-standard-common/src/main/java/com/webank/wedatasphere/dss/standard/common/utils/AppStandardClassUtils.java @@ -17,23 +17,37 @@ package com.webank.wedatasphere.dss.standard.common.utils; import com.webank.wedatasphere.dss.common.utils.ClassUtils.ClassHelper; -import java.util.HashMap; -import java.util.Map; -import java.util.function.Supplier; import org.reflections.Reflections; import org.slf4j.Logger; import org.slf4j.LoggerFactory; +import java.util.HashMap; +import java.util.Map; +import java.util.concurrent.ConcurrentHashMap; +import java.util.function.Supplier; + /** * This class is defined for AppConn jar, if some classes in AppConn want to load a class in AppConn jar. */ public class AppStandardClassUtils extends ClassHelper { - private static final Map INSTANCES = new HashMap<>(); - private static final Map CLASS_LOADER_MAP = new HashMap<>(); + private static final Map INSTANCES = new ConcurrentHashMap<>(); + private static final Map CLASS_LOADER_MAP = new ConcurrentHashMap<>(); private static final Logger LOGGER = LoggerFactory.getLogger(AppStandardClassUtils.class); + public static ClassLoader refreshClassloader(String appConnName, Supplier createClassLoader) { + if(CLASS_LOADER_MAP.containsKey(appConnName)) { + synchronized (AppStandardClassUtils.class) { + if(CLASS_LOADER_MAP.containsKey(appConnName)) { + CLASS_LOADER_MAP.remove(appConnName); + INSTANCES.remove(appConnName); + } + } + } + return getClassLoader(appConnName, createClassLoader); + } + public static ClassLoader getClassLoader(String appConnName, Supplier createClassLoader) { if(!CLASS_LOADER_MAP.containsKey(appConnName)) { synchronized (AppStandardClassUtils.class) { @@ -46,6 +60,7 @@ public static ClassLoader getClassLoader(String appConnName, Supplier RequestRef的具体实现类 + */ public static T getRequestRef(Object operation) { ParameterizedType parameterizedType = (ParameterizedType) operation.getClass().getGenericSuperclass(); return newInstance(operation, parameterizedType); diff --git a/dss-standard/pom.xml b/dss-standard/pom.xml index 6131545d2a..077e5f04e7 100644 --- a/dss-standard/pom.xml +++ b/dss-standard/pom.xml @@ -21,7 +21,7 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 ../pom.xml 4.0.0 diff --git a/dss-standard/sso-standard/origin-sso-integration-standard/pom.xml b/dss-standard/sso-standard/origin-sso-integration-standard/pom.xml index b1a8ffdb6f..daba175339 100644 --- a/dss-standard/sso-standard/origin-sso-integration-standard/pom.xml +++ b/dss-standard/sso-standard/origin-sso-integration-standard/pom.xml @@ -21,7 +21,7 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 ../../../pom.xml 4.0.0 diff --git a/dss-standard/sso-standard/origin-sso-integration-standard/src/main/scala/com/webank/wedatasphere/dss/standard/app/sso/origin/HttpSSOIntegrationStandard.scala b/dss-standard/sso-standard/origin-sso-integration-standard/src/main/scala/com/webank/wedatasphere/dss/standard/app/sso/origin/HttpSSOIntegrationStandard.scala index d87eb415f8..edee8b3da1 100644 --- a/dss-standard/sso-standard/origin-sso-integration-standard/src/main/scala/com/webank/wedatasphere/dss/standard/app/sso/origin/HttpSSOIntegrationStandard.scala +++ b/dss-standard/sso-standard/origin-sso-integration-standard/src/main/scala/com/webank/wedatasphere/dss/standard/app/sso/origin/HttpSSOIntegrationStandard.scala @@ -25,10 +25,12 @@ import scala.collection.mutable */ class HttpSSOIntegrationStandard extends SSOIntegrationStandard { - private val ssoPluginService: SSOPluginService = new OriginSSOPluginServiceImpl + private val ssoPluginService: SSOPluginService = createSSOPluginService() private val ssoRequestService: SSORequestService = createSSORequestService() private val ssoUserServices = new mutable.ArrayBuffer[SSOUserService]() + protected def createSSOPluginService(): SSOPluginService = new OriginSSOPluginServiceImpl + protected def createSSORequestService(): SSORequestService = new HttpSSORequestServiceImpl override def getSSORequestService: SSORequestService = ssoRequestService @@ -39,6 +41,7 @@ class HttpSSOIntegrationStandard extends SSOIntegrationStandard { ssoUserServices.find(_.getAppInstance == appInstance).getOrElse(ssoUserServices synchronized { ssoUserServices.find(_.getAppInstance == appInstance).getOrElse { val service = new SSOUserServiceImpl + service.setSSORequestService(getSSORequestService) service.setAppInstance(appInstance) ssoUserServices += service service diff --git a/dss-standard/sso-standard/origin-sso-integration-standard/src/main/scala/com/webank/wedatasphere/dss/standard/app/sso/origin/plugin/OriginSSOMsgParseOperation.scala b/dss-standard/sso-standard/origin-sso-integration-standard/src/main/scala/com/webank/wedatasphere/dss/standard/app/sso/origin/plugin/OriginSSOMsgParseOperation.scala index 7597e8a39f..aa8fa68443 100644 --- a/dss-standard/sso-standard/origin-sso-integration-standard/src/main/scala/com/webank/wedatasphere/dss/standard/app/sso/origin/plugin/OriginSSOMsgParseOperation.scala +++ b/dss-standard/sso-standard/origin-sso-integration-standard/src/main/scala/com/webank/wedatasphere/dss/standard/app/sso/origin/plugin/OriginSSOMsgParseOperation.scala @@ -19,24 +19,26 @@ package com.webank.wedatasphere.dss.standard.app.sso.origin.plugin import com.webank.wedatasphere.dss.standard.app.sso.builder.DssMsgBuilderOperation.DSSMsg import com.webank.wedatasphere.dss.standard.app.sso.origin.client.HttpClient import com.webank.wedatasphere.dss.standard.app.sso.plugin.AbstractSSOMsgParseOperation +import com.webank.wedatasphere.dss.standard.sso.utils.ProxyUserSSOUtils import org.apache.linkis.common.utils.Utils -import org.apache.linkis.httpclient.Client class OriginSSOMsgParseOperation extends AbstractSSOMsgParseOperation { override protected def getUser(dssMsg: DSSMsg): String = { val dssUrl = dssMsg.getDSSUrl - val dwsHttpClient:Client=null - Utils.tryFinally({ - val dwsHttpClient = HttpClient.getHttpClient(dssUrl, "DSS") - val userInfoAction = new UserInfoAction + val dwsHttpClient = HttpClient.getHttpClient(dssUrl, "DSS") + val userInfoAction = if(ProxyUserSSOUtils.existsProxyUser(dssMsg)) new ProxyUserInfoAction + else new UserInfoAction + Utils.tryFinally { HttpClient.addCookies(dssMsg, userInfoAction) dwsHttpClient.execute(userInfoAction) match { case userInfoResult: UserInfoResult => userInfoResult.getUserName + case proxyUserInfoResult: ProxyUserInfoResult => + ProxyUserSSOUtils.setUserAndProxyUser(proxyUserInfoResult.getUserName, proxyUserInfoResult.getProxyUser) } - })(Utils.tryQuietly(dwsHttpClient.close())) + }(Utils.tryQuietly(dwsHttpClient.close())) } } diff --git a/dss-standard/sso-standard/origin-sso-integration-standard/src/main/scala/com/webank/wedatasphere/dss/standard/app/sso/origin/plugin/OriginSSOPluginFilter.scala b/dss-standard/sso-standard/origin-sso-integration-standard/src/main/scala/com/webank/wedatasphere/dss/standard/app/sso/origin/plugin/OriginSSOPluginFilter.scala index f4979db570..e6524e28dd 100644 --- a/dss-standard/sso-standard/origin-sso-integration-standard/src/main/scala/com/webank/wedatasphere/dss/standard/app/sso/origin/plugin/OriginSSOPluginFilter.scala +++ b/dss-standard/sso-standard/origin-sso-integration-standard/src/main/scala/com/webank/wedatasphere/dss/standard/app/sso/origin/plugin/OriginSSOPluginFilter.scala @@ -25,6 +25,7 @@ import org.apache.linkis.common.utils.Logging abstract class OriginSSOPluginFilter extends SSOPluginFilter with Logging{ private val factory = new OriginSSOIntegrationStandardFactory + factory.init() override def info(str: String): Unit = logger.info(str) diff --git a/dss-standard/sso-standard/origin-sso-integration-standard/src/main/scala/com/webank/wedatasphere/dss/standard/app/sso/origin/plugin/UserInfoAction.scala b/dss-standard/sso-standard/origin-sso-integration-standard/src/main/scala/com/webank/wedatasphere/dss/standard/app/sso/origin/plugin/UserInfoAction.scala index 01be797564..901a1d3185 100644 --- a/dss-standard/sso-standard/origin-sso-integration-standard/src/main/scala/com/webank/wedatasphere/dss/standard/app/sso/origin/plugin/UserInfoAction.scala +++ b/dss-standard/sso-standard/origin-sso-integration-standard/src/main/scala/com/webank/wedatasphere/dss/standard/app/sso/origin/plugin/UserInfoAction.scala @@ -24,10 +24,17 @@ import org.apache.linkis.httpclient.request.POSTAction class UserInfoAction extends GetAction with DWSHttpAction { override def suffixURLs: Array[String] = Array("user", "userInfo") } + class WorkspaceUsersAction extends POSTAction with DWSHttpAction { override def getRequestPayload: String = "" override def suffixURLs: Array[String] = Array("dss", "getUsersOfWorkspace") def setWorkspace(workspace: String): Unit = addRequestPayload("workspaceName", workspace) +} + +class ProxyUserInfoAction extends GetAction with DWSHttpAction { + + override def suffixURLs: Array[String] = Array("dss", "framework", "proxy", "getProxyUser") + } \ No newline at end of file diff --git a/dss-standard/sso-standard/origin-sso-integration-standard/src/main/scala/com/webank/wedatasphere/dss/standard/app/sso/origin/plugin/UserInfoResult.scala b/dss-standard/sso-standard/origin-sso-integration-standard/src/main/scala/com/webank/wedatasphere/dss/standard/app/sso/origin/plugin/UserInfoResult.scala index 834b3e03fa..ac45cba171 100644 --- a/dss-standard/sso-standard/origin-sso-integration-standard/src/main/scala/com/webank/wedatasphere/dss/standard/app/sso/origin/plugin/UserInfoResult.scala +++ b/dss-standard/sso-standard/origin-sso-integration-standard/src/main/scala/com/webank/wedatasphere/dss/standard/app/sso/origin/plugin/UserInfoResult.scala @@ -34,4 +34,13 @@ class WorkspaceUsersResult extends DWSResult { @BeanProperty var users: java.util.List[String] = _ +} + +@DWSHttpMessageResult("/api/rest_j/v\\d+/dss/framework/proxy/getProxyUser") +class ProxyUserInfoResult extends DWSResult { + + @BeanProperty var userName: String = _ + + @BeanProperty var proxyUser: String = _ + } \ No newline at end of file diff --git a/dss-standard/sso-standard/spring-origin-sso-integration-plugin/pom.xml b/dss-standard/sso-standard/spring-origin-sso-integration-plugin/pom.xml index e6e31322d5..d9db5974cf 100644 --- a/dss-standard/sso-standard/spring-origin-sso-integration-plugin/pom.xml +++ b/dss-standard/sso-standard/spring-origin-sso-integration-plugin/pom.xml @@ -21,7 +21,7 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 ../../../pom.xml 4.0.0 @@ -42,19 +42,19 @@ org.springframework spring-core - ${spring.version} + ${spring-framework.version} compile org.springframework spring-context - ${spring.version} + ${spring-framework.version} compile org.springframework spring-web - ${spring.version} + ${spring-framework.version} compile diff --git a/dss-standard/sso-standard/sso-integration-standard/pom.xml b/dss-standard/sso-standard/sso-integration-standard/pom.xml index 139c364d95..0cf758c8c3 100644 --- a/dss-standard/sso-standard/sso-integration-standard/pom.xml +++ b/dss-standard/sso-standard/sso-integration-standard/pom.xml @@ -21,7 +21,7 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 ../../../pom.xml 4.0.0 diff --git a/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/sso/builder/impl/DssMsgBuilderOperationImpl.java b/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/sso/builder/impl/DssMsgBuilderOperationImpl.java index c12cce70f9..95ff53f39a 100644 --- a/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/sso/builder/impl/DssMsgBuilderOperationImpl.java +++ b/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/sso/builder/impl/DssMsgBuilderOperationImpl.java @@ -76,7 +76,7 @@ public DSSMsg getBuiltMsg() { cookies.put(key, value); }); } - LOGGER.info("Set cookies from dssMsg: "+cookies.toString()); +// LOGGER.info("Set cookies from dssMsg: "+cookies.toString()); dssMsg.setCookies(cookies); return dssMsg; } diff --git a/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/sso/builder/impl/SSOBuilderServiceImplImpl.java b/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/sso/builder/impl/SSOBuilderServiceImplImpl.java index 10ed2d681b..a1f284823c 100644 --- a/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/sso/builder/impl/SSOBuilderServiceImplImpl.java +++ b/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/sso/builder/impl/SSOBuilderServiceImplImpl.java @@ -24,7 +24,7 @@ public class SSOBuilderServiceImplImpl extends AppServiceImpl implements SSOBuilderService { - private static SSOBuilderService ssoBuilderService; + private static volatile SSOBuilderService ssoBuilderService; public static SSOBuilderService getSSOBuilderService() { if(ssoBuilderService == null) { diff --git a/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/sso/plugin/DssMsgCacheOperation.java b/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/sso/plugin/DssMsgCacheOperation.java index 04fdf906f6..f8dfdffe07 100644 --- a/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/sso/plugin/DssMsgCacheOperation.java +++ b/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/sso/plugin/DssMsgCacheOperation.java @@ -31,4 +31,8 @@ public interface DssMsgCacheOperation extends Operation { void setDSSMsgToSession(DSSMsg dssMsg, HttpServletRequest request); + void setExistsProxyUser(HttpServletRequest req); + + boolean isExistsProxyUser(HttpServletRequest req); + } diff --git a/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/sso/plugin/filter/SSOPluginFilter.scala b/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/sso/plugin/filter/SSOPluginFilter.scala index 7edc6d7c97..8cc966675f 100644 --- a/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/sso/plugin/filter/SSOPluginFilter.scala +++ b/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/sso/plugin/filter/SSOPluginFilter.scala @@ -16,10 +16,14 @@ package com.webank.wedatasphere.dss.standard.app.sso.plugin.filter +import com.webank.wedatasphere.dss.common.conf.DSSCommonConf.DSS_TOKEN_TICKET_KEY import com.webank.wedatasphere.dss.standard.app.sso.SSOIntegrationStandard +import com.webank.wedatasphere.dss.standard.app.sso.plugin.filter.proxy.{DSSInternalProxyUserInterceptor, HttpRequestProxyUserInterceptor, HttpSessionProxyUserInterceptor} +import com.webank.wedatasphere.dss.standard.sso.utils.ProxyUserSSOUtils import javax.servlet._ import javax.servlet.http.{HttpServletRequest, HttpServletResponse} import org.apache.commons.lang.StringUtils +import org.apache.linkis.protocol.util.ImmutablePair abstract class SSOPluginFilter extends Filter { @@ -43,33 +47,58 @@ abstract class SSOPluginFilter extends Filter { val resp = servletResponse.asInstanceOf[HttpServletResponse] val ssoPluginService = getSSOIntegrationStandard.getSSOPluginService val dssMsg = ssoPluginService.createSSOMsgParseOperation().getDSSMsg(req) + val workspaceOperation = ssoPluginService.createDssMsgCacheOperation() if (ssoPluginService.createSSOMsgParseOperation.isDssRequest(req)) { - val (redirectUrl, workspaceName, wrappedReq) = if(!userInterceptor.isUserExistInSession(req)) { + val isFirstLogin = !userInterceptor.isUserExistInSession(req) || (workspaceOperation.getDSSMsgInSession(req) != null && + req.getCookies.find(_.getName == DSS_TOKEN_TICKET_KEY.getValue).map(_.getValue).exists(_ != workspaceOperation.getDSSMsgInSession(req).getCookies.get(DSS_TOKEN_TICKET_KEY.getValue))) + // 存在一种场景:第一次登陆时没有带上代理用户(DSS登陆成功后就开始请求第三方节点),这时为了使第三方节点能够正常使用,允许完成SSO认证。 + // 这时,如果DSS绑定了代理用户,再来请求这个第三方节点时,由于已经完成认证会直接放行,所以这里加上一个判断逻辑, + // 就是就算完成了登录认证,但是如果之前的登录认证没有包含代理用户,而本次请求的cookie中包含代理用户,则触发再次认证。 + val isSecondLoginWithProxyUser = userInterceptor.isUserExistInSession(req) && !workspaceOperation.isExistsProxyUser(req) && + ProxyUserSSOUtils.existsProxyUser(dssMsg) + val (redirectUrl, workspaceName, wrappedReq) = if(isFirstLogin || isSecondLoginWithProxyUser) { val ssoMsg = ssoPluginService.createSSOMsgParseOperation.getSSOMsg(req) val username = ssoMsg.getUser + val pair = if(ProxyUserSSOUtils.existsProxyUser(username)) { + workspaceOperation.setExistsProxyUser(req) + ProxyUserSSOUtils.getUserAndProxyUser(username) + } else new ImmutablePair[String, String](username, username) val wrappedReq = userInterceptor match { + case interceptor: HttpSessionProxyUserInterceptor => + interceptor.addUserToSession(pair.getKey, pair.getValue, req) + req + case interceptor: HttpRequestProxyUserInterceptor => + interceptor.addUserToRequest(pair.getKey, pair.getValue, req) + case interceptor: DSSInternalProxyUserInterceptor => + interceptor.addCookiesToRequest(dssMsg, pair.getKey, pair.getValue, req) case interceptor: HttpSessionUserInterceptor => - interceptor.addUserToSession(username, req) + interceptor.addUserToSession(pair.getValue, req) req - case interceptor: HttpRequestUserInterceptor => interceptor.addUserToRequest(username, req) - case interceptor: DSSInternalUserInterceptor => interceptor.addCookiesToRequest(dssMsg, req) + case interceptor: HttpRequestUserInterceptor => + interceptor.addUserToRequest(pair.getValue, req) + case interceptor: DSSInternalUserInterceptor => + interceptor.addCookiesToRequest(dssMsg, req) } if(wrappedReq != req) { wrappedReq.getCookies.foreach(resp.addCookie) } - info(s"DSS User: $username succeed to login. ") + if(ProxyUserSSOUtils.existsProxyUser(username)) { + info(s"DSS proxy user mode is opened. DSS user: ${pair.getKey} proxy to proxyUser: ${pair.getValue} login succeed.") + } else { + info(s"DSS User: $username succeed to login. ") + } (ssoMsg.getRedirectUrl, ssoMsg.getWorkspaceName, wrappedReq) } else { (dssMsg.getRedirectUrl, dssMsg.getWorkspaceName, req) } - val workspaceOperation = ssoPluginService.createDssMsgCacheOperation() + + if(workspaceOperation.getDSSMsgInSession(req) == null) { + workspaceOperation.setDSSMsgToSession(dssMsg, req) + } if(workspaceName != workspaceOperation.getWorkspaceInSession(req)) { info(s"Set DSS workspace to: $workspaceName.") workspaceOperation.setWorkspaceToSession(req, workspaceName) } - if(workspaceOperation.getDSSMsgInSession(req) == null) { - workspaceOperation.setDSSMsgToSession(dssMsg, req) - } if(StringUtils.isNotBlank(redirectUrl)) resp.sendRedirect(redirectUrl) else filterChain.doFilter(wrappedReq, servletResponse) } else { diff --git a/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/sso/plugin/filter/proxy/DSSInternalProxyUserInterceptor.java b/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/sso/plugin/filter/proxy/DSSInternalProxyUserInterceptor.java new file mode 100644 index 0000000000..4b235f918e --- /dev/null +++ b/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/sso/plugin/filter/proxy/DSSInternalProxyUserInterceptor.java @@ -0,0 +1,20 @@ +package com.webank.wedatasphere.dss.standard.app.sso.plugin.filter.proxy; + +import com.webank.wedatasphere.dss.standard.app.sso.builder.DssMsgBuilderOperation; + +import javax.servlet.http.HttpServletRequest; + +/** + * @author enjoyyin + * @date 2022-09-02 + * @since 0.5.0 + */ +public interface DSSInternalProxyUserInterceptor extends ProxyUserInterceptor { + + HttpServletRequest addCookiesToRequest(DssMsgBuilderOperation.DSSMsg dssMsg, String user, String proxyUser, HttpServletRequest req); + + @Override + default ProxyUserType getProxyUserType() { + return ProxyUserType.USER_WITH_PROXY_USER; + } +} diff --git a/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/sso/plugin/filter/proxy/HttpRequestProxyUserInterceptor.java b/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/sso/plugin/filter/proxy/HttpRequestProxyUserInterceptor.java new file mode 100644 index 0000000000..3d60a1c218 --- /dev/null +++ b/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/sso/plugin/filter/proxy/HttpRequestProxyUserInterceptor.java @@ -0,0 +1,18 @@ +package com.webank.wedatasphere.dss.standard.app.sso.plugin.filter.proxy; + +import javax.servlet.http.HttpServletRequest; + +/** + * @author enjoyyin + * @date 2022-09-02 + * @since 0.5.0 + */ +public interface HttpRequestProxyUserInterceptor extends ProxyUserInterceptor { + + HttpServletRequest addUserToRequest(String user, String proxyUser, HttpServletRequest req); + + @Override + default ProxyUserType getProxyUserType() { + return ProxyUserType.USER_WITH_PROXY_USER; + } +} diff --git a/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/sso/plugin/filter/proxy/HttpSessionProxyUserInterceptor.java b/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/sso/plugin/filter/proxy/HttpSessionProxyUserInterceptor.java new file mode 100644 index 0000000000..836a15c42f --- /dev/null +++ b/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/sso/plugin/filter/proxy/HttpSessionProxyUserInterceptor.java @@ -0,0 +1,18 @@ +package com.webank.wedatasphere.dss.standard.app.sso.plugin.filter.proxy; + +import javax.servlet.http.HttpServletRequest; + +/** + * @author enjoyyin + * @date 2022-09-02 + * @since 0.5.0 + */ +public interface HttpSessionProxyUserInterceptor extends ProxyUserInterceptor { + + void addUserToSession(String user, String proxyUser, HttpServletRequest req); + + @Override + default ProxyUserType getProxyUserType() { + return ProxyUserType.USER_WITH_PROXY_USER; + } +} diff --git a/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/sso/plugin/filter/proxy/ProxyUserInterceptor.java b/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/sso/plugin/filter/proxy/ProxyUserInterceptor.java new file mode 100644 index 0000000000..e084479497 --- /dev/null +++ b/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/sso/plugin/filter/proxy/ProxyUserInterceptor.java @@ -0,0 +1,29 @@ +package com.webank.wedatasphere.dss.standard.app.sso.plugin.filter.proxy; + +import com.webank.wedatasphere.dss.standard.app.sso.plugin.filter.UserInterceptor; + +/** + * @author enjoyyin + * @date 2022-09-02 + * @since 0.5.0 + */ +public interface ProxyUserInterceptor extends UserInterceptor { + + default ProxyUserType getProxyUserType() { + return ProxyUserType.ONLY_PROXY_USER; + } + + enum ProxyUserType { + /** + * 该第三方应用只支持代理用户,即:如果DSS开启了用户登录+切换到代理用户的双用户体系之后,该第三方应用为了 + * 简化使用,只接受DSS传递代理用户给第三方系统,将不接受实际登录的用户,只使用代理用户进行所有操作。 + * 注意:如果用户不显示指定,则该模式为第三方应用的默认模式。 + */ + ONLY_PROXY_USER, + /** + * 该第三方应用支持 登录用户 + 切换到代理用户的双用户体系。 + * DSS 会同时传递登录用户和代理用户给第三方应用,由第三方应用自行决定如何使用这两个用户。 + */ + USER_WITH_PROXY_USER + } +} diff --git a/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/sso/utils/ProxyUserSSOUtils.java b/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/sso/utils/ProxyUserSSOUtils.java new file mode 100644 index 0000000000..7d8db7245b --- /dev/null +++ b/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/sso/utils/ProxyUserSSOUtils.java @@ -0,0 +1,65 @@ +package com.webank.wedatasphere.dss.standard.sso.utils; + + +import com.webank.wedatasphere.dss.common.exception.DSSRuntimeException; +import com.webank.wedatasphere.dss.standard.app.sso.builder.DssMsgBuilderOperation; +import org.apache.linkis.protocol.util.ImmutablePair; + +public class ProxyUserSSOUtils { + + private static final String PROXY_USER_TICKET_ID_STRING = "dss_user_session_proxy_ticket_id_v1"; + private static final String PROXY_USER_PREFIX = "proxy_"; + + /** + * 从 cookie 中查找是否包含了代理用户的 cookie,用以表示 DSS 是否开启了代理功能 + * @param dssMsg + * @return 返回 true 如果包含了代理用户,否则返回 false + */ + public static boolean existsProxyUser(DssMsgBuilderOperation.DSSMsg dssMsg) { + return dssMsg.getCookies().containsKey(PROXY_USER_TICKET_ID_STRING); + } + + /** + * 通过 {@code SSOMsgParseOperation} 的 getUser 方法中获得的 user,是否是由 userName + proxyUser 组成的 + * @param user {@code SSOMsgParseOperation} 的 getUser 方法中获得的 user + * @return true 如果是 user 是由userName + proxyUser 组成,否则返回 false + */ + public static boolean existsProxyUser(String user) { + return user.startsWith(PROXY_USER_PREFIX); + } + + /** + * 通过 {@code SSOMsgParseOperation} 的 getUser 方法中获得的 user,获取 userName 和 proxyUser + * @param user {@code SSOMsgParseOperation} 的 getUser 方法中获得的 user + * @return key 为 userName,value 为 proxyUser + */ + public static ImmutablePair getUserAndProxyUser(String user) { + if(!user.startsWith(PROXY_USER_PREFIX)) { + throw new DSSRuntimeException(56000, "not exists proxyUser."); + } + String userAndProxyUser = user.substring(PROXY_USER_PREFIX.length()); + int length = Integer.parseInt(userAndProxyUser.substring(0, 2)); + String userName = userAndProxyUser.substring(2, 2 + length); + String proxyUser = userAndProxyUser.substring(2 + length); + return new ImmutablePair<>(userName, proxyUser); + } + + /** + * 返回一个经过特殊处理的、包含了 userName 和 proxyUser 的特殊字符串,可通过 {@code getUserAndProxyUser} 解密获取 userName 和 proxyUser + * @param userName 登录用户名 + * @param proxyUser 代理用户 + * @return 返回包含了 userName 和 proxyUser 的特殊字符串 + */ + public static String setUserAndProxyUser(String userName, String proxyUser) { + int length = userName.length(); + if(length > 99) { + throw new DSSRuntimeException(56000, "the length of userName is too long, at most 99 characters are supported."); + } + String lengthStr = String.valueOf(length); + if(length < 10) { + lengthStr = "0" + lengthStr; + } + return String.format("%s%s%s%s", PROXY_USER_PREFIX, lengthStr, userName, proxyUser); + } + +} diff --git a/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/sso/utils/SSOHelper.java b/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/sso/utils/SSOHelper.java index 1f965486d8..a26f615233 100644 --- a/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/sso/utils/SSOHelper.java +++ b/dss-standard/sso-standard/sso-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/sso/utils/SSOHelper.java @@ -17,6 +17,7 @@ package com.webank.wedatasphere.dss.standard.sso.utils; import com.webank.wedatasphere.dss.common.conf.DSSCommonConf; +import com.webank.wedatasphere.dss.common.utils.DomainUtils; import com.webank.wedatasphere.dss.standard.app.sso.Workspace; import com.webank.wedatasphere.dss.standard.app.sso.builder.SSOBuilderService; import com.webank.wedatasphere.dss.standard.app.sso.builder.SSOUrlBuilderOperation; @@ -107,7 +108,7 @@ public static Workspace setAndGetWorkspace(HttpServletRequest request, HttpServl return getWorkspace(request); } String workspaceIdStr = String.valueOf(workspaceId); - String domain = getCookieDomain(request.getHeader("Referer")); + String domain = getCookieDomain(DomainUtils.getCookieDomain(request)); Cookie workspaceIdCookie = new Cookie(WORKSPACE_ID_COOKIE_KEY, workspaceIdStr); workspaceIdCookie.setPath("/"); // workspaceIdCookie.setDomain(domain); diff --git a/dss-standard/sso-standard/sso-integration-standard/src/main/scala/com/webank/wedatasphere/dss/standard/app/sso/plugin/DssMsgCacheOperationImpl.scala b/dss-standard/sso-standard/sso-integration-standard/src/main/scala/com/webank/wedatasphere/dss/standard/app/sso/plugin/DssMsgCacheOperationImpl.scala index f6d941fb28..9cc3b3e28a 100644 --- a/dss-standard/sso-standard/sso-integration-standard/src/main/scala/com/webank/wedatasphere/dss/standard/app/sso/plugin/DssMsgCacheOperationImpl.scala +++ b/dss-standard/sso-standard/sso-integration-standard/src/main/scala/com/webank/wedatasphere/dss/standard/app/sso/plugin/DssMsgCacheOperationImpl.scala @@ -16,10 +16,6 @@ package com.webank.wedatasphere.dss.standard.app.sso.plugin -import java.lang -import java.lang.reflect.Type - -import com.google.gson.{GsonBuilder, JsonElement, JsonPrimitive, JsonSerializationContext, JsonSerializer} import com.webank.wedatasphere.dss.standard.app.sso.builder.DssMsgBuilderOperation import com.webank.wedatasphere.dss.standard.app.sso.builder.DssMsgBuilderOperation.DSSMsg import com.webank.wedatasphere.dss.standard.app.sso.builder.impl.DSSMsgImpl @@ -60,6 +56,10 @@ class DssMsgCacheOperationImpl private() extends DssMsgCacheOperation { } request.getSession.setAttribute(DssMsgCacheOperationImpl.DSS_MSG_KEY, newDSSMsg) } + + override def setExistsProxyUser(req: HttpServletRequest): Unit = req.getSession.setAttribute("existsDSSProxyUser", true) + + override def isExistsProxyUser(req: HttpServletRequest): Boolean = req.getSession.getAttribute("existsDSSProxyUser") != null } object DssMsgCacheOperationImpl { private val DSS_MSG_KEY = "dss_msg_key" diff --git a/dss-standard/structure-standard/dss-project-plugin/pom.xml b/dss-standard/structure-standard/dss-project-plugin/pom.xml index 8923fa4f5a..1a917f3d6b 100644 --- a/dss-standard/structure-standard/dss-project-plugin/pom.xml +++ b/dss-standard/structure-standard/dss-project-plugin/pom.xml @@ -21,7 +21,7 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 ../../../pom.xml 4.0.0 diff --git a/dss-standard/structure-standard/dss-project-plugin/src/main/scala/com/webank/wedatasphere/dss/standard/app/structure/project/plugin/filter/ProjectCooperationFilter.scala b/dss-standard/structure-standard/dss-project-plugin/src/main/scala/com/webank/wedatasphere/dss/standard/app/structure/project/plugin/filter/ProjectCooperationFilter.scala index 7882b6d833..036afe498f 100644 --- a/dss-standard/structure-standard/dss-project-plugin/src/main/scala/com/webank/wedatasphere/dss/standard/app/structure/project/plugin/filter/ProjectCooperationFilter.scala +++ b/dss-standard/structure-standard/dss-project-plugin/src/main/scala/com/webank/wedatasphere/dss/standard/app/structure/project/plugin/filter/ProjectCooperationFilter.scala @@ -25,6 +25,7 @@ import com.webank.wedatasphere.dss.standard.app.sso.plugin.filter.UserIntercepto import com.webank.wedatasphere.dss.standard.app.structure.project.plugin.conf.ProjectCooperateConfiguration import com.webank.wedatasphere.dss.standard.app.structure.project.plugin.filter.ProjectRequestType.{Access, Delete, Edit, Execute} import com.webank.wedatasphere.dss.standard.app.structure.project.plugin.{ProjectAuth, ProjectCooperationPlugin} +import com.webank.wedatasphere.dss.standard.sso.utils.ProxyUserSSOUtils import org.apache.linkis.common.conf.Configuration import org.apache.linkis.common.utils.{Logging, Utils} import org.apache.linkis.httpclient.exception.HttpClientResultException @@ -93,8 +94,14 @@ abstract class ProjectCooperationFilter extends Filter with Logging { return } val projectRequestType = projectAuthInterceptor.getProjectRequestType(req) - debug(s"RequestURI: $uri, ProjectRequestType: $projectRequestType, ProjectAuth: $projectAuth.") - val user = userInterceptor.getUser(req) + var user = userInterceptor.getUser(req) + info(s"RequestURI: $uri, ProjectRequestType: $projectRequestType, ProjectAuth: $projectAuth, user: $user.") + if(ProxyUserSSOUtils.existsProxyUser(user)) { + // 这里使用代理用户的权限来表示实际的工程权限 + val pair = ProxyUserSSOUtils.getUserAndProxyUser(user) + info(s"use proxy user ${pair.getValue} to decide if user ${pair.getKey} can $projectRequestType cooperation project ${projectAuth.getProjectName}.") + user = pair.getValue + } val passed = projectRequestType match { case Edit => projectAuth.getEditUsers.contains(user) case Access => projectAuth.getAccessUsers.contains(user) @@ -106,7 +113,7 @@ abstract class ProjectCooperationFilter extends Filter with Logging { filterChain.doFilter(req, resp) } else { val msg = projectAuthInterceptor.getForbiddenMsg(projectAuth, projectRequestType, req) - info(s"Cooperation Project request $uri has been failed with auth validation.") + info(s"Cooperation Project request $uri has been failed with auth validation for user $user.") resp.getOutputStream.write(msg.getBytes(Configuration.BDP_ENCODING.getValue)) resp.setStatus(403) resp.getOutputStream.flush() diff --git a/dss-standard/structure-standard/dss-role-plugin/pom.xml b/dss-standard/structure-standard/dss-role-plugin/pom.xml index 841f66742a..a3c35f3cb2 100644 --- a/dss-standard/structure-standard/dss-role-plugin/pom.xml +++ b/dss-standard/structure-standard/dss-role-plugin/pom.xml @@ -21,7 +21,7 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 ../../../pom.xml 4.0.0 diff --git a/dss-standard/structure-standard/dss-structure-integration-standard/pom.xml b/dss-standard/structure-standard/dss-structure-integration-standard/pom.xml index 47c8e97b0e..9e146cdd05 100644 --- a/dss-standard/structure-standard/dss-structure-integration-standard/pom.xml +++ b/dss-standard/structure-standard/dss-structure-integration-standard/pom.xml @@ -21,7 +21,7 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 ../../../pom.xml 4.0.0 diff --git a/dss-standard/structure-standard/dss-structure-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/structure/utils/StructureOperationUtils.java b/dss-standard/structure-standard/dss-structure-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/structure/utils/StructureOperationUtils.java index 746102b837..2586ba3c79 100644 --- a/dss-standard/structure-standard/dss-structure-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/structure/utils/StructureOperationUtils.java +++ b/dss-standard/structure-standard/dss-structure-integration-standard/src/main/java/com/webank/wedatasphere/dss/standard/app/structure/utils/StructureOperationUtils.java @@ -9,7 +9,11 @@ import com.webank.wedatasphere.dss.standard.app.structure.project.ref.RefProjectContentRequestRef; import com.webank.wedatasphere.dss.standard.common.entity.ref.ResponseRef; import com.webank.wedatasphere.dss.standard.common.exception.AppStandardWarnException; +import com.webank.wedatasphere.dss.standard.common.exception.operation.ExternalOperationFailedException; import com.webank.wedatasphere.dss.standard.common.utils.RequestRefUtils; +import org.apache.commons.lang3.StringUtils; +import org.apache.commons.lang3.exception.ExceptionUtils; +import org.apache.linkis.common.exception.WarnException; import org.slf4j.Logger; import org.slf4j.LoggerFactory; @@ -46,7 +50,21 @@ public static dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 ../../../pom.xml 4.0.0 @@ -37,19 +37,19 @@ org.springframework spring-core - ${spring.version} + ${spring-framework.version} compile org.springframework spring-context - ${spring.version} + ${spring-framework.version} compile org.springframework spring-web - ${spring.version} + ${spring-framework.version} compile diff --git a/plugins/azkaban/linkis-jobtype/pom.xml b/plugins/azkaban/linkis-jobtype/pom.xml index eaf875129a..b287e838a5 100644 --- a/plugins/azkaban/linkis-jobtype/pom.xml +++ b/plugins/azkaban/linkis-jobtype/pom.xml @@ -22,7 +22,8 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 + ../../../pom.xml linkis-jobtype diff --git a/plugins/azkaban/linkis-jobtype/src/main/java/com/webank/wedatasphere/dss/plugins/azkaban/linkis/jobtype/AzkabanDssJobType.java b/plugins/azkaban/linkis-jobtype/src/main/java/com/webank/wedatasphere/dss/plugins/azkaban/linkis/jobtype/AzkabanDssJobType.java index e897f4b27e..0007b771d0 100644 --- a/plugins/azkaban/linkis-jobtype/src/main/java/com/webank/wedatasphere/dss/plugins/azkaban/linkis/jobtype/AzkabanDssJobType.java +++ b/plugins/azkaban/linkis-jobtype/src/main/java/com/webank/wedatasphere/dss/plugins/azkaban/linkis/jobtype/AzkabanDssJobType.java @@ -25,8 +25,9 @@ import com.webank.wedatasphere.dss.linkis.node.execution.job.JobTypeEnum; import com.webank.wedatasphere.dss.linkis.node.execution.job.LinkisJob; import com.webank.wedatasphere.dss.linkis.node.execution.listener.LinkisExecutionListener; +import com.webank.wedatasphere.dss.plugins.azkaban.linkis.jobtype.conf.LinkisJobTypeConf; import com.webank.wedatasphere.dss.plugins.azkaban.linkis.jobtype.job.JobBuilder; -import com.webank.wedatasphere.dss.plugins.azkaban.linkis.jobtype.log.AzkabanAppConnLog; +import com.webank.wedatasphere.dss.plugins.azkaban.linkis.jobtype.log.AzkabanJobLog; import org.apache.commons.lang.StringUtils; import org.slf4j.Logger; @@ -89,13 +90,17 @@ public void run() throws Exception { info("Start to execute job"); logJobProperties(); String runDate = getRunDate(); - if (StringUtils.isNotBlank(runDate)){ + if (StringUtils.isNotBlank(runDate)) { this.jobPropsMap.put("run_date", runDate); } + String runTodayH = getRunTodayh(false); + if (StringUtils.isNotBlank(runTodayH)) { + this.jobPropsMap.put("run_today_h", runTodayH); + } this.job = JobBuilder.getAzkanbanBuilder().setJobProps(this.jobPropsMap).build(); - this.job.setLogObj(new AzkabanAppConnLog(this.log)); + this.job.setLogObj(new AzkabanJobLog(this)); if(JobTypeEnum.EmptyJob == ((LinkisJob)this.job).getJobType()){ - this.log.warn("This node is empty type"); + warn("This node is empty type"); return; } // info("runtimeMap is " + job.getRuntimeParams()); @@ -106,16 +111,16 @@ public void run() throws Exception { try { LinkisNodeExecutionImpl.getLinkisNodeExecution().waitForComplete(this.job); } catch (Exception e) { - this.log.warn("Failed to execute job", e); + warn("Failed to execute job", e); //String reason = LinkisNodeExecutionImpl.getLinkisNodeExecution().getLog(this.job); //this.log.error("Reason for failure: " + reason); throw e; } try { String endLog = LinkisNodeExecutionImpl.getLinkisNodeExecution().getLog(this.job); - this.log.info(endLog); + info(endLog); } catch (Throwable e){ - this.log.info("Failed to get log", e); + info("Failed to get log", e); } LinkisExecutionListener listener = (LinkisExecutionListener)LinkisNodeExecutionImpl.getLinkisNodeExecution(); @@ -124,12 +129,15 @@ public void run() throws Exception { try{ resultSize = LinkisNodeExecutionImpl.getLinkisNodeExecution().getResultSize(this.job); }catch(final Throwable t){ - this.log.error("failed to get result size"); + error("failed to get result size"); resultSize = -1; } - for(int i =0; i < resultSize; i++){ - this.log.info("The content of the " + (i + 1) + "th resultset is :" - + LinkisNodeExecutionImpl.getLinkisNodeExecution().getResult(this.job, i, LinkisJobExecutionConfiguration.RESULT_PRINT_SIZE.getValue(this.jobPropsMap))); + for (int i = 0; i < resultSize; i++) { + String result = LinkisNodeExecutionImpl.getLinkisNodeExecution().getResult(this.job, i, LinkisJobExecutionConfiguration.RESULT_PRINT_SIZE.getValue(this.jobPropsMap)); + if (result.length() > LinkisJobTypeConf.LOG_MAX_RESULTSIZE.getValue()) { + result = result.substring(0, LinkisJobTypeConf.LOG_MAX_RESULTSIZE.getValue()); + } + info("The content of the " + (i + 1) + "th resultset is :" + result); } info("Finished to execute job"); @@ -204,7 +212,35 @@ private String getRunDate(){ } } } catch (final Exception ex) { - this.log.error("failed to log job properties ", ex); + this.log.error("failed to get run date ", ex); + } + } + return null; + } + + private String getRunTodayh(boolean stdFormat) { + this.info("begin to get run_today_h"); + if (this.jobProps != null && + this.jobProps.getBoolean(JOB_DUMP_PROPERTIES_IN_LOG, true)) { + try { + for (final Map.Entry entry : this.jobPropsMap.entrySet()) { + final String key = entry.getKey(); + final String value = key.endsWith(SENSITIVE_JOB_PROP_NAME_SUFFIX) ? + SENSITIVE_JOB_PROP_VALUE_PLACEHOLDER : + entry.getValue(); + if ("azkaban.flow.start.timestamp".equals(key)) { + this.info("run time is " + value); + String runTodayh = value.substring(0, 13).replaceAll("-", "").replaceAll("T", ""); + this.info("run today h is " + runTodayh); + //for std +// SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd HH"); + if(!stdFormat){ + return runTodayh; + } + } + } + } catch (final Exception ex) { + this.log.error("failed to get run_today_h ", ex); } } return null; diff --git a/plugins/azkaban/linkis-jobtype/src/main/java/com/webank/wedatasphere/dss/plugins/azkaban/linkis/jobtype/conf/LinkisJobTypeConf.java b/plugins/azkaban/linkis-jobtype/src/main/java/com/webank/wedatasphere/dss/plugins/azkaban/linkis/jobtype/conf/LinkisJobTypeConf.java index d607f45128..8412b79770 100644 --- a/plugins/azkaban/linkis-jobtype/src/main/java/com/webank/wedatasphere/dss/plugins/azkaban/linkis/jobtype/conf/LinkisJobTypeConf.java +++ b/plugins/azkaban/linkis-jobtype/src/main/java/com/webank/wedatasphere/dss/plugins/azkaban/linkis/jobtype/conf/LinkisJobTypeConf.java @@ -43,5 +43,8 @@ public class LinkisJobTypeConf { public static final String MSG_SAVE_KEY = "msg.savekey"; public final static CommonVars SIGNAL_NODES = CommonVars.apply("wds.dss.flow.signal.nodes","linkis.appconn.eventchecker.eventreceiver"); + public final static CommonVars LOG_MAX_RESULTSIZE = CommonVars.apply("wds.dss.log.max.resultsize",1024); + public final static CommonVars CONTEXT_ENV_DEV = CommonVars.apply("wds.dss.flow.env.dev","BDAP_DEV"); + public final static CommonVars CONTEXT_ENV_PROD = CommonVars.apply("wds.dss.flow.env.prod","BDAP_PROD"); } diff --git a/plugins/azkaban/linkis-jobtype/src/main/java/com/webank/wedatasphere/dss/plugins/azkaban/linkis/jobtype/job/AzkanbanBuilder.java b/plugins/azkaban/linkis-jobtype/src/main/java/com/webank/wedatasphere/dss/plugins/azkaban/linkis/jobtype/job/AzkanbanBuilder.java index b6af36d63e..3bc7fe1c26 100644 --- a/plugins/azkaban/linkis-jobtype/src/main/java/com/webank/wedatasphere/dss/plugins/azkaban/linkis/jobtype/job/AzkanbanBuilder.java +++ b/plugins/azkaban/linkis-jobtype/src/main/java/com/webank/wedatasphere/dss/plugins/azkaban/linkis/jobtype/job/AzkanbanBuilder.java @@ -38,7 +38,7 @@ public class AzkanbanBuilder extends Builder { private static final Logger LOGGER = LoggerFactory.getLogger(AzkanbanBuilder.class); private static final String RUN_DATE_KEY = "run_date"; - private static final String RUN_DATE_HOUR_KEY = "run_date_h"; + private static final String RUN_DATE_HOUR_KEY = "run_today_h"; private Map jobProps; public AzkanbanBuilder setJobProps(Map jobProps) { @@ -88,7 +88,12 @@ protected void fillJobInfo(Job job) { @Override protected String getContextID(Job job) { - return jobProps.get(LinkisJobExecutionConfiguration.FLOW_CONTEXTID); + String contextID = jobProps.get(LinkisJobExecutionConfiguration.FLOW_CONTEXTID); + //将部分老的工作流的BDAP_DEV标签替换为BDAP_PROD + if (null != contextID) { + contextID = contextID.replace(LinkisJobTypeConf.CONTEXT_ENV_DEV.getValue(), LinkisJobTypeConf.CONTEXT_ENV_PROD.getValue()); + } + return contextID; } @Override diff --git a/plugins/azkaban/linkis-jobtype/src/main/java/com/webank/wedatasphere/dss/plugins/azkaban/linkis/jobtype/log/AzkabanJobLog.java b/plugins/azkaban/linkis-jobtype/src/main/java/com/webank/wedatasphere/dss/plugins/azkaban/linkis/jobtype/log/AzkabanJobLog.java new file mode 100644 index 0000000000..7d2db3a034 --- /dev/null +++ b/plugins/azkaban/linkis-jobtype/src/main/java/com/webank/wedatasphere/dss/plugins/azkaban/linkis/jobtype/log/AzkabanJobLog.java @@ -0,0 +1,57 @@ +/* + * Copyright 2019 WeBank + * 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. + * + */ + +package com.webank.wedatasphere.dss.plugins.azkaban.linkis.jobtype.log; + +import azkaban.jobExecutor.AbstractJob; +import com.webank.wedatasphere.dss.linkis.node.execution.log.LinkisJobExecutionLog; + + +public class AzkabanJobLog extends LinkisJobExecutionLog { + + private AbstractJob job; + + public AzkabanJobLog(AbstractJob job){ + this.job = job; + } + + public void info(Object message) { + job.info(message.toString()); + } + + public void warn(Object message) { + job.warn(message.toString()); + } + + public void error(Object message) { + job.error(message.toString()); + } + + @Override + public void info(Object message, Throwable t) { + job.info(message.toString(), t); + } + + @Override + public void warn(Object message, Throwable t) { + job.warn(message.toString(), t); + } + + @Override + public void error(Object message, Throwable t) { + job.error(message.toString(), t); + } +} diff --git a/plugins/dolphinscheduler/dolphinscheduler-prod-metrics/pom.xml b/plugins/dolphinscheduler/dolphinscheduler-prod-metrics/pom.xml index fdad1eadd9..0e77ed6220 100644 --- a/plugins/dolphinscheduler/dolphinscheduler-prod-metrics/pom.xml +++ b/plugins/dolphinscheduler/dolphinscheduler-prod-metrics/pom.xml @@ -22,7 +22,8 @@ dss-plugin-dolphinscheduler com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 + ../pom.xml 4.0.0 diff --git a/plugins/dolphinscheduler/dss-dolphinscheduler-client/pom.xml b/plugins/dolphinscheduler/dss-dolphinscheduler-client/pom.xml index 2f9180eee4..a2f5814bac 100644 --- a/plugins/dolphinscheduler/dss-dolphinscheduler-client/pom.xml +++ b/plugins/dolphinscheduler/dss-dolphinscheduler-client/pom.xml @@ -22,7 +22,8 @@ dss-plugin-dolphinscheduler com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 + ../pom.xml 4.0.0 diff --git a/plugins/dolphinscheduler/dss-dolphinscheduler-client/src/main/scala/com/webank/wedatasphere/dss/plugins/dolphinscheduler/linkis/client/DSSDolphinSchedulerClient.scala b/plugins/dolphinscheduler/dss-dolphinscheduler-client/src/main/scala/com/webank/wedatasphere/dss/plugins/dolphinscheduler/linkis/client/DSSDolphinSchedulerClient.scala index 9fdee6ad5b..99aea40bc7 100644 --- a/plugins/dolphinscheduler/dss-dolphinscheduler-client/src/main/scala/com/webank/wedatasphere/dss/plugins/dolphinscheduler/linkis/client/DSSDolphinSchedulerClient.scala +++ b/plugins/dolphinscheduler/dss-dolphinscheduler-client/src/main/scala/com/webank/wedatasphere/dss/plugins/dolphinscheduler/linkis/client/DSSDolphinSchedulerClient.scala @@ -9,7 +9,7 @@ import com.webank.wedatasphere.dss.linkis.node.execution.listener.LinkisExecutio import com.webank.wedatasphere.dss.linkis.node.execution.log.LinkisJobExecutionLog import com.webank.wedatasphere.dss.plugins.dolphinscheduler.linkis.client.conf.LinkisJobTypeConf import com.webank.wedatasphere.dss.plugins.dolphinscheduler.linkis.client.job.DolphinSchedulerJobBuilder -import org.apache.commons.lang.StringUtils +import org.apache.commons.lang3.StringUtils import org.apache.linkis.common.conf.CommonVars import org.apache.linkis.common.utils.{Logging, Utils} diff --git a/plugins/dolphinscheduler/dss-dolphinscheduler-client/src/main/scala/com/webank/wedatasphere/dss/plugins/dolphinscheduler/linkis/client/job/DolphinSchedulerJobBuilder.scala b/plugins/dolphinscheduler/dss-dolphinscheduler-client/src/main/scala/com/webank/wedatasphere/dss/plugins/dolphinscheduler/linkis/client/job/DolphinSchedulerJobBuilder.scala index f039985b7d..8660c1177b 100644 --- a/plugins/dolphinscheduler/dss-dolphinscheduler-client/src/main/scala/com/webank/wedatasphere/dss/plugins/dolphinscheduler/linkis/client/job/DolphinSchedulerJobBuilder.scala +++ b/plugins/dolphinscheduler/dss-dolphinscheduler-client/src/main/scala/com/webank/wedatasphere/dss/plugins/dolphinscheduler/linkis/client/job/DolphinSchedulerJobBuilder.scala @@ -8,7 +8,7 @@ import com.webank.wedatasphere.dss.linkis.node.execution.entity.BMLResource import com.webank.wedatasphere.dss.linkis.node.execution.job.{Builder, CommonLinkisJob, Job, LinkisJob} import com.webank.wedatasphere.dss.linkis.node.execution.utils.LinkisJobExecutionUtils import com.webank.wedatasphere.dss.plugins.dolphinscheduler.linkis.client.conf.LinkisJobTypeConf -import org.apache.commons.lang.StringUtils +import org.apache.commons.lang3.StringUtils import org.apache.linkis.common.utils.JsonUtils import scala.collection.JavaConverters._ diff --git a/plugins/dolphinscheduler/dss-dolphinscheduler-token/pom.xml b/plugins/dolphinscheduler/dss-dolphinscheduler-token/pom.xml index c88da378ff..6e8ec09975 100644 --- a/plugins/dolphinscheduler/dss-dolphinscheduler-token/pom.xml +++ b/plugins/dolphinscheduler/dss-dolphinscheduler-token/pom.xml @@ -22,7 +22,8 @@ dss-plugin-dolphinscheduler com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 + ../pom.xml 4.0.0 diff --git a/plugins/dolphinscheduler/pom.xml b/plugins/dolphinscheduler/pom.xml index 049de359c2..45d31c0b41 100644 --- a/plugins/dolphinscheduler/pom.xml +++ b/plugins/dolphinscheduler/pom.xml @@ -21,7 +21,7 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 ../../pom.xml 4.0.0 diff --git a/plugins/linkis/dss-gateway-support/pom.xml b/plugins/linkis/dss-gateway-support/pom.xml index 06616b65e1..0bbb17af62 100644 --- a/plugins/linkis/dss-gateway-support/pom.xml +++ b/plugins/linkis/dss-gateway-support/pom.xml @@ -21,7 +21,8 @@ dss com.webank.wedatasphere.dss - 1.1.1 + 1.1.2 + ../../../pom.xml 4.0.0 diff --git a/plugins/linkis/dss-gateway-support/src/main/scala/org/apache/linkis/gateway/parser/DSSRouteLabelParser.scala b/plugins/linkis/dss-gateway-support/src/main/scala/org/apache/linkis/gateway/parser/DSSRouteLabelParser.scala index a56dbe5944..40403a784b 100644 --- a/plugins/linkis/dss-gateway-support/src/main/scala/org/apache/linkis/gateway/parser/DSSRouteLabelParser.scala +++ b/plugins/linkis/dss-gateway-support/src/main/scala/org/apache/linkis/gateway/parser/DSSRouteLabelParser.scala @@ -17,7 +17,6 @@ package org.apache.linkis.gateway.parser import org.apache.linkis.gateway.http.GatewayContext -import org.apache.linkis.gateway.ujes.route.label.RouteLabelParser import org.apache.linkis.manager.label.entity.route.RouteLabel import org.springframework.stereotype.Component diff --git a/pom.xml b/pom.xml index 80ffb18d1d..75d47ca09d 100644 --- a/pom.xml +++ b/pom.xml @@ -23,7 +23,7 @@ pom com.webank.wedatasphere.dss dss - 1.1.1 + 1.1.2 dss-commons @@ -31,32 +31,42 @@ dss-orchestrator dss-appconn dss-framework - dss-apps/dss-apiservice-server - dss-apps/dss-scriptis-server - dss-apps/dss-user-guide + dss-apps + dss-server plugins/azkaban/linkis-jobtype plugins/linkis/dss-gateway-support assembly plugins/dolphinscheduler + + + apache.snapshots + Apache Snapshot Repository + https://repository.apache.org/content/repositories/snapshots/ + + true + + + + - 1.1.1 - 1.1.1 - 2.11.12 + 1.1.2 + 1.4.0-SNAPSHOT + + 2.12.17 1.8 3.3.3 2.8.5 - 2.11.3 - 1.9.13 + 2.14.1 3.1.1 4.5.4 4.5.4 1.9.4 UTF-8 - 5.2.22.RELEASE + 5.2.23.RELEASE 2.1.2 - 2.3.7.RELEASE + 2.3.12.RELEASE 2.2.6.RELEASE 3.1.1 3.8.1 @@ -107,11 +117,6 @@ jackson-databind ${fasterxml.jackson.version} - - org.codehaus.jackson - jackson-mapper-asl - ${org.codehaus.jackson.version} - org.apache.commons commons-math3 @@ -122,22 +127,44 @@ com.thoughtworks.xstream ${xstream.version} + + org.springframework + spring-framework-bom + ${spring-framework.version} + pom + import + + + org.springframework.boot + spring-boot-starter + ${spring.boot.version} + + + org.springframework.boot + spring-boot-starter-logging + + + + + org.springframework.boot + spring-boot-starter-web + ${spring.boot.version} + + + org.springframework.boot + spring-boot-starter-tomcat + + + + + org.springframework.boot + spring-boot-dependencies + ${spring.boot.version} + pom + import + - - - - releases - Releases - https://127.0.0.1/repository/maven-releases/ - - - snapshots - Snapshot - https://127.0.0.1/repository/maven-snapshots/ - - - @@ -226,6 +253,11 @@ ${scala.version} all + + -nobootcp + -target:jvm-${jdk.compile.version} + + diff --git a/sbin/common.sh b/sbin/common.sh index 1fde094935..e0b5f6bb6c 100644 --- a/sbin/common.sh +++ b/sbin/common.sh @@ -161,27 +161,13 @@ status() } function setServerName(){ - if [[ $PROJECT_NAME == *"project"* ]]; then - SERVER_NAME=dss-framework-project-server - elif [[ $PROJECT_NAME == *"orchestrator"* ]]; then - SERVER_NAME=dss-framework-orchestrator-server - elif [[ $PROJECT_NAME == *"apiservice"* ]]; then - SERVER_NAME=dss-apiservice-server - elif [[ $PROJECT_NAME == *"scriptis"* ]]; then - SERVER_NAME=dss-scriptis-server - elif [[ $PROJECT_NAME == *"workflow"* ]]; then - SERVER_NAME=dss-workflow-server - elif [[ $PROJECT_NAME == *"execution"* ]]; then - SERVER_NAME=dss-flow-execution-server - elif [[ $PROJECT_NAME == *"data-api"* ]]; then - SERVER_NAME=dss-data-api-server - elif [[ $PROJECT_NAME == *"governance"* ]]; then - SERVER_NAME=dss-data-governance-server - elif [[ $PROJECT_NAME == *"guide"* ]]; then - SERVER_NAME=dss-guide-server + if [[ $PROJECT_NAME == *"dss-server"* ]]; then + SERVER_NAME=dss-server + elif [[ $PROJECT_NAME == *"apps"* ]]; then + SERVER_NAME=dss-apps-server else - echo "please input: sh dss-daemon.sh [start,restart,stop] [server name]; for example : sh dss-daemon.sh restart project " - echo "server name : project、orchestrator、apiservice、scriptis、workflow、execution、data-api、governance、guide" + echo "please input: sh dss-daemon.sh [start,restart,stop] [server name]; for example : sh dss-daemon.sh restart dss-server " + echo "server name : dss-server、apps-server" exit 1 fi } diff --git a/sbin/dss-start-all.sh b/sbin/dss-start-all.sh index dea17d6322..309dae8cee 100644 --- a/sbin/dss-start-all.sh +++ b/sbin/dss-start-all.sh @@ -80,102 +80,28 @@ echo "<-------------------------------->" sleep 3 } - function startDssProject(){ - SERVER_NAME=dss-framework-project-server - SERVER_IP=$DSS_FRAMEWORK_PROJECT_SERVER_INSTALL_IP - startApp - sleep 5 - -# echo "------------------------Start to check whether the project service is registered to eureka successfully-----------------------------" -# #project服务启动并注册到eureka后再启动其他服务 -# i=1 -# while [[ -z $result ]] && [[ $i -le 24 ]] -# do -# sleep 5 -# if [ -z $EUREKA_USERNAME ] || [ -z $EUREKA_PASSWORD ];then -# response=`curl http://${EUREKA_INSTALL_IP}:${EUREKA_PORT}/eureka/apps/DSS-FRAMEWORK-PROJECT-SERVER` -# else -# response=`curl http://${EUREKA_USENAME}:${EUREKA_PASSWORD}@${EUREKA_INSTALL_IP}:${EUREKA_PORT}/eureka/apps/DSS-FRAMEWORK-PROJECT-SERVER` -# fi -# let i++ -# result=$(echo $response |grep 'DSS-FRAMEWORK-PROJECT-SERVER') -# done -# if [[ $i -eq 25 ]]; then -# echo "the project server start failed in two minutes,please check the log to find more error details." -# exit -# fi -# echo "------------------------the project service is registered to eureka successfully------------------------------------------------" - - SERVER_NAME=dss-apiservice-server - SERVER_IP=$DSS_APISERVICE_SERVER_INSTALL_IP - startApp - - SERVER_NAME=dss-scriptis-server - SERVER_IP=$DSS_SCRIPTIS_SERVER_INSTALL_IP - startApp - - SERVER_NAME=dss-flow-execution-server - SERVER_IP=$DSS_FLOW_EXECUTION_SERVER_INSTALL_IP - startApp - - SERVER_NAME=dss-data-api-server - SERVER_IP=$DSS_DATA_API_SERVER_INSTALL_IP - startApp - - SERVER_NAME=dss-data-governance-server - SERVER_IP=$DSS_DATA_GOVERNANCE_SERVER_INSTALL_IP - startApp - - SERVER_NAME=dss-guide-server - SERVER_IP=$DSS_GUIDE_SERVER_INSTALL_IP - startApp - - SERVER_NAME=dss-framework-orchestrator-server - SERVER_IP=$DSS_FRAMEWORK_ORCHESTRATOR_SERVER_INSTALL_IP + SERVER_NAME=dss-apps-server + SERVER_IP=$DSS_APPS_SERVER_INSTALL_IP startApp - SERVER_NAME=dss-workflow-server - SERVER_IP=$DSS_WORKFLOW_SERVER_INSTALL_IP + SERVER_NAME=dss-server + SERVER_IP=$DSS_SERVER_INSTALL_IP startApp } function checkDssService(){ - SERVER_NAME=dss-framework-project-server - SERVER_IP=$DSS_FRAMEWORK_PROJECT_SERVER_INSTALL_IP - checkServer - - SERVER_NAME=dss-framework-orchestrator-server - SERVER_IP=$DSS_FRAMEWORK_ORCHESTRATOR_SERVER_INSTALL_IP - checkServer - - SERVER_NAME=dss-apiservice-server - SERVER_IP=$DSS_APISERVICE_SERVER_INSTALL_IP + SERVER_NAME=dss-apps-server + SERVER_IP=$DSS_APPS_SERVER_INSTALL_IP checkServer - SERVER_NAME=dss-scriptis-server - SERVER_IP=$DSS_SCRIPTIS_SERVER_INSTALL_IP - checkServer - - SERVER_NAME=dss-workflow-server - SERVER_IP=$DSS_WORKFLOW_SERVER_INSTALL_IP - checkServer - - SERVER_NAME=dss-flow-execution-server - SERVER_IP=$DSS_FLOW_EXECUTION_SERVER_INSTALL_IP + SERVER_NAME=dss-server + SERVER_IP=$DSS_SERVER_INSTALL_IP checkServer - SERVER_NAME=dss-data-api-server - SERVER_IP=$DSS_DATA_API_SERVER_INSTALL_IP - checkServer - - SERVER_NAME=dss-guide-server - SERVER_IP=$DSS_GUIDE_SERVER_INSTALL_IP - checkServer } - startDssProject checkDssService \ No newline at end of file diff --git a/sbin/dss-stop-all.sh b/sbin/dss-stop-all.sh index 55c5c806bd..b6f0b36d92 100644 --- a/sbin/dss-stop-all.sh +++ b/sbin/dss-stop-all.sh @@ -51,40 +51,12 @@ echo "<-------------------------------->" } function stopDssProject(){ - SERVER_NAME=dss-framework-project-server - SERVER_IP=$DSS_FRAMEWORK_PROJECT_SERVER_INSTALL_IP + SERVER_NAME=dss-server + SERVER_IP=$DSS_SERVER_INSTALL_IP stopApp - SERVER_NAME=dss-framework-orchestrator-server - SERVER_IP=$DSS_FRAMEWORK_ORCHESTRATOR_SERVER_INSTALL_IP - stopApp - - SERVER_NAME=dss-apiservice-server - SERVER_IP=$DSS_APISERVICE_SERVER_INSTALL_IP - stopApp - - SERVER_NAME=dss-scriptis-server - SERVER_IP=$DSS_DATAPIPE_SERVER_INSTALL_IP - stopApp - - SERVER_NAME=dss-workflow-server - SERVER_IP=$DSS_WORKFLOW_SERVER_INSTALL_IP - stopApp - - SERVER_NAME=dss-flow-execution-server - SERVER_IP=$DSS_FLOW_EXECUTION_SERVER_INSTALL_IP - stopApp - - SERVER_NAME=dss-data-api-server - SERVER_IP=$DSS_DATA_API_SERVER_INSTALL_IP - stopApp - - SERVER_NAME=dss-data-governance-server - SERVER_IP=$DSS_DATA_GOVERNANCE_SERVER_INSTALL_IP - stopApp - - SERVER_NAME=dss-guide-server - SERVER_IP=$DSS_GUIDE_SERVER_INSTALL_IP + SERVER_NAME=dss-apps-server + SERVER_IP=$DSS_APPS_SERVER_INSTALL_IP stopApp } diff --git a/sbin/ext/dss-apiservice-server b/sbin/ext/dss-apiservice-server deleted file mode 100644 index 58ad10ff0c..0000000000 --- a/sbin/ext/dss-apiservice-server +++ /dev/null @@ -1,86 +0,0 @@ -#!/bin/bash -# -# description: ecm start cmd -# - -# get log directory -cd `dirname $0` -cd .. -INSTALL_HOME=`pwd` - -# set DSS_HOME -if [ "$DSS_HOME" = "" ]; then - export DSS_HOME=$INSTALL_HOME -fi - -# set DSS_CONF_DIR -if [ "$DSS_CONF_DIR" = "" ]; then - export DSS_CONF_DIR=$DSS_HOME/conf -fi - -SERVER_SUFFIX="dss-apiservice-server" -## set log -if [ "$DSS_LOG_DIR" = "" ]; then - export DSS_LOG_DIR="$DSS_HOME/logs" -fi -export SERVER_LOG_PATH=$DSS_LOG_DIR -if [ ! -w "$SERVER_LOG_PATH" ] ; then - mkdir -p "$SERVER_LOG_PATH" -fi - -if test -z "$SERVER_HEAP_SIZE" -then - export SERVER_HEAP_SIZE="512M" -fi - -DEBUG_PORT= -if [ "$DEBUG_PORT" ]; -then - export DEBUG_CMD="-agentlib:jdwp=transport=dt_socket,server=y,suspend=n,address=$DEBUG_PORT" -fi - -if test -z "$SERVER_JAVA_OPTS" -then - export SERVER_JAVA_OPTS="-DserviceName=$SERVER_SUFFIX -Xmx$SERVER_HEAP_SIZE -XX:+UseG1GC -Xloggc:$SERVER_LOG_PATH/$SERVER_SUFFIX.gc $DEBUG_CMD" -fi - -export SERVER_CLASS=com.webank.wedatasphere.dss.apiservice.DSSApiServiceServerApplication - -## conf dir -export SERVER_CONF_PATH=$DSS_CONF_DIR:$DSS_CONF_DIR/$SERVER_SUFFIX - -## commons lib -export DSS_COMMONS_LIB="$DSS_HOME/lib/dss-commons" - - -if [ ! -r "$DSS_COMMONS_LIB" ] ; then - echo "dss commons lib not exists $DSS_COMMONS_LIB" - exit 1 -fi - -## server lib -export SERVER_LIB=$DSS_HOME/lib/dss-apps/$SERVER_SUFFIX - - -if [ ! -r "$SERVER_LIB" ] ; then - echo "server lib not exists $SERVER_LIB" - exit 1 -fi - -export SERVER_PID=$DSS_HOME/pid/${SERVER_SUFFIX}.pid -## set class path -export SERVER_CLASS_PATH=$SERVER_CONF_PATH:$DSS_COMMONS_LIB/*:$SERVER_LIB/* - -nohup java $SERVER_JAVA_OPTS -cp $SERVER_CLASS_PATH $SERVER_CLASS 2>&1 > $SERVER_LOG_PATH/$SERVER_SUFFIX.out & -pid=$! - -sleep 2 -if [[ -z "${pid}" ]]; then - echo "server $SERVER_SUFFIX start failed!" - exit 1 -else - echo "server $SERVER_SUFFIX start succeeded!" - echo $pid > $SERVER_PID - sleep 1 -fi -exit 1 diff --git a/sbin/ext/dss-data-api-server b/sbin/ext/dss-apps-server similarity index 87% rename from sbin/ext/dss-data-api-server rename to sbin/ext/dss-apps-server index 4f9010b9bf..cae38643b4 100644 --- a/sbin/ext/dss-data-api-server +++ b/sbin/ext/dss-apps-server @@ -1,4 +1,7 @@ #!/bin/bash +# +# description: dss-apps-server start cmd +# # get log directory cd `dirname $0` @@ -15,7 +18,7 @@ if [ "$DSS_CONF_DIR" = "" ]; then export DSS_CONF_DIR=$DSS_HOME/conf fi -SERVER_SUFFIX="dss-data-api-server" +SERVER_SUFFIX="dss-apps-server" ## set log if [ "$DSS_LOG_DIR" = "" ]; then export DSS_LOG_DIR="$DSS_HOME/logs" @@ -27,7 +30,7 @@ fi if test -z "$SERVER_HEAP_SIZE" then - export SERVER_HEAP_SIZE="512M" + export SERVER_HEAP_SIZE="1G" fi DEBUG_PORT= @@ -41,7 +44,7 @@ then export SERVER_JAVA_OPTS="-DserviceName=$SERVER_SUFFIX -Xmx$SERVER_HEAP_SIZE -XX:+UseG1GC -Xloggc:$SERVER_LOG_PATH/$SERVER_SUFFIX.gc $DEBUG_CMD" fi -export SERVER_CLASS=com.webank.wedatasphere.dss.data.api.server.DSSDataApiApplication +export SERVER_CLASS=com.webank.wedatasphere.dss.apps.DSSAppsServerApplication ## conf dir export SERVER_CONF_PATH=$DSS_CONF_DIR:$DSS_CONF_DIR/$SERVER_SUFFIX @@ -56,7 +59,7 @@ if [ ! -r "$DSS_COMMONS_LIB" ] ; then fi ## server lib -export SERVER_LIB=$DSS_HOME/lib/dss-data-api/$SERVER_SUFFIX +export SERVER_LIB=$DSS_HOME/lib/$SERVER_SUFFIX if [ ! -r "$SERVER_LIB" ] ; then diff --git a/sbin/ext/dss-data-governance-server b/sbin/ext/dss-data-governance-server deleted file mode 100644 index 8b259fe749..0000000000 --- a/sbin/ext/dss-data-governance-server +++ /dev/null @@ -1,85 +0,0 @@ -#!/bin/bash -# -# description: ecm start cmd -# -# get log directory -cd `dirname $0` -cd .. -INSTALL_HOME=`pwd` - -# set DSS_HOME -if [ "$DSS_HOME" = "" ]; then - export DSS_HOME=$INSTALL_HOME -fi - -# set DSS_CONF_DIR -if [ "$DSS_CONF_DIR" = "" ]; then - export DSS_CONF_DIR=$DSS_HOME/conf -fi - -SERVER_SUFFIX="dss-data-governance-server" -## set log -if [ "$DSS_LOG_DIR" = "" ]; then - export DSS_LOG_DIR="$DSS_HOME/logs" -fi -export SERVER_LOG_PATH=$DSS_LOG_DIR -if [ ! -w "$SERVER_LOG_PATH" ] ; then - mkdir -p "$SERVER_LOG_PATH" -fi - -if test -z "$SERVER_HEAP_SIZE" -then - export SERVER_HEAP_SIZE="512M" -fi - -DEBUG_PORT= -if [ "$DEBUG_PORT" ]; -then - export DEBUG_CMD="-agentlib:jdwp=transport=dt_socket,server=y,suspend=n,address=$DEBUG_PORT" -fi - -if test -z "$SERVER_JAVA_OPTS" -then - export SERVER_JAVA_OPTS="-DserviceName=$SERVER_SUFFIX -Xmx$SERVER_HEAP_SIZE -XX:+UseG1GC -Xloggc:$SERVER_LOG_PATH/$SERVER_SUFFIX.gc $DEBUG_CMD" -fi - -export SERVER_CLASS=com.webank.wedatasphere.dss.data.governance.DSSDataGovernanceApplication - -## conf dir -export SERVER_CONF_PATH=$DSS_CONF_DIR:$DSS_CONF_DIR/$SERVER_SUFFIX - -## commons lib -export DSS_COMMONS_LIB="$DSS_HOME/lib/dss-commons" - - -if [ ! -r "$DSS_COMMONS_LIB" ] ; then - echo "dss commons lib not exists $DSS_COMMONS_LIB" - exit 1 -fi - -## server lib -export SERVER_LIB=$DSS_HOME/lib/dss-data-governance/$SERVER_SUFFIX - - -if [ ! -r "$SERVER_LIB" ] ; then - echo "server lib not exists $SERVER_LIB" - exit 1 -fi - -export SERVER_PID=$DSS_HOME/pid/${SERVER_SUFFIX}.pid -## set class path -export SERVER_CLASS_PATH=$SERVER_CONF_PATH:$DSS_COMMONS_LIB/*:$SERVER_LIB/* - -nohup java $SERVER_JAVA_OPTS -cp $SERVER_CLASS_PATH $SERVER_CLASS 2>&1 > $SERVER_LOG_PATH/$SERVER_SUFFIX.out & -pid=$! - -sleep 2 -if [[ -z "${pid}" ]]; then - echo "server $SERVER_SUFFIX start failed!" - exit 1 -else - echo "server $SERVER_SUFFIX start succeeded!" - echo $pid > $SERVER_PID - sleep 1 -fi -exit 1 diff --git a/sbin/ext/dss-flow-execution-server b/sbin/ext/dss-flow-execution-server deleted file mode 100644 index 4b652c945d..0000000000 --- a/sbin/ext/dss-flow-execution-server +++ /dev/null @@ -1,83 +0,0 @@ -#!/bin/bash - -# get log directory -cd `dirname $0` -cd .. -INSTALL_HOME=`pwd` - -# set DSS_HOME -if [ "$DSS_HOME" = "" ]; then - export DSS_HOME=$INSTALL_HOME -fi - -# set DSS_CONF_DIR -if [ "$DSS_CONF_DIR" = "" ]; then - export DSS_CONF_DIR=$DSS_HOME/conf -fi - -SERVER_SUFFIX="dss-flow-execution-server" -## set log -if [ "$DSS_LOG_DIR" = "" ]; then - export DSS_LOG_DIR="$DSS_HOME/logs" -fi -export SERVER_LOG_PATH=$DSS_LOG_DIR -if [ ! -w "$SERVER_LOG_PATH" ] ; then - mkdir -p "$SERVER_LOG_PATH" -fi - -if test -z "$SERVER_HEAP_SIZE" -then - export SERVER_HEAP_SIZE="512M" -fi - -DEBUG_PORT= -if [ "$DEBUG_PORT" ]; -then - export DEBUG_CMD="-agentlib:jdwp=transport=dt_socket,server=y,suspend=n,address=$DEBUG_PORT" -fi - -if test -z "$SERVER_JAVA_OPTS" -then - export SERVER_JAVA_OPTS="-DserviceName=$SERVER_SUFFIX -Xmx$SERVER_HEAP_SIZE -XX:+UseG1GC -Xloggc:$SERVER_LOG_PATH/$SERVER_SUFFIX.gc $DEBUG_CMD" -fi - -export SERVER_CLASS=com.webank.wedatasphere.dss.flow.execution.entrance.DSSFowExecutionServerApplication - -## conf dir -export SERVER_CONF_PATH=$DSS_CONF_DIR:$DSS_CONF_DIR/$SERVER_SUFFIX - -## commons lib -export DSS_COMMONS_LIB="$DSS_HOME/lib/dss-commons" - - -if [ ! -r "$DSS_COMMONS_LIB" ] ; then - echo "dss commons lib not exists $DSS_COMMONS_LIB" - exit 1 -fi - -## server lib -export SERVER_LIB=$DSS_HOME/lib/dss-orchestrator/$SERVER_SUFFIX - - -if [ ! -r "$SERVER_LIB" ] ; then - echo "server lib not exists $SERVER_LIB" - exit 1 -fi - -export SERVER_PID=$DSS_HOME/pid/${SERVER_SUFFIX}.pid -## set class path -export SERVER_CLASS_PATH=$SERVER_CONF_PATH:$DSS_COMMONS_LIB/*:$SERVER_LIB/* - -nohup java $SERVER_JAVA_OPTS -cp $SERVER_CLASS_PATH $SERVER_CLASS 2>&1 > $SERVER_LOG_PATH/$SERVER_SUFFIX.out & -pid=$! - -sleep 2 -if [[ -z "${pid}" ]]; then - echo "server $SERVER_SUFFIX start failed!" - exit 1 -else - echo "server $SERVER_SUFFIX start succeeded!" - echo $pid > $SERVER_PID - sleep 1 -fi -exit 1 diff --git a/sbin/ext/dss-guide-server b/sbin/ext/dss-guide-server deleted file mode 100644 index a6eb968088..0000000000 --- a/sbin/ext/dss-guide-server +++ /dev/null @@ -1,85 +0,0 @@ -#!/bin/bash -# -# description: ecm start cmd -# -# get log directory -cd `dirname $0` -cd .. -INSTALL_HOME=`pwd` - -# set DSS_HOME -if [ "$DSS_HOME" = "" ]; then - export DSS_HOME=$INSTALL_HOME -fi - -# set DSS_CONF_DIR -if [ "$DSS_CONF_DIR" = "" ]; then - export DSS_CONF_DIR=$DSS_HOME/conf -fi - -SERVER_SUFFIX="dss-guide-server" -## set log -if [ "$DSS_LOG_DIR" = "" ]; then - export DSS_LOG_DIR="$DSS_HOME/logs" -fi -export SERVER_LOG_PATH=$DSS_LOG_DIR -if [ ! -w "$SERVER_LOG_PATH" ] ; then - mkdir -p "$SERVER_LOG_PATH" -fi - -if test -z "$SERVER_HEAP_SIZE" -then - export SERVER_HEAP_SIZE="512M" -fi - -DEBUG_PORT= -if [ "$DEBUG_PORT" ]; -then - export DEBUG_CMD="-agentlib:jdwp=transport=dt_socket,server=y,suspend=n,address=$DEBUG_PORT" -fi - -if test -z "$SERVER_JAVA_OPTS" -then - export SERVER_JAVA_OPTS="-DserviceName=$SERVER_SUFFIX -Xmx$SERVER_HEAP_SIZE -XX:+UseG1GC -Xloggc:$SERVER_LOG_PATH/$SERVER_SUFFIX.gc $DEBUG_CMD" -fi - -export SERVER_CLASS=com.webank.wedatasphere.dss.guide.server.DSSGuideApplication - -## conf dir -export SERVER_CONF_PATH=$DSS_CONF_DIR:$DSS_CONF_DIR/$SERVER_SUFFIX - -## commons lib -export DSS_COMMONS_LIB="$DSS_HOME/lib/dss-commons" - - -if [ ! -r "$DSS_COMMONS_LIB" ] ; then - echo "dss commons lib not exists $DSS_COMMONS_LIB" - exit 1 -fi - -## server lib -export SERVER_LIB=$DSS_HOME/lib/dss-guide/$SERVER_SUFFIX - - -if [ ! -r "$SERVER_LIB" ] ; then - echo "server lib not exists $SERVER_LIB" - exit 1 -fi - -export SERVER_PID=$DSS_HOME/pid/${SERVER_SUFFIX}.pid -## set class path -export SERVER_CLASS_PATH=$SERVER_CONF_PATH:$DSS_COMMONS_LIB/*:$SERVER_LIB/* - -nohup java $SERVER_JAVA_OPTS -cp $SERVER_CLASS_PATH $SERVER_CLASS 2>&1 > $SERVER_LOG_PATH/$SERVER_SUFFIX.out & -pid=$! - -sleep 2 -if [[ -z "${pid}" ]]; then - echo "server $SERVER_SUFFIX start failed!" - exit 1 -else - echo "server $SERVER_SUFFIX start succeeded!" - echo $pid > $SERVER_PID - sleep 1 -fi -exit 1 diff --git a/sbin/ext/dss-scriptis-server b/sbin/ext/dss-server similarity index 88% rename from sbin/ext/dss-scriptis-server rename to sbin/ext/dss-server index ac4e479247..fc33f2bb53 100644 --- a/sbin/ext/dss-scriptis-server +++ b/sbin/ext/dss-server @@ -1,4 +1,7 @@ #!/bin/bash +# +# description: dss-server start cmd +# # get log directory cd `dirname $0` @@ -15,7 +18,7 @@ if [ "$DSS_CONF_DIR" = "" ]; then export DSS_CONF_DIR=$DSS_HOME/conf fi -SERVER_SUFFIX="dss-scriptis-server" +SERVER_SUFFIX="dss-server" ## set log if [ "$DSS_LOG_DIR" = "" ]; then export DSS_LOG_DIR="$DSS_HOME/logs" @@ -27,7 +30,7 @@ fi if test -z "$SERVER_HEAP_SIZE" then - export SERVER_HEAP_SIZE="512M" + export SERVER_HEAP_SIZE="6G" fi DEBUG_PORT= @@ -41,7 +44,7 @@ then export SERVER_JAVA_OPTS="-DserviceName=$SERVER_SUFFIX -Xmx$SERVER_HEAP_SIZE -XX:+UseG1GC -Xloggc:$SERVER_LOG_PATH/$SERVER_SUFFIX.gc $DEBUG_CMD" fi -export SERVER_CLASS=com.webank.wedatasphere.dss.scriptis.DSSScriptisServerApplication +export SERVER_CLASS=com.webank.wedatasphere.dss.DSSServerApplication ## conf dir export SERVER_CONF_PATH=$DSS_CONF_DIR:$DSS_CONF_DIR/$SERVER_SUFFIX @@ -56,7 +59,7 @@ if [ ! -r "$DSS_COMMONS_LIB" ] ; then fi ## server lib -export SERVER_LIB=$DSS_HOME/lib/dss-apps/$SERVER_SUFFIX +export SERVER_LIB=$DSS_HOME/lib/$SERVER_SUFFIX if [ ! -r "$SERVER_LIB" ] ; then diff --git a/sbin/ext/dss-workflow-server b/sbin/ext/dss-workflow-server deleted file mode 100644 index 12b3c0be96..0000000000 --- a/sbin/ext/dss-workflow-server +++ /dev/null @@ -1,86 +0,0 @@ -#!/bin/bash -# -# description: ecm start cmd -# - -# get log directory -cd `dirname $0` -cd .. -INSTALL_HOME=`pwd` - -# set DSS_HOME -if [ "$DSS_HOME" = "" ]; then - export DSS_HOME=$INSTALL_HOME -fi - -# set DSS_CONF_DIR -if [ "$DSS_CONF_DIR" = "" ]; then - export DSS_CONF_DIR=$DSS_HOME/conf -fi - -SERVER_SUFFIX="dss-workflow-server" -## set log -if [ "$DSS_LOG_DIR" = "" ]; then - export DSS_LOG_DIR="$DSS_HOME/logs" -fi -export SERVER_LOG_PATH=$DSS_LOG_DIR -if [ ! -w "$SERVER_LOG_PATH" ] ; then - mkdir -p "$SERVER_LOG_PATH" -fi - -if test -z "$SERVER_HEAP_SIZE" -then - export SERVER_HEAP_SIZE="512M" -fi - -DEBUG_PORT= -if [ "$DEBUG_PORT" ]; -then - export DEBUG_CMD="-agentlib:jdwp=transport=dt_socket,server=y,suspend=n,address=$DEBUG_PORT" -fi - -if test -z "$SERVER_JAVA_OPTS" -then - export SERVER_JAVA_OPTS="-DserviceName=$SERVER_SUFFIX -Xmx$SERVER_HEAP_SIZE -XX:+UseG1GC -Xloggc:$SERVER_LOG_PATH/$SERVER_SUFFIX.gc $DEBUG_CMD" -fi - -export SERVER_CLASS=com.webank.wedatasphere.dss.workflow.DSSWorkflowServerApplication - -## conf dir -export SERVER_CONF_PATH=$DSS_CONF_DIR:$DSS_CONF_DIR/$SERVER_SUFFIX - -## commons lib -export DSS_COMMONS_LIB="$DSS_HOME/lib/dss-commons" - - -if [ ! -r "$DSS_COMMONS_LIB" ] ; then - echo "dss commons lib not exists $DSS_COMMONS_LIB" - exit 1 -fi - -## server lib -export SERVER_LIB=$DSS_HOME/lib/dss-orchestrator/$SERVER_SUFFIX - - -if [ ! -r "$SERVER_LIB" ] ; then - echo "server lib not exists $SERVER_LIB" - exit 1 -fi - -export SERVER_PID=$DSS_HOME/pid/${SERVER_SUFFIX}.pid -## set class path -export SERVER_CLASS_PATH=$SERVER_CONF_PATH:$DSS_COMMONS_LIB/*:$SERVER_LIB/* - -nohup java $SERVER_JAVA_OPTS -cp $SERVER_CLASS_PATH $SERVER_CLASS 2>&1 > $SERVER_LOG_PATH/$SERVER_SUFFIX.out & -pid=$! - -sleep 2 -if [[ -z "${pid}" ]]; then - echo "server $SERVER_SUFFIX start failed!" - exit 1 -else - echo "server $SERVER_SUFFIX start succeeded!" - echo $pid > $SERVER_PID - sleep 1 -fi -exit 1 diff --git a/sbin/k8s/dss-apiservice-server.sh b/sbin/k8s/dss-apiservice-server.sh deleted file mode 100644 index 9629b24013..0000000000 --- a/sbin/k8s/dss-apiservice-server.sh +++ /dev/null @@ -1,25 +0,0 @@ -#!/bin/bash - -SERVER_SUFFIX="dss-apiservice-server" - - -export SERVER_CONF_PATH=/opt/dss/$SERVER_SUFFIX/conf -export SERVER_LOG_PATH=/opt/dss/$SERVER_SUFFIX/logs - -if [ ! -w "$SERVER_LOG_PATH" ] ; then - mkdir -p "$SERVER_LOG_PATH" -fi - -export SERVER_JAVA_OPTS="-DserviceName=$SERVER_SUFFIX -Xmx512M -XX:+UseG1GC -Xloggc:$SERVER_LOG_PATH/linkis.log" - -export SERVER_CLASS=com.webank.wedatasphere.dss.apiservice.DSSApiServiceServerApplication - - -export DSS_COMMONS_LIB=/opt/dss/dss-commons - -export SERVER_LIB=/opt/dss/$SERVER_SUFFIX/lib - - -export SERVER_CLASS_PATH=$SERVER_CONF_PATH:$DSS_COMMONS_LIB/*:$SERVER_LIB/* - -java $SERVER_JAVA_OPTS -cp $SERVER_CLASS_PATH $SERVER_CLASS \ No newline at end of file diff --git a/sbin/k8s/dss-data-api-server.sh b/sbin/k8s/dss-data-api-server.sh deleted file mode 100644 index ef6e440839..0000000000 --- a/sbin/k8s/dss-data-api-server.sh +++ /dev/null @@ -1,25 +0,0 @@ -#!/bin/bash - -SERVER_SUFFIX="dss-data-api-server" - - -export SERVER_CONF_PATH=/opt/dss/$SERVER_SUFFIX/conf -export SERVER_LOG_PATH=/opt/dss/$SERVER_SUFFIX/logs - -if [ ! -w "$SERVER_LOG_PATH" ] ; then - mkdir -p "$SERVER_LOG_PATH" -fi - -export SERVER_JAVA_OPTS="-DserviceName=$SERVER_SUFFIX -Xmx2048M -XX:+UseG1GC -Xloggc:$SERVER_LOG_PATH/linkis.log" - -export SERVER_CLASS=com.webank.wedatasphere.dss.data.api.server.DSSDataApiApplication - - -export DSS_COMMONS_LIB=/opt/dss/dss-commons - -export SERVER_LIB=/opt/dss/$SERVER_SUFFIX/lib - - -export SERVER_CLASS_PATH=$SERVER_CONF_PATH:$DSS_COMMONS_LIB/*:$SERVER_LIB/* - -java $SERVER_JAVA_OPTS -cp $SERVER_CLASS_PATH $SERVER_CLASS diff --git a/sbin/k8s/dss-data-governance-server.sh b/sbin/k8s/dss-data-governance-server.sh deleted file mode 100644 index 76aca5370c..0000000000 --- a/sbin/k8s/dss-data-governance-server.sh +++ /dev/null @@ -1,25 +0,0 @@ -#!/bin/bash - -SERVER_SUFFIX="dss-data-governance-server" - - -export SERVER_CONF_PATH=/opt/dss/$SERVER_SUFFIX/conf -export SERVER_LOG_PATH=/opt/dss/$SERVER_SUFFIX/logs - -if [ ! -w "$SERVER_LOG_PATH" ] ; then - mkdir -p "$SERVER_LOG_PATH" -fi - -export SERVER_JAVA_OPTS="-DserviceName=$SERVER_SUFFIX -Xmx2048M -XX:+UseG1GC -Xloggc:$SERVER_LOG_PATH/linkis.log" - -export SERVER_CLASS=com.webank.wedatasphere.dss.data.governance.DSSDataGovernanceApplication - - -export DSS_COMMONS_LIB=/opt/dss/dss-commons - -export SERVER_LIB=/opt/dss/$SERVER_SUFFIX/lib - - -export SERVER_CLASS_PATH=$SERVER_CONF_PATH:$DSS_COMMONS_LIB/*:$SERVER_LIB/* - -java $SERVER_JAVA_OPTS -cp $SERVER_CLASS_PATH $SERVER_CLASS diff --git a/sbin/k8s/dss-flow-execution-server.sh b/sbin/k8s/dss-flow-execution-server.sh deleted file mode 100644 index f6d70127fa..0000000000 --- a/sbin/k8s/dss-flow-execution-server.sh +++ /dev/null @@ -1,25 +0,0 @@ -#!/bin/bash - -SERVER_SUFFIX="dss-flow-execution-server" - - -export SERVER_CONF_PATH=/opt/dss/$SERVER_SUFFIX/conf -export SERVER_LOG_PATH=/opt/dss/$SERVER_SUFFIX/logs - -if [ ! -w "$SERVER_LOG_PATH" ] ; then - mkdir -p "$SERVER_LOG_PATH" -fi - -export SERVER_JAVA_OPTS="-DserviceName=$SERVER_SUFFIX -Xmx512M -XX:+UseG1GC -Xloggc:$SERVER_LOG_PATH/linkis.log" - -export SERVER_CLASS=com.webank.wedatasphere.dss.flow.execution.entrance.DSSFowExecutionServerApplication - - -export DSS_COMMONS_LIB=/opt/dss/dss-commons - -export SERVER_LIB=/opt/dss/$SERVER_SUFFIX/lib - - -export SERVER_CLASS_PATH=$SERVER_CONF_PATH:$DSS_COMMONS_LIB/*:$SERVER_LIB/* - -java $SERVER_JAVA_OPTS -cp $SERVER_CLASS_PATH $SERVER_CLASS \ No newline at end of file diff --git a/sbin/k8s/dss-guide-server.sh b/sbin/k8s/dss-guide-server.sh deleted file mode 100644 index 3bbff2b77d..0000000000 --- a/sbin/k8s/dss-guide-server.sh +++ /dev/null @@ -1,25 +0,0 @@ -#!/bin/bash - -SERVER_SUFFIX="dss-guide-server" - - -export SERVER_CONF_PATH=/opt/dss/$SERVER_SUFFIX/conf -export SERVER_LOG_PATH=/opt/dss/$SERVER_SUFFIX/logs - -if [ ! -w "$SERVER_LOG_PATH" ] ; then - mkdir -p "$SERVER_LOG_PATH" -fi - -export SERVER_JAVA_OPTS="-DserviceName=$SERVER_SUFFIX -Xmx2048M -XX:+UseG1GC -Xloggc:$SERVER_LOG_PATH/linkis.log" - -export SERVER_CLASS=com.webank.wedatasphere.dss.guide.server.DSSGuideApplication - - -export DSS_COMMONS_LIB=/opt/dss/dss-commons - -export SERVER_LIB=/opt/dss/$SERVER_SUFFIX/lib - - -export SERVER_CLASS_PATH=$SERVER_CONF_PATH:$DSS_COMMONS_LIB/*:$SERVER_LIB/* - -java $SERVER_JAVA_OPTS -cp $SERVER_CLASS_PATH $SERVER_CLASS diff --git a/sbin/k8s/dss-workflow-server.sh b/sbin/k8s/dss-workflow-server.sh deleted file mode 100644 index ae9a627153..0000000000 --- a/sbin/k8s/dss-workflow-server.sh +++ /dev/null @@ -1,25 +0,0 @@ -#!/bin/bash - -SERVER_SUFFIX="dss-workflow-server" - - -export SERVER_CONF_PATH=/opt/dss/$SERVER_SUFFIX/conf -export SERVER_LOG_PATH=/opt/dss/$SERVER_SUFFIX/logs - -if [ ! -w "$SERVER_LOG_PATH" ] ; then - mkdir -p "$SERVER_LOG_PATH" -fi - -export SERVER_JAVA_OPTS="-DserviceName=$SERVER_SUFFIX -Xmx512M -XX:+UseG1GC -Xloggc:$SERVER_LOG_PATH/linkis.log" - -export SERVER_CLASS=com.webank.wedatasphere.dss.workflow.DSSWorkflowServerApplication - - -export DSS_COMMONS_LIB=/opt/dss/dss-commons - -export SERVER_LIB=/opt/dss/$SERVER_SUFFIX/lib - - -export SERVER_CLASS_PATH=$SERVER_CONF_PATH:$DSS_COMMONS_LIB/*:$SERVER_LIB/* - -java $SERVER_JAVA_OPTS -cp $SERVER_CLASS_PATH $SERVER_CLASS diff --git a/web/.eslintignore b/web/.eslintignore index d1a5012559..804272ee3b 100644 --- a/web/.eslintignore +++ b/web/.eslintignore @@ -1,4 +1,4 @@ iconfont.js packages/dss/server/* -packages/dss/dist/*/* +packages/*/node_modules/*/* "*.min.js" diff --git a/web/.eslintrc.js b/web/.eslintrc.js index fe93fbe50b..f499c36102 100644 --- a/web/.eslintrc.js +++ b/web/.eslintrc.js @@ -78,6 +78,7 @@ module.exports = { 'vue/multiline-html-element-content-newline': 0, 'vue/attributes-order': 0, 'vue/html-self-closing': 0, + "vue/valid-v-model": 0, 'no-useless-constructor': 0, 'no-mixed-operators': 0, 'no-new-func': 0, diff --git a/web/README-DEV.md b/web/README-DEV.md index 75d4c5b09e..c75b99752d 100644 --- a/web/README-DEV.md +++ b/web/README-DEV.md @@ -31,7 +31,6 @@ 新增功能模块先确定涉及应用,按照上面目录结构维护代码同时建议遵守以下约束: - 子应用可以配置自己的layout需要在应用router模块导出配置subAppRoutes -- 子应用支持使用自己的header,需要在config.json里配置模块路径 - 各应用需要使用iView作为UI库,并提供路由,国际化等配置写入config.json - 各应用间不要相互直接依赖,确有依赖通过lerna管理 - 可复用组件,资源需要合理放置,packages/shared 共享组件方法,修改需要注意影响 @@ -43,9 +42,12 @@ ### 如何新增一个子应用,如何扩展 -1. config.json 新增应用配置 +1. config.json apps里新增应用配置(apps里配置的字应用模块会一起合并打包) 2. packages 下新建应用目录或者插件目录进行应用开发 +参考packages/demo: +npm run serve --configfile=config.demo.json +http://localhost:8080/#/demoHome ### 前端开发、构建打包 @@ -65,4 +67,5 @@ npm run build # 打包子应用,支持通过module组合 npm run build --module=scriptis npm run build --module=apiServices,workspace --micro_module=apiServices +npm run build --module=scheduleCenter,workflows,workspace,scriptis --micro_module=scheduleCenter ``` diff --git a/web/config.demo.json b/web/config.demo.json new file mode 100644 index 0000000000..1cab33a2df --- /dev/null +++ b/web/config.demo.json @@ -0,0 +1,39 @@ +{ + "apps": { + "demo": { + "routes": "demo/router", + "module": "demo/module", + "i18n": { + "en": "demo/i18n/en.json", + "zh-CN": "demo/i18n/zh.json" + } + }, + "sparketl": { + "routes": "sparketl/router/index", + "i18n": { + "en": "sparketl/utils/i18n/zh.json", + "zh-CN": "sparketl/utils/i18n/zh.json" + } + } + }, + "exts": { + }, + "components": { + }, + "conf": { + "app_name": "DataSphere Studio", + "app_logo": "dss/assets/images/dssLogo.png", + "user_guide": "", + "faq_link": "/_book/", + "copy_project_enable": true, + "error_report": true, + "table_transfer": true, + "watermark": { + "show": true, + "template": "${username} ${time}", + "timeupdate": 60000 + }, + "update_chrome": "/_book/知识库/DSS常见问题/其他/DSS使用推荐的浏览器版本.html" + }, + "version": "1.1.2" +} diff --git a/web/config.json b/web/config.json index 050e03d02c..d0d2cc5e35 100644 --- a/web/config.json +++ b/web/config.json @@ -67,6 +67,9 @@ "options": null } }, + "components": { + "WeEditor": "editor/index.js" + }, "conf": { "app_name": "DataSphere Studio", "app_logo": "dss/assets/images/dssLogo.png", @@ -77,7 +80,8 @@ "show": false, "template": "${username} ${time}", "timeupdate": 60000 - } + }, + "env": "opensource" }, - "version": "1.1.1" + "version": "1.1.12" } diff --git a/web/config.sandbox..json b/web/config.sandbox..json new file mode 100644 index 0000000000..8e12d56961 --- /dev/null +++ b/web/config.sandbox..json @@ -0,0 +1,98 @@ +{ + "apps": { + "workspace": { + "routes": "workspace/router", + "module": "workspace/module", + "i18n": { + "en": "workspace/i18n/en.json", + "zh-CN": "workspace/i18n/zh.json" + } + }, + "scriptis": { + "routes": "scriptis/router", + "module": "scriptis/module", + "i18n": { + "en": "scriptis/i18n/common/en.json", + "zh-CN": "scriptis/i18n/common/zh.json" + } + }, + "workflows": { + "routes": "workflows/router", + "module": "workflows/module", + "i18n": { + "en": "workflows/i18n/common/en.json", + "zh-CN": "workflows/i18n/common/zh.json" + } + }, + "apiServices": { + "routes": "apiServices/router", + "module": "apiServices/module", + "i18n": { + "en": "apiServices/i18n/en.json", + "zh-CN": "apiServices/i18n/zh.json" + } + }, + "dataService": { + "routes": "dataService/router", + "module": "dataService/module", + "i18n": { + "en": "dataService/i18n/en.json", + "zh-CN": "dataService/i18n/zh.json" + } + }, + "scheduleCenter": { + "routes": "scheduleCenter/router", + "module": "scheduleCenter/module", + "i18n": { + "en": "scheduleCenter/i18n/en.json", + "zh-CN": "scheduleCenter/i18n/zh.json" + } + }, + "dolphinScheduler": { + "routes": "dolphinScheduler/router", + "module": "dolphinScheduler/module", + "i18n": { + "en": "dolphinScheduler/i18n/en.json", + "zh-CN": "dolphinScheduler/i18n/zh.json" + } + } + }, + "exts": { + "dss-plugin-open": { + "module": "exts/open-source/index.js", + "i18n": { + "en": "exts/open-source/i18n/en.json", + "zh-CN": "exts/open-source/i18n/zh.json" + }, + "options": null + }, + "dss-plugin-sandbox": { + "module": "exts/sandbox/index.js", + "i18n": { + "en": "exts/sandbox/i18n/en.json", + "zh-CN": "exts/sandbox/i18n/zh.json" + }, + "options": null + } + }, + "components": { + "WeEditorLsp": "editorLsp/index.js", + "WeEditor": "editor/index.js" + }, + "conf": { + "app_name": "DataSphere Studio", + "app_logo": "dss/assets/images/dssLogo.png", + "user_guide": "", + "isSandbox": true, + "faq_link": "/_book/", + "copy_project_enable": true, + "error_report": true, + "watermark": { + "show": true, + "template": "${username} ${time}", + "timeupdate": 60000 + }, + "update_chrome": "/_book/知识库/DSS常见问题/其他/DSS使用推荐的浏览器版本.html" + }, + "version": "1.1.7" +} diff --git a/web/config.scriptis.json b/web/config.scriptis.json new file mode 100644 index 0000000000..7fe0c865cb --- /dev/null +++ b/web/config.scriptis.json @@ -0,0 +1,33 @@ +{ + "apps": { + "scriptis": { + "routes": "scriptis/router", + "module": "scriptis/module", + "i18n": { + "en": "scriptis/i18n/common/en.json", + "zh-CN": "scriptis/i18n/common/zh.json" + } + } + }, + "exts": { + }, + "conf": { + "app_name": "DataSphere Studio", + "app_logo": "dss/assets/images/dssLogo.png", + "user_guide": "", + "faq_link": "/_book/", + "copy_project_enable": true, + "error_report": true, + "watermark": { + "show": true, + "template": "${username} ${time}", + "timeupdate": 60000 + }, + "update_chrome": "/_book/知识库/DSS常见问题/其他/DSS使用推荐的浏览器版本.html", + "lsp_service": { + "sql": "${protocol}//${host}/server", + "py": "${protocol}//${host}/python" + } + }, + "version": "1.1.12" +} diff --git a/web/lerna.json b/web/lerna.json index 62c26edb53..7e9ca0719f 100644 --- a/web/lerna.json +++ b/web/lerna.json @@ -8,5 +8,5 @@ } }, "npmClient": "npm", - "version": "1.1.4" + "version": "1.1.6" } diff --git a/web/package-lock.json b/web/package-lock.json index 1795531264..850e83a9ee 100644 --- a/web/package-lock.json +++ b/web/package-lock.json @@ -1,8 +1,27846 @@ { "name": "dataspherestudio", - "version": "1.1.4", - "lockfileVersion": 1, + "version": "1.1.6", + "lockfileVersion": 2, "requires": true, + "packages": { + "": { + "name": "dataspherestudio", + "version": "1.1.6", + "dependencies": { + "@codemirror/lang-json": "^6.0.1", + "@riophae/vue-treeselect": "0.4.0", + "axios": "0.21.1", + "babel-plugin-transform-remove-console": "6.9.4", + "babel-polyfill": "6.26.0", + "bootstrap": "^3.4.1", + "butterfly-dag": "4.1.23", + "clipboard": "2.0.8", + "codemirror": "^5.65.13", + "core-js": "2.6.11", + "d3": "3.5.17", + "dayjs": "1.10.7", + "dexie": "2.0.4", + "dt-sql-parser": "1.2.1", + "echarts": "^4.1.0", + "highlight.js": "9.18.3", + "iview": "3.5.4", + "jquery": "3.6.0", + "jsencrypt": "^3.2.1", + "lodash": "4.17.20", + "mavon-editor": "2.10.4", + "md5": "2.3.0", + "moment": "2.29.4", + "moment-timezone": "^0.5.34", + "monaco-editor": "0.19.3", + "monaco-languageclient": "0.13.0", + "qs": "6.9.4", + "reconnecting-websocket": "4.4.0", + "svgo": "1.3.0", + "vscode-ws-jsonrpc": "0.2.0", + "vue": "2.6.12", + "vue-codemirror": "^4.0.6", + "vue-i18n": "8.22.1", + "vue-router": "3.4.8", + "vuedraggable": "2.24.3", + "vuescroll": "4.16.1", + "webpack": "^4.46.0", + "worker-loader": "2.0.0" + }, + "devDependencies": { + "@kazupon/vue-i18n-loader": "0.4.1", + "@vue/cli-plugin-babel": "3.12.1", + "@vue/cli-plugin-eslint": "3.12.1", + "@vue/cli-service": "3.12.1", + "@vue/eslint-config-standard": "4.0.0", + "archiver": "3.1.1", + "babel-eslint": "10.1.0", + "copy-webpack-plugin": "4.6.0", + "csp-html-webpack-plugin": "4.0.0", + "eslint": "6.8.0", + "eslint-plugin-vue": "6.2.2", + "filemanager-webpack-plugin": "2.0.5", + "husky": "1.3.1", + "lerna": "^4.0.0", + "less-loader": "6.1.0", + "lint-staged": "8.2.1", + "monaco-editor-webpack-plugin": "1.8.2", + "node-sass": "6.0.1", + "patch-package": "6.2.2", + "sass-loader": "10.2.1", + "speed-measure-webpack-plugin": "1.5.0", + "svg-sprite-loader": "5.0.0", + "vue-cli-plugin-mockjs": "0.1.3", + "vue-template-compiler": "2.6.12", + "webpack-virtual-modules": "0.3.2" + } + }, + "node_modules/@ampproject/remapping": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/@ampproject/remapping/-/remapping-2.1.2.tgz", + "integrity": "sha512-hoyByceqwKirw7w3Z7gnIIZC3Wx3J484Y3L/cMpXFbr7d9ZQj2mODrirNzcJa+SM3UlpWXYvKV4RlRpFXlWgXg==", + "dev": true, + "dependencies": { + "@jridgewell/trace-mapping": "^0.3.0" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@antv/hierarchy": { + "version": "0.6.8", + "resolved": "https://registry.npmmirror.com/@antv/hierarchy/-/hierarchy-0.6.8.tgz", + "integrity": "sha512-wVzUl+pxny5gyGJ2mkWx8IiEypX6bnMHgr/NILgbxY6shoy0Vf4FhZpI3CY8Ez7bQT6js8fMkB2NymPW7d7i8A==", + "dependencies": { + "@antv/util": "^2.0.7" + } + }, + "node_modules/@antv/matrix-util": { + "version": "3.0.4", + "resolved": "https://registry.npmmirror.com/@antv/matrix-util/-/matrix-util-3.0.4.tgz", + "integrity": "sha512-BAPyu6dUliHcQ7fm9hZSGKqkwcjEDVLVAstlHULLvcMZvANHeLXgHEgV7JqcAV/GIhIz8aZChIlzM1ZboiXpYQ==", + "dependencies": { + "@antv/util": "^2.0.9", + "gl-matrix": "^3.3.0", + "tslib": "^2.0.3" + } + }, + "node_modules/@antv/util": { + "version": "2.0.17", + "resolved": "https://registry.npmmirror.com/@antv/util/-/util-2.0.17.tgz", + "integrity": "sha512-o6I9hi5CIUvLGDhth0RxNSFDRwXeywmt6ExR4+RmVAzIi48ps6HUy+svxOCayvrPBN37uE6TAc2KDofRo0nK9Q==", + "dependencies": { + "csstype": "^3.0.8", + "tslib": "^2.0.3" + } + }, + "node_modules/@babel/code-frame": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/code-frame/-/code-frame-7.16.7.tgz", + "integrity": "sha512-iAXqUn8IIeBTNd72xsFlgaXHkMBMt6y4HJp1tIaK465CWLT/fG1aqB7ykr95gHHmlBdGbFeWWfyB4NJJ0nmeIg==", + "dev": true, + "dependencies": { + "@babel/highlight": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/compat-data": { + "version": "7.17.7", + "resolved": "https://registry.npmmirror.com/@babel/compat-data/-/compat-data-7.17.7.tgz", + "integrity": "sha512-p8pdE6j0a29TNGebNm7NzYZWB3xVZJBZ7XGs42uAKzQo8VQ3F0By/cQCtUEABwIqw5zo6WA4NbmxsfzADzMKnQ==", + "dev": true, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/core": { + "version": "7.17.9", + "resolved": "https://registry.npmmirror.com/@babel/core/-/core-7.17.9.tgz", + "integrity": "sha512-5ug+SfZCpDAkVp9SFIZAzlW18rlzsOcJGaetCjkySnrXXDUw9AR8cDUm1iByTmdWM6yxX6/zycaV76w3YTF2gw==", + "dev": true, + "dependencies": { + "@ampproject/remapping": "^2.1.0", + "@babel/code-frame": "^7.16.7", + "@babel/generator": "^7.17.9", + "@babel/helper-compilation-targets": "^7.17.7", + "@babel/helper-module-transforms": "^7.17.7", + "@babel/helpers": "^7.17.9", + "@babel/parser": "^7.17.9", + "@babel/template": "^7.16.7", + "@babel/traverse": "^7.17.9", + "@babel/types": "^7.17.0", + "convert-source-map": "^1.7.0", + "debug": "^4.1.0", + "gensync": "^1.0.0-beta.2", + "json5": "^2.2.1", + "semver": "^6.3.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/core/node_modules/debug": { + "version": "4.3.4", + "resolved": "https://registry.npmmirror.com/debug/-/debug-4.3.4.tgz", + "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", + "dev": true, + "dependencies": { + "ms": "2.1.2" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/@babel/core/node_modules/json5": { + "version": "2.2.1", + "resolved": "https://registry.npmmirror.com/json5/-/json5-2.2.1.tgz", + "integrity": "sha512-1hqLFMSrGHRHxav9q9gNjJ5EXznIxGVO09xQRrwplcS8qs28pZ8s8hupZAmqDwZUmVZ2Qb2jnyPOWcDH8m8dlA==", + "dev": true, + "bin": { + "json5": "lib/cli.js" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/@babel/core/node_modules/ms": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", + "dev": true + }, + "node_modules/@babel/core/node_modules/semver": { + "version": "6.3.0", + "resolved": "https://registry.npmmirror.com/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/@babel/generator": { + "version": "7.17.9", + "resolved": "https://registry.npmmirror.com/@babel/generator/-/generator-7.17.9.tgz", + "integrity": "sha512-rAdDousTwxbIxbz5I7GEQ3lUip+xVCXooZNbsydCWs3xA7ZsYOv+CFRdzGxRX78BmQHu9B1Eso59AOZQOJDEdQ==", + "dev": true, + "dependencies": { + "@babel/types": "^7.17.0", + "jsesc": "^2.5.1", + "source-map": "^0.5.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-annotate-as-pure": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/helper-annotate-as-pure/-/helper-annotate-as-pure-7.16.7.tgz", + "integrity": "sha512-s6t2w/IPQVTAET1HitoowRGXooX8mCgtuP5195wD/QJPV6wYjpujCGF7JuMODVX2ZAJOf1GT6DT9MHEZvLOFSw==", + "dev": true, + "dependencies": { + "@babel/types": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-builder-binary-assignment-operator-visitor": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/helper-builder-binary-assignment-operator-visitor/-/helper-builder-binary-assignment-operator-visitor-7.16.7.tgz", + "integrity": "sha512-C6FdbRaxYjwVu/geKW4ZeQ0Q31AftgRcdSnZ5/jsH6BzCJbtvXvhpfkbkThYSuutZA7nCXpPR6AD9zd1dprMkA==", + "dev": true, + "dependencies": { + "@babel/helper-explode-assignable-expression": "^7.16.7", + "@babel/types": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-compilation-targets": { + "version": "7.17.7", + "resolved": "https://registry.npmmirror.com/@babel/helper-compilation-targets/-/helper-compilation-targets-7.17.7.tgz", + "integrity": "sha512-UFzlz2jjd8kroj0hmCFV5zr+tQPi1dpC2cRsDV/3IEW8bJfCPrPpmcSN6ZS8RqIq4LXcmpipCQFPddyFA5Yc7w==", + "dev": true, + "dependencies": { + "@babel/compat-data": "^7.17.7", + "@babel/helper-validator-option": "^7.16.7", + "browserslist": "^4.17.5", + "semver": "^6.3.0" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/helper-compilation-targets/node_modules/semver": { + "version": "6.3.0", + "resolved": "https://registry.npmmirror.com/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/@babel/helper-create-class-features-plugin": { + "version": "7.17.9", + "resolved": "https://registry.npmmirror.com/@babel/helper-create-class-features-plugin/-/helper-create-class-features-plugin-7.17.9.tgz", + "integrity": "sha512-kUjip3gruz6AJKOq5i3nC6CoCEEF/oHH3cp6tOZhB+IyyyPyW0g1Gfsxn3mkk6S08pIA2y8GQh609v9G/5sHVQ==", + "dev": true, + "dependencies": { + "@babel/helper-annotate-as-pure": "^7.16.7", + "@babel/helper-environment-visitor": "^7.16.7", + "@babel/helper-function-name": "^7.17.9", + "@babel/helper-member-expression-to-functions": "^7.17.7", + "@babel/helper-optimise-call-expression": "^7.16.7", + "@babel/helper-replace-supers": "^7.16.7", + "@babel/helper-split-export-declaration": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/helper-create-regexp-features-plugin": { + "version": "7.17.0", + "resolved": "https://registry.npmmirror.com/@babel/helper-create-regexp-features-plugin/-/helper-create-regexp-features-plugin-7.17.0.tgz", + "integrity": "sha512-awO2So99wG6KnlE+TPs6rn83gCz5WlEePJDTnLEqbchMVrBeAujURVphRdigsk094VhvZehFoNOihSlcBjwsXA==", + "dev": true, + "dependencies": { + "@babel/helper-annotate-as-pure": "^7.16.7", + "regexpu-core": "^5.0.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/helper-define-polyfill-provider": { + "version": "0.3.1", + "resolved": "https://registry.npmmirror.com/@babel/helper-define-polyfill-provider/-/helper-define-polyfill-provider-0.3.1.tgz", + "integrity": "sha512-J9hGMpJQmtWmj46B3kBHmL38UhJGhYX7eqkcq+2gsstyYt341HmPeWspihX43yVRA0mS+8GGk2Gckc7bY/HCmA==", + "dev": true, + "dependencies": { + "@babel/helper-compilation-targets": "^7.13.0", + "@babel/helper-module-imports": "^7.12.13", + "@babel/helper-plugin-utils": "^7.13.0", + "@babel/traverse": "^7.13.0", + "debug": "^4.1.1", + "lodash.debounce": "^4.0.8", + "resolve": "^1.14.2", + "semver": "^6.1.2" + }, + "peerDependencies": { + "@babel/core": "^7.4.0-0" + } + }, + "node_modules/@babel/helper-define-polyfill-provider/node_modules/debug": { + "version": "4.3.4", + "resolved": "https://registry.npmmirror.com/debug/-/debug-4.3.4.tgz", + "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", + "dev": true, + "dependencies": { + "ms": "2.1.2" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/@babel/helper-define-polyfill-provider/node_modules/ms": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", + "dev": true + }, + "node_modules/@babel/helper-define-polyfill-provider/node_modules/semver": { + "version": "6.3.0", + "resolved": "https://registry.npmmirror.com/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/@babel/helper-environment-visitor": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/helper-environment-visitor/-/helper-environment-visitor-7.16.7.tgz", + "integrity": "sha512-SLLb0AAn6PkUeAfKJCCOl9e1R53pQlGAfc4y4XuMRZfqeMYLE0dM1LMhqbGAlGQY0lfw5/ohoYWAe9V1yibRag==", + "dev": true, + "dependencies": { + "@babel/types": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-explode-assignable-expression": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/helper-explode-assignable-expression/-/helper-explode-assignable-expression-7.16.7.tgz", + "integrity": "sha512-KyUenhWMC8VrxzkGP0Jizjo4/Zx+1nNZhgocs+gLzyZyB8SHidhoq9KK/8Ato4anhwsivfkBLftky7gvzbZMtQ==", + "dev": true, + "dependencies": { + "@babel/types": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-function-name": { + "version": "7.17.9", + "resolved": "https://registry.npmmirror.com/@babel/helper-function-name/-/helper-function-name-7.17.9.tgz", + "integrity": "sha512-7cRisGlVtiVqZ0MW0/yFB4atgpGLWEHUVYnb448hZK4x+vih0YO5UoS11XIYtZYqHd0dIPMdUSv8q5K4LdMnIg==", + "dev": true, + "dependencies": { + "@babel/template": "^7.16.7", + "@babel/types": "^7.17.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-hoist-variables": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/helper-hoist-variables/-/helper-hoist-variables-7.16.7.tgz", + "integrity": "sha512-m04d/0Op34H5v7pbZw6pSKP7weA6lsMvfiIAMeIvkY/R4xQtBSMFEigu9QTZ2qB/9l22vsxtM8a+Q8CzD255fg==", + "dev": true, + "dependencies": { + "@babel/types": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-member-expression-to-functions": { + "version": "7.17.7", + "resolved": "https://registry.npmmirror.com/@babel/helper-member-expression-to-functions/-/helper-member-expression-to-functions-7.17.7.tgz", + "integrity": "sha512-thxXgnQ8qQ11W2wVUObIqDL4p148VMxkt5T/qpN5k2fboRyzFGFmKsTGViquyM5QHKUy48OZoca8kw4ajaDPyw==", + "dev": true, + "dependencies": { + "@babel/types": "^7.17.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-module-imports": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/helper-module-imports/-/helper-module-imports-7.16.7.tgz", + "integrity": "sha512-LVtS6TqjJHFc+nYeITRo6VLXve70xmq7wPhWTqDJusJEgGmkAACWwMiTNrvfoQo6hEhFwAIixNkvB0jPXDL8Wg==", + "dev": true, + "dependencies": { + "@babel/types": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-module-transforms": { + "version": "7.17.7", + "resolved": "https://registry.npmmirror.com/@babel/helper-module-transforms/-/helper-module-transforms-7.17.7.tgz", + "integrity": "sha512-VmZD99F3gNTYB7fJRDTi+u6l/zxY0BE6OIxPSU7a50s6ZUQkHwSDmV92FfM+oCG0pZRVojGYhkR8I0OGeCVREw==", + "dev": true, + "dependencies": { + "@babel/helper-environment-visitor": "^7.16.7", + "@babel/helper-module-imports": "^7.16.7", + "@babel/helper-simple-access": "^7.17.7", + "@babel/helper-split-export-declaration": "^7.16.7", + "@babel/helper-validator-identifier": "^7.16.7", + "@babel/template": "^7.16.7", + "@babel/traverse": "^7.17.3", + "@babel/types": "^7.17.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-optimise-call-expression": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/helper-optimise-call-expression/-/helper-optimise-call-expression-7.16.7.tgz", + "integrity": "sha512-EtgBhg7rd/JcnpZFXpBy0ze1YRfdm7BnBX4uKMBd3ixa3RGAE002JZB66FJyNH7g0F38U05pXmA5P8cBh7z+1w==", + "dev": true, + "dependencies": { + "@babel/types": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-plugin-utils": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/helper-plugin-utils/-/helper-plugin-utils-7.16.7.tgz", + "integrity": "sha512-Qg3Nk7ZxpgMrsox6HreY1ZNKdBq7K72tDSliA6dCl5f007jR4ne8iD5UzuNnCJH2xBf2BEEVGr+/OL6Gdp7RxA==", + "dev": true, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-remap-async-to-generator": { + "version": "7.16.8", + "resolved": "https://registry.npmmirror.com/@babel/helper-remap-async-to-generator/-/helper-remap-async-to-generator-7.16.8.tgz", + "integrity": "sha512-fm0gH7Flb8H51LqJHy3HJ3wnE1+qtYR2A99K06ahwrawLdOFsCEWjZOrYricXJHoPSudNKxrMBUPEIPxiIIvBw==", + "dev": true, + "dependencies": { + "@babel/helper-annotate-as-pure": "^7.16.7", + "@babel/helper-wrap-function": "^7.16.8", + "@babel/types": "^7.16.8" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-replace-supers": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/helper-replace-supers/-/helper-replace-supers-7.16.7.tgz", + "integrity": "sha512-y9vsWilTNaVnVh6xiJfABzsNpgDPKev9HnAgz6Gb1p6UUwf9NepdlsV7VXGCftJM+jqD5f7JIEubcpLjZj5dBw==", + "dev": true, + "dependencies": { + "@babel/helper-environment-visitor": "^7.16.7", + "@babel/helper-member-expression-to-functions": "^7.16.7", + "@babel/helper-optimise-call-expression": "^7.16.7", + "@babel/traverse": "^7.16.7", + "@babel/types": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-simple-access": { + "version": "7.17.7", + "resolved": "https://registry.npmmirror.com/@babel/helper-simple-access/-/helper-simple-access-7.17.7.tgz", + "integrity": "sha512-txyMCGroZ96i+Pxr3Je3lzEJjqwaRC9buMUgtomcrLe5Nd0+fk1h0LLA+ixUF5OW7AhHuQ7Es1WcQJZmZsz2XA==", + "dev": true, + "dependencies": { + "@babel/types": "^7.17.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-skip-transparent-expression-wrappers": { + "version": "7.16.0", + "resolved": "https://registry.npmmirror.com/@babel/helper-skip-transparent-expression-wrappers/-/helper-skip-transparent-expression-wrappers-7.16.0.tgz", + "integrity": "sha512-+il1gTy0oHwUsBQZyJvukbB4vPMdcYBrFHa0Uc4AizLxbq6BOYC51Rv4tWocX9BLBDLZ4kc6qUFpQ6HRgL+3zw==", + "dev": true, + "dependencies": { + "@babel/types": "^7.16.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-split-export-declaration": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/helper-split-export-declaration/-/helper-split-export-declaration-7.16.7.tgz", + "integrity": "sha512-xbWoy/PFoxSWazIToT9Sif+jJTlrMcndIsaOKvTA6u7QEo7ilkRZpjew18/W3c7nm8fXdUDXh02VXTbZ0pGDNw==", + "dev": true, + "dependencies": { + "@babel/types": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-validator-identifier": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/helper-validator-identifier/-/helper-validator-identifier-7.16.7.tgz", + "integrity": "sha512-hsEnFemeiW4D08A5gUAZxLBTXpZ39P+a+DGDsHw1yxqyQ/jzFEnxf5uTEGp+3bzAbNOxU1paTgYS4ECU/IgfDw==", + "dev": true, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-validator-option": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/helper-validator-option/-/helper-validator-option-7.16.7.tgz", + "integrity": "sha512-TRtenOuRUVo9oIQGPC5G9DgK4743cdxvtOw0weQNpZXaS16SCBi5MNjZF8vba3ETURjZpTbVn7Vvcf2eAwFozQ==", + "dev": true, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-wrap-function": { + "version": "7.16.8", + "resolved": "https://registry.npmmirror.com/@babel/helper-wrap-function/-/helper-wrap-function-7.16.8.tgz", + "integrity": "sha512-8RpyRVIAW1RcDDGTA+GpPAwV22wXCfKOoM9bet6TLkGIFTkRQSkH1nMQ5Yet4MpoXe1ZwHPVtNasc2w0uZMqnw==", + "dev": true, + "dependencies": { + "@babel/helper-function-name": "^7.16.7", + "@babel/template": "^7.16.7", + "@babel/traverse": "^7.16.8", + "@babel/types": "^7.16.8" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helpers": { + "version": "7.17.9", + "resolved": "https://registry.npmmirror.com/@babel/helpers/-/helpers-7.17.9.tgz", + "integrity": "sha512-cPCt915ShDWUEzEp3+UNRktO2n6v49l5RSnG9M5pS24hA+2FAc5si+Pn1i4VVbQQ+jh+bIZhPFQOJOzbrOYY1Q==", + "dev": true, + "dependencies": { + "@babel/template": "^7.16.7", + "@babel/traverse": "^7.17.9", + "@babel/types": "^7.17.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/highlight": { + "version": "7.17.9", + "resolved": "https://registry.npmmirror.com/@babel/highlight/-/highlight-7.17.9.tgz", + "integrity": "sha512-J9PfEKCbFIv2X5bjTMiZu6Vf341N05QIY+d6FvVKynkG1S7G0j3I0QoRtWIrXhZ+/Nlb5Q0MzqL7TokEJ5BNHg==", + "dev": true, + "dependencies": { + "@babel/helper-validator-identifier": "^7.16.7", + "chalk": "^2.0.0", + "js-tokens": "^4.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/parser": { + "version": "7.17.9", + "resolved": "https://registry.npmmirror.com/@babel/parser/-/parser-7.17.9.tgz", + "integrity": "sha512-vqUSBLP8dQHFPdPi9bc5GK9vRkYHJ49fsZdtoJ8EQ8ibpwk5rPKfvNIwChB0KVXcIjcepEBBd2VHC5r9Gy8ueg==", + "dev": true, + "bin": { + "parser": "bin/babel-parser.js" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@babel/plugin-proposal-async-generator-functions": { + "version": "7.16.8", + "resolved": "https://registry.npmmirror.com/@babel/plugin-proposal-async-generator-functions/-/plugin-proposal-async-generator-functions-7.16.8.tgz", + "integrity": "sha512-71YHIvMuiuqWJQkebWJtdhQTfd4Q4mF76q2IX37uZPkG9+olBxsX+rH1vkhFto4UeJZ9dPY2s+mDvhDm1u2BGQ==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.16.7", + "@babel/helper-remap-async-to-generator": "^7.16.8", + "@babel/plugin-syntax-async-generators": "^7.8.4" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-proposal-class-properties": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/plugin-proposal-class-properties/-/plugin-proposal-class-properties-7.16.7.tgz", + "integrity": "sha512-IobU0Xme31ewjYOShSIqd/ZGM/r/cuOz2z0MDbNrhF5FW+ZVgi0f2lyeoj9KFPDOAqsYxmLWZte1WOwlvY9aww==", + "dev": true, + "dependencies": { + "@babel/helper-create-class-features-plugin": "^7.16.7", + "@babel/helper-plugin-utils": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-proposal-decorators": { + "version": "7.17.9", + "resolved": "https://registry.npmmirror.com/@babel/plugin-proposal-decorators/-/plugin-proposal-decorators-7.17.9.tgz", + "integrity": "sha512-EfH2LZ/vPa2wuPwJ26j+kYRkaubf89UlwxKXtxqEm57HrgSEYDB8t4swFP+p8LcI9yiP9ZRJJjo/58hS6BnaDA==", + "dev": true, + "dependencies": { + "@babel/helper-create-class-features-plugin": "^7.17.9", + "@babel/helper-plugin-utils": "^7.16.7", + "@babel/helper-replace-supers": "^7.16.7", + "@babel/helper-split-export-declaration": "^7.16.7", + "@babel/plugin-syntax-decorators": "^7.17.0", + "charcodes": "^0.2.0" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-proposal-json-strings": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/plugin-proposal-json-strings/-/plugin-proposal-json-strings-7.16.7.tgz", + "integrity": "sha512-lNZ3EEggsGY78JavgbHsK9u5P3pQaW7k4axlgFLYkMd7UBsiNahCITShLjNQschPyjtO6dADrL24757IdhBrsQ==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.16.7", + "@babel/plugin-syntax-json-strings": "^7.8.3" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-proposal-object-rest-spread": { + "version": "7.17.3", + "resolved": "https://registry.npmmirror.com/@babel/plugin-proposal-object-rest-spread/-/plugin-proposal-object-rest-spread-7.17.3.tgz", + "integrity": "sha512-yuL5iQA/TbZn+RGAfxQXfi7CNLmKi1f8zInn4IgobuCWcAb7i+zj4TYzQ9l8cEzVyJ89PDGuqxK1xZpUDISesw==", + "dev": true, + "dependencies": { + "@babel/compat-data": "^7.17.0", + "@babel/helper-compilation-targets": "^7.16.7", + "@babel/helper-plugin-utils": "^7.16.7", + "@babel/plugin-syntax-object-rest-spread": "^7.8.3", + "@babel/plugin-transform-parameters": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-proposal-optional-catch-binding": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/plugin-proposal-optional-catch-binding/-/plugin-proposal-optional-catch-binding-7.16.7.tgz", + "integrity": "sha512-eMOH/L4OvWSZAE1VkHbr1vckLG1WUcHGJSLqqQwl2GaUqG6QjddvrOaTUMNYiv77H5IKPMZ9U9P7EaHwvAShfA==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.16.7", + "@babel/plugin-syntax-optional-catch-binding": "^7.8.3" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-proposal-unicode-property-regex": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/plugin-proposal-unicode-property-regex/-/plugin-proposal-unicode-property-regex-7.16.7.tgz", + "integrity": "sha512-QRK0YI/40VLhNVGIjRNAAQkEHws0cswSdFFjpFyt943YmJIU1da9uW63Iu6NFV6CxTZW5eTDCrwZUstBWgp/Rg==", + "dev": true, + "dependencies": { + "@babel/helper-create-regexp-features-plugin": "^7.16.7", + "@babel/helper-plugin-utils": "^7.16.7" + }, + "engines": { + "node": ">=4" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-async-generators": { + "version": "7.8.4", + "resolved": "https://registry.npmmirror.com/@babel/plugin-syntax-async-generators/-/plugin-syntax-async-generators-7.8.4.tgz", + "integrity": "sha512-tycmZxkGfZaxhMRbXlPXuVFpdWlXpir2W4AMhSJgRKzk/eDlIXOhb2LHWoLpDF7TEHylV5zNhykX6KAgHJmTNw==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-decorators": { + "version": "7.17.0", + "resolved": "https://registry.npmmirror.com/@babel/plugin-syntax-decorators/-/plugin-syntax-decorators-7.17.0.tgz", + "integrity": "sha512-qWe85yCXsvDEluNP0OyeQjH63DlhAR3W7K9BxxU1MvbDb48tgBG+Ao6IJJ6smPDrrVzSQZrbF6donpkFBMcs3A==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-dynamic-import": { + "version": "7.8.3", + "resolved": "https://registry.npmmirror.com/@babel/plugin-syntax-dynamic-import/-/plugin-syntax-dynamic-import-7.8.3.tgz", + "integrity": "sha512-5gdGbFon+PszYzqs83S3E5mpi7/y/8M9eC90MRTZfduQOYW76ig6SOSPNe41IG5LoP3FGBn2N0RjVDSQiS94kQ==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-json-strings": { + "version": "7.8.3", + "resolved": "https://registry.npmmirror.com/@babel/plugin-syntax-json-strings/-/plugin-syntax-json-strings-7.8.3.tgz", + "integrity": "sha512-lY6kdGpWHvjoe2vk4WrAapEuBR69EMxZl+RoGRhrFGNYVK8mOPAW8VfbT/ZgrFbXlDNiiaxQnAtgVCZ6jv30EA==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-jsx": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/plugin-syntax-jsx/-/plugin-syntax-jsx-7.16.7.tgz", + "integrity": "sha512-Esxmk7YjA8QysKeT3VhTXvF6y77f/a91SIs4pWb4H2eWGQkCKFgQaG6hdoEVZtGsrAcb2K5BW66XsOErD4WU3Q==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-object-rest-spread": { + "version": "7.8.3", + "resolved": "https://registry.npmmirror.com/@babel/plugin-syntax-object-rest-spread/-/plugin-syntax-object-rest-spread-7.8.3.tgz", + "integrity": "sha512-XoqMijGZb9y3y2XskN+P1wUGiVwWZ5JmoDRwx5+3GmEplNyVM2s2Dg8ILFQm8rWM48orGy5YpI5Bl8U1y7ydlA==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-optional-catch-binding": { + "version": "7.8.3", + "resolved": "https://registry.npmmirror.com/@babel/plugin-syntax-optional-catch-binding/-/plugin-syntax-optional-catch-binding-7.8.3.tgz", + "integrity": "sha512-6VPD0Pc1lpTqw0aKoeRTMiB+kWhAoT24PA+ksWSBrFtl5SIRVpZlwN3NNPQjehA2E/91FV3RjLWoVTglWcSV3Q==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-arrow-functions": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/plugin-transform-arrow-functions/-/plugin-transform-arrow-functions-7.16.7.tgz", + "integrity": "sha512-9ffkFFMbvzTvv+7dTp/66xvZAWASuPD5Tl9LK3Z9vhOmANo6j94rik+5YMBt4CwHVMWLWpMsriIc2zsa3WW3xQ==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-async-to-generator": { + "version": "7.16.8", + "resolved": "https://registry.npmmirror.com/@babel/plugin-transform-async-to-generator/-/plugin-transform-async-to-generator-7.16.8.tgz", + "integrity": "sha512-MtmUmTJQHCnyJVrScNzNlofQJ3dLFuobYn3mwOTKHnSCMtbNsqvF71GQmJfFjdrXSsAA7iysFmYWw4bXZ20hOg==", + "dev": true, + "dependencies": { + "@babel/helper-module-imports": "^7.16.7", + "@babel/helper-plugin-utils": "^7.16.7", + "@babel/helper-remap-async-to-generator": "^7.16.8" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-block-scoped-functions": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/plugin-transform-block-scoped-functions/-/plugin-transform-block-scoped-functions-7.16.7.tgz", + "integrity": "sha512-JUuzlzmF40Z9cXyytcbZEZKckgrQzChbQJw/5PuEHYeqzCsvebDx0K0jWnIIVcmmDOAVctCgnYs0pMcrYj2zJg==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-block-scoping": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/plugin-transform-block-scoping/-/plugin-transform-block-scoping-7.16.7.tgz", + "integrity": "sha512-ObZev2nxVAYA4bhyusELdo9hb3H+A56bxH3FZMbEImZFiEDYVHXQSJ1hQKFlDnlt8G9bBrCZ5ZpURZUrV4G5qQ==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-classes": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/plugin-transform-classes/-/plugin-transform-classes-7.16.7.tgz", + "integrity": "sha512-WY7og38SFAGYRe64BrjKf8OrE6ulEHtr5jEYaZMwox9KebgqPi67Zqz8K53EKk1fFEJgm96r32rkKZ3qA2nCWQ==", + "dev": true, + "dependencies": { + "@babel/helper-annotate-as-pure": "^7.16.7", + "@babel/helper-environment-visitor": "^7.16.7", + "@babel/helper-function-name": "^7.16.7", + "@babel/helper-optimise-call-expression": "^7.16.7", + "@babel/helper-plugin-utils": "^7.16.7", + "@babel/helper-replace-supers": "^7.16.7", + "@babel/helper-split-export-declaration": "^7.16.7", + "globals": "^11.1.0" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-computed-properties": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/plugin-transform-computed-properties/-/plugin-transform-computed-properties-7.16.7.tgz", + "integrity": "sha512-gN72G9bcmenVILj//sv1zLNaPyYcOzUho2lIJBMh/iakJ9ygCo/hEF9cpGb61SCMEDxbbyBoVQxrt+bWKu5KGw==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-destructuring": { + "version": "7.17.7", + "resolved": "https://registry.npmmirror.com/@babel/plugin-transform-destructuring/-/plugin-transform-destructuring-7.17.7.tgz", + "integrity": "sha512-XVh0r5yq9sLR4vZ6eVZe8FKfIcSgaTBxVBRSYokRj2qksf6QerYnTxz9/GTuKTH/n/HwLP7t6gtlybHetJ/6hQ==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-dotall-regex": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/plugin-transform-dotall-regex/-/plugin-transform-dotall-regex-7.16.7.tgz", + "integrity": "sha512-Lyttaao2SjZF6Pf4vk1dVKv8YypMpomAbygW+mU5cYP3S5cWTfCJjG8xV6CFdzGFlfWK81IjL9viiTvpb6G7gQ==", + "dev": true, + "dependencies": { + "@babel/helper-create-regexp-features-plugin": "^7.16.7", + "@babel/helper-plugin-utils": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-duplicate-keys": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/plugin-transform-duplicate-keys/-/plugin-transform-duplicate-keys-7.16.7.tgz", + "integrity": "sha512-03DvpbRfvWIXyK0/6QiR1KMTWeT6OcQ7tbhjrXyFS02kjuX/mu5Bvnh5SDSWHxyawit2g5aWhKwI86EE7GUnTw==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-exponentiation-operator": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/plugin-transform-exponentiation-operator/-/plugin-transform-exponentiation-operator-7.16.7.tgz", + "integrity": "sha512-8UYLSlyLgRixQvlYH3J2ekXFHDFLQutdy7FfFAMm3CPZ6q9wHCwnUyiXpQCe3gVVnQlHc5nsuiEVziteRNTXEA==", + "dev": true, + "dependencies": { + "@babel/helper-builder-binary-assignment-operator-visitor": "^7.16.7", + "@babel/helper-plugin-utils": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-for-of": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/plugin-transform-for-of/-/plugin-transform-for-of-7.16.7.tgz", + "integrity": "sha512-/QZm9W92Ptpw7sjI9Nx1mbcsWz33+l8kuMIQnDwgQBG5s3fAfQvkRjQ7NqXhtNcKOnPkdICmUHyCaWW06HCsqg==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-function-name": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/plugin-transform-function-name/-/plugin-transform-function-name-7.16.7.tgz", + "integrity": "sha512-SU/C68YVwTRxqWj5kgsbKINakGag0KTgq9f2iZEXdStoAbOzLHEBRYzImmA6yFo8YZhJVflvXmIHUO7GWHmxxA==", + "dev": true, + "dependencies": { + "@babel/helper-compilation-targets": "^7.16.7", + "@babel/helper-function-name": "^7.16.7", + "@babel/helper-plugin-utils": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-literals": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/plugin-transform-literals/-/plugin-transform-literals-7.16.7.tgz", + "integrity": "sha512-6tH8RTpTWI0s2sV6uq3e/C9wPo4PTqqZps4uF0kzQ9/xPLFQtipynvmT1g/dOfEJ+0EQsHhkQ/zyRId8J2b8zQ==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-modules-amd": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/plugin-transform-modules-amd/-/plugin-transform-modules-amd-7.16.7.tgz", + "integrity": "sha512-KaaEtgBL7FKYwjJ/teH63oAmE3lP34N3kshz8mm4VMAw7U3PxjVwwUmxEFksbgsNUaO3wId9R2AVQYSEGRa2+g==", + "dev": true, + "dependencies": { + "@babel/helper-module-transforms": "^7.16.7", + "@babel/helper-plugin-utils": "^7.16.7", + "babel-plugin-dynamic-import-node": "^2.3.3" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-modules-commonjs": { + "version": "7.17.9", + "resolved": "https://registry.npmmirror.com/@babel/plugin-transform-modules-commonjs/-/plugin-transform-modules-commonjs-7.17.9.tgz", + "integrity": "sha512-2TBFd/r2I6VlYn0YRTz2JdazS+FoUuQ2rIFHoAxtyP/0G3D82SBLaRq9rnUkpqlLg03Byfl/+M32mpxjO6KaPw==", + "dev": true, + "dependencies": { + "@babel/helper-module-transforms": "^7.17.7", + "@babel/helper-plugin-utils": "^7.16.7", + "@babel/helper-simple-access": "^7.17.7", + "babel-plugin-dynamic-import-node": "^2.3.3" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-modules-systemjs": { + "version": "7.17.8", + "resolved": "https://registry.npmmirror.com/@babel/plugin-transform-modules-systemjs/-/plugin-transform-modules-systemjs-7.17.8.tgz", + "integrity": "sha512-39reIkMTUVagzgA5x88zDYXPCMT6lcaRKs1+S9K6NKBPErbgO/w/kP8GlNQTC87b412ZTlmNgr3k2JrWgHH+Bw==", + "dev": true, + "dependencies": { + "@babel/helper-hoist-variables": "^7.16.7", + "@babel/helper-module-transforms": "^7.17.7", + "@babel/helper-plugin-utils": "^7.16.7", + "@babel/helper-validator-identifier": "^7.16.7", + "babel-plugin-dynamic-import-node": "^2.3.3" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-modules-umd": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/plugin-transform-modules-umd/-/plugin-transform-modules-umd-7.16.7.tgz", + "integrity": "sha512-EMh7uolsC8O4xhudF2F6wedbSHm1HHZ0C6aJ7K67zcDNidMzVcxWdGr+htW9n21klm+bOn+Rx4CBsAntZd3rEQ==", + "dev": true, + "dependencies": { + "@babel/helper-module-transforms": "^7.16.7", + "@babel/helper-plugin-utils": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-named-capturing-groups-regex": { + "version": "7.16.8", + "resolved": "https://registry.npmmirror.com/@babel/plugin-transform-named-capturing-groups-regex/-/plugin-transform-named-capturing-groups-regex-7.16.8.tgz", + "integrity": "sha512-j3Jw+n5PvpmhRR+mrgIh04puSANCk/T/UA3m3P1MjJkhlK906+ApHhDIqBQDdOgL/r1UYpz4GNclTXxyZrYGSw==", + "dev": true, + "dependencies": { + "@babel/helper-create-regexp-features-plugin": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/plugin-transform-new-target": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/plugin-transform-new-target/-/plugin-transform-new-target-7.16.7.tgz", + "integrity": "sha512-xiLDzWNMfKoGOpc6t3U+etCE2yRnn3SM09BXqWPIZOBpL2gvVrBWUKnsJx0K/ADi5F5YC5f8APFfWrz25TdlGg==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-object-super": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/plugin-transform-object-super/-/plugin-transform-object-super-7.16.7.tgz", + "integrity": "sha512-14J1feiQVWaGvRxj2WjyMuXS2jsBkgB3MdSN5HuC2G5nRspa5RK9COcs82Pwy5BuGcjb+fYaUj94mYcOj7rCvw==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.16.7", + "@babel/helper-replace-supers": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-parameters": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/plugin-transform-parameters/-/plugin-transform-parameters-7.16.7.tgz", + "integrity": "sha512-AT3MufQ7zZEhU2hwOA11axBnExW0Lszu4RL/tAlUJBuNoRak+wehQW8h6KcXOcgjY42fHtDxswuMhMjFEuv/aw==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-regenerator": { + "version": "7.17.9", + "resolved": "https://registry.npmmirror.com/@babel/plugin-transform-regenerator/-/plugin-transform-regenerator-7.17.9.tgz", + "integrity": "sha512-Lc2TfbxR1HOyn/c6b4Y/b6NHoTb67n/IoWLxTu4kC7h4KQnWlhCq2S8Tx0t2SVvv5Uu87Hs+6JEJ5kt2tYGylQ==", + "dev": true, + "dependencies": { + "regenerator-transform": "^0.15.0" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-runtime": { + "version": "7.17.0", + "resolved": "https://registry.npmmirror.com/@babel/plugin-transform-runtime/-/plugin-transform-runtime-7.17.0.tgz", + "integrity": "sha512-fr7zPWnKXNc1xoHfrIU9mN/4XKX4VLZ45Q+oMhfsYIaHvg7mHgmhfOy/ckRWqDK7XF3QDigRpkh5DKq6+clE8A==", + "dev": true, + "dependencies": { + "@babel/helper-module-imports": "^7.16.7", + "@babel/helper-plugin-utils": "^7.16.7", + "babel-plugin-polyfill-corejs2": "^0.3.0", + "babel-plugin-polyfill-corejs3": "^0.5.0", + "babel-plugin-polyfill-regenerator": "^0.3.0", + "semver": "^6.3.0" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-runtime/node_modules/semver": { + "version": "6.3.0", + "resolved": "https://registry.npmmirror.com/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/@babel/plugin-transform-shorthand-properties": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/plugin-transform-shorthand-properties/-/plugin-transform-shorthand-properties-7.16.7.tgz", + "integrity": "sha512-hah2+FEnoRoATdIb05IOXf+4GzXYTq75TVhIn1PewihbpyrNWUt2JbudKQOETWw6QpLe+AIUpJ5MVLYTQbeeUg==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-spread": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/plugin-transform-spread/-/plugin-transform-spread-7.16.7.tgz", + "integrity": "sha512-+pjJpgAngb53L0iaA5gU/1MLXJIfXcYepLgXB3esVRf4fqmj8f2cxM3/FKaHsZms08hFQJkFccEWuIpm429TXg==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.16.7", + "@babel/helper-skip-transparent-expression-wrappers": "^7.16.0" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-sticky-regex": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/plugin-transform-sticky-regex/-/plugin-transform-sticky-regex-7.16.7.tgz", + "integrity": "sha512-NJa0Bd/87QV5NZZzTuZG5BPJjLYadeSZ9fO6oOUoL4iQx+9EEuw/eEM92SrsT19Yc2jgB1u1hsjqDtH02c3Drw==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-template-literals": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/plugin-transform-template-literals/-/plugin-transform-template-literals-7.16.7.tgz", + "integrity": "sha512-VwbkDDUeenlIjmfNeDX/V0aWrQH2QiVyJtwymVQSzItFDTpxfyJh3EVaQiS0rIN/CqbLGr0VcGmuwyTdZtdIsA==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-typeof-symbol": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/plugin-transform-typeof-symbol/-/plugin-transform-typeof-symbol-7.16.7.tgz", + "integrity": "sha512-p2rOixCKRJzpg9JB4gjnG4gjWkWa89ZoYUnl9snJ1cWIcTH/hvxZqfO+WjG6T8DRBpctEol5jw1O5rA8gkCokQ==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-unicode-regex": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/plugin-transform-unicode-regex/-/plugin-transform-unicode-regex-7.16.7.tgz", + "integrity": "sha512-oC5tYYKw56HO75KZVLQ+R/Nl3Hro9kf8iG0hXoaHP7tjAyCpvqBiSNe6vGrZni1Z6MggmUOC6A7VP7AVmw225Q==", + "dev": true, + "dependencies": { + "@babel/helper-create-regexp-features-plugin": "^7.16.7", + "@babel/helper-plugin-utils": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/preset-env": { + "version": "7.3.4", + "resolved": "https://registry.npmmirror.com/@babel/preset-env/-/preset-env-7.3.4.tgz", + "integrity": "sha512-2mwqfYMK8weA0g0uBKOt4FE3iEodiHy9/CW0b+nWXcbL+pGzLx8ESYc+j9IIxr6LTDHWKgPm71i9smo02bw+gA==", + "dev": true, + "dependencies": { + "@babel/helper-module-imports": "^7.0.0", + "@babel/helper-plugin-utils": "^7.0.0", + "@babel/plugin-proposal-async-generator-functions": "^7.2.0", + "@babel/plugin-proposal-json-strings": "^7.2.0", + "@babel/plugin-proposal-object-rest-spread": "^7.3.4", + "@babel/plugin-proposal-optional-catch-binding": "^7.2.0", + "@babel/plugin-proposal-unicode-property-regex": "^7.2.0", + "@babel/plugin-syntax-async-generators": "^7.2.0", + "@babel/plugin-syntax-json-strings": "^7.2.0", + "@babel/plugin-syntax-object-rest-spread": "^7.2.0", + "@babel/plugin-syntax-optional-catch-binding": "^7.2.0", + "@babel/plugin-transform-arrow-functions": "^7.2.0", + "@babel/plugin-transform-async-to-generator": "^7.3.4", + "@babel/plugin-transform-block-scoped-functions": "^7.2.0", + "@babel/plugin-transform-block-scoping": "^7.3.4", + "@babel/plugin-transform-classes": "^7.3.4", + "@babel/plugin-transform-computed-properties": "^7.2.0", + "@babel/plugin-transform-destructuring": "^7.2.0", + "@babel/plugin-transform-dotall-regex": "^7.2.0", + "@babel/plugin-transform-duplicate-keys": "^7.2.0", + "@babel/plugin-transform-exponentiation-operator": "^7.2.0", + "@babel/plugin-transform-for-of": "^7.2.0", + "@babel/plugin-transform-function-name": "^7.2.0", + "@babel/plugin-transform-literals": "^7.2.0", + "@babel/plugin-transform-modules-amd": "^7.2.0", + "@babel/plugin-transform-modules-commonjs": "^7.2.0", + "@babel/plugin-transform-modules-systemjs": "^7.3.4", + "@babel/plugin-transform-modules-umd": "^7.2.0", + "@babel/plugin-transform-named-capturing-groups-regex": "^7.3.0", + "@babel/plugin-transform-new-target": "^7.0.0", + "@babel/plugin-transform-object-super": "^7.2.0", + "@babel/plugin-transform-parameters": "^7.2.0", + "@babel/plugin-transform-regenerator": "^7.3.4", + "@babel/plugin-transform-shorthand-properties": "^7.2.0", + "@babel/plugin-transform-spread": "^7.2.0", + "@babel/plugin-transform-sticky-regex": "^7.2.0", + "@babel/plugin-transform-template-literals": "^7.2.0", + "@babel/plugin-transform-typeof-symbol": "^7.2.0", + "@babel/plugin-transform-unicode-regex": "^7.2.0", + "browserslist": "^4.3.4", + "invariant": "^2.2.2", + "js-levenshtein": "^1.1.3", + "semver": "^5.3.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/runtime": { + "version": "7.17.9", + "resolved": "https://registry.npmmirror.com/@babel/runtime/-/runtime-7.17.9.tgz", + "integrity": "sha512-lSiBBvodq29uShpWGNbgFdKYNiFDo5/HIYsaCEY9ff4sb10x9jizo2+pRrSyF4jKZCXqgzuqBOQKbUm90gQwJg==", + "dependencies": { + "regenerator-runtime": "^0.13.4" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/runtime-corejs2": { + "version": "7.17.9", + "resolved": "https://registry.npmmirror.com/@babel/runtime-corejs2/-/runtime-corejs2-7.17.9.tgz", + "integrity": "sha512-+QThIsnjVY12uURTvmnW33risFZ7ulq6OWw0VJL08UwiYiWVp9PM63s+W1L2ppajYyKAYKb7afcGYSHzA0k04Q==", + "dev": true, + "dependencies": { + "core-js": "^2.6.5", + "regenerator-runtime": "^0.13.4" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/template": { + "version": "7.16.7", + "resolved": "https://registry.npmmirror.com/@babel/template/-/template-7.16.7.tgz", + "integrity": "sha512-I8j/x8kHUrbYRTUxXrrMbfCa7jxkE7tZre39x3kjr9hvI82cK1FfqLygotcWN5kdPGWcLdWMHpSBavse5tWw3w==", + "dev": true, + "dependencies": { + "@babel/code-frame": "^7.16.7", + "@babel/parser": "^7.16.7", + "@babel/types": "^7.16.7" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/traverse": { + "version": "7.17.9", + "resolved": "https://registry.npmmirror.com/@babel/traverse/-/traverse-7.17.9.tgz", + "integrity": "sha512-PQO8sDIJ8SIwipTPiR71kJQCKQYB5NGImbOviK8K+kg5xkNSYXLBupuX9QhatFowrsvo9Hj8WgArg3W7ijNAQw==", + "dev": true, + "dependencies": { + "@babel/code-frame": "^7.16.7", + "@babel/generator": "^7.17.9", + "@babel/helper-environment-visitor": "^7.16.7", + "@babel/helper-function-name": "^7.17.9", + "@babel/helper-hoist-variables": "^7.16.7", + "@babel/helper-split-export-declaration": "^7.16.7", + "@babel/parser": "^7.17.9", + "@babel/types": "^7.17.0", + "debug": "^4.1.0", + "globals": "^11.1.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/traverse/node_modules/debug": { + "version": "4.3.4", + "resolved": "https://registry.npmmirror.com/debug/-/debug-4.3.4.tgz", + "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", + "dev": true, + "dependencies": { + "ms": "2.1.2" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/@babel/traverse/node_modules/ms": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", + "dev": true + }, + "node_modules/@babel/types": { + "version": "7.17.0", + "resolved": "https://registry.npmmirror.com/@babel/types/-/types-7.17.0.tgz", + "integrity": "sha512-TmKSNO4D5rzhL5bjWFcVHHLETzfQ/AmbKpKPOSjlP0WoHZ6L911fgoOKY4Alp/emzG4cHJdyN49zpgkbXFEHHw==", + "dev": true, + "dependencies": { + "@babel/helper-validator-identifier": "^7.16.7", + "to-fast-properties": "^2.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@codemirror/lang-json": { + "version": "6.0.1", + "resolved": "https://registry.npmmirror.com/@codemirror/lang-json/-/lang-json-6.0.1.tgz", + "integrity": "sha512-+T1flHdgpqDDlJZ2Lkil/rLiRy684WMLc74xUnjJH48GQdfJo/pudlTRreZmKwzP8/tGdKf83wlbAdOCzlJOGQ==", + "dependencies": { + "@codemirror/language": "^6.0.0", + "@lezer/json": "^1.0.0" + } + }, + "node_modules/@codemirror/language": { + "version": "6.8.0", + "resolved": "https://registry.npmmirror.com/@codemirror/language/-/language-6.8.0.tgz", + "integrity": "sha512-r1paAyWOZkfY0RaYEZj3Kul+MiQTEbDvYqf8gPGaRvNneHXCmfSaAVFjwRUPlgxS8yflMxw2CTu6uCMp8R8A2g==", + "dependencies": { + "@codemirror/state": "^6.0.0", + "@codemirror/view": "^6.0.0", + "@lezer/common": "^1.0.0", + "@lezer/highlight": "^1.0.0", + "@lezer/lr": "^1.0.0", + "style-mod": "^4.0.0" + } + }, + "node_modules/@codemirror/state": { + "version": "6.2.1", + "resolved": "https://registry.npmmirror.com/@codemirror/state/-/state-6.2.1.tgz", + "integrity": "sha512-RupHSZ8+OjNT38zU9fKH2sv+Dnlr8Eb8sl4NOnnqz95mCFTZUaiRP8Xv5MeeaG0px2b8Bnfe7YGwCV3nsBhbuw==" + }, + "node_modules/@codemirror/view": { + "version": "6.14.0", + "resolved": "https://registry.npmmirror.com/@codemirror/view/-/view-6.14.0.tgz", + "integrity": "sha512-I263FPs4In42MNmrdwN2DfmYPFMVMXgT7o/mxdGp4jv5LPs8i0FOxzmxF5yeeQdYSTztb2ZhmPIu0ahveInVTg==", + "dependencies": { + "@codemirror/state": "^6.1.4", + "style-mod": "^4.0.0", + "w3c-keyname": "^2.2.4" + } + }, + "node_modules/@gar/promisify": { + "version": "1.1.3", + "resolved": "https://registry.npmmirror.com/@gar/promisify/-/promisify-1.1.3.tgz", + "integrity": "sha512-k2Ty1JcVojjJFwrg/ThKi2ujJ7XNLYaFGNB/bWT9wGR+oSMJHMa5w+CUq6p/pVrKeNNgA7pCqEcjSnHVoqJQFw==", + "dev": true + }, + "node_modules/@hapi/address": { + "version": "2.1.4", + "resolved": "https://registry.npmmirror.com/@hapi/address/-/address-2.1.4.tgz", + "integrity": "sha512-QD1PhQk+s31P1ixsX0H0Suoupp3VMXzIVMSwobR3F3MSUO2YCV0B7xqLcUw/Bh8yuvd3LhpyqLQWTNcRmp6IdQ==", + "deprecated": "Moved to 'npm install @sideway/address'", + "dev": true + }, + "node_modules/@hapi/bourne": { + "version": "1.3.2", + "resolved": "https://registry.npmmirror.com/@hapi/bourne/-/bourne-1.3.2.tgz", + "integrity": "sha512-1dVNHT76Uu5N3eJNTYcvxee+jzX4Z9lfciqRRHCU27ihbUcYi+iSc2iml5Ke1LXe1SyJCLA0+14Jh4tXJgOppA==", + "deprecated": "This version has been deprecated and is no longer supported or maintained", + "dev": true + }, + "node_modules/@hapi/hoek": { + "version": "8.5.1", + "resolved": "https://registry.npmmirror.com/@hapi/hoek/-/hoek-8.5.1.tgz", + "integrity": "sha512-yN7kbciD87WzLGc5539Tn0sApjyiGHAJgKvG9W8C7O+6c7qmoQMfVs0W4bX17eqz6C78QJqqFrtgdK5EWf6Qow==", + "deprecated": "This version has been deprecated and is no longer supported or maintained", + "dev": true + }, + "node_modules/@hapi/joi": { + "version": "15.1.1", + "resolved": "https://registry.npmmirror.com/@hapi/joi/-/joi-15.1.1.tgz", + "integrity": "sha512-entf8ZMOK8sc+8YfeOlM8pCfg3b5+WZIKBfUaaJT8UsjAAPjartzxIYm3TIbjvA4u+u++KbcXD38k682nVHDAQ==", + "deprecated": "Switch to 'npm install joi'", + "dev": true, + "dependencies": { + "@hapi/address": "2.x.x", + "@hapi/bourne": "1.x.x", + "@hapi/hoek": "8.x.x", + "@hapi/topo": "3.x.x" + } + }, + "node_modules/@hapi/topo": { + "version": "3.1.6", + "resolved": "https://registry.npmmirror.com/@hapi/topo/-/topo-3.1.6.tgz", + "integrity": "sha512-tAag0jEcjwH+P2quUfipd7liWCNX2F8NvYjQp2wtInsZxnMlypdw0FtAOLxtvvkO+GSRRbmNi8m/5y42PQJYCQ==", + "deprecated": "This version has been deprecated and is no longer supported or maintained", + "dev": true, + "dependencies": { + "@hapi/hoek": "^8.3.0" + } + }, + "node_modules/@hutson/parse-repository-url": { + "version": "3.0.2", + "resolved": "https://registry.npmmirror.com/@hutson/parse-repository-url/-/parse-repository-url-3.0.2.tgz", + "integrity": "sha512-H9XAx3hc0BQHY6l+IFSWHDySypcXsvsuLhgYLUGywmJ5pswRVQJUHpOsobnLYp2ZUaUlKiKDrgWWhosOwAEM8Q==", + "dev": true, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@intervolga/optimize-cssnano-plugin": { + "version": "1.0.6", + "resolved": "https://registry.npmmirror.com/@intervolga/optimize-cssnano-plugin/-/optimize-cssnano-plugin-1.0.6.tgz", + "integrity": "sha512-zN69TnSr0viRSU6cEDIcuPcP67QcpQ6uHACg58FiN9PDrU6SLyGW3MR4tiISbYxy1kDWAVPwD+XwQTWE5cigAA==", + "dev": true, + "dependencies": { + "cssnano": "^4.0.0", + "cssnano-preset-default": "^4.0.0", + "postcss": "^7.0.0" + }, + "peerDependencies": { + "webpack": "^4.0.0" + } + }, + "node_modules/@jridgewell/resolve-uri": { + "version": "3.0.6", + "resolved": "https://registry.npmmirror.com/@jridgewell/resolve-uri/-/resolve-uri-3.0.6.tgz", + "integrity": "sha512-R7xHtBSNm+9SyvpJkdQl+qrM3Hm2fea3Ef197M3mUug+v+yR+Rhfbs7PBtcBUVnIWJ4JcAdjvij+c8hXS9p5aw==", + "dev": true, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@jridgewell/sourcemap-codec": { + "version": "1.4.11", + "resolved": "https://registry.npmmirror.com/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.4.11.tgz", + "integrity": "sha512-Fg32GrJo61m+VqYSdRSjRXMjQ06j8YIYfcTqndLYVAaHmroZHLJZCydsWBOTDqXS2v+mjxohBWEMfg97GXmYQg==", + "dev": true + }, + "node_modules/@jridgewell/trace-mapping": { + "version": "0.3.9", + "resolved": "https://registry.npmmirror.com/@jridgewell/trace-mapping/-/trace-mapping-0.3.9.tgz", + "integrity": "sha512-3Belt6tdc8bPgAtbcmdtNJlirVoTmEb5e2gC94PnkwEW9jI6CAHUeoG85tjWP5WquqfavoMtMwiG4P926ZKKuQ==", + "dev": true, + "dependencies": { + "@jridgewell/resolve-uri": "^3.0.3", + "@jridgewell/sourcemap-codec": "^1.4.10" + } + }, + "node_modules/@kazupon/vue-i18n-loader": { + "version": "0.4.1", + "resolved": "https://registry.npmmirror.com/@kazupon/vue-i18n-loader/-/vue-i18n-loader-0.4.1.tgz", + "integrity": "sha512-hVznmhnyoUKozGY7pwq/UtPL76UDzb+aiN2YksZZIzCY/MkEqih0MSyEmTGw7+HVWzJRPAlDyoRNR4tWKmkCRw==", + "deprecated": "WARNING: If you would like to use @kazupon/vue-i18n-loader that is released new features and bug fixes, you need to install @intlify/vue-i18n-loader.", + "dev": true, + "dependencies": { + "js-yaml": "^3.13.1", + "json5": "^2.1.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@kazupon/vue-i18n-loader/node_modules/json5": { + "version": "2.2.1", + "resolved": "https://registry.npmmirror.com/json5/-/json5-2.2.1.tgz", + "integrity": "sha512-1hqLFMSrGHRHxav9q9gNjJ5EXznIxGVO09xQRrwplcS8qs28pZ8s8hupZAmqDwZUmVZ2Qb2jnyPOWcDH8m8dlA==", + "dev": true, + "bin": { + "json5": "lib/cli.js" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/@lerna/add": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/add/-/add-4.0.0.tgz", + "integrity": "sha512-cpmAH1iS3k8JBxNvnMqrGTTjbY/ZAiKa1ChJzFevMYY3eeqbvhsBKnBcxjRXtdrJ6bd3dCQM+ZtK+0i682Fhng==", + "dev": true, + "dependencies": { + "@lerna/bootstrap": "4.0.0", + "@lerna/command": "4.0.0", + "@lerna/filter-options": "4.0.0", + "@lerna/npm-conf": "4.0.0", + "@lerna/validation-error": "4.0.0", + "dedent": "^0.7.0", + "npm-package-arg": "^8.1.0", + "p-map": "^4.0.0", + "pacote": "^11.2.6", + "semver": "^7.3.4" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/add/node_modules/lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/add/node_modules/p-map": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/p-map/-/p-map-4.0.0.tgz", + "integrity": "sha512-/bjOqmgETBYB5BoEeGVea8dmvHb2m9GLy1E9W43yeyfP6QQCZGFNa+XRceJEuDB6zqr+gKpIAmlLebMpykw/MQ==", + "dev": true, + "dependencies": { + "aggregate-error": "^3.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/add/node_modules/semver": { + "version": "7.3.7", + "resolved": "https://registry.npmmirror.com/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/add/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/@lerna/bootstrap": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/bootstrap/-/bootstrap-4.0.0.tgz", + "integrity": "sha512-RkS7UbeM2vu+kJnHzxNRCLvoOP9yGNgkzRdy4UV2hNalD7EP41bLvRVOwRYQ7fhc2QcbhnKNdOBihYRL0LcKtw==", + "dev": true, + "dependencies": { + "@lerna/command": "4.0.0", + "@lerna/filter-options": "4.0.0", + "@lerna/has-npm-version": "4.0.0", + "@lerna/npm-install": "4.0.0", + "@lerna/package-graph": "4.0.0", + "@lerna/pulse-till-done": "4.0.0", + "@lerna/rimraf-dir": "4.0.0", + "@lerna/run-lifecycle": "4.0.0", + "@lerna/run-topologically": "4.0.0", + "@lerna/symlink-binary": "4.0.0", + "@lerna/symlink-dependencies": "4.0.0", + "@lerna/validation-error": "4.0.0", + "dedent": "^0.7.0", + "get-port": "^5.1.1", + "multimatch": "^5.0.0", + "npm-package-arg": "^8.1.0", + "npmlog": "^4.1.2", + "p-map": "^4.0.0", + "p-map-series": "^2.1.0", + "p-waterfall": "^2.1.1", + "read-package-tree": "^5.3.1", + "semver": "^7.3.4" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/bootstrap/node_modules/lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/bootstrap/node_modules/p-map": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/p-map/-/p-map-4.0.0.tgz", + "integrity": "sha512-/bjOqmgETBYB5BoEeGVea8dmvHb2m9GLy1E9W43yeyfP6QQCZGFNa+XRceJEuDB6zqr+gKpIAmlLebMpykw/MQ==", + "dev": true, + "dependencies": { + "aggregate-error": "^3.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/bootstrap/node_modules/semver": { + "version": "7.3.7", + "resolved": "https://registry.npmmirror.com/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/bootstrap/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/@lerna/changed": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/changed/-/changed-4.0.0.tgz", + "integrity": "sha512-cD+KuPRp6qiPOD+BO6S6SN5cARspIaWSOqGBpGnYzLb4uWT8Vk4JzKyYtc8ym1DIwyoFXHosXt8+GDAgR8QrgQ==", + "dev": true, + "dependencies": { + "@lerna/collect-updates": "4.0.0", + "@lerna/command": "4.0.0", + "@lerna/listable": "4.0.0", + "@lerna/output": "4.0.0" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/check-working-tree": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/check-working-tree/-/check-working-tree-4.0.0.tgz", + "integrity": "sha512-/++bxM43jYJCshBiKP5cRlCTwSJdRSxVmcDAXM+1oUewlZJVSVlnks5eO0uLxokVFvLhHlC5kHMc7gbVFPHv6Q==", + "dev": true, + "dependencies": { + "@lerna/collect-uncommitted": "4.0.0", + "@lerna/describe-ref": "4.0.0", + "@lerna/validation-error": "4.0.0" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/child-process": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/child-process/-/child-process-4.0.0.tgz", + "integrity": "sha512-XtCnmCT9eyVsUUHx6y/CTBYdV9g2Cr/VxyseTWBgfIur92/YKClfEtJTbOh94jRT62hlKLqSvux/UhxXVh613Q==", + "dev": true, + "dependencies": { + "chalk": "^4.1.0", + "execa": "^5.0.0", + "strong-log-transformer": "^2.1.0" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/child-process/node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmmirror.com/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/child-process/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmmirror.com/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/child-process/node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/@lerna/child-process/node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmmirror.com/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "node_modules/@lerna/child-process/node_modules/cross-spawn": { + "version": "7.0.3", + "resolved": "https://registry.npmmirror.com/cross-spawn/-/cross-spawn-7.0.3.tgz", + "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", + "dev": true, + "dependencies": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@lerna/child-process/node_modules/execa": { + "version": "5.1.1", + "resolved": "https://registry.npmmirror.com/execa/-/execa-5.1.1.tgz", + "integrity": "sha512-8uSpZZocAZRBAPIEINJj3Lo9HyGitllczc27Eh5YYojjMFMn8yHMDMaUHE2Jqfq05D/wucwI4JGURyXt1vchyg==", + "dev": true, + "dependencies": { + "cross-spawn": "^7.0.3", + "get-stream": "^6.0.0", + "human-signals": "^2.1.0", + "is-stream": "^2.0.0", + "merge-stream": "^2.0.0", + "npm-run-path": "^4.0.1", + "onetime": "^5.1.2", + "signal-exit": "^3.0.3", + "strip-final-newline": "^2.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/child-process/node_modules/get-stream": { + "version": "6.0.1", + "resolved": "https://registry.npmmirror.com/get-stream/-/get-stream-6.0.1.tgz", + "integrity": "sha512-ts6Wi+2j3jQjqi70w5AlN8DFnkSwC+MqmxEzdEALB2qXZYV3X/b1CTfgPLGJNMeAWxdPfU8FO1ms3NUfaHCPYg==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/child-process/node_modules/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/child-process/node_modules/human-signals": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/human-signals/-/human-signals-2.1.0.tgz", + "integrity": "sha512-B4FFZ6q/T2jhhksgkbEW3HBvWIfDW85snkQgawt07S7J5QXTk6BkNV+0yAeZrM5QpMAdYlocGoljn0sJ/WQkFw==", + "dev": true, + "engines": { + "node": ">=10.17.0" + } + }, + "node_modules/@lerna/child-process/node_modules/is-stream": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/is-stream/-/is-stream-2.0.1.tgz", + "integrity": "sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/child-process/node_modules/mimic-fn": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/mimic-fn/-/mimic-fn-2.1.0.tgz", + "integrity": "sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/@lerna/child-process/node_modules/npm-run-path": { + "version": "4.0.1", + "resolved": "https://registry.npmmirror.com/npm-run-path/-/npm-run-path-4.0.1.tgz", + "integrity": "sha512-S48WzZW777zhNIrn7gxOlISNAqi9ZC/uQFnRdbeIHhZhCA6UqpkOT8T1G7BvfdgP4Er8gF4sUbaS0i7QvIfCWw==", + "dev": true, + "dependencies": { + "path-key": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/child-process/node_modules/onetime": { + "version": "5.1.2", + "resolved": "https://registry.npmmirror.com/onetime/-/onetime-5.1.2.tgz", + "integrity": "sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==", + "dev": true, + "dependencies": { + "mimic-fn": "^2.1.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/@lerna/child-process/node_modules/path-key": { + "version": "3.1.1", + "resolved": "https://registry.npmmirror.com/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/child-process/node_modules/shebang-command": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/shebang-command/-/shebang-command-2.0.0.tgz", + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", + "dev": true, + "dependencies": { + "shebang-regex": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/child-process/node_modules/shebang-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/shebang-regex/-/shebang-regex-3.0.0.tgz", + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/child-process/node_modules/supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmmirror.com/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/child-process/node_modules/which": { + "version": "2.0.2", + "resolved": "https://registry.npmmirror.com/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "node-which": "bin/node-which" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@lerna/clean": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/clean/-/clean-4.0.0.tgz", + "integrity": "sha512-uugG2iN9k45ITx2jtd8nEOoAtca8hNlDCUM0N3lFgU/b1mEQYAPRkqr1qs4FLRl/Y50ZJ41wUz1eazS+d/0osA==", + "dev": true, + "dependencies": { + "@lerna/command": "4.0.0", + "@lerna/filter-options": "4.0.0", + "@lerna/prompt": "4.0.0", + "@lerna/pulse-till-done": "4.0.0", + "@lerna/rimraf-dir": "4.0.0", + "p-map": "^4.0.0", + "p-map-series": "^2.1.0", + "p-waterfall": "^2.1.1" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/clean/node_modules/p-map": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/p-map/-/p-map-4.0.0.tgz", + "integrity": "sha512-/bjOqmgETBYB5BoEeGVea8dmvHb2m9GLy1E9W43yeyfP6QQCZGFNa+XRceJEuDB6zqr+gKpIAmlLebMpykw/MQ==", + "dev": true, + "dependencies": { + "aggregate-error": "^3.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/cli": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/cli/-/cli-4.0.0.tgz", + "integrity": "sha512-Neaw3GzFrwZiRZv2g7g6NwFjs3er1vhraIniEs0jjVLPMNC4eata0na3GfE5yibkM/9d3gZdmihhZdZ3EBdvYA==", + "dev": true, + "dependencies": { + "@lerna/global-options": "4.0.0", + "dedent": "^0.7.0", + "npmlog": "^4.1.2", + "yargs": "^16.2.0" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/collect-uncommitted": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/collect-uncommitted/-/collect-uncommitted-4.0.0.tgz", + "integrity": "sha512-ufSTfHZzbx69YNj7KXQ3o66V4RC76ffOjwLX0q/ab//61bObJ41n03SiQEhSlmpP+gmFbTJ3/7pTe04AHX9m/g==", + "dev": true, + "dependencies": { + "@lerna/child-process": "4.0.0", + "chalk": "^4.1.0", + "npmlog": "^4.1.2" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/collect-uncommitted/node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmmirror.com/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/collect-uncommitted/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmmirror.com/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/collect-uncommitted/node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/@lerna/collect-uncommitted/node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmmirror.com/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "node_modules/@lerna/collect-uncommitted/node_modules/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/collect-uncommitted/node_modules/supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmmirror.com/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/collect-updates": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/collect-updates/-/collect-updates-4.0.0.tgz", + "integrity": "sha512-bnNGpaj4zuxsEkyaCZLka9s7nMs58uZoxrRIPJ+nrmrZYp1V5rrd+7/NYTuunOhY2ug1sTBvTAxj3NZQ+JKnOw==", + "dev": true, + "dependencies": { + "@lerna/child-process": "4.0.0", + "@lerna/describe-ref": "4.0.0", + "minimatch": "^3.0.4", + "npmlog": "^4.1.2", + "slash": "^3.0.0" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/collect-updates/node_modules/slash": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/slash/-/slash-3.0.0.tgz", + "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/command": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/command/-/command-4.0.0.tgz", + "integrity": "sha512-LM9g3rt5FsPNFqIHUeRwWXLNHJ5NKzOwmVKZ8anSp4e1SPrv2HNc1V02/9QyDDZK/w+5POXH5lxZUI1CHaOK/A==", + "dev": true, + "dependencies": { + "@lerna/child-process": "4.0.0", + "@lerna/package-graph": "4.0.0", + "@lerna/project": "4.0.0", + "@lerna/validation-error": "4.0.0", + "@lerna/write-log-file": "4.0.0", + "clone-deep": "^4.0.1", + "dedent": "^0.7.0", + "execa": "^5.0.0", + "is-ci": "^2.0.0", + "npmlog": "^4.1.2" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/command/node_modules/ci-info": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/ci-info/-/ci-info-2.0.0.tgz", + "integrity": "sha512-5tK7EtrZ0N+OLFMthtqOj4fI2Jeb88C4CAZPu25LDVUgXJ0A3Js4PMGqrn0JU1W0Mh1/Z8wZzYPxqUrXeBboCQ==", + "dev": true + }, + "node_modules/@lerna/command/node_modules/cross-spawn": { + "version": "7.0.3", + "resolved": "https://registry.npmmirror.com/cross-spawn/-/cross-spawn-7.0.3.tgz", + "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", + "dev": true, + "dependencies": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@lerna/command/node_modules/execa": { + "version": "5.1.1", + "resolved": "https://registry.npmmirror.com/execa/-/execa-5.1.1.tgz", + "integrity": "sha512-8uSpZZocAZRBAPIEINJj3Lo9HyGitllczc27Eh5YYojjMFMn8yHMDMaUHE2Jqfq05D/wucwI4JGURyXt1vchyg==", + "dev": true, + "dependencies": { + "cross-spawn": "^7.0.3", + "get-stream": "^6.0.0", + "human-signals": "^2.1.0", + "is-stream": "^2.0.0", + "merge-stream": "^2.0.0", + "npm-run-path": "^4.0.1", + "onetime": "^5.1.2", + "signal-exit": "^3.0.3", + "strip-final-newline": "^2.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/command/node_modules/get-stream": { + "version": "6.0.1", + "resolved": "https://registry.npmmirror.com/get-stream/-/get-stream-6.0.1.tgz", + "integrity": "sha512-ts6Wi+2j3jQjqi70w5AlN8DFnkSwC+MqmxEzdEALB2qXZYV3X/b1CTfgPLGJNMeAWxdPfU8FO1ms3NUfaHCPYg==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/command/node_modules/human-signals": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/human-signals/-/human-signals-2.1.0.tgz", + "integrity": "sha512-B4FFZ6q/T2jhhksgkbEW3HBvWIfDW85snkQgawt07S7J5QXTk6BkNV+0yAeZrM5QpMAdYlocGoljn0sJ/WQkFw==", + "dev": true, + "engines": { + "node": ">=10.17.0" + } + }, + "node_modules/@lerna/command/node_modules/is-ci": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/is-ci/-/is-ci-2.0.0.tgz", + "integrity": "sha512-YfJT7rkpQB0updsdHLGWrvhBJfcfzNNawYDNIyQXJz0IViGf75O8EBPKSdvw2rF+LGCsX4FZ8tcr3b19LcZq4w==", + "dev": true, + "dependencies": { + "ci-info": "^2.0.0" + }, + "bin": { + "is-ci": "bin.js" + } + }, + "node_modules/@lerna/command/node_modules/is-stream": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/is-stream/-/is-stream-2.0.1.tgz", + "integrity": "sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/command/node_modules/mimic-fn": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/mimic-fn/-/mimic-fn-2.1.0.tgz", + "integrity": "sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/@lerna/command/node_modules/npm-run-path": { + "version": "4.0.1", + "resolved": "https://registry.npmmirror.com/npm-run-path/-/npm-run-path-4.0.1.tgz", + "integrity": "sha512-S48WzZW777zhNIrn7gxOlISNAqi9ZC/uQFnRdbeIHhZhCA6UqpkOT8T1G7BvfdgP4Er8gF4sUbaS0i7QvIfCWw==", + "dev": true, + "dependencies": { + "path-key": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/command/node_modules/onetime": { + "version": "5.1.2", + "resolved": "https://registry.npmmirror.com/onetime/-/onetime-5.1.2.tgz", + "integrity": "sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==", + "dev": true, + "dependencies": { + "mimic-fn": "^2.1.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/@lerna/command/node_modules/path-key": { + "version": "3.1.1", + "resolved": "https://registry.npmmirror.com/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/command/node_modules/shebang-command": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/shebang-command/-/shebang-command-2.0.0.tgz", + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", + "dev": true, + "dependencies": { + "shebang-regex": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/command/node_modules/shebang-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/shebang-regex/-/shebang-regex-3.0.0.tgz", + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/command/node_modules/which": { + "version": "2.0.2", + "resolved": "https://registry.npmmirror.com/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "node-which": "bin/node-which" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@lerna/conventional-commits": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/conventional-commits/-/conventional-commits-4.0.0.tgz", + "integrity": "sha512-CSUQRjJHFrH8eBn7+wegZLV3OrNc0Y1FehYfYGhjLE2SIfpCL4bmfu/ViYuHh9YjwHaA+4SX6d3hR+xkeseKmw==", + "dev": true, + "dependencies": { + "@lerna/validation-error": "4.0.0", + "conventional-changelog-angular": "^5.0.12", + "conventional-changelog-core": "^4.2.2", + "conventional-recommended-bump": "^6.1.0", + "fs-extra": "^9.1.0", + "get-stream": "^6.0.0", + "lodash.template": "^4.5.0", + "npm-package-arg": "^8.1.0", + "npmlog": "^4.1.2", + "pify": "^5.0.0", + "semver": "^7.3.4" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/conventional-commits/node_modules/fs-extra": { + "version": "9.1.0", + "resolved": "https://registry.npmmirror.com/fs-extra/-/fs-extra-9.1.0.tgz", + "integrity": "sha512-hcg3ZmepS30/7BSFqRvoo3DOMQu7IjqxO5nCDt+zM9XWjb33Wg7ziNT+Qvqbuc3+gWpzO02JubVyk2G4Zvo1OQ==", + "dev": true, + "dependencies": { + "at-least-node": "^1.0.0", + "graceful-fs": "^4.2.0", + "jsonfile": "^6.0.1", + "universalify": "^2.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/conventional-commits/node_modules/get-stream": { + "version": "6.0.1", + "resolved": "https://registry.npmmirror.com/get-stream/-/get-stream-6.0.1.tgz", + "integrity": "sha512-ts6Wi+2j3jQjqi70w5AlN8DFnkSwC+MqmxEzdEALB2qXZYV3X/b1CTfgPLGJNMeAWxdPfU8FO1ms3NUfaHCPYg==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/conventional-commits/node_modules/jsonfile": { + "version": "6.1.0", + "resolved": "https://registry.npmmirror.com/jsonfile/-/jsonfile-6.1.0.tgz", + "integrity": "sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==", + "dev": true, + "dependencies": { + "universalify": "^2.0.0" + }, + "optionalDependencies": { + "graceful-fs": "^4.1.6" + } + }, + "node_modules/@lerna/conventional-commits/node_modules/lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/conventional-commits/node_modules/pify": { + "version": "5.0.0", + "resolved": "https://registry.npmmirror.com/pify/-/pify-5.0.0.tgz", + "integrity": "sha512-eW/gHNMlxdSP6dmG6uJip6FXN0EQBwm2clYYd8Wul42Cwu/DK8HEftzsapcNdYe2MfLiIwZqsDk2RDEsTE79hA==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/conventional-commits/node_modules/semver": { + "version": "7.3.7", + "resolved": "https://registry.npmmirror.com/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/conventional-commits/node_modules/universalify": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/universalify/-/universalify-2.0.0.tgz", + "integrity": "sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ==", + "dev": true, + "engines": { + "node": ">= 10.0.0" + } + }, + "node_modules/@lerna/conventional-commits/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/@lerna/create": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/create/-/create-4.0.0.tgz", + "integrity": "sha512-mVOB1niKByEUfxlbKTM1UNECWAjwUdiioIbRQZEeEabtjCL69r9rscIsjlGyhGWCfsdAG5wfq4t47nlDXdLLag==", + "dev": true, + "dependencies": { + "@lerna/child-process": "4.0.0", + "@lerna/command": "4.0.0", + "@lerna/npm-conf": "4.0.0", + "@lerna/validation-error": "4.0.0", + "dedent": "^0.7.0", + "fs-extra": "^9.1.0", + "globby": "^11.0.2", + "init-package-json": "^2.0.2", + "npm-package-arg": "^8.1.0", + "p-reduce": "^2.1.0", + "pacote": "^11.2.6", + "pify": "^5.0.0", + "semver": "^7.3.4", + "slash": "^3.0.0", + "validate-npm-package-license": "^3.0.4", + "validate-npm-package-name": "^3.0.0", + "whatwg-url": "^8.4.0", + "yargs-parser": "20.2.4" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/create-symlink": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/create-symlink/-/create-symlink-4.0.0.tgz", + "integrity": "sha512-I0phtKJJdafUiDwm7BBlEUOtogmu8+taxq6PtIrxZbllV9hWg59qkpuIsiFp+no7nfRVuaasNYHwNUhDAVQBig==", + "dev": true, + "dependencies": { + "cmd-shim": "^4.1.0", + "fs-extra": "^9.1.0", + "npmlog": "^4.1.2" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/create-symlink/node_modules/fs-extra": { + "version": "9.1.0", + "resolved": "https://registry.npmmirror.com/fs-extra/-/fs-extra-9.1.0.tgz", + "integrity": "sha512-hcg3ZmepS30/7BSFqRvoo3DOMQu7IjqxO5nCDt+zM9XWjb33Wg7ziNT+Qvqbuc3+gWpzO02JubVyk2G4Zvo1OQ==", + "dev": true, + "dependencies": { + "at-least-node": "^1.0.0", + "graceful-fs": "^4.2.0", + "jsonfile": "^6.0.1", + "universalify": "^2.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/create-symlink/node_modules/jsonfile": { + "version": "6.1.0", + "resolved": "https://registry.npmmirror.com/jsonfile/-/jsonfile-6.1.0.tgz", + "integrity": "sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==", + "dev": true, + "dependencies": { + "universalify": "^2.0.0" + }, + "optionalDependencies": { + "graceful-fs": "^4.1.6" + } + }, + "node_modules/@lerna/create-symlink/node_modules/universalify": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/universalify/-/universalify-2.0.0.tgz", + "integrity": "sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ==", + "dev": true, + "engines": { + "node": ">= 10.0.0" + } + }, + "node_modules/@lerna/create/node_modules/@nodelib/fs.stat": { + "version": "2.0.5", + "resolved": "https://registry.npmmirror.com/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz", + "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==", + "dev": true, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@lerna/create/node_modules/array-union": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/array-union/-/array-union-2.1.0.tgz", + "integrity": "sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/create/node_modules/braces": { + "version": "3.0.2", + "resolved": "https://registry.npmmirror.com/braces/-/braces-3.0.2.tgz", + "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", + "dev": true, + "dependencies": { + "fill-range": "^7.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/create/node_modules/dir-glob": { + "version": "3.0.1", + "resolved": "https://registry.npmmirror.com/dir-glob/-/dir-glob-3.0.1.tgz", + "integrity": "sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA==", + "dev": true, + "dependencies": { + "path-type": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/create/node_modules/fast-glob": { + "version": "3.2.11", + "resolved": "https://registry.npmmirror.com/fast-glob/-/fast-glob-3.2.11.tgz", + "integrity": "sha512-xrO3+1bxSo3ZVHAnqzyuewYT6aMFHRAd4Kcs92MAonjwQZLsK9d0SF1IyQ3k5PoirxTW0Oe/RqFgMQ6TcNE5Ew==", + "dev": true, + "dependencies": { + "@nodelib/fs.stat": "^2.0.2", + "@nodelib/fs.walk": "^1.2.3", + "glob-parent": "^5.1.2", + "merge2": "^1.3.0", + "micromatch": "^4.0.4" + }, + "engines": { + "node": ">=8.6.0" + } + }, + "node_modules/@lerna/create/node_modules/fill-range": { + "version": "7.0.1", + "resolved": "https://registry.npmmirror.com/fill-range/-/fill-range-7.0.1.tgz", + "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", + "dev": true, + "dependencies": { + "to-regex-range": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/create/node_modules/fs-extra": { + "version": "9.1.0", + "resolved": "https://registry.npmmirror.com/fs-extra/-/fs-extra-9.1.0.tgz", + "integrity": "sha512-hcg3ZmepS30/7BSFqRvoo3DOMQu7IjqxO5nCDt+zM9XWjb33Wg7ziNT+Qvqbuc3+gWpzO02JubVyk2G4Zvo1OQ==", + "dev": true, + "dependencies": { + "at-least-node": "^1.0.0", + "graceful-fs": "^4.2.0", + "jsonfile": "^6.0.1", + "universalify": "^2.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/create/node_modules/globby": { + "version": "11.1.0", + "resolved": "https://registry.npmmirror.com/globby/-/globby-11.1.0.tgz", + "integrity": "sha512-jhIXaOzy1sb8IyocaruWSn1TjmnBVs8Ayhcy83rmxNJ8q2uWKCAj3CnJY+KpGSXCueAPc0i05kVvVKtP1t9S3g==", + "dev": true, + "dependencies": { + "array-union": "^2.1.0", + "dir-glob": "^3.0.1", + "fast-glob": "^3.2.9", + "ignore": "^5.2.0", + "merge2": "^1.4.1", + "slash": "^3.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/create/node_modules/ignore": { + "version": "5.2.0", + "resolved": "https://registry.npmmirror.com/ignore/-/ignore-5.2.0.tgz", + "integrity": "sha512-CmxgYGiEPCLhfLnpPp1MoRmifwEIOgjcHXxOBjv7mY96c+eWScsOP9c112ZyLdWHi0FxHjI+4uVhKYp/gcdRmQ==", + "dev": true, + "engines": { + "node": ">= 4" + } + }, + "node_modules/@lerna/create/node_modules/is-number": { + "version": "7.0.0", + "resolved": "https://registry.npmmirror.com/is-number/-/is-number-7.0.0.tgz", + "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", + "dev": true, + "engines": { + "node": ">=0.12.0" + } + }, + "node_modules/@lerna/create/node_modules/jsonfile": { + "version": "6.1.0", + "resolved": "https://registry.npmmirror.com/jsonfile/-/jsonfile-6.1.0.tgz", + "integrity": "sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==", + "dev": true, + "dependencies": { + "universalify": "^2.0.0" + }, + "optionalDependencies": { + "graceful-fs": "^4.1.6" + } + }, + "node_modules/@lerna/create/node_modules/lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/create/node_modules/micromatch": { + "version": "4.0.5", + "resolved": "https://registry.npmmirror.com/micromatch/-/micromatch-4.0.5.tgz", + "integrity": "sha512-DMy+ERcEW2q8Z2Po+WNXuw3c5YaUSFjAO5GsJqfEl7UjvtIuFKO6ZrKvcItdy98dwFI2N1tg3zNIdKaQT+aNdA==", + "dev": true, + "dependencies": { + "braces": "^3.0.2", + "picomatch": "^2.3.1" + }, + "engines": { + "node": ">=8.6" + } + }, + "node_modules/@lerna/create/node_modules/path-type": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/path-type/-/path-type-4.0.0.tgz", + "integrity": "sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/create/node_modules/pify": { + "version": "5.0.0", + "resolved": "https://registry.npmmirror.com/pify/-/pify-5.0.0.tgz", + "integrity": "sha512-eW/gHNMlxdSP6dmG6uJip6FXN0EQBwm2clYYd8Wul42Cwu/DK8HEftzsapcNdYe2MfLiIwZqsDk2RDEsTE79hA==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/create/node_modules/semver": { + "version": "7.3.7", + "resolved": "https://registry.npmmirror.com/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/create/node_modules/slash": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/slash/-/slash-3.0.0.tgz", + "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/create/node_modules/to-regex-range": { + "version": "5.0.1", + "resolved": "https://registry.npmmirror.com/to-regex-range/-/to-regex-range-5.0.1.tgz", + "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", + "dev": true, + "dependencies": { + "is-number": "^7.0.0" + }, + "engines": { + "node": ">=8.0" + } + }, + "node_modules/@lerna/create/node_modules/universalify": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/universalify/-/universalify-2.0.0.tgz", + "integrity": "sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ==", + "dev": true, + "engines": { + "node": ">= 10.0.0" + } + }, + "node_modules/@lerna/create/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/@lerna/create/node_modules/yargs-parser": { + "version": "20.2.4", + "resolved": "https://registry.npmmirror.com/yargs-parser/-/yargs-parser-20.2.4.tgz", + "integrity": "sha512-WOkpgNhPTlE73h4VFAFsOnomJVaovO8VqLDzy5saChRBFQFBoMYirowyW+Q9HB4HFF4Z7VZTiG3iSzJJA29yRA==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/describe-ref": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/describe-ref/-/describe-ref-4.0.0.tgz", + "integrity": "sha512-eTU5+xC4C5Gcgz+Ey4Qiw9nV2B4JJbMulsYJMW8QjGcGh8zudib7Sduj6urgZXUYNyhYpRs+teci9M2J8u+UvQ==", + "dev": true, + "dependencies": { + "@lerna/child-process": "4.0.0", + "npmlog": "^4.1.2" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/diff": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/diff/-/diff-4.0.0.tgz", + "integrity": "sha512-jYPKprQVg41+MUMxx6cwtqsNm0Yxx9GDEwdiPLwcUTFx+/qKCEwifKNJ1oGIPBxyEHX2PFCOjkK39lHoj2qiag==", + "dev": true, + "dependencies": { + "@lerna/child-process": "4.0.0", + "@lerna/command": "4.0.0", + "@lerna/validation-error": "4.0.0", + "npmlog": "^4.1.2" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/exec": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/exec/-/exec-4.0.0.tgz", + "integrity": "sha512-VGXtL/b/JfY84NB98VWZpIExfhLOzy0ozm/0XaS4a2SmkAJc5CeUfrhvHxxkxiTBLkU+iVQUyYEoAT0ulQ8PCw==", + "dev": true, + "dependencies": { + "@lerna/child-process": "4.0.0", + "@lerna/command": "4.0.0", + "@lerna/filter-options": "4.0.0", + "@lerna/profiler": "4.0.0", + "@lerna/run-topologically": "4.0.0", + "@lerna/validation-error": "4.0.0", + "p-map": "^4.0.0" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/exec/node_modules/p-map": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/p-map/-/p-map-4.0.0.tgz", + "integrity": "sha512-/bjOqmgETBYB5BoEeGVea8dmvHb2m9GLy1E9W43yeyfP6QQCZGFNa+XRceJEuDB6zqr+gKpIAmlLebMpykw/MQ==", + "dev": true, + "dependencies": { + "aggregate-error": "^3.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/filter-options": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/filter-options/-/filter-options-4.0.0.tgz", + "integrity": "sha512-vV2ANOeZhOqM0rzXnYcFFCJ/kBWy/3OA58irXih9AMTAlQLymWAK0akWybl++sUJ4HB9Hx12TOqaXbYS2NM5uw==", + "dev": true, + "dependencies": { + "@lerna/collect-updates": "4.0.0", + "@lerna/filter-packages": "4.0.0", + "dedent": "^0.7.0", + "npmlog": "^4.1.2" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/filter-packages": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/filter-packages/-/filter-packages-4.0.0.tgz", + "integrity": "sha512-+4AJIkK7iIiOaqCiVTYJxh/I9qikk4XjNQLhE3kixaqgMuHl1NQ99qXRR0OZqAWB9mh8Z1HA9bM5K1HZLBTOqA==", + "dev": true, + "dependencies": { + "@lerna/validation-error": "4.0.0", + "multimatch": "^5.0.0", + "npmlog": "^4.1.2" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/get-npm-exec-opts": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/get-npm-exec-opts/-/get-npm-exec-opts-4.0.0.tgz", + "integrity": "sha512-yvmkerU31CTWS2c7DvmAWmZVeclPBqI7gPVr5VATUKNWJ/zmVcU4PqbYoLu92I9Qc4gY1TuUplMNdNuZTSL7IQ==", + "dev": true, + "dependencies": { + "npmlog": "^4.1.2" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/get-packed": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/get-packed/-/get-packed-4.0.0.tgz", + "integrity": "sha512-rfWONRsEIGyPJTxFzC8ECb3ZbsDXJbfqWYyeeQQDrJRPnEJErlltRLPLgC2QWbxFgFPsoDLeQmFHJnf0iDfd8w==", + "dev": true, + "dependencies": { + "fs-extra": "^9.1.0", + "ssri": "^8.0.1", + "tar": "^6.1.0" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/get-packed/node_modules/chownr": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/chownr/-/chownr-2.0.0.tgz", + "integrity": "sha512-bIomtDF5KGpdogkLd9VspvFzk9KfpyyGlS8YFVZl7TGPBHL5snIOnxeshwVgPteQ9b4Eydl+pVbIyE1DcvCWgQ==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/get-packed/node_modules/fs-extra": { + "version": "9.1.0", + "resolved": "https://registry.npmmirror.com/fs-extra/-/fs-extra-9.1.0.tgz", + "integrity": "sha512-hcg3ZmepS30/7BSFqRvoo3DOMQu7IjqxO5nCDt+zM9XWjb33Wg7ziNT+Qvqbuc3+gWpzO02JubVyk2G4Zvo1OQ==", + "dev": true, + "dependencies": { + "at-least-node": "^1.0.0", + "graceful-fs": "^4.2.0", + "jsonfile": "^6.0.1", + "universalify": "^2.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/get-packed/node_modules/fs-minipass": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/fs-minipass/-/fs-minipass-2.1.0.tgz", + "integrity": "sha512-V/JgOLFCS+R6Vcq0slCuaeWEdNC3ouDlJMNIsacH2VtALiu9mV4LPrHc5cDl8k5aw6J8jwgWWpiTo5RYhmIzvg==", + "dev": true, + "dependencies": { + "minipass": "^3.0.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@lerna/get-packed/node_modules/jsonfile": { + "version": "6.1.0", + "resolved": "https://registry.npmmirror.com/jsonfile/-/jsonfile-6.1.0.tgz", + "integrity": "sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==", + "dev": true, + "dependencies": { + "universalify": "^2.0.0" + }, + "optionalDependencies": { + "graceful-fs": "^4.1.6" + } + }, + "node_modules/@lerna/get-packed/node_modules/minipass": { + "version": "3.1.6", + "resolved": "https://registry.npmmirror.com/minipass/-/minipass-3.1.6.tgz", + "integrity": "sha512-rty5kpw9/z8SX9dmxblFA6edItUmwJgMeYDZRrwlIVN27i8gysGbznJwUggw2V/FVqFSDdWy040ZPS811DYAqQ==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/get-packed/node_modules/minizlib": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/minizlib/-/minizlib-2.1.2.tgz", + "integrity": "sha512-bAxsR8BVfj60DWXHE3u30oHzfl4G7khkSuPW+qvpd7jFRHm7dLxOjUk1EHACJ/hxLY8phGJ0YhYHZo7jil7Qdg==", + "dev": true, + "dependencies": { + "minipass": "^3.0.0", + "yallist": "^4.0.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@lerna/get-packed/node_modules/mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmmirror.com/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true, + "bin": { + "mkdirp": "bin/cmd.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/get-packed/node_modules/ssri": { + "version": "8.0.1", + "resolved": "https://registry.npmmirror.com/ssri/-/ssri-8.0.1.tgz", + "integrity": "sha512-97qShzy1AiyxvPNIkLWoGua7xoQzzPjQ0HAH4B0rWKo7SZ6USuPcrUiAFrws0UH8RrbWmgq3LMTObhPIHbbBeQ==", + "dev": true, + "dependencies": { + "minipass": "^3.1.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@lerna/get-packed/node_modules/tar": { + "version": "6.1.11", + "resolved": "https://registry.npmmirror.com/tar/-/tar-6.1.11.tgz", + "integrity": "sha512-an/KZQzQUkZCkuoAA64hM92X0Urb6VpRhAFllDzz44U2mcD5scmT3zBc4VgVpkugF580+DQn8eAFSyoQt0tznA==", + "dev": true, + "dependencies": { + "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" + }, + "engines": { + "node": ">= 10" + } + }, + "node_modules/@lerna/get-packed/node_modules/universalify": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/universalify/-/universalify-2.0.0.tgz", + "integrity": "sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ==", + "dev": true, + "engines": { + "node": ">= 10.0.0" + } + }, + "node_modules/@lerna/get-packed/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/@lerna/github-client": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/github-client/-/github-client-4.0.0.tgz", + "integrity": "sha512-2jhsldZtTKXYUBnOm23Lb0Fx8G4qfSXF9y7UpyUgWUj+YZYd+cFxSuorwQIgk5P4XXrtVhsUesIsli+BYSThiw==", + "dev": true, + "dependencies": { + "@lerna/child-process": "4.0.0", + "@octokit/plugin-enterprise-rest": "^6.0.1", + "@octokit/rest": "^18.1.0", + "git-url-parse": "^11.4.4", + "npmlog": "^4.1.2" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/gitlab-client": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/gitlab-client/-/gitlab-client-4.0.0.tgz", + "integrity": "sha512-OMUpGSkeDWFf7BxGHlkbb35T7YHqVFCwBPSIR6wRsszY8PAzCYahtH3IaJzEJyUg6vmZsNl0FSr3pdA2skhxqA==", + "dev": true, + "dependencies": { + "node-fetch": "^2.6.1", + "npmlog": "^4.1.2", + "whatwg-url": "^8.4.0" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/global-options": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/global-options/-/global-options-4.0.0.tgz", + "integrity": "sha512-TRMR8afAHxuYBHK7F++Ogop2a82xQjoGna1dvPOY6ltj/pEx59pdgcJfYcynYqMkFIk8bhLJJN9/ndIfX29FTQ==", + "dev": true, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/has-npm-version": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/has-npm-version/-/has-npm-version-4.0.0.tgz", + "integrity": "sha512-LQ3U6XFH8ZmLCsvsgq1zNDqka0Xzjq5ibVN+igAI5ccRWNaUsE/OcmsyMr50xAtNQMYMzmpw5GVLAivT2/YzCg==", + "dev": true, + "dependencies": { + "@lerna/child-process": "4.0.0", + "semver": "^7.3.4" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/has-npm-version/node_modules/lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/has-npm-version/node_modules/semver": { + "version": "7.3.7", + "resolved": "https://registry.npmmirror.com/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/has-npm-version/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/@lerna/import": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/import/-/import-4.0.0.tgz", + "integrity": "sha512-FaIhd+4aiBousKNqC7TX1Uhe97eNKf5/SC7c5WZANVWtC7aBWdmswwDt3usrzCNpj6/Wwr9EtEbYROzxKH8ffg==", + "dev": true, + "dependencies": { + "@lerna/child-process": "4.0.0", + "@lerna/command": "4.0.0", + "@lerna/prompt": "4.0.0", + "@lerna/pulse-till-done": "4.0.0", + "@lerna/validation-error": "4.0.0", + "dedent": "^0.7.0", + "fs-extra": "^9.1.0", + "p-map-series": "^2.1.0" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/import/node_modules/fs-extra": { + "version": "9.1.0", + "resolved": "https://registry.npmmirror.com/fs-extra/-/fs-extra-9.1.0.tgz", + "integrity": "sha512-hcg3ZmepS30/7BSFqRvoo3DOMQu7IjqxO5nCDt+zM9XWjb33Wg7ziNT+Qvqbuc3+gWpzO02JubVyk2G4Zvo1OQ==", + "dev": true, + "dependencies": { + "at-least-node": "^1.0.0", + "graceful-fs": "^4.2.0", + "jsonfile": "^6.0.1", + "universalify": "^2.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/import/node_modules/jsonfile": { + "version": "6.1.0", + "resolved": "https://registry.npmmirror.com/jsonfile/-/jsonfile-6.1.0.tgz", + "integrity": "sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==", + "dev": true, + "dependencies": { + "universalify": "^2.0.0" + }, + "optionalDependencies": { + "graceful-fs": "^4.1.6" + } + }, + "node_modules/@lerna/import/node_modules/universalify": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/universalify/-/universalify-2.0.0.tgz", + "integrity": "sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ==", + "dev": true, + "engines": { + "node": ">= 10.0.0" + } + }, + "node_modules/@lerna/info": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/info/-/info-4.0.0.tgz", + "integrity": "sha512-8Uboa12kaCSZEn4XRfPz5KU9XXoexSPS4oeYGj76s2UQb1O1GdnEyfjyNWoUl1KlJ2i/8nxUskpXIftoFYH0/Q==", + "dev": true, + "dependencies": { + "@lerna/command": "4.0.0", + "@lerna/output": "4.0.0", + "envinfo": "^7.7.4" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/init": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/init/-/init-4.0.0.tgz", + "integrity": "sha512-wY6kygop0BCXupzWj5eLvTUqdR7vIAm0OgyV9WHpMYQGfs1V22jhztt8mtjCloD/O0nEe4tJhdG62XU5aYmPNQ==", + "dev": true, + "dependencies": { + "@lerna/child-process": "4.0.0", + "@lerna/command": "4.0.0", + "fs-extra": "^9.1.0", + "p-map": "^4.0.0", + "write-json-file": "^4.3.0" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/init/node_modules/fs-extra": { + "version": "9.1.0", + "resolved": "https://registry.npmmirror.com/fs-extra/-/fs-extra-9.1.0.tgz", + "integrity": "sha512-hcg3ZmepS30/7BSFqRvoo3DOMQu7IjqxO5nCDt+zM9XWjb33Wg7ziNT+Qvqbuc3+gWpzO02JubVyk2G4Zvo1OQ==", + "dev": true, + "dependencies": { + "at-least-node": "^1.0.0", + "graceful-fs": "^4.2.0", + "jsonfile": "^6.0.1", + "universalify": "^2.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/init/node_modules/jsonfile": { + "version": "6.1.0", + "resolved": "https://registry.npmmirror.com/jsonfile/-/jsonfile-6.1.0.tgz", + "integrity": "sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==", + "dev": true, + "dependencies": { + "universalify": "^2.0.0" + }, + "optionalDependencies": { + "graceful-fs": "^4.1.6" + } + }, + "node_modules/@lerna/init/node_modules/p-map": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/p-map/-/p-map-4.0.0.tgz", + "integrity": "sha512-/bjOqmgETBYB5BoEeGVea8dmvHb2m9GLy1E9W43yeyfP6QQCZGFNa+XRceJEuDB6zqr+gKpIAmlLebMpykw/MQ==", + "dev": true, + "dependencies": { + "aggregate-error": "^3.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/init/node_modules/universalify": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/universalify/-/universalify-2.0.0.tgz", + "integrity": "sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ==", + "dev": true, + "engines": { + "node": ">= 10.0.0" + } + }, + "node_modules/@lerna/link": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/link/-/link-4.0.0.tgz", + "integrity": "sha512-KlvPi7XTAcVOByfaLlOeYOfkkDcd+bejpHMCd1KcArcFTwijOwXOVi24DYomIeHvy6HsX/IUquJ4PPUJIeB4+w==", + "dev": true, + "dependencies": { + "@lerna/command": "4.0.0", + "@lerna/package-graph": "4.0.0", + "@lerna/symlink-dependencies": "4.0.0", + "p-map": "^4.0.0", + "slash": "^3.0.0" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/link/node_modules/p-map": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/p-map/-/p-map-4.0.0.tgz", + "integrity": "sha512-/bjOqmgETBYB5BoEeGVea8dmvHb2m9GLy1E9W43yeyfP6QQCZGFNa+XRceJEuDB6zqr+gKpIAmlLebMpykw/MQ==", + "dev": true, + "dependencies": { + "aggregate-error": "^3.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/link/node_modules/slash": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/slash/-/slash-3.0.0.tgz", + "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/list": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/list/-/list-4.0.0.tgz", + "integrity": "sha512-L2B5m3P+U4Bif5PultR4TI+KtW+SArwq1i75QZ78mRYxPc0U/piau1DbLOmwrdqr99wzM49t0Dlvl6twd7GHFg==", + "dev": true, + "dependencies": { + "@lerna/command": "4.0.0", + "@lerna/filter-options": "4.0.0", + "@lerna/listable": "4.0.0", + "@lerna/output": "4.0.0" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/listable": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/listable/-/listable-4.0.0.tgz", + "integrity": "sha512-/rPOSDKsOHs5/PBLINZOkRIX1joOXUXEtyUs5DHLM8q6/RP668x/1lFhw6Dx7/U+L0+tbkpGtZ1Yt0LewCLgeQ==", + "dev": true, + "dependencies": { + "@lerna/query-graph": "4.0.0", + "chalk": "^4.1.0", + "columnify": "^1.5.4" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/listable/node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmmirror.com/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/listable/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmmirror.com/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/listable/node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/@lerna/listable/node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmmirror.com/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "node_modules/@lerna/listable/node_modules/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/listable/node_modules/supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmmirror.com/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/log-packed": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/log-packed/-/log-packed-4.0.0.tgz", + "integrity": "sha512-+dpCiWbdzgMAtpajLToy9PO713IHoE6GV/aizXycAyA07QlqnkpaBNZ8DW84gHdM1j79TWockGJo9PybVhrrZQ==", + "dev": true, + "dependencies": { + "byte-size": "^7.0.0", + "columnify": "^1.5.4", + "has-unicode": "^2.0.1", + "npmlog": "^4.1.2" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/npm-conf": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/npm-conf/-/npm-conf-4.0.0.tgz", + "integrity": "sha512-uS7H02yQNq3oejgjxAxqq/jhwGEE0W0ntr8vM3EfpCW1F/wZruwQw+7bleJQ9vUBjmdXST//tk8mXzr5+JXCfw==", + "dev": true, + "dependencies": { + "config-chain": "^1.1.12", + "pify": "^5.0.0" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/npm-conf/node_modules/pify": { + "version": "5.0.0", + "resolved": "https://registry.npmmirror.com/pify/-/pify-5.0.0.tgz", + "integrity": "sha512-eW/gHNMlxdSP6dmG6uJip6FXN0EQBwm2clYYd8Wul42Cwu/DK8HEftzsapcNdYe2MfLiIwZqsDk2RDEsTE79hA==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/npm-dist-tag": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/npm-dist-tag/-/npm-dist-tag-4.0.0.tgz", + "integrity": "sha512-F20sg28FMYTgXqEQihgoqSfwmq+Id3zT23CnOwD+XQMPSy9IzyLf1fFVH319vXIw6NF6Pgs4JZN2Qty6/CQXGw==", + "dev": true, + "dependencies": { + "@lerna/otplease": "4.0.0", + "npm-package-arg": "^8.1.0", + "npm-registry-fetch": "^9.0.0", + "npmlog": "^4.1.2" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/npm-dist-tag/node_modules/cacache": { + "version": "15.3.0", + "resolved": "https://registry.npmmirror.com/cacache/-/cacache-15.3.0.tgz", + "integrity": "sha512-VVdYzXEn+cnbXpFgWs5hTT7OScegHVmLhJIR8Ufqk3iFD6A6j5iSX1KuBTfNEv4tdJWE2PzA6IVFtcLC7fN9wQ==", + "dev": true, + "dependencies": { + "@npmcli/fs": "^1.0.0", + "@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.1", + "tar": "^6.0.2", + "unique-filename": "^1.1.1" + }, + "engines": { + "node": ">= 10" + } + }, + "node_modules/@lerna/npm-dist-tag/node_modules/chownr": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/chownr/-/chownr-2.0.0.tgz", + "integrity": "sha512-bIomtDF5KGpdogkLd9VspvFzk9KfpyyGlS8YFVZl7TGPBHL5snIOnxeshwVgPteQ9b4Eydl+pVbIyE1DcvCWgQ==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/npm-dist-tag/node_modules/debug": { + "version": "4.3.4", + "resolved": "https://registry.npmmirror.com/debug/-/debug-4.3.4.tgz", + "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", + "dev": true, + "dependencies": { + "ms": "2.1.2" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/@lerna/npm-dist-tag/node_modules/fs-minipass": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/fs-minipass/-/fs-minipass-2.1.0.tgz", + "integrity": "sha512-V/JgOLFCS+R6Vcq0slCuaeWEdNC3ouDlJMNIsacH2VtALiu9mV4LPrHc5cDl8k5aw6J8jwgWWpiTo5RYhmIzvg==", + "dev": true, + "dependencies": { + "minipass": "^3.0.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@lerna/npm-dist-tag/node_modules/lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/npm-dist-tag/node_modules/make-fetch-happen": { + "version": "8.0.14", + "resolved": "https://registry.npmmirror.com/make-fetch-happen/-/make-fetch-happen-8.0.14.tgz", + "integrity": "sha512-EsS89h6l4vbfJEtBZnENTOFk8mCRpY5ru36Xe5bcX1KYIli2mkSHqoFsp5O1wMDvTJJzxe/4THpCTtygjeeGWQ==", + "dev": true, + "dependencies": { + "agentkeepalive": "^4.1.3", + "cacache": "^15.0.5", + "http-cache-semantics": "^4.1.0", + "http-proxy-agent": "^4.0.1", + "https-proxy-agent": "^5.0.0", + "is-lambda": "^1.0.1", + "lru-cache": "^6.0.0", + "minipass": "^3.1.3", + "minipass-collect": "^1.0.2", + "minipass-fetch": "^1.3.2", + "minipass-flush": "^1.0.5", + "minipass-pipeline": "^1.2.4", + "promise-retry": "^2.0.1", + "socks-proxy-agent": "^5.0.0", + "ssri": "^8.0.0" + }, + "engines": { + "node": ">= 10" + } + }, + "node_modules/@lerna/npm-dist-tag/node_modules/minipass": { + "version": "3.1.6", + "resolved": "https://registry.npmmirror.com/minipass/-/minipass-3.1.6.tgz", + "integrity": "sha512-rty5kpw9/z8SX9dmxblFA6edItUmwJgMeYDZRrwlIVN27i8gysGbznJwUggw2V/FVqFSDdWy040ZPS811DYAqQ==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/npm-dist-tag/node_modules/minizlib": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/minizlib/-/minizlib-2.1.2.tgz", + "integrity": "sha512-bAxsR8BVfj60DWXHE3u30oHzfl4G7khkSuPW+qvpd7jFRHm7dLxOjUk1EHACJ/hxLY8phGJ0YhYHZo7jil7Qdg==", + "dev": true, + "dependencies": { + "minipass": "^3.0.0", + "yallist": "^4.0.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@lerna/npm-dist-tag/node_modules/mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmmirror.com/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true, + "bin": { + "mkdirp": "bin/cmd.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/npm-dist-tag/node_modules/ms": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", + "dev": true + }, + "node_modules/@lerna/npm-dist-tag/node_modules/npm-registry-fetch": { + "version": "9.0.0", + "resolved": "https://registry.npmmirror.com/npm-registry-fetch/-/npm-registry-fetch-9.0.0.tgz", + "integrity": "sha512-PuFYYtnQ8IyVl6ib9d3PepeehcUeHN9IO5N/iCRhyg9tStQcqGQBRVHmfmMWPDERU3KwZoHFvbJ4FPXPspvzbA==", + "dev": true, + "dependencies": { + "@npmcli/ci-detect": "^1.0.0", + "lru-cache": "^6.0.0", + "make-fetch-happen": "^8.0.9", + "minipass": "^3.1.3", + "minipass-fetch": "^1.3.0", + "minipass-json-stream": "^1.0.1", + "minizlib": "^2.0.0", + "npm-package-arg": "^8.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/npm-dist-tag/node_modules/p-map": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/p-map/-/p-map-4.0.0.tgz", + "integrity": "sha512-/bjOqmgETBYB5BoEeGVea8dmvHb2m9GLy1E9W43yeyfP6QQCZGFNa+XRceJEuDB6zqr+gKpIAmlLebMpykw/MQ==", + "dev": true, + "dependencies": { + "aggregate-error": "^3.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/npm-dist-tag/node_modules/rimraf": { + "version": "3.0.2", + "resolved": "https://registry.npmmirror.com/rimraf/-/rimraf-3.0.2.tgz", + "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", + "dev": true, + "dependencies": { + "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" + } + }, + "node_modules/@lerna/npm-dist-tag/node_modules/socks-proxy-agent": { + "version": "5.0.1", + "resolved": "https://registry.npmmirror.com/socks-proxy-agent/-/socks-proxy-agent-5.0.1.tgz", + "integrity": "sha512-vZdmnjb9a2Tz6WEQVIurybSwElwPxMZaIc7PzqbJTrezcKNznv6giT7J7tZDZ1BojVaa1jvO/UiUdhDVB0ACoQ==", + "dev": true, + "dependencies": { + "agent-base": "^6.0.2", + "debug": "4", + "socks": "^2.3.3" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/@lerna/npm-dist-tag/node_modules/ssri": { + "version": "8.0.1", + "resolved": "https://registry.npmmirror.com/ssri/-/ssri-8.0.1.tgz", + "integrity": "sha512-97qShzy1AiyxvPNIkLWoGua7xoQzzPjQ0HAH4B0rWKo7SZ6USuPcrUiAFrws0UH8RrbWmgq3LMTObhPIHbbBeQ==", + "dev": true, + "dependencies": { + "minipass": "^3.1.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@lerna/npm-dist-tag/node_modules/tar": { + "version": "6.1.11", + "resolved": "https://registry.npmmirror.com/tar/-/tar-6.1.11.tgz", + "integrity": "sha512-an/KZQzQUkZCkuoAA64hM92X0Urb6VpRhAFllDzz44U2mcD5scmT3zBc4VgVpkugF580+DQn8eAFSyoQt0tznA==", + "dev": true, + "dependencies": { + "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" + }, + "engines": { + "node": ">= 10" + } + }, + "node_modules/@lerna/npm-dist-tag/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/@lerna/npm-install": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/npm-install/-/npm-install-4.0.0.tgz", + "integrity": "sha512-aKNxq2j3bCH3eXl3Fmu4D54s/YLL9WSwV8W7X2O25r98wzrO38AUN6AB9EtmAx+LV/SP15et7Yueg9vSaanRWg==", + "dev": true, + "dependencies": { + "@lerna/child-process": "4.0.0", + "@lerna/get-npm-exec-opts": "4.0.0", + "fs-extra": "^9.1.0", + "npm-package-arg": "^8.1.0", + "npmlog": "^4.1.2", + "signal-exit": "^3.0.3", + "write-pkg": "^4.0.0" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/npm-install/node_modules/fs-extra": { + "version": "9.1.0", + "resolved": "https://registry.npmmirror.com/fs-extra/-/fs-extra-9.1.0.tgz", + "integrity": "sha512-hcg3ZmepS30/7BSFqRvoo3DOMQu7IjqxO5nCDt+zM9XWjb33Wg7ziNT+Qvqbuc3+gWpzO02JubVyk2G4Zvo1OQ==", + "dev": true, + "dependencies": { + "at-least-node": "^1.0.0", + "graceful-fs": "^4.2.0", + "jsonfile": "^6.0.1", + "universalify": "^2.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/npm-install/node_modules/jsonfile": { + "version": "6.1.0", + "resolved": "https://registry.npmmirror.com/jsonfile/-/jsonfile-6.1.0.tgz", + "integrity": "sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==", + "dev": true, + "dependencies": { + "universalify": "^2.0.0" + }, + "optionalDependencies": { + "graceful-fs": "^4.1.6" + } + }, + "node_modules/@lerna/npm-install/node_modules/universalify": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/universalify/-/universalify-2.0.0.tgz", + "integrity": "sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ==", + "dev": true, + "engines": { + "node": ">= 10.0.0" + } + }, + "node_modules/@lerna/npm-publish": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/npm-publish/-/npm-publish-4.0.0.tgz", + "integrity": "sha512-vQb7yAPRo5G5r77DRjHITc9piR9gvEKWrmfCH7wkfBnGWEqu7n8/4bFQ7lhnkujvc8RXOsYpvbMQkNfkYibD/w==", + "dev": true, + "dependencies": { + "@lerna/otplease": "4.0.0", + "@lerna/run-lifecycle": "4.0.0", + "fs-extra": "^9.1.0", + "libnpmpublish": "^4.0.0", + "npm-package-arg": "^8.1.0", + "npmlog": "^4.1.2", + "pify": "^5.0.0", + "read-package-json": "^3.0.0" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/npm-publish/node_modules/fs-extra": { + "version": "9.1.0", + "resolved": "https://registry.npmmirror.com/fs-extra/-/fs-extra-9.1.0.tgz", + "integrity": "sha512-hcg3ZmepS30/7BSFqRvoo3DOMQu7IjqxO5nCDt+zM9XWjb33Wg7ziNT+Qvqbuc3+gWpzO02JubVyk2G4Zvo1OQ==", + "dev": true, + "dependencies": { + "at-least-node": "^1.0.0", + "graceful-fs": "^4.2.0", + "jsonfile": "^6.0.1", + "universalify": "^2.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/npm-publish/node_modules/hosted-git-info": { + "version": "4.1.0", + "resolved": "https://registry.npmmirror.com/hosted-git-info/-/hosted-git-info-4.1.0.tgz", + "integrity": "sha512-kyCuEOWjJqZuDbRHzL8V93NzQhwIB71oFWSyzVo+KPZI+pnQPPxucdkrOZvkLRnrf5URsQM+IJ09Dw29cRALIA==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/npm-publish/node_modules/jsonfile": { + "version": "6.1.0", + "resolved": "https://registry.npmmirror.com/jsonfile/-/jsonfile-6.1.0.tgz", + "integrity": "sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==", + "dev": true, + "dependencies": { + "universalify": "^2.0.0" + }, + "optionalDependencies": { + "graceful-fs": "^4.1.6" + } + }, + "node_modules/@lerna/npm-publish/node_modules/lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/npm-publish/node_modules/normalize-package-data": { + "version": "3.0.3", + "resolved": "https://registry.npmmirror.com/normalize-package-data/-/normalize-package-data-3.0.3.tgz", + "integrity": "sha512-p2W1sgqij3zMMyRC067Dg16bfzVH+w7hyegmpIvZ4JNjqtGOVAIvLmjBx3yP7YTe9vKJgkoNOPjwQGogDoMXFA==", + "dev": true, + "dependencies": { + "hosted-git-info": "^4.0.1", + "is-core-module": "^2.5.0", + "semver": "^7.3.4", + "validate-npm-package-license": "^3.0.1" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/npm-publish/node_modules/pify": { + "version": "5.0.0", + "resolved": "https://registry.npmmirror.com/pify/-/pify-5.0.0.tgz", + "integrity": "sha512-eW/gHNMlxdSP6dmG6uJip6FXN0EQBwm2clYYd8Wul42Cwu/DK8HEftzsapcNdYe2MfLiIwZqsDk2RDEsTE79hA==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/npm-publish/node_modules/read-package-json": { + "version": "3.0.1", + "resolved": "https://registry.npmmirror.com/read-package-json/-/read-package-json-3.0.1.tgz", + "integrity": "sha512-aLcPqxovhJTVJcsnROuuzQvv6oziQx4zd3JvG0vGCL5MjTONUc4uJ90zCBC6R7W7oUKBNoR/F8pkyfVwlbxqng==", + "dev": true, + "dependencies": { + "glob": "^7.1.1", + "json-parse-even-better-errors": "^2.3.0", + "normalize-package-data": "^3.0.0", + "npm-normalize-package-bin": "^1.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/npm-publish/node_modules/semver": { + "version": "7.3.7", + "resolved": "https://registry.npmmirror.com/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/npm-publish/node_modules/universalify": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/universalify/-/universalify-2.0.0.tgz", + "integrity": "sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ==", + "dev": true, + "engines": { + "node": ">= 10.0.0" + } + }, + "node_modules/@lerna/npm-publish/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/@lerna/npm-run-script": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/npm-run-script/-/npm-run-script-4.0.0.tgz", + "integrity": "sha512-Jmyh9/IwXJjOXqKfIgtxi0bxi1pUeKe5bD3S81tkcy+kyng/GNj9WSqD5ZggoNP2NP//s4CLDAtUYLdP7CU9rA==", + "dev": true, + "dependencies": { + "@lerna/child-process": "4.0.0", + "@lerna/get-npm-exec-opts": "4.0.0", + "npmlog": "^4.1.2" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/otplease": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/otplease/-/otplease-4.0.0.tgz", + "integrity": "sha512-Sgzbqdk1GH4psNiT6hk+BhjOfIr/5KhGBk86CEfHNJTk9BK4aZYyJD4lpDbDdMjIV4g03G7pYoqHzH765T4fxw==", + "dev": true, + "dependencies": { + "@lerna/prompt": "4.0.0" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/output": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/output/-/output-4.0.0.tgz", + "integrity": "sha512-Un1sHtO1AD7buDQrpnaYTi2EG6sLF+KOPEAMxeUYG5qG3khTs2Zgzq5WE3dt2N/bKh7naESt20JjIW6tBELP0w==", + "dev": true, + "dependencies": { + "npmlog": "^4.1.2" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/pack-directory": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/pack-directory/-/pack-directory-4.0.0.tgz", + "integrity": "sha512-NJrmZNmBHS+5aM+T8N6FVbaKFScVqKlQFJNY2k7nsJ/uklNKsLLl6VhTQBPwMTbf6Tf7l6bcKzpy7aePuq9UiQ==", + "dev": true, + "dependencies": { + "@lerna/get-packed": "4.0.0", + "@lerna/package": "4.0.0", + "@lerna/run-lifecycle": "4.0.0", + "npm-packlist": "^2.1.4", + "npmlog": "^4.1.2", + "tar": "^6.1.0", + "temp-write": "^4.0.0" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/pack-directory/node_modules/chownr": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/chownr/-/chownr-2.0.0.tgz", + "integrity": "sha512-bIomtDF5KGpdogkLd9VspvFzk9KfpyyGlS8YFVZl7TGPBHL5snIOnxeshwVgPteQ9b4Eydl+pVbIyE1DcvCWgQ==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/pack-directory/node_modules/fs-minipass": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/fs-minipass/-/fs-minipass-2.1.0.tgz", + "integrity": "sha512-V/JgOLFCS+R6Vcq0slCuaeWEdNC3ouDlJMNIsacH2VtALiu9mV4LPrHc5cDl8k5aw6J8jwgWWpiTo5RYhmIzvg==", + "dev": true, + "dependencies": { + "minipass": "^3.0.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@lerna/pack-directory/node_modules/minipass": { + "version": "3.1.6", + "resolved": "https://registry.npmmirror.com/minipass/-/minipass-3.1.6.tgz", + "integrity": "sha512-rty5kpw9/z8SX9dmxblFA6edItUmwJgMeYDZRrwlIVN27i8gysGbznJwUggw2V/FVqFSDdWy040ZPS811DYAqQ==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/pack-directory/node_modules/minizlib": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/minizlib/-/minizlib-2.1.2.tgz", + "integrity": "sha512-bAxsR8BVfj60DWXHE3u30oHzfl4G7khkSuPW+qvpd7jFRHm7dLxOjUk1EHACJ/hxLY8phGJ0YhYHZo7jil7Qdg==", + "dev": true, + "dependencies": { + "minipass": "^3.0.0", + "yallist": "^4.0.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@lerna/pack-directory/node_modules/mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmmirror.com/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true, + "bin": { + "mkdirp": "bin/cmd.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/pack-directory/node_modules/tar": { + "version": "6.1.11", + "resolved": "https://registry.npmmirror.com/tar/-/tar-6.1.11.tgz", + "integrity": "sha512-an/KZQzQUkZCkuoAA64hM92X0Urb6VpRhAFllDzz44U2mcD5scmT3zBc4VgVpkugF580+DQn8eAFSyoQt0tznA==", + "dev": true, + "dependencies": { + "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" + }, + "engines": { + "node": ">= 10" + } + }, + "node_modules/@lerna/pack-directory/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/@lerna/package": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/package/-/package-4.0.0.tgz", + "integrity": "sha512-l0M/izok6FlyyitxiQKr+gZLVFnvxRQdNhzmQ6nRnN9dvBJWn+IxxpM+cLqGACatTnyo9LDzNTOj2Db3+s0s8Q==", + "dev": true, + "dependencies": { + "load-json-file": "^6.2.0", + "npm-package-arg": "^8.1.0", + "write-pkg": "^4.0.0" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/package-graph": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/package-graph/-/package-graph-4.0.0.tgz", + "integrity": "sha512-QED2ZCTkfXMKFoTGoccwUzjHtZMSf3UKX14A4/kYyBms9xfFsesCZ6SLI5YeySEgcul8iuIWfQFZqRw+Qrjraw==", + "dev": true, + "dependencies": { + "@lerna/prerelease-id-from-version": "4.0.0", + "@lerna/validation-error": "4.0.0", + "npm-package-arg": "^8.1.0", + "npmlog": "^4.1.2", + "semver": "^7.3.4" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/package-graph/node_modules/lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/package-graph/node_modules/semver": { + "version": "7.3.7", + "resolved": "https://registry.npmmirror.com/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/package-graph/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/@lerna/prerelease-id-from-version": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/prerelease-id-from-version/-/prerelease-id-from-version-4.0.0.tgz", + "integrity": "sha512-GQqguzETdsYRxOSmdFZ6zDBXDErIETWOqomLERRY54f4p+tk4aJjoVdd9xKwehC9TBfIFvlRbL1V9uQGHh1opg==", + "dev": true, + "dependencies": { + "semver": "^7.3.4" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/prerelease-id-from-version/node_modules/lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/prerelease-id-from-version/node_modules/semver": { + "version": "7.3.7", + "resolved": "https://registry.npmmirror.com/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/prerelease-id-from-version/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/@lerna/profiler": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/profiler/-/profiler-4.0.0.tgz", + "integrity": "sha512-/BaEbqnVh1LgW/+qz8wCuI+obzi5/vRE8nlhjPzdEzdmWmZXuCKyWSEzAyHOJWw1ntwMiww5dZHhFQABuoFz9Q==", + "dev": true, + "dependencies": { + "fs-extra": "^9.1.0", + "npmlog": "^4.1.2", + "upath": "^2.0.1" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/profiler/node_modules/fs-extra": { + "version": "9.1.0", + "resolved": "https://registry.npmmirror.com/fs-extra/-/fs-extra-9.1.0.tgz", + "integrity": "sha512-hcg3ZmepS30/7BSFqRvoo3DOMQu7IjqxO5nCDt+zM9XWjb33Wg7ziNT+Qvqbuc3+gWpzO02JubVyk2G4Zvo1OQ==", + "dev": true, + "dependencies": { + "at-least-node": "^1.0.0", + "graceful-fs": "^4.2.0", + "jsonfile": "^6.0.1", + "universalify": "^2.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/profiler/node_modules/jsonfile": { + "version": "6.1.0", + "resolved": "https://registry.npmmirror.com/jsonfile/-/jsonfile-6.1.0.tgz", + "integrity": "sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==", + "dev": true, + "dependencies": { + "universalify": "^2.0.0" + }, + "optionalDependencies": { + "graceful-fs": "^4.1.6" + } + }, + "node_modules/@lerna/profiler/node_modules/universalify": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/universalify/-/universalify-2.0.0.tgz", + "integrity": "sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ==", + "dev": true, + "engines": { + "node": ">= 10.0.0" + } + }, + "node_modules/@lerna/profiler/node_modules/upath": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/upath/-/upath-2.0.1.tgz", + "integrity": "sha512-1uEe95xksV1O0CYKXo8vQvN1JEbtJp7lb7C5U9HMsIp6IVwntkH/oNUzyVNQSd4S1sYk2FpSSW44FqMc8qee5w==", + "dev": true, + "engines": { + "node": ">=4", + "yarn": "*" + } + }, + "node_modules/@lerna/project": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/project/-/project-4.0.0.tgz", + "integrity": "sha512-o0MlVbDkD5qRPkFKlBZsXZjoNTWPyuL58564nSfZJ6JYNmgAptnWPB2dQlAc7HWRZkmnC2fCkEdoU+jioPavbg==", + "dev": true, + "dependencies": { + "@lerna/package": "4.0.0", + "@lerna/validation-error": "4.0.0", + "cosmiconfig": "^7.0.0", + "dedent": "^0.7.0", + "dot-prop": "^6.0.1", + "glob-parent": "^5.1.1", + "globby": "^11.0.2", + "load-json-file": "^6.2.0", + "npmlog": "^4.1.2", + "p-map": "^4.0.0", + "resolve-from": "^5.0.0", + "write-json-file": "^4.3.0" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/project/node_modules/@nodelib/fs.stat": { + "version": "2.0.5", + "resolved": "https://registry.npmmirror.com/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz", + "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==", + "dev": true, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@lerna/project/node_modules/array-union": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/array-union/-/array-union-2.1.0.tgz", + "integrity": "sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/project/node_modules/braces": { + "version": "3.0.2", + "resolved": "https://registry.npmmirror.com/braces/-/braces-3.0.2.tgz", + "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", + "dev": true, + "dependencies": { + "fill-range": "^7.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/project/node_modules/cosmiconfig": { + "version": "7.0.1", + "resolved": "https://registry.npmmirror.com/cosmiconfig/-/cosmiconfig-7.0.1.tgz", + "integrity": "sha512-a1YWNUV2HwGimB7dU2s1wUMurNKjpx60HxBB6xUM8Re+2s1g1IIfJvFR0/iCF+XHdE0GMTKTuLR32UQff4TEyQ==", + "dev": true, + "dependencies": { + "@types/parse-json": "^4.0.0", + "import-fresh": "^3.2.1", + "parse-json": "^5.0.0", + "path-type": "^4.0.0", + "yaml": "^1.10.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/project/node_modules/dir-glob": { + "version": "3.0.1", + "resolved": "https://registry.npmmirror.com/dir-glob/-/dir-glob-3.0.1.tgz", + "integrity": "sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA==", + "dev": true, + "dependencies": { + "path-type": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/project/node_modules/dot-prop": { + "version": "6.0.1", + "resolved": "https://registry.npmmirror.com/dot-prop/-/dot-prop-6.0.1.tgz", + "integrity": "sha512-tE7ztYzXHIeyvc7N+hR3oi7FIbf/NIjVP9hmAt3yMXzrQ072/fpjGLx2GxNxGxUl5V73MEqYzioOMoVhGMJ5cA==", + "dev": true, + "dependencies": { + "is-obj": "^2.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/project/node_modules/fast-glob": { + "version": "3.2.11", + "resolved": "https://registry.npmmirror.com/fast-glob/-/fast-glob-3.2.11.tgz", + "integrity": "sha512-xrO3+1bxSo3ZVHAnqzyuewYT6aMFHRAd4Kcs92MAonjwQZLsK9d0SF1IyQ3k5PoirxTW0Oe/RqFgMQ6TcNE5Ew==", + "dev": true, + "dependencies": { + "@nodelib/fs.stat": "^2.0.2", + "@nodelib/fs.walk": "^1.2.3", + "glob-parent": "^5.1.2", + "merge2": "^1.3.0", + "micromatch": "^4.0.4" + }, + "engines": { + "node": ">=8.6.0" + } + }, + "node_modules/@lerna/project/node_modules/fill-range": { + "version": "7.0.1", + "resolved": "https://registry.npmmirror.com/fill-range/-/fill-range-7.0.1.tgz", + "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", + "dev": true, + "dependencies": { + "to-regex-range": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/project/node_modules/globby": { + "version": "11.1.0", + "resolved": "https://registry.npmmirror.com/globby/-/globby-11.1.0.tgz", + "integrity": "sha512-jhIXaOzy1sb8IyocaruWSn1TjmnBVs8Ayhcy83rmxNJ8q2uWKCAj3CnJY+KpGSXCueAPc0i05kVvVKtP1t9S3g==", + "dev": true, + "dependencies": { + "array-union": "^2.1.0", + "dir-glob": "^3.0.1", + "fast-glob": "^3.2.9", + "ignore": "^5.2.0", + "merge2": "^1.4.1", + "slash": "^3.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/project/node_modules/ignore": { + "version": "5.2.0", + "resolved": "https://registry.npmmirror.com/ignore/-/ignore-5.2.0.tgz", + "integrity": "sha512-CmxgYGiEPCLhfLnpPp1MoRmifwEIOgjcHXxOBjv7mY96c+eWScsOP9c112ZyLdWHi0FxHjI+4uVhKYp/gcdRmQ==", + "dev": true, + "engines": { + "node": ">= 4" + } + }, + "node_modules/@lerna/project/node_modules/import-fresh": { + "version": "3.3.0", + "resolved": "https://registry.npmmirror.com/import-fresh/-/import-fresh-3.3.0.tgz", + "integrity": "sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==", + "dev": true, + "dependencies": { + "parent-module": "^1.0.0", + "resolve-from": "^4.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/@lerna/project/node_modules/import-fresh/node_modules/resolve-from": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/resolve-from/-/resolve-from-4.0.0.tgz", + "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/@lerna/project/node_modules/is-number": { + "version": "7.0.0", + "resolved": "https://registry.npmmirror.com/is-number/-/is-number-7.0.0.tgz", + "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", + "dev": true, + "engines": { + "node": ">=0.12.0" + } + }, + "node_modules/@lerna/project/node_modules/micromatch": { + "version": "4.0.5", + "resolved": "https://registry.npmmirror.com/micromatch/-/micromatch-4.0.5.tgz", + "integrity": "sha512-DMy+ERcEW2q8Z2Po+WNXuw3c5YaUSFjAO5GsJqfEl7UjvtIuFKO6ZrKvcItdy98dwFI2N1tg3zNIdKaQT+aNdA==", + "dev": true, + "dependencies": { + "braces": "^3.0.2", + "picomatch": "^2.3.1" + }, + "engines": { + "node": ">=8.6" + } + }, + "node_modules/@lerna/project/node_modules/p-map": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/p-map/-/p-map-4.0.0.tgz", + "integrity": "sha512-/bjOqmgETBYB5BoEeGVea8dmvHb2m9GLy1E9W43yeyfP6QQCZGFNa+XRceJEuDB6zqr+gKpIAmlLebMpykw/MQ==", + "dev": true, + "dependencies": { + "aggregate-error": "^3.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/project/node_modules/parse-json": { + "version": "5.2.0", + "resolved": "https://registry.npmmirror.com/parse-json/-/parse-json-5.2.0.tgz", + "integrity": "sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==", + "dev": true, + "dependencies": { + "@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" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/project/node_modules/path-type": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/path-type/-/path-type-4.0.0.tgz", + "integrity": "sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/project/node_modules/resolve-from": { + "version": "5.0.0", + "resolved": "https://registry.npmmirror.com/resolve-from/-/resolve-from-5.0.0.tgz", + "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/project/node_modules/slash": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/slash/-/slash-3.0.0.tgz", + "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/project/node_modules/to-regex-range": { + "version": "5.0.1", + "resolved": "https://registry.npmmirror.com/to-regex-range/-/to-regex-range-5.0.1.tgz", + "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", + "dev": true, + "dependencies": { + "is-number": "^7.0.0" + }, + "engines": { + "node": ">=8.0" + } + }, + "node_modules/@lerna/prompt": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/prompt/-/prompt-4.0.0.tgz", + "integrity": "sha512-4Ig46oCH1TH5M7YyTt53fT6TuaKMgqUUaqdgxvp6HP6jtdak6+amcsqB8YGz2eQnw/sdxunx84DfI9XpoLj4bQ==", + "dev": true, + "dependencies": { + "inquirer": "^7.3.3", + "npmlog": "^4.1.2" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/prompt/node_modules/ansi-escapes": { + "version": "4.3.2", + "resolved": "https://registry.npmmirror.com/ansi-escapes/-/ansi-escapes-4.3.2.tgz", + "integrity": "sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ==", + "dev": true, + "dependencies": { + "type-fest": "^0.21.3" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/prompt/node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmmirror.com/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/prompt/node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmmirror.com/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/prompt/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmmirror.com/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/prompt/node_modules/chardet": { + "version": "0.7.0", + "resolved": "https://registry.npmmirror.com/chardet/-/chardet-0.7.0.tgz", + "integrity": "sha512-mT8iDcrh03qDGRRmoA2hmBJnxpllMR+0/0qlzjqZES6NdiWDcZkCNAk4rPFZ9Q85r27unkiNNg8ZOiwZXBHwcA==", + "dev": true + }, + "node_modules/@lerna/prompt/node_modules/cli-cursor": { + "version": "3.1.0", + "resolved": "https://registry.npmmirror.com/cli-cursor/-/cli-cursor-3.1.0.tgz", + "integrity": "sha512-I/zHAwsKf9FqGoXM4WWRACob9+SNukZTd94DWF57E4toouRulbCxcUh6RKUEOQlYTHJnzkPMySvPNaaSLNfLZw==", + "dev": true, + "dependencies": { + "restore-cursor": "^3.1.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/prompt/node_modules/cli-width": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/cli-width/-/cli-width-3.0.0.tgz", + "integrity": "sha512-FxqpkPPwu1HjuN93Omfm4h8uIanXofW0RxVEW3k5RKx+mJJYSthzNhp32Kzxxy3YAEZ/Dc/EWN1vZRY0+kOhbw==", + "dev": true, + "engines": { + "node": ">= 10" + } + }, + "node_modules/@lerna/prompt/node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/@lerna/prompt/node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmmirror.com/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "node_modules/@lerna/prompt/node_modules/external-editor": { + "version": "3.1.0", + "resolved": "https://registry.npmmirror.com/external-editor/-/external-editor-3.1.0.tgz", + "integrity": "sha512-hMQ4CX1p1izmuLYyZqLMO/qGNw10wSv9QDCPfzXfyFrOaCSSoRfqE1Kf1s5an66J5JZC62NewG+mK49jOCtQew==", + "dev": true, + "dependencies": { + "chardet": "^0.7.0", + "iconv-lite": "^0.4.24", + "tmp": "^0.0.33" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/@lerna/prompt/node_modules/figures": { + "version": "3.2.0", + "resolved": "https://registry.npmmirror.com/figures/-/figures-3.2.0.tgz", + "integrity": "sha512-yaduQFRKLXYOGgEn6AZau90j3ggSOyiqXU0F9JZfeXYhNa+Jk4X+s45A2zg5jns87GAFa34BBm2kXw4XpNcbdg==", + "dev": true, + "dependencies": { + "escape-string-regexp": "^1.0.5" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/prompt/node_modules/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/prompt/node_modules/inquirer": { + "version": "7.3.3", + "resolved": "https://registry.npmmirror.com/inquirer/-/inquirer-7.3.3.tgz", + "integrity": "sha512-JG3eIAj5V9CwcGvuOmoo6LB9kbAYT8HXffUl6memuszlwDC/qvFAJw49XJ5NROSFNPxp3iQg1GqkFhaY/CR0IA==", + "dev": true, + "dependencies": { + "ansi-escapes": "^4.2.1", + "chalk": "^4.1.0", + "cli-cursor": "^3.1.0", + "cli-width": "^3.0.0", + "external-editor": "^3.0.3", + "figures": "^3.0.0", + "lodash": "^4.17.19", + "mute-stream": "0.0.8", + "run-async": "^2.4.0", + "rxjs": "^6.6.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0", + "through": "^2.3.6" + }, + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/@lerna/prompt/node_modules/is-fullwidth-code-point": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", + "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/prompt/node_modules/mimic-fn": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/mimic-fn/-/mimic-fn-2.1.0.tgz", + "integrity": "sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/@lerna/prompt/node_modules/mute-stream": { + "version": "0.0.8", + "resolved": "https://registry.npmmirror.com/mute-stream/-/mute-stream-0.0.8.tgz", + "integrity": "sha512-nnbWWOkoWyUsTjKrhgD0dcz22mdkSnpYqbEjIm2nhwhuxlSkpywJmBo8h0ZqJdkp73mb90SssHkN4rsRaBAfAA==", + "dev": true + }, + "node_modules/@lerna/prompt/node_modules/onetime": { + "version": "5.1.2", + "resolved": "https://registry.npmmirror.com/onetime/-/onetime-5.1.2.tgz", + "integrity": "sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==", + "dev": true, + "dependencies": { + "mimic-fn": "^2.1.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/@lerna/prompt/node_modules/restore-cursor": { + "version": "3.1.0", + "resolved": "https://registry.npmmirror.com/restore-cursor/-/restore-cursor-3.1.0.tgz", + "integrity": "sha512-l+sSefzHpj5qimhFSE5a8nufZYAM3sBSVMAPtYkmC+4EH2anSGaEMXSD0izRQbu9nfyQ9y5JrVmp7E8oZrUjvA==", + "dev": true, + "dependencies": { + "onetime": "^5.1.0", + "signal-exit": "^3.0.2" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/prompt/node_modules/string-width": { + "version": "4.2.3", + "resolved": "https://registry.npmmirror.com/string-width/-/string-width-4.2.3.tgz", + "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", + "dev": true, + "dependencies": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/prompt/node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmmirror.com/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/prompt/node_modules/supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmmirror.com/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/prompt/node_modules/type-fest": { + "version": "0.21.3", + "resolved": "https://registry.npmmirror.com/type-fest/-/type-fest-0.21.3.tgz", + "integrity": "sha512-t0rzBq87m3fVcduHDUFhKmyyX+9eo6WQjZvf51Ea/M0Q7+T374Jp1aUiyUl0GKxp8M/OETVHSDvmkyPgvX+X2w==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/publish": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/publish/-/publish-4.0.0.tgz", + "integrity": "sha512-K8jpqjHrChH22qtkytA5GRKIVFEtqBF6JWj1I8dWZtHs4Jywn8yB1jQ3BAMLhqmDJjWJtRck0KXhQQKzDK2UPg==", + "dev": true, + "dependencies": { + "@lerna/check-working-tree": "4.0.0", + "@lerna/child-process": "4.0.0", + "@lerna/collect-updates": "4.0.0", + "@lerna/command": "4.0.0", + "@lerna/describe-ref": "4.0.0", + "@lerna/log-packed": "4.0.0", + "@lerna/npm-conf": "4.0.0", + "@lerna/npm-dist-tag": "4.0.0", + "@lerna/npm-publish": "4.0.0", + "@lerna/otplease": "4.0.0", + "@lerna/output": "4.0.0", + "@lerna/pack-directory": "4.0.0", + "@lerna/prerelease-id-from-version": "4.0.0", + "@lerna/prompt": "4.0.0", + "@lerna/pulse-till-done": "4.0.0", + "@lerna/run-lifecycle": "4.0.0", + "@lerna/run-topologically": "4.0.0", + "@lerna/validation-error": "4.0.0", + "@lerna/version": "4.0.0", + "fs-extra": "^9.1.0", + "libnpmaccess": "^4.0.1", + "npm-package-arg": "^8.1.0", + "npm-registry-fetch": "^9.0.0", + "npmlog": "^4.1.2", + "p-map": "^4.0.0", + "p-pipe": "^3.1.0", + "pacote": "^11.2.6", + "semver": "^7.3.4" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/publish/node_modules/cacache": { + "version": "15.3.0", + "resolved": "https://registry.npmmirror.com/cacache/-/cacache-15.3.0.tgz", + "integrity": "sha512-VVdYzXEn+cnbXpFgWs5hTT7OScegHVmLhJIR8Ufqk3iFD6A6j5iSX1KuBTfNEv4tdJWE2PzA6IVFtcLC7fN9wQ==", + "dev": true, + "dependencies": { + "@npmcli/fs": "^1.0.0", + "@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.1", + "tar": "^6.0.2", + "unique-filename": "^1.1.1" + }, + "engines": { + "node": ">= 10" + } + }, + "node_modules/@lerna/publish/node_modules/chownr": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/chownr/-/chownr-2.0.0.tgz", + "integrity": "sha512-bIomtDF5KGpdogkLd9VspvFzk9KfpyyGlS8YFVZl7TGPBHL5snIOnxeshwVgPteQ9b4Eydl+pVbIyE1DcvCWgQ==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/publish/node_modules/debug": { + "version": "4.3.4", + "resolved": "https://registry.npmmirror.com/debug/-/debug-4.3.4.tgz", + "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", + "dev": true, + "dependencies": { + "ms": "2.1.2" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/@lerna/publish/node_modules/fs-extra": { + "version": "9.1.0", + "resolved": "https://registry.npmmirror.com/fs-extra/-/fs-extra-9.1.0.tgz", + "integrity": "sha512-hcg3ZmepS30/7BSFqRvoo3DOMQu7IjqxO5nCDt+zM9XWjb33Wg7ziNT+Qvqbuc3+gWpzO02JubVyk2G4Zvo1OQ==", + "dev": true, + "dependencies": { + "at-least-node": "^1.0.0", + "graceful-fs": "^4.2.0", + "jsonfile": "^6.0.1", + "universalify": "^2.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/publish/node_modules/fs-minipass": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/fs-minipass/-/fs-minipass-2.1.0.tgz", + "integrity": "sha512-V/JgOLFCS+R6Vcq0slCuaeWEdNC3ouDlJMNIsacH2VtALiu9mV4LPrHc5cDl8k5aw6J8jwgWWpiTo5RYhmIzvg==", + "dev": true, + "dependencies": { + "minipass": "^3.0.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@lerna/publish/node_modules/jsonfile": { + "version": "6.1.0", + "resolved": "https://registry.npmmirror.com/jsonfile/-/jsonfile-6.1.0.tgz", + "integrity": "sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==", + "dev": true, + "dependencies": { + "universalify": "^2.0.0" + }, + "optionalDependencies": { + "graceful-fs": "^4.1.6" + } + }, + "node_modules/@lerna/publish/node_modules/lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/publish/node_modules/make-fetch-happen": { + "version": "8.0.14", + "resolved": "https://registry.npmmirror.com/make-fetch-happen/-/make-fetch-happen-8.0.14.tgz", + "integrity": "sha512-EsS89h6l4vbfJEtBZnENTOFk8mCRpY5ru36Xe5bcX1KYIli2mkSHqoFsp5O1wMDvTJJzxe/4THpCTtygjeeGWQ==", + "dev": true, + "dependencies": { + "agentkeepalive": "^4.1.3", + "cacache": "^15.0.5", + "http-cache-semantics": "^4.1.0", + "http-proxy-agent": "^4.0.1", + "https-proxy-agent": "^5.0.0", + "is-lambda": "^1.0.1", + "lru-cache": "^6.0.0", + "minipass": "^3.1.3", + "minipass-collect": "^1.0.2", + "minipass-fetch": "^1.3.2", + "minipass-flush": "^1.0.5", + "minipass-pipeline": "^1.2.4", + "promise-retry": "^2.0.1", + "socks-proxy-agent": "^5.0.0", + "ssri": "^8.0.0" + }, + "engines": { + "node": ">= 10" + } + }, + "node_modules/@lerna/publish/node_modules/minipass": { + "version": "3.1.6", + "resolved": "https://registry.npmmirror.com/minipass/-/minipass-3.1.6.tgz", + "integrity": "sha512-rty5kpw9/z8SX9dmxblFA6edItUmwJgMeYDZRrwlIVN27i8gysGbznJwUggw2V/FVqFSDdWy040ZPS811DYAqQ==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/publish/node_modules/minizlib": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/minizlib/-/minizlib-2.1.2.tgz", + "integrity": "sha512-bAxsR8BVfj60DWXHE3u30oHzfl4G7khkSuPW+qvpd7jFRHm7dLxOjUk1EHACJ/hxLY8phGJ0YhYHZo7jil7Qdg==", + "dev": true, + "dependencies": { + "minipass": "^3.0.0", + "yallist": "^4.0.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@lerna/publish/node_modules/mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmmirror.com/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true, + "bin": { + "mkdirp": "bin/cmd.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/publish/node_modules/ms": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", + "dev": true + }, + "node_modules/@lerna/publish/node_modules/npm-registry-fetch": { + "version": "9.0.0", + "resolved": "https://registry.npmmirror.com/npm-registry-fetch/-/npm-registry-fetch-9.0.0.tgz", + "integrity": "sha512-PuFYYtnQ8IyVl6ib9d3PepeehcUeHN9IO5N/iCRhyg9tStQcqGQBRVHmfmMWPDERU3KwZoHFvbJ4FPXPspvzbA==", + "dev": true, + "dependencies": { + "@npmcli/ci-detect": "^1.0.0", + "lru-cache": "^6.0.0", + "make-fetch-happen": "^8.0.9", + "minipass": "^3.1.3", + "minipass-fetch": "^1.3.0", + "minipass-json-stream": "^1.0.1", + "minizlib": "^2.0.0", + "npm-package-arg": "^8.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/publish/node_modules/p-map": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/p-map/-/p-map-4.0.0.tgz", + "integrity": "sha512-/bjOqmgETBYB5BoEeGVea8dmvHb2m9GLy1E9W43yeyfP6QQCZGFNa+XRceJEuDB6zqr+gKpIAmlLebMpykw/MQ==", + "dev": true, + "dependencies": { + "aggregate-error": "^3.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/publish/node_modules/rimraf": { + "version": "3.0.2", + "resolved": "https://registry.npmmirror.com/rimraf/-/rimraf-3.0.2.tgz", + "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", + "dev": true, + "dependencies": { + "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" + } + }, + "node_modules/@lerna/publish/node_modules/semver": { + "version": "7.3.7", + "resolved": "https://registry.npmmirror.com/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/publish/node_modules/socks-proxy-agent": { + "version": "5.0.1", + "resolved": "https://registry.npmmirror.com/socks-proxy-agent/-/socks-proxy-agent-5.0.1.tgz", + "integrity": "sha512-vZdmnjb9a2Tz6WEQVIurybSwElwPxMZaIc7PzqbJTrezcKNznv6giT7J7tZDZ1BojVaa1jvO/UiUdhDVB0ACoQ==", + "dev": true, + "dependencies": { + "agent-base": "^6.0.2", + "debug": "4", + "socks": "^2.3.3" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/@lerna/publish/node_modules/ssri": { + "version": "8.0.1", + "resolved": "https://registry.npmmirror.com/ssri/-/ssri-8.0.1.tgz", + "integrity": "sha512-97qShzy1AiyxvPNIkLWoGua7xoQzzPjQ0HAH4B0rWKo7SZ6USuPcrUiAFrws0UH8RrbWmgq3LMTObhPIHbbBeQ==", + "dev": true, + "dependencies": { + "minipass": "^3.1.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@lerna/publish/node_modules/tar": { + "version": "6.1.11", + "resolved": "https://registry.npmmirror.com/tar/-/tar-6.1.11.tgz", + "integrity": "sha512-an/KZQzQUkZCkuoAA64hM92X0Urb6VpRhAFllDzz44U2mcD5scmT3zBc4VgVpkugF580+DQn8eAFSyoQt0tznA==", + "dev": true, + "dependencies": { + "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" + }, + "engines": { + "node": ">= 10" + } + }, + "node_modules/@lerna/publish/node_modules/universalify": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/universalify/-/universalify-2.0.0.tgz", + "integrity": "sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ==", + "dev": true, + "engines": { + "node": ">= 10.0.0" + } + }, + "node_modules/@lerna/publish/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/@lerna/pulse-till-done": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/pulse-till-done/-/pulse-till-done-4.0.0.tgz", + "integrity": "sha512-Frb4F7QGckaybRhbF7aosLsJ5e9WuH7h0KUkjlzSByVycxY91UZgaEIVjS2oN9wQLrheLMHl6SiFY0/Pvo0Cxg==", + "dev": true, + "dependencies": { + "npmlog": "^4.1.2" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/query-graph": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/query-graph/-/query-graph-4.0.0.tgz", + "integrity": "sha512-YlP6yI3tM4WbBmL9GCmNDoeQyzcyg1e4W96y/PKMZa5GbyUvkS2+Jc2kwPD+5KcXou3wQZxSPzR3Te5OenaDdg==", + "dev": true, + "dependencies": { + "@lerna/package-graph": "4.0.0" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/resolve-symlink": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/resolve-symlink/-/resolve-symlink-4.0.0.tgz", + "integrity": "sha512-RtX8VEUzqT+uLSCohx8zgmjc6zjyRlh6i/helxtZTMmc4+6O4FS9q5LJas2uGO2wKvBlhcD6siibGt7dIC3xZA==", + "dev": true, + "dependencies": { + "fs-extra": "^9.1.0", + "npmlog": "^4.1.2", + "read-cmd-shim": "^2.0.0" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/resolve-symlink/node_modules/fs-extra": { + "version": "9.1.0", + "resolved": "https://registry.npmmirror.com/fs-extra/-/fs-extra-9.1.0.tgz", + "integrity": "sha512-hcg3ZmepS30/7BSFqRvoo3DOMQu7IjqxO5nCDt+zM9XWjb33Wg7ziNT+Qvqbuc3+gWpzO02JubVyk2G4Zvo1OQ==", + "dev": true, + "dependencies": { + "at-least-node": "^1.0.0", + "graceful-fs": "^4.2.0", + "jsonfile": "^6.0.1", + "universalify": "^2.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/resolve-symlink/node_modules/jsonfile": { + "version": "6.1.0", + "resolved": "https://registry.npmmirror.com/jsonfile/-/jsonfile-6.1.0.tgz", + "integrity": "sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==", + "dev": true, + "dependencies": { + "universalify": "^2.0.0" + }, + "optionalDependencies": { + "graceful-fs": "^4.1.6" + } + }, + "node_modules/@lerna/resolve-symlink/node_modules/universalify": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/universalify/-/universalify-2.0.0.tgz", + "integrity": "sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ==", + "dev": true, + "engines": { + "node": ">= 10.0.0" + } + }, + "node_modules/@lerna/rimraf-dir": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/rimraf-dir/-/rimraf-dir-4.0.0.tgz", + "integrity": "sha512-QNH9ABWk9mcMJh2/muD9iYWBk1oQd40y6oH+f3wwmVGKYU5YJD//+zMiBI13jxZRtwBx0vmBZzkBkK1dR11cBg==", + "dev": true, + "dependencies": { + "@lerna/child-process": "4.0.0", + "npmlog": "^4.1.2", + "path-exists": "^4.0.0", + "rimraf": "^3.0.2" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/rimraf-dir/node_modules/path-exists": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/path-exists/-/path-exists-4.0.0.tgz", + "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/rimraf-dir/node_modules/rimraf": { + "version": "3.0.2", + "resolved": "https://registry.npmmirror.com/rimraf/-/rimraf-3.0.2.tgz", + "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", + "dev": true, + "dependencies": { + "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" + } + }, + "node_modules/@lerna/run": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/run/-/run-4.0.0.tgz", + "integrity": "sha512-9giulCOzlMPzcZS/6Eov6pxE9gNTyaXk0Man+iCIdGJNMrCnW7Dme0Z229WWP/UoxDKg71F2tMsVVGDiRd8fFQ==", + "dev": true, + "dependencies": { + "@lerna/command": "4.0.0", + "@lerna/filter-options": "4.0.0", + "@lerna/npm-run-script": "4.0.0", + "@lerna/output": "4.0.0", + "@lerna/profiler": "4.0.0", + "@lerna/run-topologically": "4.0.0", + "@lerna/timer": "4.0.0", + "@lerna/validation-error": "4.0.0", + "p-map": "^4.0.0" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/run-lifecycle": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/run-lifecycle/-/run-lifecycle-4.0.0.tgz", + "integrity": "sha512-IwxxsajjCQQEJAeAaxF8QdEixfI7eLKNm4GHhXHrgBu185JcwScFZrj9Bs+PFKxwb+gNLR4iI5rpUdY8Y0UdGQ==", + "dev": true, + "dependencies": { + "@lerna/npm-conf": "4.0.0", + "npm-lifecycle": "^3.1.5", + "npmlog": "^4.1.2" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/run-topologically": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/run-topologically/-/run-topologically-4.0.0.tgz", + "integrity": "sha512-EVZw9hGwo+5yp+VL94+NXRYisqgAlj0jWKWtAIynDCpghRxCE5GMO3xrQLmQgqkpUl9ZxQFpICgYv5DW4DksQA==", + "dev": true, + "dependencies": { + "@lerna/query-graph": "4.0.0", + "p-queue": "^6.6.2" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/run/node_modules/p-map": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/p-map/-/p-map-4.0.0.tgz", + "integrity": "sha512-/bjOqmgETBYB5BoEeGVea8dmvHb2m9GLy1E9W43yeyfP6QQCZGFNa+XRceJEuDB6zqr+gKpIAmlLebMpykw/MQ==", + "dev": true, + "dependencies": { + "aggregate-error": "^3.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/symlink-binary": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/symlink-binary/-/symlink-binary-4.0.0.tgz", + "integrity": "sha512-zualodWC4q1QQc1pkz969hcFeWXOsVYZC5AWVtAPTDfLl+TwM7eG/O6oP+Rr3fFowspxo6b1TQ6sYfDV6HXNWA==", + "dev": true, + "dependencies": { + "@lerna/create-symlink": "4.0.0", + "@lerna/package": "4.0.0", + "fs-extra": "^9.1.0", + "p-map": "^4.0.0" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/symlink-binary/node_modules/fs-extra": { + "version": "9.1.0", + "resolved": "https://registry.npmmirror.com/fs-extra/-/fs-extra-9.1.0.tgz", + "integrity": "sha512-hcg3ZmepS30/7BSFqRvoo3DOMQu7IjqxO5nCDt+zM9XWjb33Wg7ziNT+Qvqbuc3+gWpzO02JubVyk2G4Zvo1OQ==", + "dev": true, + "dependencies": { + "at-least-node": "^1.0.0", + "graceful-fs": "^4.2.0", + "jsonfile": "^6.0.1", + "universalify": "^2.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/symlink-binary/node_modules/jsonfile": { + "version": "6.1.0", + "resolved": "https://registry.npmmirror.com/jsonfile/-/jsonfile-6.1.0.tgz", + "integrity": "sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==", + "dev": true, + "dependencies": { + "universalify": "^2.0.0" + }, + "optionalDependencies": { + "graceful-fs": "^4.1.6" + } + }, + "node_modules/@lerna/symlink-binary/node_modules/p-map": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/p-map/-/p-map-4.0.0.tgz", + "integrity": "sha512-/bjOqmgETBYB5BoEeGVea8dmvHb2m9GLy1E9W43yeyfP6QQCZGFNa+XRceJEuDB6zqr+gKpIAmlLebMpykw/MQ==", + "dev": true, + "dependencies": { + "aggregate-error": "^3.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/symlink-binary/node_modules/universalify": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/universalify/-/universalify-2.0.0.tgz", + "integrity": "sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ==", + "dev": true, + "engines": { + "node": ">= 10.0.0" + } + }, + "node_modules/@lerna/symlink-dependencies": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/symlink-dependencies/-/symlink-dependencies-4.0.0.tgz", + "integrity": "sha512-BABo0MjeUHNAe2FNGty1eantWp8u83BHSeIMPDxNq0MuW2K3CiQRaeWT3EGPAzXpGt0+hVzBrA6+OT0GPn7Yuw==", + "dev": true, + "dependencies": { + "@lerna/create-symlink": "4.0.0", + "@lerna/resolve-symlink": "4.0.0", + "@lerna/symlink-binary": "4.0.0", + "fs-extra": "^9.1.0", + "p-map": "^4.0.0", + "p-map-series": "^2.1.0" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/symlink-dependencies/node_modules/fs-extra": { + "version": "9.1.0", + "resolved": "https://registry.npmmirror.com/fs-extra/-/fs-extra-9.1.0.tgz", + "integrity": "sha512-hcg3ZmepS30/7BSFqRvoo3DOMQu7IjqxO5nCDt+zM9XWjb33Wg7ziNT+Qvqbuc3+gWpzO02JubVyk2G4Zvo1OQ==", + "dev": true, + "dependencies": { + "at-least-node": "^1.0.0", + "graceful-fs": "^4.2.0", + "jsonfile": "^6.0.1", + "universalify": "^2.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/symlink-dependencies/node_modules/jsonfile": { + "version": "6.1.0", + "resolved": "https://registry.npmmirror.com/jsonfile/-/jsonfile-6.1.0.tgz", + "integrity": "sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==", + "dev": true, + "dependencies": { + "universalify": "^2.0.0" + }, + "optionalDependencies": { + "graceful-fs": "^4.1.6" + } + }, + "node_modules/@lerna/symlink-dependencies/node_modules/p-map": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/p-map/-/p-map-4.0.0.tgz", + "integrity": "sha512-/bjOqmgETBYB5BoEeGVea8dmvHb2m9GLy1E9W43yeyfP6QQCZGFNa+XRceJEuDB6zqr+gKpIAmlLebMpykw/MQ==", + "dev": true, + "dependencies": { + "aggregate-error": "^3.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/symlink-dependencies/node_modules/universalify": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/universalify/-/universalify-2.0.0.tgz", + "integrity": "sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ==", + "dev": true, + "engines": { + "node": ">= 10.0.0" + } + }, + "node_modules/@lerna/timer": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/timer/-/timer-4.0.0.tgz", + "integrity": "sha512-WFsnlaE7SdOvjuyd05oKt8Leg3ENHICnvX3uYKKdByA+S3g+TCz38JsNs7OUZVt+ba63nC2nbXDlUnuT2Xbsfg==", + "dev": true, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/validation-error": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/validation-error/-/validation-error-4.0.0.tgz", + "integrity": "sha512-1rBOM5/koiVWlRi3V6dB863E1YzJS8v41UtsHgMr6gB2ncJ2LsQtMKlJpi3voqcgh41H8UsPXR58RrrpPpufyw==", + "dev": true, + "dependencies": { + "npmlog": "^4.1.2" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/version": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/version/-/version-4.0.0.tgz", + "integrity": "sha512-otUgiqs5W9zGWJZSCCMRV/2Zm2A9q9JwSDS7s/tlKq4mWCYriWo7+wsHEA/nPTMDyYyBO5oyZDj+3X50KDUzeA==", + "dev": true, + "dependencies": { + "@lerna/check-working-tree": "4.0.0", + "@lerna/child-process": "4.0.0", + "@lerna/collect-updates": "4.0.0", + "@lerna/command": "4.0.0", + "@lerna/conventional-commits": "4.0.0", + "@lerna/github-client": "4.0.0", + "@lerna/gitlab-client": "4.0.0", + "@lerna/output": "4.0.0", + "@lerna/prerelease-id-from-version": "4.0.0", + "@lerna/prompt": "4.0.0", + "@lerna/run-lifecycle": "4.0.0", + "@lerna/run-topologically": "4.0.0", + "@lerna/validation-error": "4.0.0", + "chalk": "^4.1.0", + "dedent": "^0.7.0", + "load-json-file": "^6.2.0", + "minimatch": "^3.0.4", + "npmlog": "^4.1.2", + "p-map": "^4.0.0", + "p-pipe": "^3.1.0", + "p-reduce": "^2.1.0", + "p-waterfall": "^2.1.1", + "semver": "^7.3.4", + "slash": "^3.0.0", + "temp-write": "^4.0.0", + "write-json-file": "^4.3.0" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/version/node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmmirror.com/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/version/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmmirror.com/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/version/node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/@lerna/version/node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmmirror.com/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "node_modules/@lerna/version/node_modules/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/version/node_modules/lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/version/node_modules/p-map": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/p-map/-/p-map-4.0.0.tgz", + "integrity": "sha512-/bjOqmgETBYB5BoEeGVea8dmvHb2m9GLy1E9W43yeyfP6QQCZGFNa+XRceJEuDB6zqr+gKpIAmlLebMpykw/MQ==", + "dev": true, + "dependencies": { + "aggregate-error": "^3.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/version/node_modules/semver": { + "version": "7.3.7", + "resolved": "https://registry.npmmirror.com/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@lerna/version/node_modules/slash": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/slash/-/slash-3.0.0.tgz", + "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/version/node_modules/supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmmirror.com/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@lerna/version/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/@lerna/write-log-file": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@lerna/write-log-file/-/write-log-file-4.0.0.tgz", + "integrity": "sha512-XRG5BloiArpXRakcnPHmEHJp+4AtnhRtpDIHSghmXD5EichI1uD73J7FgPp30mm2pDRq3FdqB0NbwSEsJ9xFQg==", + "dev": true, + "dependencies": { + "npmlog": "^4.1.2", + "write-file-atomic": "^3.0.3" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/@lerna/write-log-file/node_modules/write-file-atomic": { + "version": "3.0.3", + "resolved": "https://registry.npmmirror.com/write-file-atomic/-/write-file-atomic-3.0.3.tgz", + "integrity": "sha512-AvHcyZ5JnSfq3ioSyjrBkH9yW4m7Ayk8/9My/DD9onKeu/94fwrMocemO2QAJFAlnnDN+ZDS+ZjAR5ua1/PV/Q==", + "dev": true, + "dependencies": { + "imurmurhash": "^0.1.4", + "is-typedarray": "^1.0.0", + "signal-exit": "^3.0.2", + "typedarray-to-buffer": "^3.1.5" + } + }, + "node_modules/@lezer/common": { + "version": "1.0.3", + "resolved": "https://registry.npmmirror.com/@lezer/common/-/common-1.0.3.tgz", + "integrity": "sha512-JH4wAXCgUOcCGNekQPLhVeUtIqjH0yPBs7vvUdSjyQama9618IOKFJwkv2kcqdhF0my8hQEgCTEJU0GIgnahvA==" + }, + "node_modules/@lezer/highlight": { + "version": "1.1.6", + "resolved": "https://registry.npmmirror.com/@lezer/highlight/-/highlight-1.1.6.tgz", + "integrity": "sha512-cmSJYa2us+r3SePpRCjN5ymCqCPv+zyXmDl0ciWtVaNiORT/MxM7ZgOMQZADD0o51qOaOg24qc/zBViOIwAjJg==", + "dependencies": { + "@lezer/common": "^1.0.0" + } + }, + "node_modules/@lezer/json": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/@lezer/json/-/json-1.0.1.tgz", + "integrity": "sha512-nkVC27qiEZEjySbi6gQRuMwa2sDu2PtfjSgz0A4QF81QyRGm3kb2YRzLcOPcTEtmcwvrX/cej7mlhbwViA4WJw==", + "dependencies": { + "@lezer/highlight": "^1.0.0", + "@lezer/lr": "^1.0.0" + } + }, + "node_modules/@lezer/lr": { + "version": "1.3.9", + "resolved": "https://registry.npmmirror.com/@lezer/lr/-/lr-1.3.9.tgz", + "integrity": "sha512-XPz6dzuTHlnsbA5M2DZgjflNQ+9Hi5Swhic0RULdp3oOs3rh6bqGZolosVqN/fQIT8uNiepzINJDnS39oweTHQ==", + "dependencies": { + "@lezer/common": "^1.0.0" + } + }, + "node_modules/@mrmlnc/readdir-enhanced": { + "version": "2.2.1", + "resolved": "https://registry.npmmirror.com/@mrmlnc/readdir-enhanced/-/readdir-enhanced-2.2.1.tgz", + "integrity": "sha512-bPHp6Ji8b41szTOcaP63VlnbbO5Ny6dwAATtY6JTjh5N2OLrb5Qk/Th5cRkRQhkWCt+EJsYrNB0MiL+Gpn6e3g==", + "dev": true, + "dependencies": { + "call-me-maybe": "^1.0.1", + "glob-to-regexp": "^0.3.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/@nodelib/fs.scandir": { + "version": "2.1.5", + "resolved": "https://registry.npmmirror.com/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", + "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==", + "dev": true, + "dependencies": { + "@nodelib/fs.stat": "2.0.5", + "run-parallel": "^1.1.9" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@nodelib/fs.scandir/node_modules/@nodelib/fs.stat": { + "version": "2.0.5", + "resolved": "https://registry.npmmirror.com/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz", + "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==", + "dev": true, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@nodelib/fs.stat": { + "version": "1.1.3", + "resolved": "https://registry.npmmirror.com/@nodelib/fs.stat/-/fs.stat-1.1.3.tgz", + "integrity": "sha512-shAmDyaQC4H92APFoIaVDHCx5bStIocgvbwQyxPRrbUY20V1EYTbSDchWbuwlMG3V17cprZhA6+78JfB+3DTPw==", + "dev": true, + "engines": { + "node": ">= 6" + } + }, + "node_modules/@nodelib/fs.walk": { + "version": "1.2.8", + "resolved": "https://registry.npmmirror.com/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz", + "integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==", + "dev": true, + "dependencies": { + "@nodelib/fs.scandir": "2.1.5", + "fastq": "^1.6.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@npmcli/ci-detect": { + "version": "1.4.0", + "resolved": "https://registry.npmmirror.com/@npmcli/ci-detect/-/ci-detect-1.4.0.tgz", + "integrity": "sha512-3BGrt6FLjqM6br5AhWRKTr3u5GIVkjRYeAFrMp3HjnfICrg4xOrVRwFavKT6tsp++bq5dluL5t8ME/Nha/6c1Q==", + "dev": true + }, + "node_modules/@npmcli/fs": { + "version": "1.1.1", + "resolved": "https://registry.npmmirror.com/@npmcli/fs/-/fs-1.1.1.tgz", + "integrity": "sha512-8KG5RD0GVP4ydEzRn/I4BNDuxDtqVbOdm8675T49OIG/NGhaK0pjPX7ZcDlvKYbA+ulvVK3ztfcF4uBdOxuJbQ==", + "dev": true, + "dependencies": { + "@gar/promisify": "^1.0.1", + "semver": "^7.3.5" + } + }, + "node_modules/@npmcli/fs/node_modules/lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@npmcli/fs/node_modules/semver": { + "version": "7.3.7", + "resolved": "https://registry.npmmirror.com/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@npmcli/fs/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/@npmcli/git": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/@npmcli/git/-/git-2.1.0.tgz", + "integrity": "sha512-/hBFX/QG1b+N7PZBFs0bi+evgRZcK9nWBxQKZkGoXUT5hJSwl5c4d7y8/hm+NQZRPhQ67RzFaj5UM9YeyKoryw==", + "dev": true, + "dependencies": { + "@npmcli/promise-spawn": "^1.3.2", + "lru-cache": "^6.0.0", + "mkdirp": "^1.0.4", + "npm-pick-manifest": "^6.1.1", + "promise-inflight": "^1.0.1", + "promise-retry": "^2.0.1", + "semver": "^7.3.5", + "which": "^2.0.2" + } + }, + "node_modules/@npmcli/git/node_modules/lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@npmcli/git/node_modules/mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmmirror.com/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true, + "bin": { + "mkdirp": "bin/cmd.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@npmcli/git/node_modules/semver": { + "version": "7.3.7", + "resolved": "https://registry.npmmirror.com/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@npmcli/git/node_modules/which": { + "version": "2.0.2", + "resolved": "https://registry.npmmirror.com/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "node-which": "bin/node-which" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@npmcli/git/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/@npmcli/installed-package-contents": { + "version": "1.0.7", + "resolved": "https://registry.npmmirror.com/@npmcli/installed-package-contents/-/installed-package-contents-1.0.7.tgz", + "integrity": "sha512-9rufe0wnJusCQoLpV9ZPKIVP55itrM5BxOXs10DmdbRfgWtHy1LDyskbwRnBghuB0PrF7pNPOqREVtpz4HqzKw==", + "dev": true, + "dependencies": { + "npm-bundled": "^1.1.1", + "npm-normalize-package-bin": "^1.0.1" + }, + "bin": { + "installed-package-contents": "index.js" + }, + "engines": { + "node": ">= 10" + } + }, + "node_modules/@npmcli/move-file": { + "version": "1.1.2", + "resolved": "https://registry.npmmirror.com/@npmcli/move-file/-/move-file-1.1.2.tgz", + "integrity": "sha512-1SUf/Cg2GzGDyaf15aR9St9TWlb+XvbZXWpDx8YKs7MLzMH/BCeopv+y9vzrzgkfykCGuWOlSu3mZhj2+FQcrg==", + "deprecated": "This functionality has been moved to @npmcli/fs", + "dev": true, + "dependencies": { + "mkdirp": "^1.0.4", + "rimraf": "^3.0.2" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@npmcli/move-file/node_modules/mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmmirror.com/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true, + "bin": { + "mkdirp": "bin/cmd.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@npmcli/move-file/node_modules/rimraf": { + "version": "3.0.2", + "resolved": "https://registry.npmmirror.com/rimraf/-/rimraf-3.0.2.tgz", + "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", + "dev": true, + "dependencies": { + "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" + } + }, + "node_modules/@npmcli/node-gyp": { + "version": "1.0.3", + "resolved": "https://registry.npmmirror.com/@npmcli/node-gyp/-/node-gyp-1.0.3.tgz", + "integrity": "sha512-fnkhw+fmX65kiLqk6E3BFLXNC26rUhK90zVwe2yncPliVT/Qos3xjhTLE59Df8KnPlcwIERXKVlU1bXoUQ+liA==", + "dev": true + }, + "node_modules/@npmcli/promise-spawn": { + "version": "1.3.2", + "resolved": "https://registry.npmmirror.com/@npmcli/promise-spawn/-/promise-spawn-1.3.2.tgz", + "integrity": "sha512-QyAGYo/Fbj4MXeGdJcFzZ+FkDkomfRBrPM+9QYJSg+PxgAUL+LU3FneQk37rKR2/zjqkCV1BLHccX98wRXG3Sg==", + "dev": true, + "dependencies": { + "infer-owner": "^1.0.4" + } + }, + "node_modules/@npmcli/run-script": { + "version": "1.8.6", + "resolved": "https://registry.npmmirror.com/@npmcli/run-script/-/run-script-1.8.6.tgz", + "integrity": "sha512-e42bVZnC6VluBZBAFEr3YrdqSspG3bgilyg4nSLBJ7TRGNCzxHa92XAHxQBLYg0BmgwO4b2mf3h/l5EkEWRn3g==", + "dev": true, + "dependencies": { + "@npmcli/node-gyp": "^1.0.2", + "@npmcli/promise-spawn": "^1.3.2", + "node-gyp": "^7.1.0", + "read-package-json-fast": "^2.0.1" + } + }, + "node_modules/@npmcli/run-script/node_modules/chownr": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/chownr/-/chownr-2.0.0.tgz", + "integrity": "sha512-bIomtDF5KGpdogkLd9VspvFzk9KfpyyGlS8YFVZl7TGPBHL5snIOnxeshwVgPteQ9b4Eydl+pVbIyE1DcvCWgQ==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/@npmcli/run-script/node_modules/fs-minipass": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/fs-minipass/-/fs-minipass-2.1.0.tgz", + "integrity": "sha512-V/JgOLFCS+R6Vcq0slCuaeWEdNC3ouDlJMNIsacH2VtALiu9mV4LPrHc5cDl8k5aw6J8jwgWWpiTo5RYhmIzvg==", + "dev": true, + "dependencies": { + "minipass": "^3.0.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@npmcli/run-script/node_modules/lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@npmcli/run-script/node_modules/minipass": { + "version": "3.1.6", + "resolved": "https://registry.npmmirror.com/minipass/-/minipass-3.1.6.tgz", + "integrity": "sha512-rty5kpw9/z8SX9dmxblFA6edItUmwJgMeYDZRrwlIVN27i8gysGbznJwUggw2V/FVqFSDdWy040ZPS811DYAqQ==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@npmcli/run-script/node_modules/minizlib": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/minizlib/-/minizlib-2.1.2.tgz", + "integrity": "sha512-bAxsR8BVfj60DWXHE3u30oHzfl4G7khkSuPW+qvpd7jFRHm7dLxOjUk1EHACJ/hxLY8phGJ0YhYHZo7jil7Qdg==", + "dev": true, + "dependencies": { + "minipass": "^3.0.0", + "yallist": "^4.0.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@npmcli/run-script/node_modules/mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmmirror.com/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true, + "bin": { + "mkdirp": "bin/cmd.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@npmcli/run-script/node_modules/node-gyp": { + "version": "7.1.2", + "resolved": "https://registry.npmmirror.com/node-gyp/-/node-gyp-7.1.2.tgz", + "integrity": "sha512-CbpcIo7C3eMu3dL1c3d0xw449fHIGALIJsRP4DDPHpyiW8vcriNY7ubh9TE4zEKfSxscY7PjeFnshE7h75ynjQ==", + "dev": true, + "dependencies": { + "env-paths": "^2.2.0", + "glob": "^7.1.4", + "graceful-fs": "^4.2.3", + "nopt": "^5.0.0", + "npmlog": "^4.1.2", + "request": "^2.88.2", + "rimraf": "^3.0.2", + "semver": "^7.3.2", + "tar": "^6.0.2", + "which": "^2.0.2" + }, + "bin": { + "node-gyp": "bin/node-gyp.js" + }, + "engines": { + "node": ">= 10.12.0" + } + }, + "node_modules/@npmcli/run-script/node_modules/nopt": { + "version": "5.0.0", + "resolved": "https://registry.npmmirror.com/nopt/-/nopt-5.0.0.tgz", + "integrity": "sha512-Tbj67rffqceeLpcRXrT7vKAN8CwfPeIBgM7E6iBkmKLV7bEMwpGgYLGv0jACUsECaa/vuxP0IjEont6umdMgtQ==", + "dev": true, + "dependencies": { + "abbrev": "1" + }, + "bin": { + "nopt": "bin/nopt.js" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/@npmcli/run-script/node_modules/rimraf": { + "version": "3.0.2", + "resolved": "https://registry.npmmirror.com/rimraf/-/rimraf-3.0.2.tgz", + "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", + "dev": true, + "dependencies": { + "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" + } + }, + "node_modules/@npmcli/run-script/node_modules/semver": { + "version": "7.3.7", + "resolved": "https://registry.npmmirror.com/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@npmcli/run-script/node_modules/tar": { + "version": "6.1.11", + "resolved": "https://registry.npmmirror.com/tar/-/tar-6.1.11.tgz", + "integrity": "sha512-an/KZQzQUkZCkuoAA64hM92X0Urb6VpRhAFllDzz44U2mcD5scmT3zBc4VgVpkugF580+DQn8eAFSyoQt0tznA==", + "dev": true, + "dependencies": { + "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" + }, + "engines": { + "node": ">= 10" + } + }, + "node_modules/@npmcli/run-script/node_modules/which": { + "version": "2.0.2", + "resolved": "https://registry.npmmirror.com/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "node-which": "bin/node-which" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@npmcli/run-script/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/@octokit/auth-token": { + "version": "2.5.0", + "resolved": "https://registry.npmmirror.com/@octokit/auth-token/-/auth-token-2.5.0.tgz", + "integrity": "sha512-r5FVUJCOLl19AxiuZD2VRZ/ORjp/4IN98Of6YJoJOkY75CIBuYfmiNHGrDwXr+aLGG55igl9QrxX3hbiXlLb+g==", + "dev": true, + "dependencies": { + "@octokit/types": "^6.0.3" + } + }, + "node_modules/@octokit/core": { + "version": "3.6.0", + "resolved": "https://registry.npmmirror.com/@octokit/core/-/core-3.6.0.tgz", + "integrity": "sha512-7RKRKuA4xTjMhY+eG3jthb3hlZCsOwg3rztWh75Xc+ShDWOfDDATWbeZpAHBNRpm4Tv9WgBMOy1zEJYXG6NJ7Q==", + "dev": true, + "dependencies": { + "@octokit/auth-token": "^2.4.4", + "@octokit/graphql": "^4.5.8", + "@octokit/request": "^5.6.3", + "@octokit/request-error": "^2.0.5", + "@octokit/types": "^6.0.3", + "before-after-hook": "^2.2.0", + "universal-user-agent": "^6.0.0" + } + }, + "node_modules/@octokit/endpoint": { + "version": "6.0.12", + "resolved": "https://registry.npmmirror.com/@octokit/endpoint/-/endpoint-6.0.12.tgz", + "integrity": "sha512-lF3puPwkQWGfkMClXb4k/eUT/nZKQfxinRWJrdZaJO85Dqwo/G0yOC434Jr2ojwafWJMYqFGFa5ms4jJUgujdA==", + "dev": true, + "dependencies": { + "@octokit/types": "^6.0.3", + "is-plain-object": "^5.0.0", + "universal-user-agent": "^6.0.0" + } + }, + "node_modules/@octokit/endpoint/node_modules/is-plain-object": { + "version": "5.0.0", + "resolved": "https://registry.npmmirror.com/is-plain-object/-/is-plain-object-5.0.0.tgz", + "integrity": "sha512-VRSzKkbMm5jMDoKLbltAkFQ5Qr7VDiTFGXxYFXXowVj387GeGNOCsOH6Msy00SGZ3Fp84b1Naa1psqgcCIEP5Q==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/@octokit/graphql": { + "version": "4.8.0", + "resolved": "https://registry.npmmirror.com/@octokit/graphql/-/graphql-4.8.0.tgz", + "integrity": "sha512-0gv+qLSBLKF0z8TKaSKTsS39scVKF9dbMxJpj3U0vC7wjNWFuIpL/z76Qe2fiuCbDRcJSavkXsVtMS6/dtQQsg==", + "dev": true, + "dependencies": { + "@octokit/request": "^5.6.0", + "@octokit/types": "^6.0.3", + "universal-user-agent": "^6.0.0" + } + }, + "node_modules/@octokit/openapi-types": { + "version": "11.2.0", + "resolved": "https://registry.npmmirror.com/@octokit/openapi-types/-/openapi-types-11.2.0.tgz", + "integrity": "sha512-PBsVO+15KSlGmiI8QAzaqvsNlZlrDlyAJYcrXBCvVUxCp7VnXjkwPoFHgjEJXx3WF9BAwkA6nfCUA7i9sODzKA==", + "dev": true + }, + "node_modules/@octokit/plugin-enterprise-rest": { + "version": "6.0.1", + "resolved": "https://registry.npmmirror.com/@octokit/plugin-enterprise-rest/-/plugin-enterprise-rest-6.0.1.tgz", + "integrity": "sha512-93uGjlhUD+iNg1iWhUENAtJata6w5nE+V4urXOAlIXdco6xNZtUSfYY8dzp3Udy74aqO/B5UZL80x/YMa5PKRw==", + "dev": true + }, + "node_modules/@octokit/plugin-paginate-rest": { + "version": "2.17.0", + "resolved": "https://registry.npmmirror.com/@octokit/plugin-paginate-rest/-/plugin-paginate-rest-2.17.0.tgz", + "integrity": "sha512-tzMbrbnam2Mt4AhuyCHvpRkS0oZ5MvwwcQPYGtMv4tUa5kkzG58SVB0fcsLulOZQeRnOgdkZWkRUiyBlh0Bkyw==", + "dev": true, + "dependencies": { + "@octokit/types": "^6.34.0" + }, + "peerDependencies": { + "@octokit/core": ">=2" + } + }, + "node_modules/@octokit/plugin-request-log": { + "version": "1.0.4", + "resolved": "https://registry.npmmirror.com/@octokit/plugin-request-log/-/plugin-request-log-1.0.4.tgz", + "integrity": "sha512-mLUsMkgP7K/cnFEw07kWqXGF5LKrOkD+lhCrKvPHXWDywAwuDUeDwWBpc69XK3pNX0uKiVt8g5z96PJ6z9xCFA==", + "dev": true, + "peerDependencies": { + "@octokit/core": ">=3" + } + }, + "node_modules/@octokit/plugin-rest-endpoint-methods": { + "version": "5.13.0", + "resolved": "https://registry.npmmirror.com/@octokit/plugin-rest-endpoint-methods/-/plugin-rest-endpoint-methods-5.13.0.tgz", + "integrity": "sha512-uJjMTkN1KaOIgNtUPMtIXDOjx6dGYysdIFhgA52x4xSadQCz3b/zJexvITDVpANnfKPW/+E0xkOvLntqMYpviA==", + "dev": true, + "dependencies": { + "@octokit/types": "^6.34.0", + "deprecation": "^2.3.1" + }, + "peerDependencies": { + "@octokit/core": ">=3" + } + }, + "node_modules/@octokit/request": { + "version": "5.6.3", + "resolved": "https://registry.npmmirror.com/@octokit/request/-/request-5.6.3.tgz", + "integrity": "sha512-bFJl0I1KVc9jYTe9tdGGpAMPy32dLBXXo1dS/YwSCTL/2nd9XeHsY616RE3HPXDVk+a+dBuzyz5YdlXwcDTr2A==", + "dev": true, + "dependencies": { + "@octokit/endpoint": "^6.0.1", + "@octokit/request-error": "^2.1.0", + "@octokit/types": "^6.16.1", + "is-plain-object": "^5.0.0", + "node-fetch": "^2.6.7", + "universal-user-agent": "^6.0.0" + } + }, + "node_modules/@octokit/request-error": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/@octokit/request-error/-/request-error-2.1.0.tgz", + "integrity": "sha512-1VIvgXxs9WHSjicsRwq8PlR2LR2x6DwsJAaFgzdi0JfJoGSO8mYI/cHJQ+9FbN21aa+DrgNLnwObmyeSC8Rmpg==", + "dev": true, + "dependencies": { + "@octokit/types": "^6.0.3", + "deprecation": "^2.0.0", + "once": "^1.4.0" + } + }, + "node_modules/@octokit/request/node_modules/is-plain-object": { + "version": "5.0.0", + "resolved": "https://registry.npmmirror.com/is-plain-object/-/is-plain-object-5.0.0.tgz", + "integrity": "sha512-VRSzKkbMm5jMDoKLbltAkFQ5Qr7VDiTFGXxYFXXowVj387GeGNOCsOH6Msy00SGZ3Fp84b1Naa1psqgcCIEP5Q==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/@octokit/rest": { + "version": "18.12.0", + "resolved": "https://registry.npmmirror.com/@octokit/rest/-/rest-18.12.0.tgz", + "integrity": "sha512-gDPiOHlyGavxr72y0guQEhLsemgVjwRePayJ+FcKc2SJqKUbxbkvf5kAZEWA/MKvsfYlQAMVzNJE3ezQcxMJ2Q==", + "dev": true, + "dependencies": { + "@octokit/core": "^3.5.1", + "@octokit/plugin-paginate-rest": "^2.16.8", + "@octokit/plugin-request-log": "^1.0.4", + "@octokit/plugin-rest-endpoint-methods": "^5.12.0" + } + }, + "node_modules/@octokit/types": { + "version": "6.34.0", + "resolved": "https://registry.npmmirror.com/@octokit/types/-/types-6.34.0.tgz", + "integrity": "sha512-s1zLBjWhdEI2zwaoSgyOFoKSl109CUcVBCc7biPJ3aAf6LGLU6szDvi31JPU7bxfla2lqfhjbbg/5DdFNxOwHw==", + "dev": true, + "dependencies": { + "@octokit/openapi-types": "^11.2.0" + } + }, + "node_modules/@riophae/vue-treeselect": { + "version": "0.4.0", + "resolved": "https://registry.npmmirror.com/@riophae/vue-treeselect/-/vue-treeselect-0.4.0.tgz", + "integrity": "sha512-J4atYmBqXQmiPFK/0B5sXKjtnGc21mBJEiyKIDZwk0Q9XuynVFX6IJ4EpaLmUgL5Tve7HAS7wkiGGSti6Uaxcg==", + "dependencies": { + "@babel/runtime": "^7.3.1", + "babel-helper-vue-jsx-merge-props": "^2.0.3", + "easings-css": "^1.0.0", + "fuzzysearch": "^1.0.3", + "is-promise": "^2.1.0", + "lodash": "^4.0.0", + "material-colors": "^1.2.6", + "watch-size": "^2.0.0" + }, + "peerDependencies": { + "vue": "^2.2.0" + } + }, + "node_modules/@samverschueren/stream-to-observable": { + "version": "0.3.1", + "resolved": "https://registry.npmmirror.com/@samverschueren/stream-to-observable/-/stream-to-observable-0.3.1.tgz", + "integrity": "sha512-c/qwwcHyafOQuVQJj0IlBjf5yYgBI7YPJ77k4fOJYesb41jio65eaJODRUmfYKhTOFBrIZ66kgvGPlNbjuoRdQ==", + "dev": true, + "dependencies": { + "any-observable": "^0.3.0" + }, + "engines": { + "node": ">=6" + }, + "peerDependenciesMeta": { + "rxjs": { + "optional": true + }, + "zen-observable": { + "optional": true + } + } + }, + "node_modules/@soda/friendly-errors-webpack-plugin": { + "version": "1.8.1", + "resolved": "https://registry.npmmirror.com/@soda/friendly-errors-webpack-plugin/-/friendly-errors-webpack-plugin-1.8.1.tgz", + "integrity": "sha512-h2ooWqP8XuFqTXT+NyAFbrArzfQA7R6HTezADrvD9Re8fxMLTPPniLdqVTdDaO0eIoLaAwKT+d6w+5GeTk7Vbg==", + "dev": true, + "dependencies": { + "chalk": "^3.0.0", + "error-stack-parser": "^2.0.6", + "string-width": "^4.2.3", + "strip-ansi": "^6.0.1" + }, + "engines": { + "node": ">=8.0.0" + }, + "peerDependencies": { + "webpack": "^4.0.0 || ^5.0.0" + } + }, + "node_modules/@soda/friendly-errors-webpack-plugin/node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmmirror.com/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/@soda/friendly-errors-webpack-plugin/node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmmirror.com/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@soda/friendly-errors-webpack-plugin/node_modules/chalk": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/chalk/-/chalk-3.0.0.tgz", + "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@soda/friendly-errors-webpack-plugin/node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/@soda/friendly-errors-webpack-plugin/node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmmirror.com/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "node_modules/@soda/friendly-errors-webpack-plugin/node_modules/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/@soda/friendly-errors-webpack-plugin/node_modules/is-fullwidth-code-point": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", + "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/@soda/friendly-errors-webpack-plugin/node_modules/string-width": { + "version": "4.2.3", + "resolved": "https://registry.npmmirror.com/string-width/-/string-width-4.2.3.tgz", + "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", + "dev": true, + "dependencies": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@soda/friendly-errors-webpack-plugin/node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmmirror.com/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@soda/friendly-errors-webpack-plugin/node_modules/supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmmirror.com/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@tootallnate/once": { + "version": "1.1.2", + "resolved": "https://registry.npmmirror.com/@tootallnate/once/-/once-1.1.2.tgz", + "integrity": "sha512-RbzJvlNzmRq5c3O09UipeuXno4tA1FE6ikOjxZK0tuxVv3412l64l5t1W5pj4+rJq9vpkm/kwiR07aZXnsKPxw==", + "dev": true, + "engines": { + "node": ">= 6" + } + }, + "node_modules/@types/glob": { + "version": "7.2.0", + "resolved": "https://registry.npmmirror.com/@types/glob/-/glob-7.2.0.tgz", + "integrity": "sha512-ZUxbzKl0IfJILTS6t7ip5fQQM/J3TJYubDm3nMbgubNNYS62eXeUpoLUC8/7fJNiFYHTrGPQn7hspDUzIHX3UA==", + "dev": true, + "dependencies": { + "@types/minimatch": "*", + "@types/node": "*" + } + }, + "node_modules/@types/json-schema": { + "version": "7.0.11", + "resolved": "https://registry.npmmirror.com/@types/json-schema/-/json-schema-7.0.11.tgz", + "integrity": "sha512-wOuvG1SN4Us4rez+tylwwwCV1psiNVOkJeM3AUWUNWg/jDQY2+HE/444y5gc+jBmRqASOm2Oeh5c1axHobwRKQ==", + "dev": true + }, + "node_modules/@types/json5": { + "version": "0.0.29", + "resolved": "https://registry.npmmirror.com/@types/json5/-/json5-0.0.29.tgz", + "integrity": "sha512-dRLjCWHYg4oaA77cxO64oO+7JwCwnIzkZPdrrC71jQmQtlhM556pwKo5bUzqvZndkVbeFLIIi+9TC40JNF5hNQ==", + "dev": true + }, + "node_modules/@types/minimatch": { + "version": "3.0.5", + "resolved": "https://registry.npmmirror.com/@types/minimatch/-/minimatch-3.0.5.tgz", + "integrity": "sha512-Klz949h02Gz2uZCMGwDUSDS1YBlTdDDgbWHi+81l29tQALUtvz4rAYi5uoVhE5Lagoq6DeqAUlbrHvW/mXDgdQ==", + "dev": true + }, + "node_modules/@types/minimist": { + "version": "1.2.2", + "resolved": "https://registry.npmmirror.com/@types/minimist/-/minimist-1.2.2.tgz", + "integrity": "sha512-jhuKLIRrhvCPLqwPcx6INqmKeiA5EWrsCOPhrlFSrbrmU4ZMPjj5Ul/oLCMDO98XRUIwVm78xICz4EPCektzeQ==", + "dev": true + }, + "node_modules/@types/node": { + "version": "17.0.27", + "resolved": "https://registry.npmmirror.com/@types/node/-/node-17.0.27.tgz", + "integrity": "sha512-4/Ke7bbWOasuT3kceBZFGakP1dYN2XFd8v2l9bqF2LNWrmeU07JLpp56aEeG6+Q3olqO5TvXpW0yaiYnZJ5CXg==", + "dev": true + }, + "node_modules/@types/normalize-package-data": { + "version": "2.4.1", + "resolved": "https://registry.npmmirror.com/@types/normalize-package-data/-/normalize-package-data-2.4.1.tgz", + "integrity": "sha512-Gj7cI7z+98M282Tqmp2K5EIsoouUEzbBJhQQzDE3jSIRk6r9gsz0oUokqIUR4u1R3dMHo0pDHM7sNOHyhulypw==", + "dev": true + }, + "node_modules/@types/parse-json": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@types/parse-json/-/parse-json-4.0.0.tgz", + "integrity": "sha512-//oorEZjL6sbPcKUaCdIGlIUeH26mgzimjBB77G6XRgnDl/L5wOnpyBGRe/Mmf5CVW3PwEBE1NjiMZ/ssFh4wA==", + "dev": true + }, + "node_modules/@types/q": { + "version": "1.5.5", + "resolved": "https://registry.npmmirror.com/@types/q/-/q-1.5.5.tgz", + "integrity": "sha512-L28j2FcJfSZOnL1WBjDYp2vUHCeIFlyYI/53EwD/rKUBQ7MtUUfbQWiyKJGpcnv4/WgrhWsFKrcPstcAt/J0tQ==" + }, + "node_modules/@vue/babel-helper-vue-jsx-merge-props": { + "version": "1.2.1", + "resolved": "https://registry.npmmirror.com/@vue/babel-helper-vue-jsx-merge-props/-/babel-helper-vue-jsx-merge-props-1.2.1.tgz", + "integrity": "sha512-QOi5OW45e2R20VygMSNhyQHvpdUwQZqGPc748JLGCYEy+yp8fNFNdbNIGAgZmi9e+2JHPd6i6idRuqivyicIkA==", + "dev": true + }, + "node_modules/@vue/babel-plugin-transform-vue-jsx": { + "version": "1.2.1", + "resolved": "https://registry.npmmirror.com/@vue/babel-plugin-transform-vue-jsx/-/babel-plugin-transform-vue-jsx-1.2.1.tgz", + "integrity": "sha512-HJuqwACYehQwh1fNT8f4kyzqlNMpBuUK4rSiSES5D4QsYncv5fxFsLyrxFPG2ksO7t5WP+Vgix6tt6yKClwPzA==", + "dev": true, + "dependencies": { + "@babel/helper-module-imports": "^7.0.0", + "@babel/plugin-syntax-jsx": "^7.2.0", + "@vue/babel-helper-vue-jsx-merge-props": "^1.2.1", + "html-tags": "^2.0.0", + "lodash.kebabcase": "^4.1.1", + "svg-tags": "^1.0.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@vue/babel-preset-app": { + "version": "3.12.1", + "resolved": "https://registry.npmmirror.com/@vue/babel-preset-app/-/babel-preset-app-3.12.1.tgz", + "integrity": "sha512-Zjy5jQaikV1Pz+ri0YgXFS7q4/5wCxB5tRkDOEIt5+4105u0Feb/pvH20nVL6nx9GyXrECFfcm7Yxr/z++OaPQ==", + "dev": true, + "dependencies": { + "@babel/helper-module-imports": "^7.0.0", + "@babel/plugin-proposal-class-properties": "^7.0.0", + "@babel/plugin-proposal-decorators": "^7.1.0", + "@babel/plugin-syntax-dynamic-import": "^7.0.0", + "@babel/plugin-syntax-jsx": "^7.0.0", + "@babel/plugin-transform-runtime": "^7.4.0", + "@babel/preset-env": "^7.0.0 < 7.4.0", + "@babel/runtime": "^7.0.0", + "@babel/runtime-corejs2": "^7.2.0", + "@vue/babel-preset-jsx": "^1.0.0", + "babel-plugin-dynamic-import-node": "^2.2.0", + "babel-plugin-module-resolver": "3.2.0", + "core-js": "^2.6.5" + } + }, + "node_modules/@vue/babel-preset-jsx": { + "version": "1.2.4", + "resolved": "https://registry.npmmirror.com/@vue/babel-preset-jsx/-/babel-preset-jsx-1.2.4.tgz", + "integrity": "sha512-oRVnmN2a77bYDJzeGSt92AuHXbkIxbf/XXSE3klINnh9AXBmVS1DGa1f0d+dDYpLfsAKElMnqKTQfKn7obcL4w==", + "dev": true, + "dependencies": { + "@vue/babel-helper-vue-jsx-merge-props": "^1.2.1", + "@vue/babel-plugin-transform-vue-jsx": "^1.2.1", + "@vue/babel-sugar-composition-api-inject-h": "^1.2.1", + "@vue/babel-sugar-composition-api-render-instance": "^1.2.4", + "@vue/babel-sugar-functional-vue": "^1.2.2", + "@vue/babel-sugar-inject-h": "^1.2.2", + "@vue/babel-sugar-v-model": "^1.2.3", + "@vue/babel-sugar-v-on": "^1.2.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@vue/babel-sugar-composition-api-inject-h": { + "version": "1.2.1", + "resolved": "https://registry.npmmirror.com/@vue/babel-sugar-composition-api-inject-h/-/babel-sugar-composition-api-inject-h-1.2.1.tgz", + "integrity": "sha512-4B3L5Z2G+7s+9Bwbf+zPIifkFNcKth7fQwekVbnOA3cr3Pq71q71goWr97sk4/yyzH8phfe5ODVzEjX7HU7ItQ==", + "dev": true, + "dependencies": { + "@babel/plugin-syntax-jsx": "^7.2.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@vue/babel-sugar-composition-api-render-instance": { + "version": "1.2.4", + "resolved": "https://registry.npmmirror.com/@vue/babel-sugar-composition-api-render-instance/-/babel-sugar-composition-api-render-instance-1.2.4.tgz", + "integrity": "sha512-joha4PZznQMsxQYXtR3MnTgCASC9u3zt9KfBxIeuI5g2gscpTsSKRDzWQt4aqNIpx6cv8On7/m6zmmovlNsG7Q==", + "dev": true, + "dependencies": { + "@babel/plugin-syntax-jsx": "^7.2.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@vue/babel-sugar-functional-vue": { + "version": "1.2.2", + "resolved": "https://registry.npmmirror.com/@vue/babel-sugar-functional-vue/-/babel-sugar-functional-vue-1.2.2.tgz", + "integrity": "sha512-JvbgGn1bjCLByIAU1VOoepHQ1vFsroSA/QkzdiSs657V79q6OwEWLCQtQnEXD/rLTA8rRit4rMOhFpbjRFm82w==", + "dev": true, + "dependencies": { + "@babel/plugin-syntax-jsx": "^7.2.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@vue/babel-sugar-inject-h": { + "version": "1.2.2", + "resolved": "https://registry.npmmirror.com/@vue/babel-sugar-inject-h/-/babel-sugar-inject-h-1.2.2.tgz", + "integrity": "sha512-y8vTo00oRkzQTgufeotjCLPAvlhnpSkcHFEp60+LJUwygGcd5Chrpn5480AQp/thrxVm8m2ifAk0LyFel9oCnw==", + "dev": true, + "dependencies": { + "@babel/plugin-syntax-jsx": "^7.2.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@vue/babel-sugar-v-model": { + "version": "1.2.3", + "resolved": "https://registry.npmmirror.com/@vue/babel-sugar-v-model/-/babel-sugar-v-model-1.2.3.tgz", + "integrity": "sha512-A2jxx87mySr/ulAsSSyYE8un6SIH0NWHiLaCWpodPCVOlQVODCaSpiR4+IMsmBr73haG+oeCuSvMOM+ttWUqRQ==", + "dev": true, + "dependencies": { + "@babel/plugin-syntax-jsx": "^7.2.0", + "@vue/babel-helper-vue-jsx-merge-props": "^1.2.1", + "@vue/babel-plugin-transform-vue-jsx": "^1.2.1", + "camelcase": "^5.0.0", + "html-tags": "^2.0.0", + "svg-tags": "^1.0.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@vue/babel-sugar-v-on": { + "version": "1.2.3", + "resolved": "https://registry.npmmirror.com/@vue/babel-sugar-v-on/-/babel-sugar-v-on-1.2.3.tgz", + "integrity": "sha512-kt12VJdz/37D3N3eglBywV8GStKNUhNrsxChXIV+o0MwVXORYuhDTHJRKPgLJRb/EY3vM2aRFQdxJBp9CLikjw==", + "dev": true, + "dependencies": { + "@babel/plugin-syntax-jsx": "^7.2.0", + "@vue/babel-plugin-transform-vue-jsx": "^1.2.1", + "camelcase": "^5.0.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@vue/cli-overlay": { + "version": "3.12.1", + "resolved": "https://registry.npmmirror.com/@vue/cli-overlay/-/cli-overlay-3.12.1.tgz", + "integrity": "sha512-Bym92EN+lj+cNRN2ozbYyH+V8DMXWGbCDUk+hiJ4EYDBZfBkZKvalk1/mOBFwyxiopnnbOEBAAhL/UuMQ1xARg==", + "dev": true + }, + "node_modules/@vue/cli-plugin-babel": { + "version": "3.12.1", + "resolved": "https://registry.npmmirror.com/@vue/cli-plugin-babel/-/cli-plugin-babel-3.12.1.tgz", + "integrity": "sha512-Zetvz8PikLCGomeKOKu8pC9YQ7cfxs7pGpvEOzaxGdhMnebhjAYR6i6dOB57A6N5lhxQksXCtYTv26QgfiIpdg==", + "dev": true, + "dependencies": { + "@babel/core": "^7.0.0", + "@vue/babel-preset-app": "^3.12.1", + "@vue/cli-shared-utils": "^3.12.1", + "babel-loader": "^8.0.5", + "webpack": "^4.0.0" + } + }, + "node_modules/@vue/cli-plugin-eslint": { + "version": "3.12.1", + "resolved": "https://registry.npmmirror.com/@vue/cli-plugin-eslint/-/cli-plugin-eslint-3.12.1.tgz", + "integrity": "sha512-tVTZlEZsy3sQbO4LLWFK11yzlWwqVAqaM+IY+BeWHITBzEJKh2KmouG+x6x/reXiU3qROsMJ4Ej3Hs8buSMWyQ==", + "dev": true, + "dependencies": { + "@vue/cli-shared-utils": "^3.12.1", + "babel-eslint": "^10.0.1", + "eslint-loader": "^2.1.2", + "globby": "^9.2.0", + "webpack": "^4.0.0", + "yorkie": "^2.0.0" + }, + "optionalDependencies": { + "eslint": "^4.19.1", + "eslint-plugin-vue": "^4.7.1" + } + }, + "node_modules/@vue/cli-plugin-eslint/node_modules/ajv": { + "version": "5.5.2", + "resolved": "https://registry.npmmirror.com/ajv/-/ajv-5.5.2.tgz", + "integrity": "sha512-Ajr4IcMXq/2QmMkEmSvxqfLN5zGmJ92gHXAeOXq1OekoH2rfDNsgdDoL2f7QaRCy7G/E6TpxBVdRuNraMztGHw==", + "dev": true, + "optional": true, + "dependencies": { + "co": "^4.6.0", + "fast-deep-equal": "^1.0.0", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.3.0" + } + }, + "node_modules/@vue/cli-plugin-eslint/node_modules/ansi-regex": { + "version": "3.0.1", + "resolved": "https://registry.npmmirror.com/ansi-regex/-/ansi-regex-3.0.1.tgz", + "integrity": "sha512-+O9Jct8wf++lXxxFc4hc8LsjaSq0HFzzL7cVsw8pRDIPdjKD2mT4ytDZlLuSBZ4cLKZFXIrMGO7DbQCtMJJMKw==", + "dev": true, + "optional": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/@vue/cli-plugin-eslint/node_modules/cross-spawn": { + "version": "5.1.0", + "resolved": "https://registry.npmmirror.com/cross-spawn/-/cross-spawn-5.1.0.tgz", + "integrity": "sha512-pTgQJ5KC0d2hcY8eyL1IzlBPYjTkyH72XRZPnLyKus2mBfNjQs3klqbJU2VILqZryAZUt9JOb3h/mWMy23/f5A==", + "dev": true, + "optional": true, + "dependencies": { + "lru-cache": "^4.0.1", + "shebang-command": "^1.2.0", + "which": "^1.2.9" + } + }, + "node_modules/@vue/cli-plugin-eslint/node_modules/debug": { + "version": "3.2.7", + "resolved": "https://registry.npmmirror.com/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dev": true, + "optional": true, + "dependencies": { + "ms": "^2.1.1" + } + }, + "node_modules/@vue/cli-plugin-eslint/node_modules/eslint": { + "version": "4.19.1", + "resolved": "https://registry.npmmirror.com/eslint/-/eslint-4.19.1.tgz", + "integrity": "sha512-bT3/1x1EbZB7phzYu7vCr1v3ONuzDtX8WjuM9c0iYxe+cq+pwcKEoQjl7zd3RpC6YOLgnSy3cTN58M2jcoPDIQ==", + "dev": true, + "optional": true, + "dependencies": { + "ajv": "^5.3.0", + "babel-code-frame": "^6.22.0", + "chalk": "^2.1.0", + "concat-stream": "^1.6.0", + "cross-spawn": "^5.1.0", + "debug": "^3.1.0", + "doctrine": "^2.1.0", + "eslint-scope": "^3.7.1", + "eslint-visitor-keys": "^1.0.0", + "espree": "^3.5.4", + "esquery": "^1.0.0", + "esutils": "^2.0.2", + "file-entry-cache": "^2.0.0", + "functional-red-black-tree": "^1.0.1", + "glob": "^7.1.2", + "globals": "^11.0.1", + "ignore": "^3.3.3", + "imurmurhash": "^0.1.4", + "inquirer": "^3.0.6", + "is-resolvable": "^1.0.0", + "js-yaml": "^3.9.1", + "json-stable-stringify-without-jsonify": "^1.0.1", + "levn": "^0.3.0", + "lodash": "^4.17.4", + "minimatch": "^3.0.2", + "mkdirp": "^0.5.1", + "natural-compare": "^1.4.0", + "optionator": "^0.8.2", + "path-is-inside": "^1.0.2", + "pluralize": "^7.0.0", + "progress": "^2.0.0", + "regexpp": "^1.0.1", + "require-uncached": "^1.0.3", + "semver": "^5.3.0", + "strip-ansi": "^4.0.0", + "strip-json-comments": "~2.0.1", + "table": "4.0.2", + "text-table": "~0.2.0" + }, + "bin": { + "eslint": "bin/eslint.js" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/@vue/cli-plugin-eslint/node_modules/eslint-plugin-vue": { + "version": "4.7.1", + "resolved": "https://registry.npmmirror.com/eslint-plugin-vue/-/eslint-plugin-vue-4.7.1.tgz", + "integrity": "sha512-esETKhVMI7Vdli70Wt4bvAwnZBJeM0pxVX9Yb0wWKxdCJc2EADalVYK/q2FzMw8oKN0wPMdqVCKS8kmR89recA==", + "dev": true, + "optional": true, + "dependencies": { + "vue-eslint-parser": "^2.0.3" + }, + "engines": { + "node": ">=4" + }, + "peerDependencies": { + "eslint": "^3.18.0 || ^4.0.0" + } + }, + "node_modules/@vue/cli-plugin-eslint/node_modules/eslint-scope": { + "version": "3.7.3", + "resolved": "https://registry.npmmirror.com/eslint-scope/-/eslint-scope-3.7.3.tgz", + "integrity": "sha512-W+B0SvF4gamyCTmUc+uITPY0989iXVfKvhwtmJocTaYoc/3khEHmEmvfY/Gn9HA9VV75jrQECsHizkNw1b68FA==", + "dev": true, + "optional": true, + "dependencies": { + "esrecurse": "^4.1.0", + "estraverse": "^4.1.1" + }, + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/@vue/cli-plugin-eslint/node_modules/fast-deep-equal": { + "version": "1.1.0", + "resolved": "https://registry.npmmirror.com/fast-deep-equal/-/fast-deep-equal-1.1.0.tgz", + "integrity": "sha512-fueX787WZKCV0Is4/T2cyAdM4+x1S3MXXOAhavE1ys/W42SHAPacLTQhucja22QBYrfGw50M2sRiXPtTGv9Ymw==", + "dev": true, + "optional": true + }, + "node_modules/@vue/cli-plugin-eslint/node_modules/json-schema-traverse": { + "version": "0.3.1", + "resolved": "https://registry.npmmirror.com/json-schema-traverse/-/json-schema-traverse-0.3.1.tgz", + "integrity": "sha512-4JD/Ivzg7PoW8NzdrBSr3UFwC9mHgvI7Z6z3QGBsSHgKaRTUDmyZAAKJo2UbG1kUVfS9WS8bi36N49U1xw43DA==", + "dev": true, + "optional": true + }, + "node_modules/@vue/cli-plugin-eslint/node_modules/lru-cache": { + "version": "4.1.5", + "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-4.1.5.tgz", + "integrity": "sha512-sWZlbEP2OsHNkXrMl5GYk/jKk70MBng6UU4YI/qGDYbgf6YbP4EvmqISbXCoJiRKs+1bSpFHVgQxvJ17F2li5g==", + "dev": true, + "optional": true, + "dependencies": { + "pseudomap": "^1.0.2", + "yallist": "^2.1.2" + } + }, + "node_modules/@vue/cli-plugin-eslint/node_modules/ms": { + "version": "2.1.3", + "resolved": "https://registry.npmmirror.com/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", + "dev": true, + "optional": true + }, + "node_modules/@vue/cli-plugin-eslint/node_modules/strip-ansi": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/strip-ansi/-/strip-ansi-4.0.0.tgz", + "integrity": "sha512-4XaJ2zQdCzROZDivEVIDPkcQn8LMFSa8kj8Gxb/Lnwzv9A8VctNZ+lfivC/sV3ivW8ElJTERXZoPBRrZKkNKow==", + "dev": true, + "optional": true, + "dependencies": { + "ansi-regex": "^3.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/@vue/cli-plugin-eslint/node_modules/yallist": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-2.1.2.tgz", + "integrity": "sha512-ncTzHV7NvsQZkYe1DW7cbDLm0YpzHmZF5r/iyP3ZnQtMiJ+pjzisCiMNI+Sj+xQF5pXhSHxSB3uDbsBTzY/c2A==", + "dev": true, + "optional": true + }, + "node_modules/@vue/cli-service": { + "version": "3.12.1", + "resolved": "https://registry.npmmirror.com/@vue/cli-service/-/cli-service-3.12.1.tgz", + "integrity": "sha512-PDxNrTGnSKzeV1ruFlsRIAO8JcPizwT0EJXq9GeyooU+p+sOkv7aKkCBJQVYNjZapD1NOGWx6CvAAC/wAW+gew==", + "dev": true, + "dependencies": { + "@intervolga/optimize-cssnano-plugin": "^1.0.5", + "@soda/friendly-errors-webpack-plugin": "^1.7.1", + "@vue/cli-overlay": "^3.12.1", + "@vue/cli-shared-utils": "^3.12.1", + "@vue/component-compiler-utils": "^3.0.0", + "@vue/preload-webpack-plugin": "^1.1.0", + "@vue/web-component-wrapper": "^1.2.0", + "acorn": "^6.1.1", + "acorn-walk": "^6.1.1", + "address": "^1.0.3", + "autoprefixer": "^9.5.1", + "browserslist": "^4.5.4", + "cache-loader": "^2.0.1", + "case-sensitive-paths-webpack-plugin": "^2.2.0", + "chalk": "^2.4.2", + "cli-highlight": "^2.1.0", + "clipboardy": "^2.0.0", + "cliui": "^5.0.0", + "copy-webpack-plugin": "^4.6.0", + "css-loader": "^1.0.1", + "cssnano": "^4.1.10", + "current-script-polyfill": "^1.0.0", + "debug": "^4.1.1", + "default-gateway": "^5.0.2", + "dotenv": "^7.0.0", + "dotenv-expand": "^5.1.0", + "escape-string-regexp": "^1.0.5", + "file-loader": "^3.0.1", + "fs-extra": "^7.0.1", + "globby": "^9.2.0", + "hash-sum": "^1.0.2", + "html-webpack-plugin": "^3.2.0", + "launch-editor-middleware": "^2.2.1", + "lodash.defaultsdeep": "^4.6.1", + "lodash.mapvalues": "^4.6.0", + "lodash.transform": "^4.6.0", + "mini-css-extract-plugin": "^0.8.0", + "minimist": "^1.2.0", + "ora": "^3.4.0", + "portfinder": "^1.0.20", + "postcss-loader": "^3.0.0", + "read-pkg": "^5.0.0", + "semver": "^6.0.0", + "slash": "^2.0.0", + "source-map-url": "^0.4.0", + "ssri": "^6.0.1", + "string.prototype.padend": "^3.0.0", + "terser-webpack-plugin": "^1.2.3", + "thread-loader": "^2.1.2", + "url-loader": "^1.1.2", + "vue-loader": "^15.7.0", + "webpack": "^4.0.0", + "webpack-bundle-analyzer": "^3.3.0", + "webpack-chain": "^4.11.0", + "webpack-dev-server": "^3.4.1", + "webpack-merge": "^4.2.1" + }, + "bin": { + "vue-cli-service": "bin/vue-cli-service.js" + }, + "engines": { + "node": ">=8" + }, + "peerDependencies": { + "vue-template-compiler": "^2.0.0" + } + }, + "node_modules/@vue/cli-service/node_modules/debug": { + "version": "4.3.4", + "resolved": "https://registry.npmmirror.com/debug/-/debug-4.3.4.tgz", + "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", + "dev": true, + "dependencies": { + "ms": "2.1.2" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/@vue/cli-service/node_modules/ms": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", + "dev": true + }, + "node_modules/@vue/cli-service/node_modules/semver": { + "version": "6.3.0", + "resolved": "https://registry.npmmirror.com/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/@vue/cli-shared-utils": { + "version": "3.12.1", + "resolved": "https://registry.npmmirror.com/@vue/cli-shared-utils/-/cli-shared-utils-3.12.1.tgz", + "integrity": "sha512-jFblzRFjutGwu5utOKdVlPlsbA1lBUNNQlAThzNqej+JtTKJjnvjlhjKX0Gq0oOny5FjKWhoyfQ74p9h1qE6JQ==", + "dev": true, + "dependencies": { + "@hapi/joi": "^15.0.1", + "chalk": "^2.4.1", + "execa": "^1.0.0", + "launch-editor": "^2.2.1", + "lru-cache": "^5.1.1", + "node-ipc": "^9.1.1", + "open": "^6.3.0", + "ora": "^3.4.0", + "request": "^2.87.0", + "request-promise-native": "^1.0.7", + "semver": "^6.0.0", + "string.prototype.padstart": "^3.0.0" + } + }, + "node_modules/@vue/cli-shared-utils/node_modules/semver": { + "version": "6.3.0", + "resolved": "https://registry.npmmirror.com/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/@vue/component-compiler-utils": { + "version": "3.3.0", + "resolved": "https://registry.npmmirror.com/@vue/component-compiler-utils/-/component-compiler-utils-3.3.0.tgz", + "integrity": "sha512-97sfH2mYNU+2PzGrmK2haqffDpVASuib9/w2/noxiFi31Z54hW+q3izKQXXQZSNhtiUpAI36uSuYepeBe4wpHQ==", + "dev": true, + "dependencies": { + "consolidate": "^0.15.1", + "hash-sum": "^1.0.2", + "lru-cache": "^4.1.2", + "merge-source-map": "^1.1.0", + "postcss": "^7.0.36", + "postcss-selector-parser": "^6.0.2", + "source-map": "~0.6.1", + "vue-template-es2015-compiler": "^1.9.0" + }, + "optionalDependencies": { + "prettier": "^1.18.2 || ^2.0.0" + } + }, + "node_modules/@vue/component-compiler-utils/node_modules/lru-cache": { + "version": "4.1.5", + "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-4.1.5.tgz", + "integrity": "sha512-sWZlbEP2OsHNkXrMl5GYk/jKk70MBng6UU4YI/qGDYbgf6YbP4EvmqISbXCoJiRKs+1bSpFHVgQxvJ17F2li5g==", + "dev": true, + "dependencies": { + "pseudomap": "^1.0.2", + "yallist": "^2.1.2" + } + }, + "node_modules/@vue/component-compiler-utils/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmmirror.com/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/@vue/component-compiler-utils/node_modules/yallist": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-2.1.2.tgz", + "integrity": "sha512-ncTzHV7NvsQZkYe1DW7cbDLm0YpzHmZF5r/iyP3ZnQtMiJ+pjzisCiMNI+Sj+xQF5pXhSHxSB3uDbsBTzY/c2A==", + "dev": true + }, + "node_modules/@vue/eslint-config-standard": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/@vue/eslint-config-standard/-/eslint-config-standard-4.0.0.tgz", + "integrity": "sha512-bQghq1cw1BuMRHNhr3tRpAJx1tpGy0QtajQX873kLtA9YVuOIoXR7nAWnTN09bBHnSUh2N288vMsqPi2fI4Hzg==", + "dev": true, + "dependencies": { + "eslint-config-standard": "^12.0.0", + "eslint-plugin-import": "^2.14.0", + "eslint-plugin-node": "^8.0.0", + "eslint-plugin-promise": "^4.0.1", + "eslint-plugin-standard": "^4.0.0" + } + }, + "node_modules/@vue/preload-webpack-plugin": { + "version": "1.1.2", + "resolved": "https://registry.npmmirror.com/@vue/preload-webpack-plugin/-/preload-webpack-plugin-1.1.2.tgz", + "integrity": "sha512-LIZMuJk38pk9U9Ur4YzHjlIyMuxPlACdBIHH9/nGYVTsaGKOSnSuELiE8vS9wa+dJpIYspYUOqk+L1Q4pgHQHQ==", + "dev": true, + "engines": { + "node": ">=6.0.0" + }, + "peerDependencies": { + "html-webpack-plugin": ">=2.26.0", + "webpack": ">=4.0.0" + } + }, + "node_modules/@vue/web-component-wrapper": { + "version": "1.3.0", + "resolved": "https://registry.npmmirror.com/@vue/web-component-wrapper/-/web-component-wrapper-1.3.0.tgz", + "integrity": "sha512-Iu8Tbg3f+emIIMmI2ycSI8QcEuAUgPTgHwesDU1eKMLE4YC/c/sFbGc70QgMq31ijRftV0R7vCm9co6rldCeOA==", + "dev": true + }, + "node_modules/@webassemblyjs/ast": { + "version": "1.9.0", + "resolved": "https://registry.npmmirror.com/@webassemblyjs/ast/-/ast-1.9.0.tgz", + "integrity": "sha512-C6wW5L+b7ogSDVqymbkkvuW9kruN//YisMED04xzeBBqjHa2FYnmvOlS6Xj68xWQRgWvI9cIglsjFowH/RJyEA==", + "dependencies": { + "@webassemblyjs/helper-module-context": "1.9.0", + "@webassemblyjs/helper-wasm-bytecode": "1.9.0", + "@webassemblyjs/wast-parser": "1.9.0" + } + }, + "node_modules/@webassemblyjs/floating-point-hex-parser": { + "version": "1.9.0", + "resolved": "https://registry.npmmirror.com/@webassemblyjs/floating-point-hex-parser/-/floating-point-hex-parser-1.9.0.tgz", + "integrity": "sha512-TG5qcFsS8QB4g4MhrxK5TqfdNe7Ey/7YL/xN+36rRjl/BlGE/NcBvJcqsRgCP6Z92mRE+7N50pRIi8SmKUbcQA==" + }, + "node_modules/@webassemblyjs/helper-api-error": { + "version": "1.9.0", + "resolved": "https://registry.npmmirror.com/@webassemblyjs/helper-api-error/-/helper-api-error-1.9.0.tgz", + "integrity": "sha512-NcMLjoFMXpsASZFxJ5h2HZRcEhDkvnNFOAKneP5RbKRzaWJN36NC4jqQHKwStIhGXu5mUWlUUk7ygdtrO8lbmw==" + }, + "node_modules/@webassemblyjs/helper-buffer": { + "version": "1.9.0", + "resolved": "https://registry.npmmirror.com/@webassemblyjs/helper-buffer/-/helper-buffer-1.9.0.tgz", + "integrity": "sha512-qZol43oqhq6yBPx7YM3m9Bv7WMV9Eevj6kMi6InKOuZxhw+q9hOkvq5e/PpKSiLfyetpaBnogSbNCfBwyB00CA==" + }, + "node_modules/@webassemblyjs/helper-code-frame": { + "version": "1.9.0", + "resolved": "https://registry.npmmirror.com/@webassemblyjs/helper-code-frame/-/helper-code-frame-1.9.0.tgz", + "integrity": "sha512-ERCYdJBkD9Vu4vtjUYe8LZruWuNIToYq/ME22igL+2vj2dQ2OOujIZr3MEFvfEaqKoVqpsFKAGsRdBSBjrIvZA==", + "dependencies": { + "@webassemblyjs/wast-printer": "1.9.0" + } + }, + "node_modules/@webassemblyjs/helper-fsm": { + "version": "1.9.0", + "resolved": "https://registry.npmmirror.com/@webassemblyjs/helper-fsm/-/helper-fsm-1.9.0.tgz", + "integrity": "sha512-OPRowhGbshCb5PxJ8LocpdX9Kl0uB4XsAjl6jH/dWKlk/mzsANvhwbiULsaiqT5GZGT9qinTICdj6PLuM5gslw==" + }, + "node_modules/@webassemblyjs/helper-module-context": { + "version": "1.9.0", + "resolved": "https://registry.npmmirror.com/@webassemblyjs/helper-module-context/-/helper-module-context-1.9.0.tgz", + "integrity": "sha512-MJCW8iGC08tMk2enck1aPW+BE5Cw8/7ph/VGZxwyvGbJwjktKkDK7vy7gAmMDx88D7mhDTCNKAW5tED+gZ0W8g==", + "dependencies": { + "@webassemblyjs/ast": "1.9.0" + } + }, + "node_modules/@webassemblyjs/helper-wasm-bytecode": { + "version": "1.9.0", + "resolved": "https://registry.npmmirror.com/@webassemblyjs/helper-wasm-bytecode/-/helper-wasm-bytecode-1.9.0.tgz", + "integrity": "sha512-R7FStIzyNcd7xKxCZH5lE0Bqy+hGTwS3LJjuv1ZVxd9O7eHCedSdrId/hMOd20I+v8wDXEn+bjfKDLzTepoaUw==" + }, + "node_modules/@webassemblyjs/helper-wasm-section": { + "version": "1.9.0", + "resolved": "https://registry.npmmirror.com/@webassemblyjs/helper-wasm-section/-/helper-wasm-section-1.9.0.tgz", + "integrity": "sha512-XnMB8l3ek4tvrKUUku+IVaXNHz2YsJyOOmz+MMkZvh8h1uSJpSen6vYnw3IoQ7WwEuAhL8Efjms1ZWjqh2agvw==", + "dependencies": { + "@webassemblyjs/ast": "1.9.0", + "@webassemblyjs/helper-buffer": "1.9.0", + "@webassemblyjs/helper-wasm-bytecode": "1.9.0", + "@webassemblyjs/wasm-gen": "1.9.0" + } + }, + "node_modules/@webassemblyjs/ieee754": { + "version": "1.9.0", + "resolved": "https://registry.npmmirror.com/@webassemblyjs/ieee754/-/ieee754-1.9.0.tgz", + "integrity": "sha512-dcX8JuYU/gvymzIHc9DgxTzUUTLexWwt8uCTWP3otys596io0L5aW02Gb1RjYpx2+0Jus1h4ZFqjla7umFniTg==", + "dependencies": { + "@xtuc/ieee754": "^1.2.0" + } + }, + "node_modules/@webassemblyjs/leb128": { + "version": "1.9.0", + "resolved": "https://registry.npmmirror.com/@webassemblyjs/leb128/-/leb128-1.9.0.tgz", + "integrity": "sha512-ENVzM5VwV1ojs9jam6vPys97B/S65YQtv/aanqnU7D8aSoHFX8GyhGg0CMfyKNIHBuAVjy3tlzd5QMMINa7wpw==", + "dependencies": { + "@xtuc/long": "4.2.2" + } + }, + "node_modules/@webassemblyjs/utf8": { + "version": "1.9.0", + "resolved": "https://registry.npmmirror.com/@webassemblyjs/utf8/-/utf8-1.9.0.tgz", + "integrity": "sha512-GZbQlWtopBTP0u7cHrEx+73yZKrQoBMpwkGEIqlacljhXCkVM1kMQge/Mf+csMJAjEdSwhOyLAS0AoR3AG5P8w==" + }, + "node_modules/@webassemblyjs/wasm-edit": { + "version": "1.9.0", + "resolved": "https://registry.npmmirror.com/@webassemblyjs/wasm-edit/-/wasm-edit-1.9.0.tgz", + "integrity": "sha512-FgHzBm80uwz5M8WKnMTn6j/sVbqilPdQXTWraSjBwFXSYGirpkSWE2R9Qvz9tNiTKQvoKILpCuTjBKzOIm0nxw==", + "dependencies": { + "@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" + } + }, + "node_modules/@webassemblyjs/wasm-gen": { + "version": "1.9.0", + "resolved": "https://registry.npmmirror.com/@webassemblyjs/wasm-gen/-/wasm-gen-1.9.0.tgz", + "integrity": "sha512-cPE3o44YzOOHvlsb4+E9qSqjc9Qf9Na1OO/BHFy4OI91XDE14MjFN4lTMezzaIWdPqHnsTodGGNP+iRSYfGkjA==", + "dependencies": { + "@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" + } + }, + "node_modules/@webassemblyjs/wasm-opt": { + "version": "1.9.0", + "resolved": "https://registry.npmmirror.com/@webassemblyjs/wasm-opt/-/wasm-opt-1.9.0.tgz", + "integrity": "sha512-Qkjgm6Anhm+OMbIL0iokO7meajkzQD71ioelnfPEj6r4eOFuqm4YC3VBPqXjFyyNwowzbMD+hizmprP/Fwkl2A==", + "dependencies": { + "@webassemblyjs/ast": "1.9.0", + "@webassemblyjs/helper-buffer": "1.9.0", + "@webassemblyjs/wasm-gen": "1.9.0", + "@webassemblyjs/wasm-parser": "1.9.0" + } + }, + "node_modules/@webassemblyjs/wasm-parser": { + "version": "1.9.0", + "resolved": "https://registry.npmmirror.com/@webassemblyjs/wasm-parser/-/wasm-parser-1.9.0.tgz", + "integrity": "sha512-9+wkMowR2AmdSWQzsPEjFU7njh8HTO5MqO8vjwEHuM+AMHioNqSBONRdr0NQQ3dVQrzp0s8lTcYqzUdb7YgELA==", + "dependencies": { + "@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" + } + }, + "node_modules/@webassemblyjs/wast-parser": { + "version": "1.9.0", + "resolved": "https://registry.npmmirror.com/@webassemblyjs/wast-parser/-/wast-parser-1.9.0.tgz", + "integrity": "sha512-qsqSAP3QQ3LyZjNC/0jBJ/ToSxfYJ8kYyuiGvtn/8MK89VrNEfwj7BPQzJVHi0jGTRK2dGdJ5PRqhtjzoww+bw==", + "dependencies": { + "@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" + } + }, + "node_modules/@webassemblyjs/wast-printer": { + "version": "1.9.0", + "resolved": "https://registry.npmmirror.com/@webassemblyjs/wast-printer/-/wast-printer-1.9.0.tgz", + "integrity": "sha512-2J0nE95rHXHyQ24cWjMKJ1tqB/ds8z/cyeOZxJhcb+rW+SQASVjuznUSmdz5GpVJTzU8JkhYut0D3siFDD6wsA==", + "dependencies": { + "@webassemblyjs/ast": "1.9.0", + "@webassemblyjs/wast-parser": "1.9.0", + "@xtuc/long": "4.2.2" + } + }, + "node_modules/@xtuc/ieee754": { + "version": "1.2.0", + "resolved": "https://registry.npmmirror.com/@xtuc/ieee754/-/ieee754-1.2.0.tgz", + "integrity": "sha512-DX8nKgqcGwsc0eJSqYt5lwP4DH5FlHnmuWWBRy7X0NcaGR0ZtuyeESgMwTYVEtxmsNGY+qit4QYT/MIYTOTPeA==" + }, + "node_modules/@xtuc/long": { + "version": "4.2.2", + "resolved": "https://registry.npmmirror.com/@xtuc/long/-/long-4.2.2.tgz", + "integrity": "sha512-NuHqBY1PB/D8xU6s/thBgOAiAP7HOYDQ32+BFZILJ8ivkUkAHQnWfn6WhL79Owj1qmUnoN/YPhktdIoucipkAQ==" + }, + "node_modules/@yarnpkg/lockfile": { + "version": "1.1.0", + "resolved": "https://registry.npmmirror.com/@yarnpkg/lockfile/-/lockfile-1.1.0.tgz", + "integrity": "sha512-GpSwvyXOcOOlV70vbnzjj4fW5xW/FdUF6nQEt1ENy7m4ZCczi1+/buVUPAqmGfqznsORNFzUMjctTIp8a9tuCQ==", + "dev": true + }, + "node_modules/abbrev": { + "version": "1.1.1", + "resolved": "https://registry.npmmirror.com/abbrev/-/abbrev-1.1.1.tgz", + "integrity": "sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q==", + "dev": true + }, + "node_modules/accepts": { + "version": "1.3.8", + "resolved": "https://registry.npmmirror.com/accepts/-/accepts-1.3.8.tgz", + "integrity": "sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw==", + "dev": true, + "dependencies": { + "mime-types": "~2.1.34", + "negotiator": "0.6.3" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/acorn": { + "version": "6.4.2", + "resolved": "https://registry.npmmirror.com/acorn/-/acorn-6.4.2.tgz", + "integrity": "sha512-XtGIhXwF8YM8bJhGxG5kXgjkEuNGLTkoYqVE+KMR+aspr4KGYmKYg7yUe3KghyQ9yheNwLnjmzh/7+gfDBmHCQ==", + "bin": { + "acorn": "bin/acorn" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/acorn-jsx": { + "version": "3.0.1", + "resolved": "https://registry.npmmirror.com/acorn-jsx/-/acorn-jsx-3.0.1.tgz", + "integrity": "sha512-AU7pnZkguthwBjKgCg6998ByQNIMjbuDQZ8bb78QAFZwPfmKia8AIzgY/gWgqCjnht8JLdXmB4YxA0KaV60ncQ==", + "dev": true, + "optional": true, + "dependencies": { + "acorn": "^3.0.4" + } + }, + "node_modules/acorn-jsx/node_modules/acorn": { + "version": "3.3.0", + "resolved": "https://registry.npmmirror.com/acorn/-/acorn-3.3.0.tgz", + "integrity": "sha512-OLUyIIZ7mF5oaAUT1w0TFqQS81q3saT46x8t7ukpPjMNk+nbs4ZHhs7ToV8EWnLYLepjETXd4XaCE4uxkMeqUw==", + "dev": true, + "optional": true, + "bin": { + "acorn": "bin/acorn" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/acorn-walk": { + "version": "6.2.0", + "resolved": "https://registry.npmmirror.com/acorn-walk/-/acorn-walk-6.2.0.tgz", + "integrity": "sha512-7evsyfH1cLOCdAzZAd43Cic04yKydNx0cF+7tiA19p1XnLLPU4dpCQOqpjqwokFe//vS0QqfqqjCS2JkiIs0cA==", + "dev": true, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/add-stream": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/add-stream/-/add-stream-1.0.0.tgz", + "integrity": "sha512-qQLMr+8o0WC4FZGQTcJiKBVC59JylcPSrTtk6usvmIDFUOCKegapy1VHQwRbFMOFyb/inzUVqHs+eMYKDM1YeQ==", + "dev": true + }, + "node_modules/address": { + "version": "1.1.2", + "resolved": "https://registry.npmmirror.com/address/-/address-1.1.2.tgz", + "integrity": "sha512-aT6camzM4xEA54YVJYSqxz1kv4IHnQZRtThJJHhUMRExaU5spC7jX5ugSwTaTgJliIgs4VhZOk7htClvQ/LmRA==", + "dev": true, + "engines": { + "node": ">= 0.12.0" + } + }, + "node_modules/agent-base": { + "version": "6.0.2", + "resolved": "https://registry.npmmirror.com/agent-base/-/agent-base-6.0.2.tgz", + "integrity": "sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==", + "dev": true, + "dependencies": { + "debug": "4" + }, + "engines": { + "node": ">= 6.0.0" + } + }, + "node_modules/agent-base/node_modules/debug": { + "version": "4.3.4", + "resolved": "https://registry.npmmirror.com/debug/-/debug-4.3.4.tgz", + "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", + "dev": true, + "dependencies": { + "ms": "2.1.2" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/agent-base/node_modules/ms": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", + "dev": true + }, + "node_modules/agentkeepalive": { + "version": "4.2.1", + "resolved": "https://registry.npmmirror.com/agentkeepalive/-/agentkeepalive-4.2.1.tgz", + "integrity": "sha512-Zn4cw2NEqd+9fiSVWMscnjyQ1a8Yfoc5oBajLeo5w+YBHgDUcEBY2hS4YpTz6iN5f/2zQiktcuM6tS8x1p9dpA==", + "dev": true, + "dependencies": { + "debug": "^4.1.0", + "depd": "^1.1.2", + "humanize-ms": "^1.2.1" + }, + "engines": { + "node": ">= 8.0.0" + } + }, + "node_modules/agentkeepalive/node_modules/debug": { + "version": "4.3.4", + "resolved": "https://registry.npmmirror.com/debug/-/debug-4.3.4.tgz", + "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", + "dev": true, + "dependencies": { + "ms": "2.1.2" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/agentkeepalive/node_modules/depd": { + "version": "1.1.2", + "resolved": "https://registry.npmmirror.com/depd/-/depd-1.1.2.tgz", + "integrity": "sha512-7emPTl6Dpo6JRXOXjLRxck+FlLRX5847cLKEn00PLAgc3g2hTZZgr+e4c2v6QpSmLeFP3n5yUo7ft6avBK/5jQ==", + "dev": true, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/agentkeepalive/node_modules/ms": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", + "dev": true + }, + "node_modules/aggregate-error": { + "version": "3.1.0", + "resolved": "https://registry.npmmirror.com/aggregate-error/-/aggregate-error-3.1.0.tgz", + "integrity": "sha512-4I7Td01quW/RpocfNayFdFVk1qSuoh0E7JrbRJ16nH01HhKFQ88INq9Sd+nd72zqRySlr9BmDA8xlEJ6vJMrYA==", + "dev": true, + "dependencies": { + "clean-stack": "^2.0.0", + "indent-string": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/ajv": { + "version": "6.12.6", + "resolved": "https://registry.npmmirror.com/ajv/-/ajv-6.12.6.tgz", + "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", + "dependencies": { + "fast-deep-equal": "^3.1.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" + } + }, + "node_modules/ajv-errors": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/ajv-errors/-/ajv-errors-1.0.1.tgz", + "integrity": "sha512-DCRfO/4nQ+89p/RK43i8Ezd41EqdGIU4ld7nGF8OQ14oc/we5rEntLCUa7+jrn3nn83BosfwZA0wb4pon2o8iQ==", + "peerDependencies": { + "ajv": ">=5.0.0" + } + }, + "node_modules/ajv-keywords": { + "version": "3.5.2", + "resolved": "https://registry.npmmirror.com/ajv-keywords/-/ajv-keywords-3.5.2.tgz", + "integrity": "sha512-5p6WTN0DdTGVQk6VjcEju19IgaHudalcfabD7yhDGeA6bcQnmL+CpveLJq/3hvfwd1aof6L386Ougkx6RfyMIQ==", + "peerDependencies": { + "ajv": "^6.9.1" + } + }, + "node_modules/alphanum-sort": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/alphanum-sort/-/alphanum-sort-1.0.2.tgz", + "integrity": "sha512-0FcBfdcmaumGPQ0qPn7Q5qTgz/ooXgIyp1rf8ik5bGX8mpE2YHjC0P/eyQvxu1GURYQgq9ozf2mteQ5ZD9YiyQ==", + "dev": true + }, + "node_modules/amdefine": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/amdefine/-/amdefine-1.0.1.tgz", + "integrity": "sha512-S2Hw0TtNkMJhIabBwIojKL9YHO5T0n5eNqWJ7Lrlel/zDbftQpxpapi8tZs3X1HWa+u+QeydGmzzNU0m09+Rcg==", + "dev": true, + "engines": { + "node": ">=0.4.2" + } + }, + "node_modules/ansi-colors": { + "version": "3.2.4", + "resolved": "https://registry.npmmirror.com/ansi-colors/-/ansi-colors-3.2.4.tgz", + "integrity": "sha512-hHUXGagefjN2iRrID63xckIvotOXOojhQKWIPUZ4mNUZ9nLZW+7FMNoE1lOkEhNWYsx/7ysGIuJYCiMAA9FnrA==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/ansi-escapes": { + "version": "3.2.0", + "resolved": "https://registry.npmmirror.com/ansi-escapes/-/ansi-escapes-3.2.0.tgz", + "integrity": "sha512-cBhpre4ma+U0T1oM5fXg7Dy1Jw7zzwv7lt/GoCpr+hDQJoYnKVPLL4dCvSEFMmQurOQvSrwT7SL/DAlhBI97RQ==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/ansi-html-community": { + "version": "0.0.8", + "resolved": "https://registry.npmmirror.com/ansi-html-community/-/ansi-html-community-0.0.8.tgz", + "integrity": "sha512-1APHAyr3+PCamwNw3bXCPp4HFLONZt/yIH0sZp0/469KWNTEy+qN5jQ3GVX6DMZ1UXAi34yVwtTeaG/HpBuuzw==", + "dev": true, + "engines": [ + "node >= 0.8.0" + ], + "bin": { + "ansi-html": "bin/ansi-html" + } + }, + "node_modules/ansi-regex": { + "version": "4.1.1", + "resolved": "https://registry.npmmirror.com/ansi-regex/-/ansi-regex-4.1.1.tgz", + "integrity": "sha512-ILlv4k/3f6vfQ4OoP2AGvirOktlQ98ZEL1k9FaQjxa3L1abBgbuTDAdPOpvbGncC0BTVQrl+OM8xZGK6tWXt7g==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/ansi-styles": { + "version": "3.2.1", + "resolved": "https://registry.npmmirror.com/ansi-styles/-/ansi-styles-3.2.1.tgz", + "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", + "dependencies": { + "color-convert": "^1.9.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/any-observable": { + "version": "0.3.0", + "resolved": "https://registry.npmmirror.com/any-observable/-/any-observable-0.3.0.tgz", + "integrity": "sha512-/FQM1EDkTsf63Ub2C6O7GuYFDsSXUwsaZDurV0np41ocwq0jthUAYCmhBX9f+KwlaCgIuWyr/4WlUQUBfKfZog==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/any-promise": { + "version": "1.3.0", + "resolved": "https://registry.npmmirror.com/any-promise/-/any-promise-1.3.0.tgz", + "integrity": "sha512-7UvmKalWRt1wgjL1RrGxoSJW/0QZFIegpeGvZG9kjp8vrRu55XTHbwnqq2GpXm9uLbcuhxm3IqX9OB4MZR1b2A==", + "dev": true + }, + "node_modules/anymatch": { + "version": "3.1.2", + "resolved": "https://registry.npmmirror.com/anymatch/-/anymatch-3.1.2.tgz", + "integrity": "sha512-P43ePfOAIupkguHUycrc4qJ9kz8ZiuOUijaETwX7THt0Y/GNK7v0aa8rY816xWjZ7rJdA5XdMcpVFTKMq+RvWg==", + "optional": true, + "dependencies": { + "normalize-path": "^3.0.0", + "picomatch": "^2.0.4" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/aproba": { + "version": "1.2.0", + "resolved": "https://registry.npmmirror.com/aproba/-/aproba-1.2.0.tgz", + "integrity": "sha512-Y9J6ZjXtoYh8RnXVCMOU/ttDmk1aBjunq9vO0ta5x85WDQiQfUF9sIPBITdbiiIVcBo03Hi3jMxigBtsddlXRw==" + }, + "node_modules/arch": { + "version": "2.2.0", + "resolved": "https://registry.npmmirror.com/arch/-/arch-2.2.0.tgz", + "integrity": "sha512-Of/R0wqp83cgHozfIYLbBMnej79U/SVGOOyuB3VVFv1NRM/PSFMK12x9KVtiYzJqmnU5WR2qp0Z5rHb7sWGnFQ==", + "dev": true + }, + "node_modules/archiver": { + "version": "3.1.1", + "resolved": "https://registry.npmmirror.com/archiver/-/archiver-3.1.1.tgz", + "integrity": "sha512-5Hxxcig7gw5Jod/8Gq0OneVgLYET+oNHcxgWItq4TbhOzRLKNAFUb9edAftiMKXvXfCB0vbGrJdZDNq0dWMsxg==", + "dev": true, + "dependencies": { + "archiver-utils": "^2.1.0", + "async": "^2.6.3", + "buffer-crc32": "^0.2.1", + "glob": "^7.1.4", + "readable-stream": "^3.4.0", + "tar-stream": "^2.1.0", + "zip-stream": "^2.1.2" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/archiver-utils": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/archiver-utils/-/archiver-utils-2.1.0.tgz", + "integrity": "sha512-bEL/yUb/fNNiNTuUz979Z0Yg5L+LzLxGJz8x79lYmR54fmTIb6ob/hNQgkQnIUDWIFjZVQwl9Xs356I6BAMHfw==", + "dev": true, + "dependencies": { + "glob": "^7.1.4", + "graceful-fs": "^4.2.0", + "lazystream": "^1.0.0", + "lodash.defaults": "^4.2.0", + "lodash.difference": "^4.5.0", + "lodash.flatten": "^4.4.0", + "lodash.isplainobject": "^4.0.6", + "lodash.union": "^4.6.0", + "normalize-path": "^3.0.0", + "readable-stream": "^2.0.0" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/archiver/node_modules/readable-stream": { + "version": "3.6.0", + "resolved": "https://registry.npmmirror.com/readable-stream/-/readable-stream-3.6.0.tgz", + "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", + "dev": true, + "dependencies": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/are-we-there-yet": { + "version": "1.1.7", + "resolved": "https://registry.npmmirror.com/are-we-there-yet/-/are-we-there-yet-1.1.7.tgz", + "integrity": "sha512-nxwy40TuMiUGqMyRHgCSWZ9FM4VAoRP4xUYSTv5ImRog+h9yISPbVH7H8fASCIzYn9wlEv4zvFL7uKDMCFQm3g==", + "dev": true, + "dependencies": { + "delegates": "^1.0.0", + "readable-stream": "^2.0.6" + } + }, + "node_modules/argparse": { + "version": "1.0.10", + "resolved": "https://registry.npmmirror.com/argparse/-/argparse-1.0.10.tgz", + "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", + "dependencies": { + "sprintf-js": "~1.0.2" + } + }, + "node_modules/arr-diff": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/arr-diff/-/arr-diff-4.0.0.tgz", + "integrity": "sha512-YVIQ82gZPGBebQV/a8dar4AitzCQs0jjXwMPZllpXMaGjXPYVUawSxQrRsjhjupyVxEvbHgUmIhKVlND+j02kA==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/arr-flatten": { + "version": "1.1.0", + "resolved": "https://registry.npmmirror.com/arr-flatten/-/arr-flatten-1.1.0.tgz", + "integrity": "sha512-L3hKV5R/p5o81R7O02IGnwpDmkp6E982XhtbuwSe3O4qOtMMMtodicASA1Cny2U+aCXcNpml+m4dPsvsJ3jatg==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/arr-union": { + "version": "3.1.0", + "resolved": "https://registry.npmmirror.com/arr-union/-/arr-union-3.1.0.tgz", + "integrity": "sha512-sKpyeERZ02v1FeCZT8lrfJq5u6goHCtpTAzPwJYe7c8SPFOboNjNg1vz2L4VTn9T4PQxEx13TbXLmYUcS6Ug7Q==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/array-differ": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/array-differ/-/array-differ-3.0.0.tgz", + "integrity": "sha512-THtfYS6KtME/yIAhKjZ2ul7XI96lQGHRputJQHO80LAWQnuGP4iCIN8vdMRboGbIEYBwU33q8Tch1os2+X0kMg==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/array-flatten": { + "version": "1.1.1", + "resolved": "https://registry.npmmirror.com/array-flatten/-/array-flatten-1.1.1.tgz", + "integrity": "sha512-PCVAQswWemu6UdxsDFFX/+gVeYqKAod3D3UVm91jHwynguOwAvYPhx8nNlM++NqRcK6CxxpUafjmhIdKiHibqg==", + "dev": true + }, + "node_modules/array-ify": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/array-ify/-/array-ify-1.0.0.tgz", + "integrity": "sha512-c5AMf34bKdvPhQ7tBGhqkgKNUzMr4WUs+WDtC2ZUGOUncbxKMTvqxYctiseW3+L4bA8ec+GcZ6/A/FW4m8ukng==", + "dev": true + }, + "node_modules/array-includes": { + "version": "3.1.4", + "resolved": "https://registry.npmmirror.com/array-includes/-/array-includes-3.1.4.tgz", + "integrity": "sha512-ZTNSQkmWumEbiHO2GF4GmWxYVTiQyJy2XOTa15sdQSrvKn7l+180egQMqlrMOUMCyLMD7pmyQe4mMDUT6Behrw==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.3", + "es-abstract": "^1.19.1", + "get-intrinsic": "^1.1.1", + "is-string": "^1.0.7" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/array-union": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/array-union/-/array-union-1.0.2.tgz", + "integrity": "sha512-Dxr6QJj/RdU/hCaBjOfxW+q6lyuVE6JFWIrAUpuOOhoJJoQ99cUn3igRaHVB5P9WrgFVN0FfArM3x0cueOU8ng==", + "dev": true, + "dependencies": { + "array-uniq": "^1.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/array-uniq": { + "version": "1.0.3", + "resolved": "https://registry.npmmirror.com/array-uniq/-/array-uniq-1.0.3.tgz", + "integrity": "sha512-MNha4BWQ6JbwhFhj03YK552f7cb3AzoE8SzeljgChvL1dl3IcvggXVz1DilzySZkCja+CXuZbdW7yATchWn8/Q==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/array-unique": { + "version": "0.3.2", + "resolved": "https://registry.npmmirror.com/array-unique/-/array-unique-0.3.2.tgz", + "integrity": "sha512-SleRWjh9JUud2wH1hPs9rZBZ33H6T9HOiL0uwGnGx9FpE6wKGyfWugmbkEOIs6qWrZhg0LWeLziLrEwQJhs5mQ==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/array.prototype.flat": { + "version": "1.3.0", + "resolved": "https://registry.npmmirror.com/array.prototype.flat/-/array.prototype.flat-1.3.0.tgz", + "integrity": "sha512-12IUEkHsAhA4DY5s0FPgNXIdc8VRSqD9Zp78a5au9abH/SOBrsp082JOWFNTjkMozh8mqcdiKuaLGhPeYztxSw==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.3", + "es-abstract": "^1.19.2", + "es-shim-unscopables": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/arrify": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/arrify/-/arrify-2.0.1.tgz", + "integrity": "sha512-3duEwti880xqi4eAMN8AyR4a0ByT90zoYdLlevfrvU43vb0YZwZVfxOgxWrLXXXpyugL0hNZc9G6BiB5B3nUug==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/asap": { + "version": "2.0.6", + "resolved": "https://registry.npmmirror.com/asap/-/asap-2.0.6.tgz", + "integrity": "sha512-BSHWgDSAiKs50o2Re8ppvp3seVHXSRM44cdSsT9FfNEUUZLOGWVCsiWaRPWM1Znn+mqZ1OfVZ3z3DWEzSp7hRA==", + "dev": true + }, + "node_modules/asn1": { + "version": "0.2.6", + "resolved": "https://registry.npmmirror.com/asn1/-/asn1-0.2.6.tgz", + "integrity": "sha512-ix/FxPn0MDjeyJ7i/yoHGFt/EX6LyNbxSEhPPXODPL+KB0VPk86UYfL0lMdy+KCnv+fmvIzySwaK5COwqVbWTQ==", + "dev": true, + "dependencies": { + "safer-buffer": "~2.1.0" + } + }, + "node_modules/asn1.js": { + "version": "5.4.1", + "resolved": "https://registry.npmmirror.com/asn1.js/-/asn1.js-5.4.1.tgz", + "integrity": "sha512-+I//4cYPccV8LdmBLiX8CYvf9Sp3vQsrqu2QNXRcrbiWvcx/UdlFiqUJJzxRQxgsZmvhXhn4cSKeSmoFjVdupA==", + "dependencies": { + "bn.js": "^4.0.0", + "inherits": "^2.0.1", + "minimalistic-assert": "^1.0.0", + "safer-buffer": "^2.1.0" + } + }, + "node_modules/asn1.js/node_modules/bn.js": { + "version": "4.12.0", + "resolved": "https://registry.npmmirror.com/bn.js/-/bn.js-4.12.0.tgz", + "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==" + }, + "node_modules/assert": { + "version": "1.5.0", + "resolved": "https://registry.npmmirror.com/assert/-/assert-1.5.0.tgz", + "integrity": "sha512-EDsgawzwoun2CZkCgtxJbv392v4nbk9XDD06zI+kQYoBM/3RBWLlEyJARDOmhAAosBjWACEkKL6S+lIZtcAubA==", + "dependencies": { + "object-assign": "^4.1.1", + "util": "0.10.3" + } + }, + "node_modules/assert-plus": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/assert-plus/-/assert-plus-1.0.0.tgz", + "integrity": "sha512-NfJ4UzBCcQGLDlQq7nHxH+tv3kyZ0hHQqF5BO6J7tNJeP5do1llPr8dZ8zHonfhAu0PHAdMkSo+8o0wxg9lZWw==", + "dev": true, + "engines": { + "node": ">=0.8" + } + }, + "node_modules/assert/node_modules/inherits": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/inherits/-/inherits-2.0.1.tgz", + "integrity": "sha512-8nWq2nLTAwd02jTqJExUYFSD/fKq6VH9Y/oG2accc/kdI0V98Bag8d5a4gi3XHz73rDWa2PvTtvcWYquKqSENA==" + }, + "node_modules/assert/node_modules/util": { + "version": "0.10.3", + "resolved": "https://registry.npmmirror.com/util/-/util-0.10.3.tgz", + "integrity": "sha512-5KiHfsmkqacuKjkRkdV7SsfDJ2EGiPsK92s2MhNSY0craxjTdKTtqKsJaCWp4LW33ZZ0OPUv1WO/TFvNQRiQxQ==", + "dependencies": { + "inherits": "2.0.1" + } + }, + "node_modules/assign-symbols": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/assign-symbols/-/assign-symbols-1.0.0.tgz", + "integrity": "sha512-Q+JC7Whu8HhmTdBph/Tq59IoRtoy6KAm5zzPv00WdujX82lbAL8K7WVjne7vdCsAmbF4AYaDOPyO3k0kl8qIrw==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/astral-regex": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/astral-regex/-/astral-regex-1.0.0.tgz", + "integrity": "sha512-+Ryf6g3BKoRc7jfp7ad8tM4TtMiaWvbF/1/sQcZPkkS7ag3D5nMBCe2UfOTONtAkaG0tO0ij3C5Lwmf1EiyjHg==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/async": { + "version": "2.6.4", + "resolved": "https://registry.npmmirror.com/async/-/async-2.6.4.tgz", + "integrity": "sha512-mzo5dfJYwAn29PeiJ0zvwTo04zj8HDJj0Mn8TD7sno7q12prdbnasKJHhkm2c1LgrhlJ0teaea8860oxi51mGA==", + "dev": true, + "dependencies": { + "lodash": "^4.17.14" + } + }, + "node_modules/async-each": { + "version": "1.0.3", + "resolved": "https://registry.npmmirror.com/async-each/-/async-each-1.0.3.tgz", + "integrity": "sha512-z/WhQ5FPySLdvREByI2vZiTWwCnF0moMJ1hK9YQwDTHKh6I7/uSckMetoRGb5UBZPC1z0jlw+n/XCgjeH7y1AQ==", + "devOptional": true + }, + "node_modules/async-foreach": { + "version": "0.1.3", + "resolved": "https://registry.npmmirror.com/async-foreach/-/async-foreach-0.1.3.tgz", + "integrity": "sha512-VUeSMD8nEGBWaZK4lizI1sf3yEC7pnAQ/mrI7pC2fBz2s/tq5jWWEngTwaf0Gruu/OoXRGLGg1XFqpYBiGTYJA==", + "dev": true, + "engines": { + "node": "*" + } + }, + "node_modules/async-limiter": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/async-limiter/-/async-limiter-1.0.1.tgz", + "integrity": "sha512-csOlWGAcRFJaI6m+F2WKdnMKr4HhdhFVBk0H/QbJFMCr+uO2kwohwXQPxw/9OCxp05r5ghVBFSyioixx3gfkNQ==", + "dev": true + }, + "node_modules/async-validator": { + "version": "1.12.2", + "resolved": "https://registry.npmmirror.com/async-validator/-/async-validator-1.12.2.tgz", + "integrity": "sha512-57EETfCPFiB7M4QscvQzWSGNsmtkjjzZv318SK1CBlstk+hycV72ocjriMOOM48HjvmoAoJGpJNjC7Z76RlnZA==" + }, + "node_modules/asynckit": { + "version": "0.4.0", + "resolved": "https://registry.npmmirror.com/asynckit/-/asynckit-0.4.0.tgz", + "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==", + "dev": true + }, + "node_modules/at-least-node": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/at-least-node/-/at-least-node-1.0.0.tgz", + "integrity": "sha512-+q/t7Ekv1EDY2l6Gda6LLiX14rU9TV20Wa3ofeQmwPFZbOMo9DXrLbOjFaaclkXKWidIaopwAObQDqwWtGUjqg==", + "dev": true, + "engines": { + "node": ">= 4.0.0" + } + }, + "node_modules/atob": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/atob/-/atob-2.1.2.tgz", + "integrity": "sha512-Wm6ukoaOGJi/73p/cl2GvLjTI5JM1k/O14isD73YML8StrH/7/lRFgmg8nICZgD3bZZvjwCGxtMOD3wWNAu8cg==", + "bin": { + "atob": "bin/atob.js" + }, + "engines": { + "node": ">= 4.5.0" + } + }, + "node_modules/autoprefixer": { + "version": "9.8.8", + "resolved": "https://registry.npmmirror.com/autoprefixer/-/autoprefixer-9.8.8.tgz", + "integrity": "sha512-eM9d/swFopRt5gdJ7jrpCwgvEMIayITpojhkkSMRsFHYuH5bkSQ4p/9qTEHtmNudUZh22Tehu7I6CxAW0IXTKA==", + "dev": true, + "dependencies": { + "browserslist": "^4.12.0", + "caniuse-lite": "^1.0.30001109", + "normalize-range": "^0.1.2", + "num2fraction": "^1.2.2", + "picocolors": "^0.2.1", + "postcss": "^7.0.32", + "postcss-value-parser": "^4.1.0" + }, + "bin": { + "autoprefixer": "bin/autoprefixer" + } + }, + "node_modules/autoprefixer/node_modules/picocolors": { + "version": "0.2.1", + "resolved": "https://registry.npmmirror.com/picocolors/-/picocolors-0.2.1.tgz", + "integrity": "sha512-cMlDqaLEqfSaW8Z7N5Jw+lyIW869EzT73/F5lhtY9cLGoVxSXznfgfXMO0Z5K0o0Q2TkTXq+0KFsdnSe3jDViA==", + "dev": true + }, + "node_modules/aws-sign2": { + "version": "0.7.0", + "resolved": "https://registry.npmmirror.com/aws-sign2/-/aws-sign2-0.7.0.tgz", + "integrity": "sha512-08kcGqnYf/YmjoRhfxyu+CLxBjUtHLXLXX/vUfx9l2LYzG3c1m61nrpyFUZI6zeS+Li/wWMMidD9KgrqtGq3mA==", + "dev": true, + "engines": { + "node": "*" + } + }, + "node_modules/aws4": { + "version": "1.11.0", + "resolved": "https://registry.npmmirror.com/aws4/-/aws4-1.11.0.tgz", + "integrity": "sha512-xh1Rl34h6Fi1DC2WWKfxUTVqRsNnr6LsKz2+hfwDxQJWmrx8+c7ylaqBMcHfl1U1r2dsifOvKX3LQuLNZ+XSvA==", + "dev": true + }, + "node_modules/axios": { + "version": "0.21.1", + "resolved": "https://registry.npmmirror.com/axios/-/axios-0.21.1.tgz", + "integrity": "sha512-dKQiRHxGD9PPRIUNIWvZhPTPpl1rf/OxTYKsqKUDjBwYylTvV7SjSHJb9ratfyzM6wCdLCOYLzs73qpg5c4iGA==", + "dependencies": { + "follow-redirects": "^1.10.0" + } + }, + "node_modules/babel-code-frame": { + "version": "6.26.0", + "resolved": "https://registry.npmmirror.com/babel-code-frame/-/babel-code-frame-6.26.0.tgz", + "integrity": "sha512-XqYMR2dfdGMW+hd0IUZ2PwK+fGeFkOxZJ0wY+JaQAHzt1Zx8LcvpiZD2NiGkEG8qx0CfkAOr5xt76d1e8vG90g==", + "dev": true, + "dependencies": { + "chalk": "^1.1.3", + "esutils": "^2.0.2", + "js-tokens": "^3.0.2" + } + }, + "node_modules/babel-code-frame/node_modules/ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmmirror.com/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha512-TIGnTpdo+E3+pCyAluZvtED5p5wCqLdezCyhPZzKPcxvFplEt4i+W7OONCKgeZFT3+y5NZZfOOS/Bdcanm1MYA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/babel-code-frame/node_modules/ansi-styles": { + "version": "2.2.1", + "resolved": "https://registry.npmmirror.com/ansi-styles/-/ansi-styles-2.2.1.tgz", + "integrity": "sha512-kmCevFghRiWM7HB5zTPULl4r9bVFSWjz62MhqizDGUrq2NWuNMQyuv4tHHoKJHs69M/MF64lEcHdYIocrdWQYA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/babel-code-frame/node_modules/chalk": { + "version": "1.1.3", + "resolved": "https://registry.npmmirror.com/chalk/-/chalk-1.1.3.tgz", + "integrity": "sha512-U3lRVLMSlsCfjqYPbLyVv11M9CPW4I728d6TCKMAOJueEeB9/8o+eSsMnxPJD+Q+K909sdESg7C+tIkoH6on1A==", + "dev": true, + "dependencies": { + "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" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/babel-code-frame/node_modules/js-tokens": { + "version": "3.0.2", + "resolved": "https://registry.npmmirror.com/js-tokens/-/js-tokens-3.0.2.tgz", + "integrity": "sha512-RjTcuD4xjtthQkaWH7dFlH85L+QaVtSoOyGdZ3g6HFhS9dFNDfLyqgm2NFe2X6cQpeFmt0452FJjFG5UameExg==", + "dev": true + }, + "node_modules/babel-code-frame/node_modules/strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmmirror.com/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha512-VhumSSbBqDTP8p2ZLKj40UjBCV4+v8bUSEpUb4KjRgWk9pbqGF4REFj6KEagidb2f/M6AzC0EmFyDNGaw9OCzg==", + "dev": true, + "dependencies": { + "ansi-regex": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/babel-code-frame/node_modules/supports-color": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/supports-color/-/supports-color-2.0.0.tgz", + "integrity": "sha512-KKNVtd6pCYgPIKU4cp2733HWYCpplQhddZLBUryaAHou723x+FRzQ5Df824Fj+IyyuiQTRoub4SnIFfIcrp70g==", + "dev": true, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/babel-eslint": { + "version": "10.1.0", + "resolved": "https://registry.npmmirror.com/babel-eslint/-/babel-eslint-10.1.0.tgz", + "integrity": "sha512-ifWaTHQ0ce+448CYop8AdrQiBsGrnC+bMgfyKFdi6EsPLTAWG+QfyDeM6OH+FmWnKvEq5NnBMLvlBUPKQZoDSg==", + "deprecated": "babel-eslint is now @babel/eslint-parser. This package will no longer receive updates.", + "dev": true, + "dependencies": { + "@babel/code-frame": "^7.0.0", + "@babel/parser": "^7.7.0", + "@babel/traverse": "^7.7.0", + "@babel/types": "^7.7.0", + "eslint-visitor-keys": "^1.0.0", + "resolve": "^1.12.0" + }, + "engines": { + "node": ">=6" + }, + "peerDependencies": { + "eslint": ">= 4.12.1" + } + }, + "node_modules/babel-helper-vue-jsx-merge-props": { + "version": "2.0.3", + "resolved": "https://registry.npmmirror.com/babel-helper-vue-jsx-merge-props/-/babel-helper-vue-jsx-merge-props-2.0.3.tgz", + "integrity": "sha512-gsLiKK7Qrb7zYJNgiXKpXblxbV5ffSwR0f5whkPAaBAR4fhi6bwRZxX9wBlIc5M/v8CCkXUbXZL4N/nSE97cqg==" + }, + "node_modules/babel-loader": { + "version": "8.2.5", + "resolved": "https://registry.npmmirror.com/babel-loader/-/babel-loader-8.2.5.tgz", + "integrity": "sha512-OSiFfH89LrEMiWd4pLNqGz4CwJDtbs2ZVc+iGu2HrkRfPxId9F2anQj38IxWpmRfsUY0aBZYi1EFcd3mhtRMLQ==", + "dev": true, + "dependencies": { + "find-cache-dir": "^3.3.1", + "loader-utils": "^2.0.0", + "make-dir": "^3.1.0", + "schema-utils": "^2.6.5" + }, + "engines": { + "node": ">= 8.9" + }, + "peerDependencies": { + "@babel/core": "^7.0.0", + "webpack": ">=2" + } + }, + "node_modules/babel-loader/node_modules/find-cache-dir": { + "version": "3.3.2", + "resolved": "https://registry.npmmirror.com/find-cache-dir/-/find-cache-dir-3.3.2.tgz", + "integrity": "sha512-wXZV5emFEjrridIgED11OoUKLxiYjAcqot/NJdAkOhlJ+vGzwhOAfcG5OX1jP+S0PcjEn8bdMJv+g2jwQ3Onig==", + "dev": true, + "dependencies": { + "commondir": "^1.0.1", + "make-dir": "^3.0.2", + "pkg-dir": "^4.1.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/babel-loader/node_modules/find-up": { + "version": "4.1.0", + "resolved": "https://registry.npmmirror.com/find-up/-/find-up-4.1.0.tgz", + "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", + "dev": true, + "dependencies": { + "locate-path": "^5.0.0", + "path-exists": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/babel-loader/node_modules/json5": { + "version": "2.2.1", + "resolved": "https://registry.npmmirror.com/json5/-/json5-2.2.1.tgz", + "integrity": "sha512-1hqLFMSrGHRHxav9q9gNjJ5EXznIxGVO09xQRrwplcS8qs28pZ8s8hupZAmqDwZUmVZ2Qb2jnyPOWcDH8m8dlA==", + "dev": true, + "bin": { + "json5": "lib/cli.js" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/babel-loader/node_modules/loader-utils": { + "version": "2.0.2", + "resolved": "https://registry.npmmirror.com/loader-utils/-/loader-utils-2.0.2.tgz", + "integrity": "sha512-TM57VeHptv569d/GKh6TAYdzKblwDNiumOdkFnejjD0XwTH87K90w3O7AiJRqdQoXygvi1VQTJTLGhJl7WqA7A==", + "dev": true, + "dependencies": { + "big.js": "^5.2.2", + "emojis-list": "^3.0.0", + "json5": "^2.1.2" + }, + "engines": { + "node": ">=8.9.0" + } + }, + "node_modules/babel-loader/node_modules/locate-path": { + "version": "5.0.0", + "resolved": "https://registry.npmmirror.com/locate-path/-/locate-path-5.0.0.tgz", + "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", + "dev": true, + "dependencies": { + "p-locate": "^4.1.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/babel-loader/node_modules/make-dir": { + "version": "3.1.0", + "resolved": "https://registry.npmmirror.com/make-dir/-/make-dir-3.1.0.tgz", + "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==", + "dev": true, + "dependencies": { + "semver": "^6.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/babel-loader/node_modules/p-locate": { + "version": "4.1.0", + "resolved": "https://registry.npmmirror.com/p-locate/-/p-locate-4.1.0.tgz", + "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", + "dev": true, + "dependencies": { + "p-limit": "^2.2.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/babel-loader/node_modules/path-exists": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/path-exists/-/path-exists-4.0.0.tgz", + "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/babel-loader/node_modules/pkg-dir": { + "version": "4.2.0", + "resolved": "https://registry.npmmirror.com/pkg-dir/-/pkg-dir-4.2.0.tgz", + "integrity": "sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ==", + "dev": true, + "dependencies": { + "find-up": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/babel-loader/node_modules/schema-utils": { + "version": "2.7.1", + "resolved": "https://registry.npmmirror.com/schema-utils/-/schema-utils-2.7.1.tgz", + "integrity": "sha512-SHiNtMOUGWBQJwzISiVYKu82GiV4QYGePp3odlY1tuKO7gPtphAT5R/py0fA6xtbgLL/RvtJZnU9b8s0F1q0Xg==", + "dev": true, + "dependencies": { + "@types/json-schema": "^7.0.5", + "ajv": "^6.12.4", + "ajv-keywords": "^3.5.2" + }, + "engines": { + "node": ">= 8.9.0" + } + }, + "node_modules/babel-loader/node_modules/semver": { + "version": "6.3.0", + "resolved": "https://registry.npmmirror.com/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/babel-plugin-dynamic-import-node": { + "version": "2.3.3", + "resolved": "https://registry.npmmirror.com/babel-plugin-dynamic-import-node/-/babel-plugin-dynamic-import-node-2.3.3.tgz", + "integrity": "sha512-jZVI+s9Zg3IqA/kdi0i6UDCybUI3aSBLnglhYbSSjKlV7yF1F/5LWv8MakQmvYpnbJDS6fcBL2KzHSxNCMtWSQ==", + "dev": true, + "dependencies": { + "object.assign": "^4.1.0" + } + }, + "node_modules/babel-plugin-module-resolver": { + "version": "3.2.0", + "resolved": "https://registry.npmmirror.com/babel-plugin-module-resolver/-/babel-plugin-module-resolver-3.2.0.tgz", + "integrity": "sha512-tjR0GvSndzPew/Iayf4uICWZqjBwnlMWjSx6brryfQ81F9rxBVqwDJtFCV8oOs0+vJeefK9TmdZtkIFdFe1UnA==", + "dev": true, + "dependencies": { + "find-babel-config": "^1.1.0", + "glob": "^7.1.2", + "pkg-up": "^2.0.0", + "reselect": "^3.0.1", + "resolve": "^1.4.0" + }, + "engines": { + "node": ">= 6.0.0" + } + }, + "node_modules/babel-plugin-polyfill-corejs2": { + "version": "0.3.1", + "resolved": "https://registry.npmmirror.com/babel-plugin-polyfill-corejs2/-/babel-plugin-polyfill-corejs2-0.3.1.tgz", + "integrity": "sha512-v7/T6EQcNfVLfcN2X8Lulb7DjprieyLWJK/zOWH5DUYcAgex9sP3h25Q+DLsX9TloXe3y1O8l2q2Jv9q8UVB9w==", + "dev": true, + "dependencies": { + "@babel/compat-data": "^7.13.11", + "@babel/helper-define-polyfill-provider": "^0.3.1", + "semver": "^6.1.1" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/babel-plugin-polyfill-corejs2/node_modules/semver": { + "version": "6.3.0", + "resolved": "https://registry.npmmirror.com/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/babel-plugin-polyfill-corejs3": { + "version": "0.5.2", + "resolved": "https://registry.npmmirror.com/babel-plugin-polyfill-corejs3/-/babel-plugin-polyfill-corejs3-0.5.2.tgz", + "integrity": "sha512-G3uJih0XWiID451fpeFaYGVuxHEjzKTHtc9uGFEjR6hHrvNzeS/PX+LLLcetJcytsB5m4j+K3o/EpXJNb/5IEQ==", + "dev": true, + "dependencies": { + "@babel/helper-define-polyfill-provider": "^0.3.1", + "core-js-compat": "^3.21.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/babel-plugin-polyfill-regenerator": { + "version": "0.3.1", + "resolved": "https://registry.npmmirror.com/babel-plugin-polyfill-regenerator/-/babel-plugin-polyfill-regenerator-0.3.1.tgz", + "integrity": "sha512-Y2B06tvgHYt1x0yz17jGkGeeMr5FeKUu+ASJ+N6nB5lQ8Dapfg42i0OVrf8PNGJ3zKL4A23snMi1IRwrqqND7A==", + "dev": true, + "dependencies": { + "@babel/helper-define-polyfill-provider": "^0.3.1" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/babel-plugin-transform-remove-console": { + "version": "6.9.4", + "resolved": "https://registry.npmmirror.com/babel-plugin-transform-remove-console/-/babel-plugin-transform-remove-console-6.9.4.tgz", + "integrity": "sha512-88blrUrMX3SPiGkT1GnvVY8E/7A+k6oj3MNvUtTIxJflFzXTw1bHkuJ/y039ouhFMp2prRn5cQGzokViYi1dsg==" + }, + "node_modules/babel-polyfill": { + "version": "6.26.0", + "resolved": "https://registry.npmmirror.com/babel-polyfill/-/babel-polyfill-6.26.0.tgz", + "integrity": "sha512-F2rZGQnAdaHWQ8YAoeRbukc7HS9QgdgeyJ0rQDd485v9opwuPvjpPFcOOT/WmkKTdgy9ESgSPXDcTNpzrGr6iQ==", + "dependencies": { + "babel-runtime": "^6.26.0", + "core-js": "^2.5.0", + "regenerator-runtime": "^0.10.5" + } + }, + "node_modules/babel-polyfill/node_modules/regenerator-runtime": { + "version": "0.10.5", + "resolved": "https://registry.npmmirror.com/regenerator-runtime/-/regenerator-runtime-0.10.5.tgz", + "integrity": "sha512-02YopEIhAgiBHWeoTiA8aitHDt8z6w+rQqNuIftlM+ZtvSl/brTouaU7DW6GO/cHtvxJvS4Hwv2ibKdxIRi24w==" + }, + "node_modules/babel-runtime": { + "version": "6.26.0", + "resolved": "https://registry.npmmirror.com/babel-runtime/-/babel-runtime-6.26.0.tgz", + "integrity": "sha512-ITKNuq2wKlW1fJg9sSW52eepoYgZBggvOAHC0u/CYu/qxQ9EVzThCgR69BnSXLHjy2f7SY5zaQ4yt7H9ZVxY2g==", + "dependencies": { + "core-js": "^2.4.0", + "regenerator-runtime": "^0.11.0" + } + }, + "node_modules/babel-runtime/node_modules/regenerator-runtime": { + "version": "0.11.1", + "resolved": "https://registry.npmmirror.com/regenerator-runtime/-/regenerator-runtime-0.11.1.tgz", + "integrity": "sha512-MguG95oij0fC3QV3URf4V2SDYGJhJnJGqvIIgdECeODCT98wSWDAJ94SSuVpYQUoTcGUIL6L4yNB7j1DFFHSBg==" + }, + "node_modules/balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==" + }, + "node_modules/base": { + "version": "0.11.2", + "resolved": "https://registry.npmmirror.com/base/-/base-0.11.2.tgz", + "integrity": "sha512-5T6P4xPgpp0YDFvSWwEZ4NoE3aM4QBQXDzmVbraCkFj8zHM+mba8SyqB5DbZWyR7mYHo6Y7BdQo3MoA4m0TeQg==", + "dependencies": { + "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" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/base/node_modules/define-property": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/define-property/-/define-property-1.0.0.tgz", + "integrity": "sha512-cZTYKFWspt9jZsMscWo8sc/5lbPC9Q0N5nBLgb+Yd915iL3udB1uFgS3B8YCx66UVHq018DAVFoee7x+gxggeA==", + "dependencies": { + "is-descriptor": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/base/node_modules/is-accessor-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", + "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", + "dependencies": { + "kind-of": "^6.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/base/node_modules/is-data-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", + "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", + "dependencies": { + "kind-of": "^6.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/base/node_modules/is-descriptor": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/is-descriptor/-/is-descriptor-1.0.2.tgz", + "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", + "dependencies": { + "is-accessor-descriptor": "^1.0.0", + "is-data-descriptor": "^1.0.0", + "kind-of": "^6.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/base64-js": { + "version": "1.5.1", + "resolved": "https://registry.npmmirror.com/base64-js/-/base64-js-1.5.1.tgz", + "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==" + }, + "node_modules/batch": { + "version": "0.6.1", + "resolved": "https://registry.npmmirror.com/batch/-/batch-0.6.1.tgz", + "integrity": "sha512-x+VAiMRL6UPkx+kudNvxTl6hB2XNNCG2r+7wixVfIYwu/2HKRXimwQyaumLjMveWvT2Hkd/cAJw+QBMfJ/EKVw==", + "dev": true + }, + "node_modules/batch-processor": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/batch-processor/-/batch-processor-1.0.0.tgz", + "integrity": "sha512-xoLQD8gmmR32MeuBHgH0Tzd5PuSZx71ZsbhVxOCRbgktZEPe4SQy7s9Z50uPp0F/f7iw2XmkHN2xkgbMfckMDA==" + }, + "node_modules/bcrypt-pbkdf": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/bcrypt-pbkdf/-/bcrypt-pbkdf-1.0.2.tgz", + "integrity": "sha512-qeFIXtP4MSoi6NLqO12WfqARWWuCKi2Rn/9hJLEmtB5yTNr9DqFWkJRCf2qShWzPeAMRnOgCrq0sg/KLv5ES9w==", + "dev": true, + "dependencies": { + "tweetnacl": "^0.14.3" + } + }, + "node_modules/before-after-hook": { + "version": "2.2.2", + "resolved": "https://registry.npmmirror.com/before-after-hook/-/before-after-hook-2.2.2.tgz", + "integrity": "sha512-3pZEU3NT5BFUo/AD5ERPWOgQOCZITni6iavr5AUw5AUwQjMlI0kzu5btnyD39AF0gUEsDPwJT+oY1ORBJijPjQ==", + "dev": true + }, + "node_modules/bfj": { + "version": "6.1.2", + "resolved": "https://registry.npmmirror.com/bfj/-/bfj-6.1.2.tgz", + "integrity": "sha512-BmBJa4Lip6BPRINSZ0BPEIfB1wUY/9rwbwvIHQA1KjX9om29B6id0wnWXq7m3bn5JrUVjeOTnVuhPT1FiHwPGw==", + "dev": true, + "dependencies": { + "bluebird": "^3.5.5", + "check-types": "^8.0.3", + "hoopy": "^0.1.4", + "tryer": "^1.0.1" + }, + "engines": { + "node": ">= 6.0.0" + } + }, + "node_modules/big.js": { + "version": "5.2.2", + "resolved": "https://registry.npmmirror.com/big.js/-/big.js-5.2.2.tgz", + "integrity": "sha512-vyL2OymJxmarO8gxMr0mhChsO9QGwhynfuu4+MHTAW6czfq9humCB7rKpUjDd9YUiDPU4mzpyupFSvOClAwbmQ==", + "engines": { + "node": "*" + } + }, + "node_modules/binary-extensions": { + "version": "2.2.0", + "resolved": "https://registry.npmmirror.com/binary-extensions/-/binary-extensions-2.2.0.tgz", + "integrity": "sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA==", + "optional": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/bindings": { + "version": "1.5.0", + "resolved": "https://registry.npmmirror.com/bindings/-/bindings-1.5.0.tgz", + "integrity": "sha512-p2q/t/mhvuOj/UeLlV6566GD/guowlr0hHxClI0W9m7MWYkL1F0hLo+0Aexs9HSPCtR1SXQ0TD3MMKrXZajbiQ==", + "optional": true, + "dependencies": { + "file-uri-to-path": "1.0.0" + } + }, + "node_modules/bl": { + "version": "4.1.0", + "resolved": "https://registry.npmmirror.com/bl/-/bl-4.1.0.tgz", + "integrity": "sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w==", + "dev": true, + "dependencies": { + "buffer": "^5.5.0", + "inherits": "^2.0.4", + "readable-stream": "^3.4.0" + } + }, + "node_modules/bl/node_modules/buffer": { + "version": "5.7.1", + "resolved": "https://registry.npmmirror.com/buffer/-/buffer-5.7.1.tgz", + "integrity": "sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==", + "dev": true, + "dependencies": { + "base64-js": "^1.3.1", + "ieee754": "^1.1.13" + } + }, + "node_modules/bl/node_modules/readable-stream": { + "version": "3.6.0", + "resolved": "https://registry.npmmirror.com/readable-stream/-/readable-stream-3.6.0.tgz", + "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", + "dev": true, + "dependencies": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/bluebird": { + "version": "3.7.2", + "resolved": "https://registry.npmmirror.com/bluebird/-/bluebird-3.7.2.tgz", + "integrity": "sha512-XpNj6GDQzdfW+r2Wnn7xiSAd7TM3jzkxGXBGTtWKuSXv1xUV+azxAm8jdWZN06QTQk+2N2XB9jRDkvbmQmcRtg==" + }, + "node_modules/bn.js": { + "version": "5.2.0", + "resolved": "https://registry.npmmirror.com/bn.js/-/bn.js-5.2.0.tgz", + "integrity": "sha512-D7iWRBvnZE8ecXiLj/9wbxH7Tk79fAh8IHaTNq1RWRixsS02W+5qS+iE9yq6RYl0asXx5tw0bLhmT5pIfbSquw==" + }, + "node_modules/body-parser": { + "version": "1.20.0", + "resolved": "https://registry.npmmirror.com/body-parser/-/body-parser-1.20.0.tgz", + "integrity": "sha512-DfJ+q6EPcGKZD1QWUjSpqp+Q7bDQTsQIF4zfUAtZ6qk+H/3/QRhg9CEp39ss+/T2vw0+HaidC0ecJj/DRLIaKg==", + "dev": true, + "dependencies": { + "bytes": "3.1.2", + "content-type": "~1.0.4", + "debug": "2.6.9", + "depd": "2.0.0", + "destroy": "1.2.0", + "http-errors": "2.0.0", + "iconv-lite": "0.4.24", + "on-finished": "2.4.1", + "qs": "6.10.3", + "raw-body": "2.5.1", + "type-is": "~1.6.18", + "unpipe": "1.0.0" + }, + "engines": { + "node": ">= 0.8", + "npm": "1.2.8000 || >= 1.4.16" + } + }, + "node_modules/body-parser/node_modules/qs": { + "version": "6.10.3", + "resolved": "https://registry.npmmirror.com/qs/-/qs-6.10.3.tgz", + "integrity": "sha512-wr7M2E0OFRfIfJZjKGieI8lBKb7fRCH4Fv5KNPEs7gJ8jadvotdsS08PzOKR7opXhZ/Xkjtt3WF9g38drmyRqQ==", + "dev": true, + "dependencies": { + "side-channel": "^1.0.4" + }, + "engines": { + "node": ">=0.6" + } + }, + "node_modules/bonjour": { + "version": "3.5.0", + "resolved": "https://registry.npmmirror.com/bonjour/-/bonjour-3.5.0.tgz", + "integrity": "sha512-RaVTblr+OnEli0r/ud8InrU7D+G0y6aJhlxaLa6Pwty4+xoxboF1BsUI45tujvRpbj9dQVoglChqonGAsjEBYg==", + "dev": true, + "dependencies": { + "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" + } + }, + "node_modules/bonjour/node_modules/array-flatten": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/array-flatten/-/array-flatten-2.1.2.tgz", + "integrity": "sha512-hNfzcOV8W4NdualtqBFPyVO+54DSJuZGY9qT4pRroB6S9e3iiido2ISIC5h9R2sPJ8H3FHCIiEnsv1lPXO3KtQ==", + "dev": true + }, + "node_modules/boolbase": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/boolbase/-/boolbase-1.0.0.tgz", + "integrity": "sha512-JZOSA7Mo9sNGB8+UjSgzdLtokWAky1zbztM3WRLCbZ70/3cTANmQmOdR7y2g+J0e2WXywy1yS468tY+IruqEww==" + }, + "node_modules/bootstrap": { + "version": "3.4.1", + "resolved": "https://registry.npmmirror.com/bootstrap/-/bootstrap-3.4.1.tgz", + "integrity": "sha512-yN5oZVmRCwe5aKwzRj6736nSmKDX7pLYwsXiCj/EYmo16hODaBiT4En5btW/jhBF/seV+XMx3aYwukYC3A49DA==", + "engines": { + "node": ">=6" + } + }, + "node_modules/brace-expansion": { + "version": "1.1.11", + "resolved": "https://registry.npmmirror.com/brace-expansion/-/brace-expansion-1.1.11.tgz", + "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/braces": { + "version": "2.3.2", + "resolved": "https://registry.npmmirror.com/braces/-/braces-2.3.2.tgz", + "integrity": "sha512-aNdbnj9P8PjdXU4ybaWLK2IF3jc/EoDYbC7AazW6to3TRsfXxscC9UXOB5iDiEQrkyIbWp2SLQda4+QAa7nc3w==", + "dependencies": { + "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" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/braces/node_modules/extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha512-zCnTtlxNoAiDc3gqY2aYAWFx7XWWiasuF2K8Me5WbN8otHKTUKBwjPtNpRs/rbUZm7KxWAaNj7P1a/p52GbVug==", + "dependencies": { + "is-extendable": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/brorand": { + "version": "1.1.0", + "resolved": "https://registry.npmmirror.com/brorand/-/brorand-1.1.0.tgz", + "integrity": "sha512-cKV8tMCEpQs4hK/ik71d6LrPOnpkpGBR0wzxqr68g2m/LB2GxVYQroAjMJZRVM1Y4BCjCKc3vAamxSzOY2RP+w==" + }, + "node_modules/browserify-aes": { + "version": "1.2.0", + "resolved": "https://registry.npmmirror.com/browserify-aes/-/browserify-aes-1.2.0.tgz", + "integrity": "sha512-+7CHXqGuspUn/Sl5aO7Ea0xWGAtETPXNSAjHo48JfLdPWcMng33Xe4znFvQweqc/uzk5zSOI3H52CYnjCfb5hA==", + "dependencies": { + "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" + } + }, + "node_modules/browserify-cipher": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/browserify-cipher/-/browserify-cipher-1.0.1.tgz", + "integrity": "sha512-sPhkz0ARKbf4rRQt2hTpAHqn47X3llLkUGn+xEJzLjwY8LRs2p0v7ljvI5EyoRO/mexrNunNECisZs+gw2zz1w==", + "dependencies": { + "browserify-aes": "^1.0.4", + "browserify-des": "^1.0.0", + "evp_bytestokey": "^1.0.0" + } + }, + "node_modules/browserify-des": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/browserify-des/-/browserify-des-1.0.2.tgz", + "integrity": "sha512-BioO1xf3hFwz4kc6iBhI3ieDFompMhrMlnDFC4/0/vd5MokpuAc3R+LYbwTA9A5Yc9pq9UYPqffKpW2ObuwX5A==", + "dependencies": { + "cipher-base": "^1.0.1", + "des.js": "^1.0.0", + "inherits": "^2.0.1", + "safe-buffer": "^5.1.2" + } + }, + "node_modules/browserify-rsa": { + "version": "4.1.0", + "resolved": "https://registry.npmmirror.com/browserify-rsa/-/browserify-rsa-4.1.0.tgz", + "integrity": "sha512-AdEER0Hkspgno2aR97SAf6vi0y0k8NuOpGnVH3O99rcA5Q6sh8QxcngtHuJ6uXwnfAXNM4Gn1Gb7/MV1+Ymbog==", + "dependencies": { + "bn.js": "^5.0.0", + "randombytes": "^2.0.1" + } + }, + "node_modules/browserify-sign": { + "version": "4.2.1", + "resolved": "https://registry.npmmirror.com/browserify-sign/-/browserify-sign-4.2.1.tgz", + "integrity": "sha512-/vrA5fguVAKKAVTNJjgSm1tRQDHUU6DbwO9IROu/0WAzC8PKhucDSh18J0RMvVeHAn5puMd+QHC2erPRNf8lmg==", + "dependencies": { + "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" + } + }, + "node_modules/browserify-sign/node_modules/readable-stream": { + "version": "3.6.0", + "resolved": "https://registry.npmmirror.com/readable-stream/-/readable-stream-3.6.0.tgz", + "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", + "dependencies": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/browserify-sign/node_modules/safe-buffer": { + "version": "5.2.1", + "resolved": "https://registry.npmmirror.com/safe-buffer/-/safe-buffer-5.2.1.tgz", + "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==" + }, + "node_modules/browserify-zlib": { + "version": "0.2.0", + "resolved": "https://registry.npmmirror.com/browserify-zlib/-/browserify-zlib-0.2.0.tgz", + "integrity": "sha512-Z942RysHXmJrhqk88FmKBVq/v5tqmSkDz7p54G/MGyjMnCFFnC79XWNbg+Vta8W6Wb2qtSZTSxIGkJrRpCFEiA==", + "dependencies": { + "pako": "~1.0.5" + } + }, + "node_modules/browserslist": { + "version": "4.20.3", + "resolved": "https://registry.npmmirror.com/browserslist/-/browserslist-4.20.3.tgz", + "integrity": "sha512-NBhymBQl1zM0Y5dQT/O+xiLP9/rzOIQdKM/eMJBAq7yBgaB6krIYLGejrwVYnSHZdqjscB1SPuAjHwxjvN6Wdg==", + "dev": true, + "dependencies": { + "caniuse-lite": "^1.0.30001332", + "electron-to-chromium": "^1.4.118", + "escalade": "^3.1.1", + "node-releases": "^2.0.3", + "picocolors": "^1.0.0" + }, + "bin": { + "browserslist": "cli.js" + }, + "engines": { + "node": "^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7" + } + }, + "node_modules/buffer": { + "version": "4.9.2", + "resolved": "https://registry.npmmirror.com/buffer/-/buffer-4.9.2.tgz", + "integrity": "sha512-xq+q3SRMOxGivLhBNaUdC64hDTQwejJ+H0T/NB1XMtTVEwNTrfFF3gAxiyW0Bu/xWEGhjVKgUcMhCrUy2+uCWg==", + "dependencies": { + "base64-js": "^1.0.2", + "ieee754": "^1.1.4", + "isarray": "^1.0.0" + } + }, + "node_modules/buffer-crc32": { + "version": "0.2.13", + "resolved": "https://registry.npmmirror.com/buffer-crc32/-/buffer-crc32-0.2.13.tgz", + "integrity": "sha512-VO9Ht/+p3SN7SKWqcrgEzjGbRSJYTx+Q1pTQC0wrWqHx0vpJraQ6GtHx8tvcg1rlK1byhU5gccxgOgj7B0TDkQ==", + "dev": true, + "engines": { + "node": "*" + } + }, + "node_modules/buffer-from": { + "version": "1.1.2", + "resolved": "https://registry.npmmirror.com/buffer-from/-/buffer-from-1.1.2.tgz", + "integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==" + }, + "node_modules/buffer-indexof": { + "version": "1.1.1", + "resolved": "https://registry.npmmirror.com/buffer-indexof/-/buffer-indexof-1.1.1.tgz", + "integrity": "sha512-4/rOEg86jivtPTeOUUT61jJO1Ya1TrR/OkqCSZDyq84WJh3LuuiphBYJN+fm5xufIk4XAFcEwte/8WzC8If/1g==", + "dev": true + }, + "node_modules/buffer-xor": { + "version": "1.0.3", + "resolved": "https://registry.npmmirror.com/buffer-xor/-/buffer-xor-1.0.3.tgz", + "integrity": "sha512-571s0T7nZWK6vB67HI5dyUF7wXiNcfaPPPTl6zYCNApANjIvYJTg7hlud/+cJpdAhS7dVzqMLmfhfHR3rAcOjQ==" + }, + "node_modules/builtin-status-codes": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/builtin-status-codes/-/builtin-status-codes-3.0.0.tgz", + "integrity": "sha512-HpGFw18DgFWlncDfjTa2rcQ4W88O1mC8e8yZ2AvQY5KDaktSTwo+KRf6nHK6FRI5FyRyb/5T6+TSxfP7QyGsmQ==" + }, + "node_modules/builtins": { + "version": "1.0.3", + "resolved": "https://registry.npmmirror.com/builtins/-/builtins-1.0.3.tgz", + "integrity": "sha512-uYBjakWipfaO/bXI7E8rq6kpwHRZK5cNYrUv2OzZSI/FvmdMyXJ2tG9dKcjEC5YHmHpUAwsargWIZNWdxb/bnQ==", + "dev": true + }, + "node_modules/butterfly-dag": { + "version": "4.1.23", + "resolved": "https://registry.npmmirror.com/butterfly-dag/-/butterfly-dag-4.1.23.tgz", + "integrity": "sha512-8B7HI5sexTQx9Wexy7pOSVwyMxzKmQxnmz6S7gWzzRuP6O/8Luny4edzkUjnnREE5PBplxK+c6PSJKrny901mQ==", + "dependencies": { + "@antv/hierarchy": "~0.6.4", + "@antv/matrix-util": "^3.0.4", + "d3-force": "~2.1.1", + "dagre": "~0.8.5", + "dom-to-image": "~2.6.0", + "eventemitter3": "4.0.7", + "ml-matrix": "^6.5.0" + }, + "peerDependencies": { + "jquery": ">=2.0.0", + "lodash": ">=4.0.0" + } + }, + "node_modules/byline": { + "version": "5.0.0", + "resolved": "https://registry.npmmirror.com/byline/-/byline-5.0.0.tgz", + "integrity": "sha512-s6webAy+R4SR8XVuJWt2V2rGvhnrhxN+9S15GNuTK3wKPOXFF6RNc+8ug2XhH+2s4f+uudG4kUVYmYOQWL2g0Q==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/byte-size": { + "version": "7.0.1", + "resolved": "https://registry.npmmirror.com/byte-size/-/byte-size-7.0.1.tgz", + "integrity": "sha512-crQdqyCwhokxwV1UyDzLZanhkugAgft7vt0qbbdt60C6Zf3CAiGmtUCylbtYwrU6loOUw3euGrNtW1J651ot1A==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/bytes": { + "version": "3.1.2", + "resolved": "https://registry.npmmirror.com/bytes/-/bytes-3.1.2.tgz", + "integrity": "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==", + "dev": true, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/cacache": { + "version": "12.0.4", + "resolved": "https://registry.npmmirror.com/cacache/-/cacache-12.0.4.tgz", + "integrity": "sha512-a0tMB40oefvuInr4Cwb3GerbL9xTj1D5yg0T5xrjGCGyfvbxseIXX7BAO/u/hIXdafzOI5JC3wDwHyf24buOAQ==", + "dependencies": { + "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" + } + }, + "node_modules/cache-base": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/cache-base/-/cache-base-1.0.1.tgz", + "integrity": "sha512-AKcdTnFSWATd5/GCPRxr2ChwIJ85CeyrEyjRHlKxQ56d4XJMGym0uAiKn0xbLOGOl3+yRpOTi484dVCEc5AUzQ==", + "dependencies": { + "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" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cache-loader": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/cache-loader/-/cache-loader-2.0.1.tgz", + "integrity": "sha512-V99T3FOynmGx26Zom+JrVBytLBsmUCzVG2/4NnUKgvXN4bEV42R1ERl1IyiH/cvFIDA1Ytq2lPZ9tXDSahcQpQ==", + "dev": true, + "dependencies": { + "loader-utils": "^1.1.0", + "mkdirp": "^0.5.1", + "neo-async": "^2.6.0", + "normalize-path": "^3.0.0", + "schema-utils": "^1.0.0" + }, + "engines": { + "node": ">= 6.9.0" + }, + "peerDependencies": { + "webpack": "^4.0.0" + } + }, + "node_modules/call-bind": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/call-bind/-/call-bind-1.0.2.tgz", + "integrity": "sha512-7O+FbCihrB5WGbFYesctwmTKae6rOiIzmz1icreWJ+0aA7LJfuqhEso2T9ncpcFtzMQtzXf2QGGueWJGTYsqrA==", + "dependencies": { + "function-bind": "^1.1.1", + "get-intrinsic": "^1.0.2" + } + }, + "node_modules/call-me-maybe": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/call-me-maybe/-/call-me-maybe-1.0.1.tgz", + "integrity": "sha512-wCyFsDQkKPwwF8BDwOiWNx/9K45L/hvggQiDbve+viMNMQnWhrlYIuBk09offfwCRtCO9P6XwUttufzU11WCVw==", + "dev": true + }, + "node_modules/caller-callsite": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/caller-callsite/-/caller-callsite-2.0.0.tgz", + "integrity": "sha512-JuG3qI4QOftFsZyOn1qq87fq5grLIyk1JYd5lJmdA+fG7aQ9pA/i3JIJGcO3q0MrRcHlOt1U+ZeHW8Dq9axALQ==", + "dev": true, + "dependencies": { + "callsites": "^2.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/caller-callsite/node_modules/callsites": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/callsites/-/callsites-2.0.0.tgz", + "integrity": "sha512-ksWePWBloaWPxJYQ8TL0JHvtci6G5QTKwQ95RcWAa/lzoAKuAOflGdAK92hpHXjkwb8zLxoLNUoNYZgVsaJzvQ==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/caller-path": { + "version": "0.1.0", + "resolved": "https://registry.npmmirror.com/caller-path/-/caller-path-0.1.0.tgz", + "integrity": "sha512-UJiE1otjXPF5/x+T3zTnSFiTOEmJoGTD9HmBoxnCUwho61a2eSNn/VwtwuIBDAo2SEOv1AJ7ARI5gCmohFLu/g==", + "dev": true, + "optional": true, + "dependencies": { + "callsites": "^0.2.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/callsites": { + "version": "0.2.0", + "resolved": "https://registry.npmmirror.com/callsites/-/callsites-0.2.0.tgz", + "integrity": "sha512-Zv4Dns9IbXXmPkgRRUjAaJQgfN4xX5p6+RQFhWUqscdvvK2xK/ZL8b3IXIJsj+4sD+f24NwnWy2BY8AJ82JB0A==", + "dev": true, + "optional": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/camel-case": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/camel-case/-/camel-case-3.0.0.tgz", + "integrity": "sha512-+MbKztAYHXPr1jNTSKQF52VpcFjwY5RkR7fxksV8Doo4KAYc5Fl4UJRgthBbTmEx8C54DqahhbLJkDwjI3PI/w==", + "dev": true, + "dependencies": { + "no-case": "^2.2.0", + "upper-case": "^1.1.1" + } + }, + "node_modules/camelcase": { + "version": "5.3.1", + "resolved": "https://registry.npmmirror.com/camelcase/-/camelcase-5.3.1.tgz", + "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/camelcase-keys": { + "version": "6.2.2", + "resolved": "https://registry.npmmirror.com/camelcase-keys/-/camelcase-keys-6.2.2.tgz", + "integrity": "sha512-YrwaA0vEKazPBkn0ipTiMpSajYDSe+KjQfrjhcBMxJt/znbvlHd8Pw/Vamaz5EB4Wfhs3SUR3Z9mwRu/P3s3Yg==", + "dev": true, + "dependencies": { + "camelcase": "^5.3.1", + "map-obj": "^4.0.0", + "quick-lru": "^4.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/caniuse-api": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/caniuse-api/-/caniuse-api-3.0.0.tgz", + "integrity": "sha512-bsTwuIg/BZZK/vreVTYYbSWoe2F+71P7K5QGEX+pT250DZbfU1MQ5prOKpPR+LL6uWKK3KMwMCAS74QB3Um1uw==", + "dev": true, + "dependencies": { + "browserslist": "^4.0.0", + "caniuse-lite": "^1.0.0", + "lodash.memoize": "^4.1.2", + "lodash.uniq": "^4.5.0" + } + }, + "node_modules/caniuse-lite": { + "version": "1.0.30001332", + "resolved": "https://registry.npmmirror.com/caniuse-lite/-/caniuse-lite-1.0.30001332.tgz", + "integrity": "sha512-10T30NYOEQtN6C11YGg411yebhvpnC6Z102+B95eAsN0oB6KUs01ivE8u+G6FMIRtIrVlYXhL+LUwQ3/hXwDWw==", + "dev": true + }, + "node_modules/case-sensitive-paths-webpack-plugin": { + "version": "2.4.0", + "resolved": "https://registry.npmmirror.com/case-sensitive-paths-webpack-plugin/-/case-sensitive-paths-webpack-plugin-2.4.0.tgz", + "integrity": "sha512-roIFONhcxog0JSSWbvVAh3OocukmSgpqOH6YpMkCvav/ySIV3JKg4Dc8vYtQjYi/UxpNE36r/9v+VqTQqgkYmw==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/caseless": { + "version": "0.12.0", + "resolved": "https://registry.npmmirror.com/caseless/-/caseless-0.12.0.tgz", + "integrity": "sha512-4tYFyifaFfGacoiObjJegolkwSU4xQNGbVgUiNYVUxbQ2x2lUsFvY4hVgVzGiIe6WLOPqycWXA40l+PWsxthUw==", + "dev": true + }, + "node_modules/chalk": { + "version": "2.4.2", + "resolved": "https://registry.npmmirror.com/chalk/-/chalk-2.4.2.tgz", + "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", + "dependencies": { + "ansi-styles": "^3.2.1", + "escape-string-regexp": "^1.0.5", + "supports-color": "^5.3.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/charcodes": { + "version": "0.2.0", + "resolved": "https://registry.npmmirror.com/charcodes/-/charcodes-0.2.0.tgz", + "integrity": "sha512-Y4kiDb+AM4Ecy58YkuZrrSRJBDQdQ2L+NyS1vHHFtNtUjgutcZfx3yp1dAONI/oPaPmyGfCLx5CxL+zauIMyKQ==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/chardet": { + "version": "0.4.2", + "resolved": "https://registry.npmmirror.com/chardet/-/chardet-0.4.2.tgz", + "integrity": "sha512-j/Toj7f1z98Hh2cYo2BVr85EpIRWqUi7rtRSGxh/cqUjqrnJe9l9UE7IUGd2vQ2p+kSHLkSzObQPZPLUC6TQwg==", + "dev": true, + "optional": true + }, + "node_modules/charenc": { + "version": "0.0.2", + "resolved": "https://registry.npmmirror.com/charenc/-/charenc-0.0.2.tgz", + "integrity": "sha512-yrLQ/yVUFXkzg7EDQsPieE/53+0RlaWTs+wBrvW36cyilJ2SaDWfl4Yj7MtLTXleV9uEKefbAGUPv2/iWSooRA==", + "engines": { + "node": "*" + } + }, + "node_modules/check-types": { + "version": "8.0.3", + "resolved": "https://registry.npmmirror.com/check-types/-/check-types-8.0.3.tgz", + "integrity": "sha512-YpeKZngUmG65rLudJ4taU7VLkOCTMhNl/u4ctNC56LQS/zJTyNH0Lrtwm1tfTsbLlwvlfsA2d1c8vCf/Kh2KwQ==", + "dev": true + }, + "node_modules/cheerio": { + "version": "1.0.0-rc.10", + "resolved": "https://registry.npmmirror.com/cheerio/-/cheerio-1.0.0-rc.10.tgz", + "integrity": "sha512-g0J0q/O6mW8z5zxQ3A8E8J1hUgp4SMOvEoW/x84OwyHKe/Zccz83PVT4y5Crcr530FV6NgmKI1qvGTKVl9XXVw==", + "dev": true, + "dependencies": { + "cheerio-select": "^1.5.0", + "dom-serializer": "^1.3.2", + "domhandler": "^4.2.0", + "htmlparser2": "^6.1.0", + "parse5": "^6.0.1", + "parse5-htmlparser2-tree-adapter": "^6.0.1", + "tslib": "^2.2.0" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/cheerio-select": { + "version": "1.6.0", + "resolved": "https://registry.npmmirror.com/cheerio-select/-/cheerio-select-1.6.0.tgz", + "integrity": "sha512-eq0GdBvxVFbqWgmCm7M3XGs1I8oLy/nExUnh6oLqmBditPO9AqQJrkslDpMun/hZ0yyTs8L0m85OHp4ho6Qm9g==", + "dev": true, + "dependencies": { + "css-select": "^4.3.0", + "css-what": "^6.0.1", + "domelementtype": "^2.2.0", + "domhandler": "^4.3.1", + "domutils": "^2.8.0" + } + }, + "node_modules/cheerio-select/node_modules/css-select": { + "version": "4.3.0", + "resolved": "https://registry.npmmirror.com/css-select/-/css-select-4.3.0.tgz", + "integrity": "sha512-wPpOYtnsVontu2mODhA19JrqWxNsfdatRKd64kmpRbQgh1KtItko5sTnEpPdpSaJszTOhEMlF/RPz28qj4HqhQ==", + "dev": true, + "dependencies": { + "boolbase": "^1.0.0", + "css-what": "^6.0.1", + "domhandler": "^4.3.1", + "domutils": "^2.8.0", + "nth-check": "^2.0.1" + } + }, + "node_modules/cheerio-select/node_modules/css-what": { + "version": "6.1.0", + "resolved": "https://registry.npmmirror.com/css-what/-/css-what-6.1.0.tgz", + "integrity": "sha512-HTUrgRJ7r4dsZKU6GjmpfRK1O76h97Z8MfS1G0FozR+oF2kG6Vfe8JE6zwrkbxigziPHinCJ+gCPjA9EaBDtRw==", + "dev": true, + "engines": { + "node": ">= 6" + } + }, + "node_modules/cheerio-select/node_modules/dom-serializer": { + "version": "1.4.1", + "resolved": "https://registry.npmmirror.com/dom-serializer/-/dom-serializer-1.4.1.tgz", + "integrity": "sha512-VHwB3KfrcOOkelEG2ZOfxqLZdfkil8PtJi4P8N2MMXucZq2yLp75ClViUlOVwyoHEDjYU433Aq+5zWP61+RGag==", + "dev": true, + "dependencies": { + "domelementtype": "^2.0.1", + "domhandler": "^4.2.0", + "entities": "^2.0.0" + } + }, + "node_modules/cheerio-select/node_modules/domelementtype": { + "version": "2.3.0", + "resolved": "https://registry.npmmirror.com/domelementtype/-/domelementtype-2.3.0.tgz", + "integrity": "sha512-OLETBj6w0OsagBwdXnPdN0cnMfF9opN69co+7ZrbfPGrdpPVNBUj02spi6B1N7wChLQiPn4CSH/zJvXw56gmHw==", + "dev": true + }, + "node_modules/cheerio-select/node_modules/domutils": { + "version": "2.8.0", + "resolved": "https://registry.npmmirror.com/domutils/-/domutils-2.8.0.tgz", + "integrity": "sha512-w96Cjofp72M5IIhpjgobBimYEfoPjx1Vx0BSX9P30WBdZW2WIKU0T1Bd0kz2eNZ9ikjKgHbEyKx8BB6H1L3h3A==", + "dev": true, + "dependencies": { + "dom-serializer": "^1.0.1", + "domelementtype": "^2.2.0", + "domhandler": "^4.2.0" + } + }, + "node_modules/cheerio-select/node_modules/nth-check": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/nth-check/-/nth-check-2.0.1.tgz", + "integrity": "sha512-it1vE95zF6dTT9lBsYbxvqh0Soy4SPowchj0UBGj/V6cTPnXXtQOPUbhZ6CmGzAD/rW22LQK6E96pcdJXk4A4w==", + "dev": true, + "dependencies": { + "boolbase": "^1.0.0" + } + }, + "node_modules/cheerio/node_modules/dom-serializer": { + "version": "1.4.1", + "resolved": "https://registry.npmmirror.com/dom-serializer/-/dom-serializer-1.4.1.tgz", + "integrity": "sha512-VHwB3KfrcOOkelEG2ZOfxqLZdfkil8PtJi4P8N2MMXucZq2yLp75ClViUlOVwyoHEDjYU433Aq+5zWP61+RGag==", + "dev": true, + "dependencies": { + "domelementtype": "^2.0.1", + "domhandler": "^4.2.0", + "entities": "^2.0.0" + } + }, + "node_modules/cheerio/node_modules/domelementtype": { + "version": "2.3.0", + "resolved": "https://registry.npmmirror.com/domelementtype/-/domelementtype-2.3.0.tgz", + "integrity": "sha512-OLETBj6w0OsagBwdXnPdN0cnMfF9opN69co+7ZrbfPGrdpPVNBUj02spi6B1N7wChLQiPn4CSH/zJvXw56gmHw==", + "dev": true + }, + "node_modules/cheerio/node_modules/parse5": { + "version": "6.0.1", + "resolved": "https://registry.npmmirror.com/parse5/-/parse5-6.0.1.tgz", + "integrity": "sha512-Ofn/CTFzRGTTxwpNEs9PP93gXShHcTq255nzRYSKe8AkVpZY7e1fpmTfOyoIvjP5HG7Z2ZM7VS9PPhQGW2pOpw==", + "dev": true + }, + "node_modules/chokidar": { + "version": "3.5.3", + "resolved": "https://registry.npmmirror.com/chokidar/-/chokidar-3.5.3.tgz", + "integrity": "sha512-Dr3sfKRP6oTcjf2JmUmFJfeVMvXBdegxB0iVQ5eb2V10uFJUCAS8OByZdVAyVb8xXNz3GjjTgj9kLWsZTqE6kw==", + "optional": true, + "dependencies": { + "anymatch": "~3.1.2", + "braces": "~3.0.2", + "glob-parent": "~5.1.2", + "is-binary-path": "~2.1.0", + "is-glob": "~4.0.1", + "normalize-path": "~3.0.0", + "readdirp": "~3.6.0" + }, + "engines": { + "node": ">= 8.10.0" + }, + "optionalDependencies": { + "fsevents": "~2.3.2" + } + }, + "node_modules/chokidar/node_modules/braces": { + "version": "3.0.2", + "resolved": "https://registry.npmmirror.com/braces/-/braces-3.0.2.tgz", + "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", + "optional": true, + "dependencies": { + "fill-range": "^7.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/chokidar/node_modules/fill-range": { + "version": "7.0.1", + "resolved": "https://registry.npmmirror.com/fill-range/-/fill-range-7.0.1.tgz", + "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", + "optional": true, + "dependencies": { + "to-regex-range": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/chokidar/node_modules/is-number": { + "version": "7.0.0", + "resolved": "https://registry.npmmirror.com/is-number/-/is-number-7.0.0.tgz", + "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", + "optional": true, + "engines": { + "node": ">=0.12.0" + } + }, + "node_modules/chokidar/node_modules/to-regex-range": { + "version": "5.0.1", + "resolved": "https://registry.npmmirror.com/to-regex-range/-/to-regex-range-5.0.1.tgz", + "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", + "optional": true, + "dependencies": { + "is-number": "^7.0.0" + }, + "engines": { + "node": ">=8.0" + } + }, + "node_modules/chownr": { + "version": "1.1.4", + "resolved": "https://registry.npmmirror.com/chownr/-/chownr-1.1.4.tgz", + "integrity": "sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg==" + }, + "node_modules/chrome-trace-event": { + "version": "1.0.3", + "resolved": "https://registry.npmmirror.com/chrome-trace-event/-/chrome-trace-event-1.0.3.tgz", + "integrity": "sha512-p3KULyQg4S7NIHixdwbGX+nFHkoBiA4YQmyWtjb8XngSKV124nJmRysgAeujbUVb15vh+RvFUfCPqU7rXk+hZg==", + "engines": { + "node": ">=6.0" + } + }, + "node_modules/ci-info": { + "version": "1.6.0", + "resolved": "https://registry.npmmirror.com/ci-info/-/ci-info-1.6.0.tgz", + "integrity": "sha512-vsGdkwSCDpWmP80ncATX7iea5DWQemg1UgCW5J8tqjU3lYw4FBYuj89J0CTVomA7BEfvSZd84GmHko+MxFQU2A==", + "dev": true + }, + "node_modules/cipher-base": { + "version": "1.0.4", + "resolved": "https://registry.npmmirror.com/cipher-base/-/cipher-base-1.0.4.tgz", + "integrity": "sha512-Kkht5ye6ZGmwv40uUDZztayT2ThLQGfnj/T71N/XzeZeo3nf8foyW7zGTsPYkEya3m5f3cAypH+qe7YOrM1U2Q==", + "dependencies": { + "inherits": "^2.0.1", + "safe-buffer": "^5.0.1" + } + }, + "node_modules/circular-json": { + "version": "0.3.3", + "resolved": "https://registry.npmmirror.com/circular-json/-/circular-json-0.3.3.tgz", + "integrity": "sha512-UZK3NBx2Mca+b5LsG7bY183pHWt5Y1xts4P3Pz7ENTwGVnJOUWbRb3ocjvX7hx9tq/yTAdclXm9sZ38gNuem4A==", + "deprecated": "CircularJSON is in maintenance only, flatted is its successor.", + "dev": true, + "optional": true + }, + "node_modules/class-utils": { + "version": "0.3.6", + "resolved": "https://registry.npmmirror.com/class-utils/-/class-utils-0.3.6.tgz", + "integrity": "sha512-qOhPa/Fj7s6TY8H8esGu5QNpMMQxz79h+urzrNYN6mn+9BnxlDGf5QZ+XeCDsxSjPqsSR56XOZOJmpeurnLMeg==", + "dependencies": { + "arr-union": "^3.1.0", + "define-property": "^0.2.5", + "isobject": "^3.0.0", + "static-extend": "^0.1.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/class-utils/node_modules/define-property": { + "version": "0.2.5", + "resolved": "https://registry.npmmirror.com/define-property/-/define-property-0.2.5.tgz", + "integrity": "sha512-Rr7ADjQZenceVOAKop6ALkkRAmH1A4Gx9hV/7ZujPUN2rkATqFO0JZLZInbAjpZYoJ1gUx8MRMQVkYemcbMSTA==", + "dependencies": { + "is-descriptor": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/clean-css": { + "version": "4.2.4", + "resolved": "https://registry.npmmirror.com/clean-css/-/clean-css-4.2.4.tgz", + "integrity": "sha512-EJUDT7nDVFDvaQgAo2G/PJvxmp1o/c6iXLbswsBbUFXi1Nr+AjA2cKmfbKDMjMvzEe75g3P6JkaDDAKk96A85A==", + "dev": true, + "dependencies": { + "source-map": "~0.6.0" + }, + "engines": { + "node": ">= 4.0" + } + }, + "node_modules/clean-css/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmmirror.com/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/clean-stack": { + "version": "2.2.0", + "resolved": "https://registry.npmmirror.com/clean-stack/-/clean-stack-2.2.0.tgz", + "integrity": "sha512-4diC9HaTE+KRAMWhDhrGOECgWZxoevMc5TlkObMqNSsVU62PYzXZ/SMTjzyGAFF1YusgxGcSWTEXBhp0CPwQ1A==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/cli-cursor": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/cli-cursor/-/cli-cursor-2.1.0.tgz", + "integrity": "sha512-8lgKz8LmCRYZZQDpRyT2m5rKJ08TnU4tR9FFFW2rxpxR1FzWi4PQ/NfyODchAatHaUgnSPVcx/R5w6NuTBzFiw==", + "dev": true, + "dependencies": { + "restore-cursor": "^2.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/cli-highlight": { + "version": "2.1.11", + "resolved": "https://registry.npmmirror.com/cli-highlight/-/cli-highlight-2.1.11.tgz", + "integrity": "sha512-9KDcoEVwyUXrjcJNvHD0NFc/hiwe/WPVYIleQh2O1N2Zro5gWJZ/K+3DGn8w8P/F6FxOgzyC5bxDyHIgCSPhGg==", + "dev": true, + "dependencies": { + "chalk": "^4.0.0", + "highlight.js": "^10.7.1", + "mz": "^2.4.0", + "parse5": "^5.1.1", + "parse5-htmlparser2-tree-adapter": "^6.0.0", + "yargs": "^16.0.0" + }, + "bin": { + "highlight": "bin/highlight" + }, + "engines": { + "node": ">=8.0.0", + "npm": ">=5.0.0" + } + }, + "node_modules/cli-highlight/node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmmirror.com/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/cli-highlight/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmmirror.com/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/cli-highlight/node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/cli-highlight/node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmmirror.com/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "node_modules/cli-highlight/node_modules/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/cli-highlight/node_modules/highlight.js": { + "version": "10.7.3", + "resolved": "https://registry.npmmirror.com/highlight.js/-/highlight.js-10.7.3.tgz", + "integrity": "sha512-tzcUFauisWKNHaRkN4Wjl/ZA07gENAjFl3J/c480dprkGTg5EQstgaNFqBfUqCq54kZRIEcreTsAgF/m2quD7A==", + "dev": true, + "engines": { + "node": "*" + } + }, + "node_modules/cli-highlight/node_modules/supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmmirror.com/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/cli-spinners": { + "version": "2.6.1", + "resolved": "https://registry.npmmirror.com/cli-spinners/-/cli-spinners-2.6.1.tgz", + "integrity": "sha512-x/5fWmGMnbKQAaNwN+UZlV79qBLM9JFnJuJ03gIi5whrob0xV0ofNVHy9DhwGdsMJQc2OKv0oGmLzvaqvAVv+g==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/cli-truncate": { + "version": "0.2.1", + "resolved": "https://registry.npmmirror.com/cli-truncate/-/cli-truncate-0.2.1.tgz", + "integrity": "sha512-f4r4yJnbT++qUPI9NR4XLDLq41gQ+uqnPItWG0F5ZkehuNiTTa3EY0S4AqTSUOeJ7/zU41oWPQSNkW5BqPL9bg==", + "dev": true, + "dependencies": { + "slice-ansi": "0.0.4", + "string-width": "^1.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cli-truncate/node_modules/ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmmirror.com/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha512-TIGnTpdo+E3+pCyAluZvtED5p5wCqLdezCyhPZzKPcxvFplEt4i+W7OONCKgeZFT3+y5NZZfOOS/Bdcanm1MYA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cli-truncate/node_modules/is-fullwidth-code-point": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz", + "integrity": "sha512-1pqUqRjkhPJ9miNq9SwMfdvi6lBJcd6eFxvfaivQhaH3SgisfiuudvFntdKOmxuee/77l+FPjKrQjWvmPjWrRw==", + "dev": true, + "dependencies": { + "number-is-nan": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cli-truncate/node_modules/slice-ansi": { + "version": "0.0.4", + "resolved": "https://registry.npmmirror.com/slice-ansi/-/slice-ansi-0.0.4.tgz", + "integrity": "sha512-up04hB2hR92PgjpyU3y/eg91yIBILyjVY26NvvciY3EVVPjybkMszMpXQ9QAkcS3I5rtJBDLoTxxg+qvW8c7rw==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cli-truncate/node_modules/string-width": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/string-width/-/string-width-1.0.2.tgz", + "integrity": "sha512-0XsVpQLnVCXHJfyEs8tC0zpTVIr5PKKsQtkT29IwupnPTjtPmQ3xT/4yCREF9hYkV/3M3kzcUTSAZT6a6h81tw==", + "dev": true, + "dependencies": { + "code-point-at": "^1.0.0", + "is-fullwidth-code-point": "^1.0.0", + "strip-ansi": "^3.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cli-truncate/node_modules/strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmmirror.com/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha512-VhumSSbBqDTP8p2ZLKj40UjBCV4+v8bUSEpUb4KjRgWk9pbqGF4REFj6KEagidb2f/M6AzC0EmFyDNGaw9OCzg==", + "dev": true, + "dependencies": { + "ansi-regex": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cli-width": { + "version": "2.2.1", + "resolved": "https://registry.npmmirror.com/cli-width/-/cli-width-2.2.1.tgz", + "integrity": "sha512-GRMWDxpOB6Dgk2E5Uo+3eEBvtOOlimMmpbFiKuLFnQzYDavtLFY3K5ona41jgN/WdRZtG7utuVSVTL4HbZHGkw==", + "dev": true, + "optional": true + }, + "node_modules/clipboard": { + "version": "2.0.8", + "resolved": "https://registry.npmmirror.com/clipboard/-/clipboard-2.0.8.tgz", + "integrity": "sha512-Y6WO0unAIQp5bLmk1zdThRhgJt/x3ks6f30s3oE3H1mgIEU33XyQjEf8gsf6DxC7NPX8Y1SsNWjUjL/ywLnnbQ==", + "dependencies": { + "good-listener": "^1.2.2", + "select": "^1.1.2", + "tiny-emitter": "^2.0.0" + } + }, + "node_modules/clipboardy": { + "version": "2.3.0", + "resolved": "https://registry.npmmirror.com/clipboardy/-/clipboardy-2.3.0.tgz", + "integrity": "sha512-mKhiIL2DrQIsuXMgBgnfEHOZOryC7kY7YO//TN6c63wlEm3NG5tz+YgY5rVi29KCmq/QQjKYvM7a19+MDOTHOQ==", + "dev": true, + "dependencies": { + "arch": "^2.1.1", + "execa": "^1.0.0", + "is-wsl": "^2.1.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/clipboardy/node_modules/is-wsl": { + "version": "2.2.0", + "resolved": "https://registry.npmmirror.com/is-wsl/-/is-wsl-2.2.0.tgz", + "integrity": "sha512-fKzAra0rGJUUBwGBgNkHZuToZcn+TtXHpeCgmkMJMMYx1sQDYaCSyjJBSCa2nH1DGm7s3n1oBnohoVTBaN7Lww==", + "dev": true, + "dependencies": { + "is-docker": "^2.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/cliui": { + "version": "5.0.0", + "resolved": "https://registry.npmmirror.com/cliui/-/cliui-5.0.0.tgz", + "integrity": "sha512-PYeGSEmmHM6zvoef2w8TPzlrnNpXIjTipYK780YswmIP9vjxmd6Y2a3CB2Ks6/AU8NHjZugXvo8w3oWM2qnwXA==", + "dev": true, + "dependencies": { + "string-width": "^3.1.0", + "strip-ansi": "^5.2.0", + "wrap-ansi": "^5.1.0" + } + }, + "node_modules/cliui/node_modules/emoji-regex": { + "version": "7.0.3", + "resolved": "https://registry.npmmirror.com/emoji-regex/-/emoji-regex-7.0.3.tgz", + "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", + "dev": true + }, + "node_modules/cliui/node_modules/string-width": { + "version": "3.1.0", + "resolved": "https://registry.npmmirror.com/string-width/-/string-width-3.1.0.tgz", + "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", + "dev": true, + "dependencies": { + "emoji-regex": "^7.0.1", + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^5.1.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/cliui/node_modules/wrap-ansi": { + "version": "5.1.0", + "resolved": "https://registry.npmmirror.com/wrap-ansi/-/wrap-ansi-5.1.0.tgz", + "integrity": "sha512-QC1/iN/2/RPVJ5jYK8BGttj5z83LmSKmvbvrXPNCLZSEb32KKVDJDl/MOt2N01qU2H/FkzEa9PKto1BqDjtd7Q==", + "dev": true, + "dependencies": { + "ansi-styles": "^3.2.0", + "string-width": "^3.0.0", + "strip-ansi": "^5.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/clone": { + "version": "1.0.4", + "resolved": "https://registry.npmmirror.com/clone/-/clone-1.0.4.tgz", + "integrity": "sha512-JQHZ2QMW6l3aH/j6xCqQThY/9OH4D/9ls34cgkUBiEeocRTU04tHfKPBsUK1PqZCUQM7GiA0IIXJSuXHI64Kbg==", + "dev": true, + "engines": { + "node": ">=0.8" + } + }, + "node_modules/clone-deep": { + "version": "4.0.1", + "resolved": "https://registry.npmmirror.com/clone-deep/-/clone-deep-4.0.1.tgz", + "integrity": "sha512-neHB9xuzh/wk0dIHweyAXv2aPGZIVk3pLMe+/RNzINf17fe0OG96QroktYAUm7SM1PBnzTabaLboqqxDyMU+SQ==", + "dev": true, + "dependencies": { + "is-plain-object": "^2.0.4", + "kind-of": "^6.0.2", + "shallow-clone": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/cmd-shim": { + "version": "4.1.0", + "resolved": "https://registry.npmmirror.com/cmd-shim/-/cmd-shim-4.1.0.tgz", + "integrity": "sha512-lb9L7EM4I/ZRVuljLPEtUJOP+xiQVknZ4ZMpMgEp4JzNldPb27HU03hi6K1/6CoIuit/Zm/LQXySErFeXxDprw==", + "dev": true, + "dependencies": { + "mkdirp-infer-owner": "^2.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/co": { + "version": "4.6.0", + "resolved": "https://registry.npmmirror.com/co/-/co-4.6.0.tgz", + "integrity": "sha512-QVb0dM5HvG+uaxitm8wONl7jltx8dqhfU33DcqtOZcLSVIKSDDLDi7+0LbAKiyI8hD9u42m2YxXSkMGWThaecQ==", + "dev": true, + "optional": true, + "engines": { + "iojs": ">= 1.0.0", + "node": ">= 0.12.0" + } + }, + "node_modules/coa": { + "version": "2.0.2", + "resolved": "https://registry.npmmirror.com/coa/-/coa-2.0.2.tgz", + "integrity": "sha512-q5/jG+YQnSy4nRTV4F7lPepBJZ8qBNJJDBuJdoejDyLXgmL7IEo+Le2JDZudFTFt7mrCqIRaSjws4ygRCTCAXA==", + "dependencies": { + "@types/q": "^1.5.1", + "chalk": "^2.4.1", + "q": "^1.1.2" + }, + "engines": { + "node": ">= 4.0" + } + }, + "node_modules/code-point-at": { + "version": "1.1.0", + "resolved": "https://registry.npmmirror.com/code-point-at/-/code-point-at-1.1.0.tgz", + "integrity": "sha512-RpAVKQA5T63xEj6/giIbUEtZwJ4UFIc3ZtvEkiaUERylqe8xb5IvqcgOurZLahv93CLKfxcw5YI+DZcUBRyLXA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/codemirror": { + "version": "5.65.13", + "resolved": "https://registry.npmmirror.com/codemirror/-/codemirror-5.65.13.tgz", + "integrity": "sha512-SVWEzKXmbHmTQQWaz03Shrh4nybG0wXx2MEu3FO4ezbPW8IbnZEd5iGHGEffSUaitKYa3i+pHpBsSvw8sPHtzg==" + }, + "node_modules/collection-visit": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/collection-visit/-/collection-visit-1.0.0.tgz", + "integrity": "sha512-lNkKvzEeMBBjUGHZ+q6z9pSJla0KWAQPvtzhEV9+iGyQYG+pBpl7xKDhxoNSOZH2hhv0v5k0y2yAM4o4SjoSkw==", + "dependencies": { + "map-visit": "^1.0.0", + "object-visit": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/color": { + "version": "3.2.1", + "resolved": "https://registry.npmmirror.com/color/-/color-3.2.1.tgz", + "integrity": "sha512-aBl7dZI9ENN6fUGC7mWpMTPNHmWUSNan9tuWN6ahh5ZLNk9baLJOnSMlrQkHcrfFgz2/RigjUVAjdx36VcemKA==", + "dev": true, + "dependencies": { + "color-convert": "^1.9.3", + "color-string": "^1.6.0" + } + }, + "node_modules/color-convert": { + "version": "1.9.3", + "resolved": "https://registry.npmmirror.com/color-convert/-/color-convert-1.9.3.tgz", + "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", + "dependencies": { + "color-name": "1.1.3" + } + }, + "node_modules/color-name": { + "version": "1.1.3", + "resolved": "https://registry.npmmirror.com/color-name/-/color-name-1.1.3.tgz", + "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==" + }, + "node_modules/color-string": { + "version": "1.9.1", + "resolved": "https://registry.npmmirror.com/color-string/-/color-string-1.9.1.tgz", + "integrity": "sha512-shrVawQFojnZv6xM40anx4CkoDP+fZsw/ZerEMsW/pyzsRbElpsL/DBVW7q3ExxwusdNXI3lXpuhEZkzs8p5Eg==", + "dev": true, + "dependencies": { + "color-name": "^1.0.0", + "simple-swizzle": "^0.2.2" + } + }, + "node_modules/columnify": { + "version": "1.6.0", + "resolved": "https://registry.npmmirror.com/columnify/-/columnify-1.6.0.tgz", + "integrity": "sha512-lomjuFZKfM6MSAnV9aCZC9sc0qGbmZdfygNv+nCpqVkSKdCxCklLtd16O0EILGkImHw9ZpHkAnHaB+8Zxq5W6Q==", + "dev": true, + "dependencies": { + "strip-ansi": "^6.0.1", + "wcwidth": "^1.0.0" + }, + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/columnify/node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmmirror.com/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/columnify/node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmmirror.com/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/combined-stream": { + "version": "1.0.8", + "resolved": "https://registry.npmmirror.com/combined-stream/-/combined-stream-1.0.8.tgz", + "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", + "dev": true, + "dependencies": { + "delayed-stream": "~1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/commander": { + "version": "2.20.3", + "resolved": "https://registry.npmmirror.com/commander/-/commander-2.20.3.tgz", + "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==" + }, + "node_modules/commondir": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/commondir/-/commondir-1.0.1.tgz", + "integrity": "sha512-W9pAhw0ja1Edb5GVdIF1mjZw/ASI0AlShXM83UUGe2DVr5TdAPEA1OA8m/g8zWp9x6On7gqufY+FatDbC3MDQg==" + }, + "node_modules/compare-func": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/compare-func/-/compare-func-2.0.0.tgz", + "integrity": "sha512-zHig5N+tPWARooBnb0Zx1MFcdfpyJrfTJ3Y5L+IFvUm8rM74hHz66z0gw0x4tijh5CorKkKUCnW82R2vmpeCRA==", + "dev": true, + "dependencies": { + "array-ify": "^1.0.0", + "dot-prop": "^5.1.0" + } + }, + "node_modules/component-emitter": { + "version": "1.3.0", + "resolved": "https://registry.npmmirror.com/component-emitter/-/component-emitter-1.3.0.tgz", + "integrity": "sha512-Rd3se6QB+sO1TwqZjscQrurpEPIfO0/yYnSin6Q/rD3mOutHvUrCAhJub3r90uNb+SESBuE0QYoB90YdfatsRg==" + }, + "node_modules/compress-commons": { + "version": "2.1.1", + "resolved": "https://registry.npmmirror.com/compress-commons/-/compress-commons-2.1.1.tgz", + "integrity": "sha512-eVw6n7CnEMFzc3duyFVrQEuY1BlHR3rYsSztyG32ibGMW722i3C6IizEGMFmfMU+A+fALvBIwxN3czffTcdA+Q==", + "dev": true, + "dependencies": { + "buffer-crc32": "^0.2.13", + "crc32-stream": "^3.0.1", + "normalize-path": "^3.0.0", + "readable-stream": "^2.3.6" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/compressible": { + "version": "2.0.18", + "resolved": "https://registry.npmmirror.com/compressible/-/compressible-2.0.18.tgz", + "integrity": "sha512-AF3r7P5dWxL8MxyITRMlORQNaOA2IkAFaTr4k7BUumjPtRpGDTZpl0Pb1XCO6JeDCBdp126Cgs9sMxqSjgYyRg==", + "dev": true, + "dependencies": { + "mime-db": ">= 1.43.0 < 2" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/compression": { + "version": "1.7.4", + "resolved": "https://registry.npmmirror.com/compression/-/compression-1.7.4.tgz", + "integrity": "sha512-jaSIDzP9pZVS4ZfQ+TzvtiWhdpFhE2RDHz8QJkpX9SIpLq88VueF5jJw6t+6CUQcAoA6t+x89MLrWAqpfDE8iQ==", + "dev": true, + "dependencies": { + "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" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/compression/node_modules/bytes": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/bytes/-/bytes-3.0.0.tgz", + "integrity": "sha512-pMhOfFDPiv9t5jjIXkHosWmkSyQbvsgEVNkz0ERHbuLh2T/7j4Mqqpz523Fe8MVY89KC6Sh/QfS2sM+SjgFDcw==", + "dev": true, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmmirror.com/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==" + }, + "node_modules/concat-stream": { + "version": "1.6.2", + "resolved": "https://registry.npmmirror.com/concat-stream/-/concat-stream-1.6.2.tgz", + "integrity": "sha512-27HBghJxjiZtIk3Ycvn/4kbJk/1uZuJFfuPEns6LaEvpvG1f0hTea8lilrouyo9mVc2GWdcEZ8OLoGmSADlrCw==", + "engines": [ + "node >= 0.8" + ], + "dependencies": { + "buffer-from": "^1.0.0", + "inherits": "^2.0.3", + "readable-stream": "^2.2.2", + "typedarray": "^0.0.6" + } + }, + "node_modules/config-chain": { + "version": "1.1.13", + "resolved": "https://registry.npmmirror.com/config-chain/-/config-chain-1.1.13.tgz", + "integrity": "sha512-qj+f8APARXHrM0hraqXYb2/bOVSV4PvJQlNZ/DVj0QrmNM2q2euizkeuVckQ57J+W0mRH6Hvi+k50M4Jul2VRQ==", + "dev": true, + "dependencies": { + "ini": "^1.3.4", + "proto-list": "~1.2.1" + } + }, + "node_modules/connect-history-api-fallback": { + "version": "1.6.0", + "resolved": "https://registry.npmmirror.com/connect-history-api-fallback/-/connect-history-api-fallback-1.6.0.tgz", + "integrity": "sha512-e54B99q/OUoH64zYYRf3HBP5z24G38h5D3qXu23JGRoigpX5Ss4r9ZnDk3g0Z8uQC2x2lPaJ+UlWBc1ZWBWdLg==", + "dev": true, + "engines": { + "node": ">=0.8" + } + }, + "node_modules/console-browserify": { + "version": "1.2.0", + "resolved": "https://registry.npmmirror.com/console-browserify/-/console-browserify-1.2.0.tgz", + "integrity": "sha512-ZMkYO/LkF17QvCPqM0gxw8yUzigAOZOSWSHg91FH6orS7vcEj5dVZTidN2fQ14yBSdg97RqhSNwLUXInd52OTA==" + }, + "node_modules/console-control-strings": { + "version": "1.1.0", + "resolved": "https://registry.npmmirror.com/console-control-strings/-/console-control-strings-1.1.0.tgz", + "integrity": "sha512-ty/fTekppD2fIwRvnZAVdeOiGd1c7YXEixbgJTNzqcxJWKQnjJ/V1bNEEE6hygpM3WjwHFUVK6HTjWSzV4a8sQ==", + "dev": true + }, + "node_modules/consolidate": { + "version": "0.15.1", + "resolved": "https://registry.npmmirror.com/consolidate/-/consolidate-0.15.1.tgz", + "integrity": "sha512-DW46nrsMJgy9kqAbPt5rKaCr7uFtpo4mSUvLHIUbJEjm0vo+aY5QLwBUq3FK4tRnJr/X0Psc0C4jf/h+HtXSMw==", + "dev": true, + "dependencies": { + "bluebird": "^3.1.1" + }, + "engines": { + "node": ">= 0.10.0" + } + }, + "node_modules/constants-browserify": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/constants-browserify/-/constants-browserify-1.0.0.tgz", + "integrity": "sha512-xFxOwqIzR/e1k1gLiWEophSCMqXcwVHIH7akf7b/vxcUeGunlj3hvZaaqxwHsTgn+IndtkQJgSztIDWeumWJDQ==" + }, + "node_modules/content-disposition": { + "version": "0.5.4", + "resolved": "https://registry.npmmirror.com/content-disposition/-/content-disposition-0.5.4.tgz", + "integrity": "sha512-FveZTNuGw04cxlAiWbzi6zTAL/lhehaWbTtgluJh4/E95DqMwTmha3KZN1aAWA8cFIhHzMZUvLevkw5Rqk+tSQ==", + "dev": true, + "dependencies": { + "safe-buffer": "5.2.1" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/content-disposition/node_modules/safe-buffer": { + "version": "5.2.1", + "resolved": "https://registry.npmmirror.com/safe-buffer/-/safe-buffer-5.2.1.tgz", + "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", + "dev": true + }, + "node_modules/content-type": { + "version": "1.0.4", + "resolved": "https://registry.npmmirror.com/content-type/-/content-type-1.0.4.tgz", + "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==", + "dev": true, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/conventional-changelog-angular": { + "version": "5.0.13", + "resolved": "https://registry.npmmirror.com/conventional-changelog-angular/-/conventional-changelog-angular-5.0.13.tgz", + "integrity": "sha512-i/gipMxs7s8L/QeuavPF2hLnJgH6pEZAttySB6aiQLWcX3puWDL3ACVmvBhJGxnAy52Qc15ua26BufY6KpmrVA==", + "dev": true, + "dependencies": { + "compare-func": "^2.0.0", + "q": "^1.5.1" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/conventional-changelog-core": { + "version": "4.2.4", + "resolved": "https://registry.npmmirror.com/conventional-changelog-core/-/conventional-changelog-core-4.2.4.tgz", + "integrity": "sha512-gDVS+zVJHE2v4SLc6B0sLsPiloR0ygU7HaDW14aNJE1v4SlqJPILPl/aJC7YdtRE4CybBf8gDwObBvKha8Xlyg==", + "dev": true, + "dependencies": { + "add-stream": "^1.0.0", + "conventional-changelog-writer": "^5.0.0", + "conventional-commits-parser": "^3.2.0", + "dateformat": "^3.0.0", + "get-pkg-repo": "^4.0.0", + "git-raw-commits": "^2.0.8", + "git-remote-origin-url": "^2.0.0", + "git-semver-tags": "^4.1.1", + "lodash": "^4.17.15", + "normalize-package-data": "^3.0.0", + "q": "^1.5.1", + "read-pkg": "^3.0.0", + "read-pkg-up": "^3.0.0", + "through2": "^4.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/conventional-changelog-core/node_modules/hosted-git-info": { + "version": "4.1.0", + "resolved": "https://registry.npmmirror.com/hosted-git-info/-/hosted-git-info-4.1.0.tgz", + "integrity": "sha512-kyCuEOWjJqZuDbRHzL8V93NzQhwIB71oFWSyzVo+KPZI+pnQPPxucdkrOZvkLRnrf5URsQM+IJ09Dw29cRALIA==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/conventional-changelog-core/node_modules/load-json-file": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/load-json-file/-/load-json-file-4.0.0.tgz", + "integrity": "sha512-Kx8hMakjX03tiGTLAIdJ+lL0htKnXjEZN6hk/tozf/WOuYGdZBJrZ+rCJRbVCugsjB3jMLn9746NsQIf5VjBMw==", + "dev": true, + "dependencies": { + "graceful-fs": "^4.1.2", + "parse-json": "^4.0.0", + "pify": "^3.0.0", + "strip-bom": "^3.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/conventional-changelog-core/node_modules/lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/conventional-changelog-core/node_modules/normalize-package-data": { + "version": "3.0.3", + "resolved": "https://registry.npmmirror.com/normalize-package-data/-/normalize-package-data-3.0.3.tgz", + "integrity": "sha512-p2W1sgqij3zMMyRC067Dg16bfzVH+w7hyegmpIvZ4JNjqtGOVAIvLmjBx3yP7YTe9vKJgkoNOPjwQGogDoMXFA==", + "dev": true, + "dependencies": { + "hosted-git-info": "^4.0.1", + "is-core-module": "^2.5.0", + "semver": "^7.3.4", + "validate-npm-package-license": "^3.0.1" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/conventional-changelog-core/node_modules/pify": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/pify/-/pify-3.0.0.tgz", + "integrity": "sha512-C3FsVNH1udSEX48gGX1xfvwTWfsYWj5U+8/uK15BGzIGrKoUpghX8hWZwa/OFnakBiiVNmBvemTJR5mcy7iPcg==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/conventional-changelog-core/node_modules/read-pkg": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/read-pkg/-/read-pkg-3.0.0.tgz", + "integrity": "sha512-BLq/cCO9two+lBgiTYNqD6GdtK8s4NpaWrl6/rCO9w0TUS8oJl7cmToOZfRYllKTISY6nt1U7jQ53brmKqY6BA==", + "dev": true, + "dependencies": { + "load-json-file": "^4.0.0", + "normalize-package-data": "^2.3.2", + "path-type": "^3.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/conventional-changelog-core/node_modules/read-pkg/node_modules/hosted-git-info": { + "version": "2.8.9", + "resolved": "https://registry.npmmirror.com/hosted-git-info/-/hosted-git-info-2.8.9.tgz", + "integrity": "sha512-mxIDAb9Lsm6DoOJ7xH+5+X4y1LU/4Hi50L9C5sIswK3JzULS4bwk1FvjdBgvYR4bzT4tuUQiC15FE2f5HbLvYw==", + "dev": true + }, + "node_modules/conventional-changelog-core/node_modules/read-pkg/node_modules/normalize-package-data": { + "version": "2.5.0", + "resolved": "https://registry.npmmirror.com/normalize-package-data/-/normalize-package-data-2.5.0.tgz", + "integrity": "sha512-/5CMN3T0R4XTj4DcGaexo+roZSdSFW/0AOOTROrjxzCG1wrWXEsGbRKevjlIL+ZDE4sZlJr5ED4YW0yqmkK+eA==", + "dev": true, + "dependencies": { + "hosted-git-info": "^2.1.4", + "resolve": "^1.10.0", + "semver": "2 || 3 || 4 || 5", + "validate-npm-package-license": "^3.0.1" + } + }, + "node_modules/conventional-changelog-core/node_modules/read-pkg/node_modules/semver": { + "version": "5.7.1", + "resolved": "https://registry.npmmirror.com/semver/-/semver-5.7.1.tgz", + "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", + "dev": true, + "bin": { + "semver": "bin/semver" + } + }, + "node_modules/conventional-changelog-core/node_modules/readable-stream": { + "version": "3.6.0", + "resolved": "https://registry.npmmirror.com/readable-stream/-/readable-stream-3.6.0.tgz", + "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", + "dev": true, + "dependencies": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/conventional-changelog-core/node_modules/semver": { + "version": "7.3.7", + "resolved": "https://registry.npmmirror.com/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/conventional-changelog-core/node_modules/through2": { + "version": "4.0.2", + "resolved": "https://registry.npmmirror.com/through2/-/through2-4.0.2.tgz", + "integrity": "sha512-iOqSav00cVxEEICeD7TjLB1sueEL+81Wpzp2bY17uZjZN0pWZPuo4suZ/61VujxmqSGFfgOcNuTZ85QJwNZQpw==", + "dev": true, + "dependencies": { + "readable-stream": "3" + } + }, + "node_modules/conventional-changelog-core/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/conventional-changelog-preset-loader": { + "version": "2.3.4", + "resolved": "https://registry.npmmirror.com/conventional-changelog-preset-loader/-/conventional-changelog-preset-loader-2.3.4.tgz", + "integrity": "sha512-GEKRWkrSAZeTq5+YjUZOYxdHq+ci4dNwHvpaBC3+ENalzFWuCWa9EZXSuZBpkr72sMdKB+1fyDV4takK1Lf58g==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/conventional-changelog-writer": { + "version": "5.0.1", + "resolved": "https://registry.npmmirror.com/conventional-changelog-writer/-/conventional-changelog-writer-5.0.1.tgz", + "integrity": "sha512-5WsuKUfxW7suLblAbFnxAcrvf6r+0b7GvNaWUwUIk0bXMnENP/PEieGKVUQrjPqwPT4o3EPAASBXiY6iHooLOQ==", + "dev": true, + "dependencies": { + "conventional-commits-filter": "^2.0.7", + "dateformat": "^3.0.0", + "handlebars": "^4.7.7", + "json-stringify-safe": "^5.0.1", + "lodash": "^4.17.15", + "meow": "^8.0.0", + "semver": "^6.0.0", + "split": "^1.0.0", + "through2": "^4.0.0" + }, + "bin": { + "conventional-changelog-writer": "cli.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/conventional-changelog-writer/node_modules/readable-stream": { + "version": "3.6.0", + "resolved": "https://registry.npmmirror.com/readable-stream/-/readable-stream-3.6.0.tgz", + "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", + "dev": true, + "dependencies": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/conventional-changelog-writer/node_modules/semver": { + "version": "6.3.0", + "resolved": "https://registry.npmmirror.com/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/conventional-changelog-writer/node_modules/through2": { + "version": "4.0.2", + "resolved": "https://registry.npmmirror.com/through2/-/through2-4.0.2.tgz", + "integrity": "sha512-iOqSav00cVxEEICeD7TjLB1sueEL+81Wpzp2bY17uZjZN0pWZPuo4suZ/61VujxmqSGFfgOcNuTZ85QJwNZQpw==", + "dev": true, + "dependencies": { + "readable-stream": "3" + } + }, + "node_modules/conventional-commits-filter": { + "version": "2.0.7", + "resolved": "https://registry.npmmirror.com/conventional-commits-filter/-/conventional-commits-filter-2.0.7.tgz", + "integrity": "sha512-ASS9SamOP4TbCClsRHxIHXRfcGCnIoQqkvAzCSbZzTFLfcTqJVugB0agRgsEELsqaeWgsXv513eS116wnlSSPA==", + "dev": true, + "dependencies": { + "lodash.ismatch": "^4.4.0", + "modify-values": "^1.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/conventional-commits-parser": { + "version": "3.2.4", + "resolved": "https://registry.npmmirror.com/conventional-commits-parser/-/conventional-commits-parser-3.2.4.tgz", + "integrity": "sha512-nK7sAtfi+QXbxHCYfhpZsfRtaitZLIA6889kFIouLvz6repszQDgxBu7wf2WbU+Dco7sAnNCJYERCwt54WPC2Q==", + "dev": true, + "dependencies": { + "is-text-path": "^1.0.1", + "JSONStream": "^1.0.4", + "lodash": "^4.17.15", + "meow": "^8.0.0", + "split2": "^3.0.0", + "through2": "^4.0.0" + }, + "bin": { + "conventional-commits-parser": "cli.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/conventional-commits-parser/node_modules/readable-stream": { + "version": "3.6.0", + "resolved": "https://registry.npmmirror.com/readable-stream/-/readable-stream-3.6.0.tgz", + "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", + "dev": true, + "dependencies": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/conventional-commits-parser/node_modules/through2": { + "version": "4.0.2", + "resolved": "https://registry.npmmirror.com/through2/-/through2-4.0.2.tgz", + "integrity": "sha512-iOqSav00cVxEEICeD7TjLB1sueEL+81Wpzp2bY17uZjZN0pWZPuo4suZ/61VujxmqSGFfgOcNuTZ85QJwNZQpw==", + "dev": true, + "dependencies": { + "readable-stream": "3" + } + }, + "node_modules/conventional-recommended-bump": { + "version": "6.1.0", + "resolved": "https://registry.npmmirror.com/conventional-recommended-bump/-/conventional-recommended-bump-6.1.0.tgz", + "integrity": "sha512-uiApbSiNGM/kkdL9GTOLAqC4hbptObFo4wW2QRyHsKciGAfQuLU1ShZ1BIVI/+K2BE/W1AWYQMCXAsv4dyKPaw==", + "dev": true, + "dependencies": { + "concat-stream": "^2.0.0", + "conventional-changelog-preset-loader": "^2.3.4", + "conventional-commits-filter": "^2.0.7", + "conventional-commits-parser": "^3.2.0", + "git-raw-commits": "^2.0.8", + "git-semver-tags": "^4.1.1", + "meow": "^8.0.0", + "q": "^1.5.1" + }, + "bin": { + "conventional-recommended-bump": "cli.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/conventional-recommended-bump/node_modules/concat-stream": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/concat-stream/-/concat-stream-2.0.0.tgz", + "integrity": "sha512-MWufYdFw53ccGjCA+Ol7XJYpAlW6/prSMzuPOTRnJGcGzuhLn4Scrz7qf6o8bROZ514ltazcIFJZevcfbo0x7A==", + "dev": true, + "engines": [ + "node >= 6.0" + ], + "dependencies": { + "buffer-from": "^1.0.0", + "inherits": "^2.0.3", + "readable-stream": "^3.0.2", + "typedarray": "^0.0.6" + } + }, + "node_modules/conventional-recommended-bump/node_modules/readable-stream": { + "version": "3.6.0", + "resolved": "https://registry.npmmirror.com/readable-stream/-/readable-stream-3.6.0.tgz", + "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", + "dev": true, + "dependencies": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/convert-source-map": { + "version": "1.8.0", + "resolved": "https://registry.npmmirror.com/convert-source-map/-/convert-source-map-1.8.0.tgz", + "integrity": "sha512-+OQdjP49zViI/6i7nIJpA8rAl4sV/JdPfU9nZs3VqOwGIgizICvuN2ru6fMd+4llL0tar18UYJXfZ/TWtmhUjA==", + "dev": true, + "dependencies": { + "safe-buffer": "~5.1.1" + } + }, + "node_modules/cookie": { + "version": "0.5.0", + "resolved": "https://registry.npmmirror.com/cookie/-/cookie-0.5.0.tgz", + "integrity": "sha512-YZ3GUyn/o8gfKJlnlX7g7xq4gyO6OSuhGPKaaGssGB2qgDUS0gPgtTvoyZLTt9Ab6dC4hfc9dV5arkvc/OCmrw==", + "dev": true, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/cookie-signature": { + "version": "1.0.6", + "resolved": "https://registry.npmmirror.com/cookie-signature/-/cookie-signature-1.0.6.tgz", + "integrity": "sha512-QADzlaHc8icV8I7vbaJXJwod9HWYp8uCqf1xa4OfNu1T7JVxQIrUgOWtHdNDtPiywmFbiS12VjotIXLrKM3orQ==", + "dev": true + }, + "node_modules/copy-anything": { + "version": "2.0.6", + "resolved": "https://registry.npmmirror.com/copy-anything/-/copy-anything-2.0.6.tgz", + "integrity": "sha512-1j20GZTsvKNkc4BY3NpMOM8tt///wY3FpIzozTOFO2ffuZcV61nojHXVKIy3WM+7ADCy5FVhdZYHYDdgTU0yJw==", + "dev": true, + "dependencies": { + "is-what": "^3.14.1" + } + }, + "node_modules/copy-concurrently": { + "version": "1.0.5", + "resolved": "https://registry.npmmirror.com/copy-concurrently/-/copy-concurrently-1.0.5.tgz", + "integrity": "sha512-f2domd9fsVDFtaFcbaRZuYXwtdmnzqbADSwhSWYxYB/Q8zsdUUFMXVRwXGDMWmbEzAn1kdRrtI1T/KTFOL4X2A==", + "dependencies": { + "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" + } + }, + "node_modules/copy-descriptor": { + "version": "0.1.1", + "resolved": "https://registry.npmmirror.com/copy-descriptor/-/copy-descriptor-0.1.1.tgz", + "integrity": "sha512-XgZ0pFcakEUlbwQEVNg3+QAis1FyTL3Qel9FYy8pSkQqoG3PNoT0bOCQtOXcOkur21r2Eq2kI+IE+gsmAEVlYw==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/copy-webpack-plugin": { + "version": "4.6.0", + "resolved": "https://registry.npmmirror.com/copy-webpack-plugin/-/copy-webpack-plugin-4.6.0.tgz", + "integrity": "sha512-Y+SQCF+0NoWQryez2zXn5J5knmr9z/9qSQt7fbL78u83rxmigOy8X5+BFn8CFSuX+nKT8gpYwJX68ekqtQt6ZA==", + "dev": true, + "dependencies": { + "cacache": "^10.0.4", + "find-cache-dir": "^1.0.0", + "globby": "^7.1.1", + "is-glob": "^4.0.0", + "loader-utils": "^1.1.0", + "minimatch": "^3.0.4", + "p-limit": "^1.0.0", + "serialize-javascript": "^1.4.0" + }, + "engines": { + "node": ">= 4" + } + }, + "node_modules/copy-webpack-plugin/node_modules/cacache": { + "version": "10.0.4", + "resolved": "https://registry.npmmirror.com/cacache/-/cacache-10.0.4.tgz", + "integrity": "sha512-Dph0MzuH+rTQzGPNT9fAnrPmMmjKfST6trxJeK7NQuHRaVw24VzPRWTmg9MpcwOVQZO0E1FBICUlFeNaKPIfHA==", + "dev": true, + "dependencies": { + "bluebird": "^3.5.1", + "chownr": "^1.0.1", + "glob": "^7.1.2", + "graceful-fs": "^4.1.11", + "lru-cache": "^4.1.1", + "mississippi": "^2.0.0", + "mkdirp": "^0.5.1", + "move-concurrently": "^1.0.1", + "promise-inflight": "^1.0.1", + "rimraf": "^2.6.2", + "ssri": "^5.2.4", + "unique-filename": "^1.1.0", + "y18n": "^4.0.0" + } + }, + "node_modules/copy-webpack-plugin/node_modules/find-cache-dir": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/find-cache-dir/-/find-cache-dir-1.0.0.tgz", + "integrity": "sha512-46TFiBOzX7xq/PcSWfFwkyjpemdRnMe31UQF+os0y+1W3k95f6R4SEt02Hj4p3X0Mir9gfrkmOtshFidS0VPUg==", + "dev": true, + "dependencies": { + "commondir": "^1.0.1", + "make-dir": "^1.0.0", + "pkg-dir": "^2.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/copy-webpack-plugin/node_modules/find-up": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/find-up/-/find-up-2.1.0.tgz", + "integrity": "sha512-NWzkk0jSJtTt08+FBFMvXoeZnOJD+jTtsRmBYbAIzJdX6l7dLgR7CTubCM5/eDdPUBvLCeVasP1brfVR/9/EZQ==", + "dev": true, + "dependencies": { + "locate-path": "^2.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/copy-webpack-plugin/node_modules/globby": { + "version": "7.1.1", + "resolved": "https://registry.npmmirror.com/globby/-/globby-7.1.1.tgz", + "integrity": "sha512-yANWAN2DUcBtuus5Cpd+SKROzXHs2iVXFZt/Ykrfz6SAXqacLX25NZpltE+39ceMexYF4TtEadjuSTw8+3wX4g==", + "dev": true, + "dependencies": { + "array-union": "^1.0.1", + "dir-glob": "^2.0.0", + "glob": "^7.1.2", + "ignore": "^3.3.5", + "pify": "^3.0.0", + "slash": "^1.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/copy-webpack-plugin/node_modules/locate-path": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/locate-path/-/locate-path-2.0.0.tgz", + "integrity": "sha512-NCI2kiDkyR7VeEKm27Kda/iQHyKJe1Bu0FlTbYp3CqJu+9IFe9bLyAjMxf5ZDDbEg+iMPzB5zYyUTSm8wVTKmA==", + "dev": true, + "dependencies": { + "p-locate": "^2.0.0", + "path-exists": "^3.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/copy-webpack-plugin/node_modules/lru-cache": { + "version": "4.1.5", + "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-4.1.5.tgz", + "integrity": "sha512-sWZlbEP2OsHNkXrMl5GYk/jKk70MBng6UU4YI/qGDYbgf6YbP4EvmqISbXCoJiRKs+1bSpFHVgQxvJ17F2li5g==", + "dev": true, + "dependencies": { + "pseudomap": "^1.0.2", + "yallist": "^2.1.2" + } + }, + "node_modules/copy-webpack-plugin/node_modules/make-dir": { + "version": "1.3.0", + "resolved": "https://registry.npmmirror.com/make-dir/-/make-dir-1.3.0.tgz", + "integrity": "sha512-2w31R7SJtieJJnQtGc7RVL2StM2vGYVfqUOvUDxH6bC6aJTxPxTF0GnIgCyu7tjockiUWAYQRbxa7vKn34s5sQ==", + "dev": true, + "dependencies": { + "pify": "^3.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/copy-webpack-plugin/node_modules/mississippi": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/mississippi/-/mississippi-2.0.0.tgz", + "integrity": "sha512-zHo8v+otD1J10j/tC+VNoGK9keCuByhKovAvdn74dmxJl9+mWHnx6EMsDN4lgRoMI/eYo2nchAxniIbUPb5onw==", + "dev": true, + "dependencies": { + "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": "^2.0.1", + "pumpify": "^1.3.3", + "stream-each": "^1.1.0", + "through2": "^2.0.0" + }, + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/copy-webpack-plugin/node_modules/p-limit": { + "version": "1.3.0", + "resolved": "https://registry.npmmirror.com/p-limit/-/p-limit-1.3.0.tgz", + "integrity": "sha512-vvcXsLAJ9Dr5rQOPk7toZQZJApBl2K4J6dANSsEuh6QI41JYcsS/qhTGa9ErIUUgK3WNQoJYvylxvjqmiqEA9Q==", + "dev": true, + "dependencies": { + "p-try": "^1.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/copy-webpack-plugin/node_modules/p-locate": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/p-locate/-/p-locate-2.0.0.tgz", + "integrity": "sha512-nQja7m7gSKuewoVRen45CtVfODR3crN3goVQ0DDZ9N3yHxgpkuBhZqsaiotSQRrADUrne346peY7kT3TSACykg==", + "dev": true, + "dependencies": { + "p-limit": "^1.1.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/copy-webpack-plugin/node_modules/p-try": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/p-try/-/p-try-1.0.0.tgz", + "integrity": "sha512-U1etNYuMJoIz3ZXSrrySFjsXQTWOx2/jdi86L+2pRvph/qMKL6sbcCYdH23fqsbm8TH2Gn0OybpT4eSFlCVHww==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/copy-webpack-plugin/node_modules/pify": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/pify/-/pify-3.0.0.tgz", + "integrity": "sha512-C3FsVNH1udSEX48gGX1xfvwTWfsYWj5U+8/uK15BGzIGrKoUpghX8hWZwa/OFnakBiiVNmBvemTJR5mcy7iPcg==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/copy-webpack-plugin/node_modules/pkg-dir": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/pkg-dir/-/pkg-dir-2.0.0.tgz", + "integrity": "sha512-ojakdnUgL5pzJYWw2AIDEupaQCX5OPbM688ZevubICjdIX01PRSYKqm33fJoCOJBRseYCTUlQRnBNX+Pchaejw==", + "dev": true, + "dependencies": { + "find-up": "^2.1.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/copy-webpack-plugin/node_modules/pump": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/pump/-/pump-2.0.1.tgz", + "integrity": "sha512-ruPMNRkN3MHP1cWJc9OWr+T/xDP0jhXYCLfJcBuX54hhfIBnaQmAUMfDcG4DM5UMWByBbJY69QSphm3jtDKIkA==", + "dev": true, + "dependencies": { + "end-of-stream": "^1.1.0", + "once": "^1.3.1" + } + }, + "node_modules/copy-webpack-plugin/node_modules/serialize-javascript": { + "version": "1.9.1", + "resolved": "https://registry.npmmirror.com/serialize-javascript/-/serialize-javascript-1.9.1.tgz", + "integrity": "sha512-0Vb/54WJ6k5v8sSWN09S0ora+Hnr+cX40r9F170nT+mSkaxltoE/7R3OrIdBSUv1OoiobH1QoWQbCnAO+e8J1A==", + "dev": true + }, + "node_modules/copy-webpack-plugin/node_modules/slash": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/slash/-/slash-1.0.0.tgz", + "integrity": "sha512-3TYDR7xWt4dIqV2JauJr+EJeW356RXijHeUlO+8djJ+uBXPn8/2dpzBc8yQhh583sVvc9CvFAeQVgijsH+PNNg==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/copy-webpack-plugin/node_modules/ssri": { + "version": "5.3.0", + "resolved": "https://registry.npmmirror.com/ssri/-/ssri-5.3.0.tgz", + "integrity": "sha512-XRSIPqLij52MtgoQavH/x/dU1qVKtWUAAZeOHsR9c2Ddi4XerFy3mc1alf+dLJKl9EUIm/Ht+EowFkTUOA6GAQ==", + "dev": true, + "dependencies": { + "safe-buffer": "^5.1.1" + } + }, + "node_modules/copy-webpack-plugin/node_modules/yallist": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-2.1.2.tgz", + "integrity": "sha512-ncTzHV7NvsQZkYe1DW7cbDLm0YpzHmZF5r/iyP3ZnQtMiJ+pjzisCiMNI+Sj+xQF5pXhSHxSB3uDbsBTzY/c2A==", + "dev": true + }, + "node_modules/core-js": { + "version": "2.6.11", + "resolved": "https://registry.npmmirror.com/core-js/-/core-js-2.6.11.tgz", + "integrity": "sha512-5wjnpaT/3dV+XB4borEsnAYQchn00XSgTAWKDkEqv+K8KevjbzmofK6hfJ9TZIlpj2N0xQpazy7PiRQiWHqzWg==", + "deprecated": "core-js@<3.23.3 is no longer maintained and not recommended for usage due to the number of issues. Because of the V8 engine whims, feature detection in old core-js versions could cause a slowdown up to 100x even if nothing is polyfilled. Some versions have web compatibility issues. Please, upgrade your dependencies to the actual version of core-js.", + "hasInstallScript": true + }, + "node_modules/core-js-compat": { + "version": "3.22.2", + "resolved": "https://registry.npmmirror.com/core-js-compat/-/core-js-compat-3.22.2.tgz", + "integrity": "sha512-Fns9lU06ZJ07pdfmPMu7OnkIKGPKDzXKIiuGlSvHHapwqMUF2QnnsWwtueFZtSyZEilP0o6iUeHQwpn7LxtLUw==", + "dev": true, + "dependencies": { + "browserslist": "^4.20.2", + "semver": "7.0.0" + } + }, + "node_modules/core-js-compat/node_modules/semver": { + "version": "7.0.0", + "resolved": "https://registry.npmmirror.com/semver/-/semver-7.0.0.tgz", + "integrity": "sha512-+GB6zVA9LWh6zovYQLALHwv5rb2PHGlJi3lfiqIHxR0uuwCgefcOJc59v9fv1w8GbStwxuuqqAjI9NMAOOgq1A==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/core-util-is": { + "version": "1.0.3", + "resolved": "https://registry.npmmirror.com/core-util-is/-/core-util-is-1.0.3.tgz", + "integrity": "sha512-ZQBvi1DcpJ4GDqanjucZ2Hj3wEO5pZDS89BWbkcrvdxksJorwUDDZamX9ldFkp9aw2lmBDLgkObEA4DWNJ9FYQ==" + }, + "node_modules/cosmiconfig": { + "version": "5.2.1", + "resolved": "https://registry.npmmirror.com/cosmiconfig/-/cosmiconfig-5.2.1.tgz", + "integrity": "sha512-H65gsXo1SKjf8zmrJ67eJk8aIRKV5ff2D4uKZIBZShbhGSpEmsQOPW/SKMKYhSTrqR7ufy6RP69rPogdaPh/kA==", + "dev": true, + "dependencies": { + "import-fresh": "^2.0.0", + "is-directory": "^0.3.1", + "js-yaml": "^3.13.1", + "parse-json": "^4.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/cpx": { + "version": "1.5.0", + "resolved": "https://registry.npmmirror.com/cpx/-/cpx-1.5.0.tgz", + "integrity": "sha512-jHTjZhsbg9xWgsP2vuNW2jnnzBX+p4T+vNI9Lbjzs1n4KhOfa22bQppiFYLsWQKd8TzmL5aSP/Me3yfsCwXbDA==", + "dev": true, + "dependencies": { + "babel-runtime": "^6.9.2", + "chokidar": "^1.6.0", + "duplexer": "^0.1.1", + "glob": "^7.0.5", + "glob2base": "^0.0.12", + "minimatch": "^3.0.2", + "mkdirp": "^0.5.1", + "resolve": "^1.1.7", + "safe-buffer": "^5.0.1", + "shell-quote": "^1.6.1", + "subarg": "^1.0.0" + }, + "bin": { + "cpx": "bin/index.js" + } + }, + "node_modules/cpx/node_modules/anymatch": { + "version": "1.3.2", + "resolved": "https://registry.npmmirror.com/anymatch/-/anymatch-1.3.2.tgz", + "integrity": "sha512-0XNayC8lTHQ2OI8aljNCN3sSx6hsr/1+rlcDAotXJR7C1oZZHCNsfpbKwMjRA3Uqb5tF1Rae2oloTr4xpq+WjA==", + "dev": true, + "dependencies": { + "micromatch": "^2.1.5", + "normalize-path": "^2.0.0" + } + }, + "node_modules/cpx/node_modules/arr-diff": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/arr-diff/-/arr-diff-2.0.0.tgz", + "integrity": "sha512-dtXTVMkh6VkEEA7OhXnN1Ecb8aAGFdZ1LFxtOCoqj4qkyOJMt7+qs6Ahdy6p/NQCPYsRSXXivhSB/J5E9jmYKA==", + "dev": true, + "dependencies": { + "arr-flatten": "^1.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/array-unique": { + "version": "0.2.1", + "resolved": "https://registry.npmmirror.com/array-unique/-/array-unique-0.2.1.tgz", + "integrity": "sha512-G2n5bG5fSUCpnsXz4+8FUkYsGPkNfLn9YvS66U5qbTIXI2Ynnlo4Bi42bWv+omKUCqz+ejzfClwne0alJWJPhg==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/binary-extensions": { + "version": "1.13.1", + "resolved": "https://registry.npmmirror.com/binary-extensions/-/binary-extensions-1.13.1.tgz", + "integrity": "sha512-Un7MIEDdUC5gNpcGDV97op1Ywk748MpHcFTHoYs6qnj1Z3j7I53VG3nwZhKzoBZmbdRNnb6WRdFlwl7tSDuZGw==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/braces": { + "version": "1.8.5", + "resolved": "https://registry.npmmirror.com/braces/-/braces-1.8.5.tgz", + "integrity": "sha512-xU7bpz2ytJl1bH9cgIurjpg/n8Gohy9GTw81heDYLJQ4RU60dlyJsa+atVF2pI0yMMvKxI9HkKwjePCj5XI1hw==", + "dev": true, + "dependencies": { + "expand-range": "^1.8.1", + "preserve": "^0.2.0", + "repeat-element": "^1.1.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/chokidar": { + "version": "1.7.0", + "resolved": "https://registry.npmmirror.com/chokidar/-/chokidar-1.7.0.tgz", + "integrity": "sha512-mk8fAWcRUOxY7btlLtitj3A45jOwSAxH4tOFOoEGbVsl6cL6pPMWUy7dwZ/canfj3QEdP6FHSnf/l1c6/WkzVg==", + "deprecated": "Chokidar 2 will break on node v14+. Upgrade to chokidar 3 with 15x less dependencies.", + "dev": true, + "dependencies": { + "anymatch": "^1.3.0", + "async-each": "^1.0.0", + "glob-parent": "^2.0.0", + "inherits": "^2.0.1", + "is-binary-path": "^1.0.0", + "is-glob": "^2.0.0", + "path-is-absolute": "^1.0.0", + "readdirp": "^2.0.0" + }, + "optionalDependencies": { + "fsevents": "^1.0.0" + } + }, + "node_modules/cpx/node_modules/expand-brackets": { + "version": "0.1.5", + "resolved": "https://registry.npmmirror.com/expand-brackets/-/expand-brackets-0.1.5.tgz", + "integrity": "sha512-hxx03P2dJxss6ceIeri9cmYOT4SRs3Zk3afZwWpOsRqLqprhTR8u++SlC+sFGsQr7WGFPdMF7Gjc1njDLDK6UA==", + "dev": true, + "dependencies": { + "is-posix-bracket": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/extglob": { + "version": "0.3.2", + "resolved": "https://registry.npmmirror.com/extglob/-/extglob-0.3.2.tgz", + "integrity": "sha512-1FOj1LOwn42TMrruOHGt18HemVnbwAmAak7krWk+wa93KXxGbK+2jpezm+ytJYDaBX0/SPLZFHKM7m+tKobWGg==", + "dev": true, + "dependencies": { + "is-extglob": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/fsevents": { + "version": "1.2.13", + "resolved": "https://registry.npmmirror.com/fsevents/-/fsevents-1.2.13.tgz", + "integrity": "sha512-oWb1Z6mkHIskLzEJ/XWX0srkpkTQ7vaopMQkyaEIoq0fmtFVxOthb8cCxeT+p3ynTdkk/RZwbgG4brR5BeWECw==", + "deprecated": "fsevents 1 will break on node v14+ and could be using insecure binaries. Upgrade to fsevents 2.", + "dev": true, + "hasInstallScript": true, + "optional": true, + "os": [ + "darwin" + ], + "dependencies": { + "bindings": "^1.5.0", + "nan": "^2.12.1" + }, + "engines": { + "node": ">= 4.0" + } + }, + "node_modules/cpx/node_modules/glob-parent": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/glob-parent/-/glob-parent-2.0.0.tgz", + "integrity": "sha512-JDYOvfxio/t42HKdxkAYaCiBN7oYiuxykOxKxdaUW5Qn0zaYN3gRQWolrwdnf0shM9/EP0ebuuTmyoXNr1cC5w==", + "dev": true, + "dependencies": { + "is-glob": "^2.0.0" + } + }, + "node_modules/cpx/node_modules/is-accessor-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", + "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", + "dev": true, + "dependencies": { + "kind-of": "^6.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/is-accessor-descriptor/node_modules/kind-of": { + "version": "6.0.3", + "resolved": "https://registry.npmmirror.com/kind-of/-/kind-of-6.0.3.tgz", + "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/is-binary-path": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/is-binary-path/-/is-binary-path-1.0.1.tgz", + "integrity": "sha512-9fRVlXc0uCxEDj1nQzaWONSpbTfx0FmJfzHF7pwlI8DkWGoHBBea4Pg5Ky0ojwwxQmnSifgbKkI06Qv0Ljgj+Q==", + "dev": true, + "dependencies": { + "binary-extensions": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/is-data-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", + "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", + "dev": true, + "dependencies": { + "kind-of": "^6.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/is-data-descriptor/node_modules/kind-of": { + "version": "6.0.3", + "resolved": "https://registry.npmmirror.com/kind-of/-/kind-of-6.0.3.tgz", + "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/is-descriptor": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/is-descriptor/-/is-descriptor-1.0.2.tgz", + "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", + "dev": true, + "dependencies": { + "is-accessor-descriptor": "^1.0.0", + "is-data-descriptor": "^1.0.0", + "kind-of": "^6.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/is-descriptor/node_modules/kind-of": { + "version": "6.0.3", + "resolved": "https://registry.npmmirror.com/kind-of/-/kind-of-6.0.3.tgz", + "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/is-extglob": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/is-extglob/-/is-extglob-1.0.0.tgz", + "integrity": "sha512-7Q+VbVafe6x2T+Tu6NcOf6sRklazEPmBoB3IWk3WdGZM2iGUwU/Oe3Wtq5lSEkDTTlpp8yx+5t4pzO/i9Ty1ww==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/is-glob": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/is-glob/-/is-glob-2.0.1.tgz", + "integrity": "sha512-a1dBeB19NXsf/E0+FHqkagizel/LQw2DjSQpvQrj3zT+jYPpaUCryPnrQajXKFLCMuf4I6FhRpaGtw4lPrG6Eg==", + "dev": true, + "dependencies": { + "is-extglob": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmmirror.com/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha512-NOW9QQXMoZGg/oqnVNoNTTIFEIid1627WCffUBJEdMxYApq7mNE7CpzucIPc+ZQg25Phej7IJSmX3hO+oblOtQ==", + "dev": true, + "dependencies": { + "is-buffer": "^1.1.5" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/micromatch": { + "version": "2.3.11", + "resolved": "https://registry.npmmirror.com/micromatch/-/micromatch-2.3.11.tgz", + "integrity": "sha512-LnU2XFEk9xxSJ6rfgAry/ty5qwUTyHYOBU0g4R6tIw5ljwgGIBmiKhRWLw5NpMOnrgUNcDJ4WMp8rl3sYVHLNA==", + "dev": true, + "dependencies": { + "arr-diff": "^2.0.0", + "array-unique": "^0.2.1", + "braces": "^1.8.2", + "expand-brackets": "^0.1.4", + "extglob": "^0.3.1", + "filename-regex": "^2.0.0", + "is-extglob": "^1.0.0", + "is-glob": "^2.0.1", + "kind-of": "^3.0.2", + "normalize-path": "^2.0.1", + "object.omit": "^2.0.0", + "parse-glob": "^3.0.4", + "regex-cache": "^0.4.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/normalize-path": { + "version": "2.1.1", + "resolved": "https://registry.npmmirror.com/normalize-path/-/normalize-path-2.1.1.tgz", + "integrity": "sha512-3pKJwH184Xo/lnH6oyP1q2pMd7HcypqqmRs91/6/i2CGtWwIKGCkOOMTm/zXbgTEWHw1uNpNi/igc3ePOYHb6w==", + "dev": true, + "dependencies": { + "remove-trailing-separator": "^1.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/readdirp": { + "version": "2.2.1", + "resolved": "https://registry.npmmirror.com/readdirp/-/readdirp-2.2.1.tgz", + "integrity": "sha512-1JU/8q+VgFZyxwrJ+SVIOsh+KywWGpds3NTqikiKpDMZWScmAYyKIgqkO+ARvNWJfXeXR1zxz7aHF4u4CyH6vQ==", + "dev": true, + "dependencies": { + "graceful-fs": "^4.1.11", + "micromatch": "^3.1.10", + "readable-stream": "^2.0.2" + }, + "engines": { + "node": ">=0.10" + } + }, + "node_modules/cpx/node_modules/readdirp/node_modules/arr-diff": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/arr-diff/-/arr-diff-4.0.0.tgz", + "integrity": "sha512-YVIQ82gZPGBebQV/a8dar4AitzCQs0jjXwMPZllpXMaGjXPYVUawSxQrRsjhjupyVxEvbHgUmIhKVlND+j02kA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/readdirp/node_modules/array-unique": { + "version": "0.3.2", + "resolved": "https://registry.npmmirror.com/array-unique/-/array-unique-0.3.2.tgz", + "integrity": "sha512-SleRWjh9JUud2wH1hPs9rZBZ33H6T9HOiL0uwGnGx9FpE6wKGyfWugmbkEOIs6qWrZhg0LWeLziLrEwQJhs5mQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/readdirp/node_modules/braces": { + "version": "2.3.2", + "resolved": "https://registry.npmmirror.com/braces/-/braces-2.3.2.tgz", + "integrity": "sha512-aNdbnj9P8PjdXU4ybaWLK2IF3jc/EoDYbC7AazW6to3TRsfXxscC9UXOB5iDiEQrkyIbWp2SLQda4+QAa7nc3w==", + "dev": true, + "dependencies": { + "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" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/readdirp/node_modules/braces/node_modules/extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha512-zCnTtlxNoAiDc3gqY2aYAWFx7XWWiasuF2K8Me5WbN8otHKTUKBwjPtNpRs/rbUZm7KxWAaNj7P1a/p52GbVug==", + "dev": true, + "dependencies": { + "is-extendable": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/readdirp/node_modules/expand-brackets": { + "version": "2.1.4", + "resolved": "https://registry.npmmirror.com/expand-brackets/-/expand-brackets-2.1.4.tgz", + "integrity": "sha512-w/ozOKR9Obk3qoWeY/WDi6MFta9AoMR+zud60mdnbniMcBxRuFJyDt2LdX/14A1UABeqk+Uk+LDfUpvoGKppZA==", + "dev": true, + "dependencies": { + "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" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/readdirp/node_modules/expand-brackets/node_modules/define-property": { + "version": "0.2.5", + "resolved": "https://registry.npmmirror.com/define-property/-/define-property-0.2.5.tgz", + "integrity": "sha512-Rr7ADjQZenceVOAKop6ALkkRAmH1A4Gx9hV/7ZujPUN2rkATqFO0JZLZInbAjpZYoJ1gUx8MRMQVkYemcbMSTA==", + "dev": true, + "dependencies": { + "is-descriptor": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/readdirp/node_modules/expand-brackets/node_modules/extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha512-zCnTtlxNoAiDc3gqY2aYAWFx7XWWiasuF2K8Me5WbN8otHKTUKBwjPtNpRs/rbUZm7KxWAaNj7P1a/p52GbVug==", + "dev": true, + "dependencies": { + "is-extendable": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/readdirp/node_modules/expand-brackets/node_modules/is-descriptor": { + "version": "0.1.6", + "resolved": "https://registry.npmmirror.com/is-descriptor/-/is-descriptor-0.1.6.tgz", + "integrity": "sha512-avDYr0SB3DwO9zsMov0gKCESFYqCnE4hq/4z3TdUlukEy5t9C0YRq7HLrsN52NAcqXKaepeCD0n+B0arnVG3Hg==", + "dev": true, + "dependencies": { + "is-accessor-descriptor": "^0.1.6", + "is-data-descriptor": "^0.1.4", + "kind-of": "^5.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/readdirp/node_modules/expand-brackets/node_modules/kind-of": { + "version": "5.1.0", + "resolved": "https://registry.npmmirror.com/kind-of/-/kind-of-5.1.0.tgz", + "integrity": "sha512-NGEErnH6F2vUuXDh+OlbcKW7/wOcfdRHaZ7VWtqCztfHri/++YKmP51OdWeGPuqCOba6kk2OTe5d02VmTB80Pw==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/readdirp/node_modules/extglob": { + "version": "2.0.4", + "resolved": "https://registry.npmmirror.com/extglob/-/extglob-2.0.4.tgz", + "integrity": "sha512-Nmb6QXkELsuBr24CJSkilo6UHHgbekK5UiZgfE6UHD3Eb27YC6oD+bhcT+tJ6cl8dmsgdQxnWlcry8ksBIBLpw==", + "dev": true, + "dependencies": { + "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" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/readdirp/node_modules/extglob/node_modules/define-property": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/define-property/-/define-property-1.0.0.tgz", + "integrity": "sha512-cZTYKFWspt9jZsMscWo8sc/5lbPC9Q0N5nBLgb+Yd915iL3udB1uFgS3B8YCx66UVHq018DAVFoee7x+gxggeA==", + "dev": true, + "dependencies": { + "is-descriptor": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/readdirp/node_modules/extglob/node_modules/extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha512-zCnTtlxNoAiDc3gqY2aYAWFx7XWWiasuF2K8Me5WbN8otHKTUKBwjPtNpRs/rbUZm7KxWAaNj7P1a/p52GbVug==", + "dev": true, + "dependencies": { + "is-extendable": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/readdirp/node_modules/is-accessor-descriptor": { + "version": "0.1.6", + "resolved": "https://registry.npmmirror.com/is-accessor-descriptor/-/is-accessor-descriptor-0.1.6.tgz", + "integrity": "sha512-e1BM1qnDbMRG3ll2U9dSK0UMHuWOs3pY3AtcFsmvwPtKL3MML/Q86i+GilLfvqEs4GW+ExB91tQ3Ig9noDIZ+A==", + "dev": true, + "dependencies": { + "kind-of": "^3.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/readdirp/node_modules/is-accessor-descriptor/node_modules/kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmmirror.com/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha512-NOW9QQXMoZGg/oqnVNoNTTIFEIid1627WCffUBJEdMxYApq7mNE7CpzucIPc+ZQg25Phej7IJSmX3hO+oblOtQ==", + "dev": true, + "dependencies": { + "is-buffer": "^1.1.5" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/readdirp/node_modules/is-data-descriptor": { + "version": "0.1.4", + "resolved": "https://registry.npmmirror.com/is-data-descriptor/-/is-data-descriptor-0.1.4.tgz", + "integrity": "sha512-+w9D5ulSoBNlmw9OHn3U2v51SyoCd0he+bB3xMl62oijhrspxowjU+AIcDY0N3iEJbUEkB15IlMASQsxYigvXg==", + "dev": true, + "dependencies": { + "kind-of": "^3.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/readdirp/node_modules/is-data-descriptor/node_modules/kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmmirror.com/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha512-NOW9QQXMoZGg/oqnVNoNTTIFEIid1627WCffUBJEdMxYApq7mNE7CpzucIPc+ZQg25Phej7IJSmX3hO+oblOtQ==", + "dev": true, + "dependencies": { + "is-buffer": "^1.1.5" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/readdirp/node_modules/kind-of": { + "version": "6.0.3", + "resolved": "https://registry.npmmirror.com/kind-of/-/kind-of-6.0.3.tgz", + "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cpx/node_modules/readdirp/node_modules/micromatch": { + "version": "3.1.10", + "resolved": "https://registry.npmmirror.com/micromatch/-/micromatch-3.1.10.tgz", + "integrity": "sha512-MWikgl9n9M3w+bpsY3He8L+w9eF9338xRl8IAO5viDizwSzziFEyUzo2xrrloB64ADbTf8uA8vRqqttDTOmccg==", + "dev": true, + "dependencies": { + "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" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/crc": { + "version": "3.8.0", + "resolved": "https://registry.npmmirror.com/crc/-/crc-3.8.0.tgz", + "integrity": "sha512-iX3mfgcTMIq3ZKLIsVFAbv7+Mc10kxabAGQb8HvjA1o3T1PIYprbakQ65d3I+2HGHt6nSKkM9PYjgoJO2KcFBQ==", + "dev": true, + "dependencies": { + "buffer": "^5.1.0" + } + }, + "node_modules/crc/node_modules/buffer": { + "version": "5.7.1", + "resolved": "https://registry.npmmirror.com/buffer/-/buffer-5.7.1.tgz", + "integrity": "sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==", + "dev": true, + "dependencies": { + "base64-js": "^1.3.1", + "ieee754": "^1.1.13" + } + }, + "node_modules/crc32-stream": { + "version": "3.0.1", + "resolved": "https://registry.npmmirror.com/crc32-stream/-/crc32-stream-3.0.1.tgz", + "integrity": "sha512-mctvpXlbzsvK+6z8kJwSJ5crm7yBwrQMTybJzMw1O4lLGJqjlDCXY2Zw7KheiA6XBEcBmfLx1D88mjRGVJtY9w==", + "dev": true, + "dependencies": { + "crc": "^3.4.4", + "readable-stream": "^3.4.0" + }, + "engines": { + "node": ">= 6.9.0" + } + }, + "node_modules/crc32-stream/node_modules/readable-stream": { + "version": "3.6.0", + "resolved": "https://registry.npmmirror.com/readable-stream/-/readable-stream-3.6.0.tgz", + "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", + "dev": true, + "dependencies": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/create-ecdh": { + "version": "4.0.4", + "resolved": "https://registry.npmmirror.com/create-ecdh/-/create-ecdh-4.0.4.tgz", + "integrity": "sha512-mf+TCx8wWc9VpuxfP2ht0iSISLZnt0JgWlrOKZiNqyUZWnjIaCIVNQArMHnCZKfEYRg6IM7A+NeJoN8gf/Ws0A==", + "dependencies": { + "bn.js": "^4.1.0", + "elliptic": "^6.5.3" + } + }, + "node_modules/create-ecdh/node_modules/bn.js": { + "version": "4.12.0", + "resolved": "https://registry.npmmirror.com/bn.js/-/bn.js-4.12.0.tgz", + "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==" + }, + "node_modules/create-hash": { + "version": "1.2.0", + "resolved": "https://registry.npmmirror.com/create-hash/-/create-hash-1.2.0.tgz", + "integrity": "sha512-z00bCGNHDG8mHAkP7CtT1qVu+bFQUPjYq/4Iv3C3kWjTFV10zIjfSoeqXo9Asws8gwSHDGj/hl2u4OGIjapeCg==", + "dependencies": { + "cipher-base": "^1.0.1", + "inherits": "^2.0.1", + "md5.js": "^1.3.4", + "ripemd160": "^2.0.1", + "sha.js": "^2.4.0" + } + }, + "node_modules/create-hmac": { + "version": "1.1.7", + "resolved": "https://registry.npmmirror.com/create-hmac/-/create-hmac-1.1.7.tgz", + "integrity": "sha512-MJG9liiZ+ogc4TzUwuvbER1JRdgvUFSB5+VR/g5h82fGaIRWMWddtKBHi7/sVhfjQZ6SehlyhvQYrcYkaUIpLg==", + "dependencies": { + "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" + } + }, + "node_modules/cross-spawn": { + "version": "6.0.5", + "resolved": "https://registry.npmmirror.com/cross-spawn/-/cross-spawn-6.0.5.tgz", + "integrity": "sha512-eTVLrBSt7fjbDygz805pMnstIs2VTBNkRm0qxZd+M7A5XDdxVRWO5MxGBXZhjY4cqLYLdtrGqRf8mBPmzwSpWQ==", + "dev": true, + "dependencies": { + "nice-try": "^1.0.4", + "path-key": "^2.0.1", + "semver": "^5.5.0", + "shebang-command": "^1.2.0", + "which": "^1.2.9" + }, + "engines": { + "node": ">=4.8" + } + }, + "node_modules/crypt": { + "version": "0.0.2", + "resolved": "https://registry.npmmirror.com/crypt/-/crypt-0.0.2.tgz", + "integrity": "sha512-mCxBlsHFYh9C+HVpiEacem8FEBnMXgU9gy4zmNC+SXAZNB/1idgp/aulFJ4FgCi7GPEVbfyng092GqL2k2rmow==", + "engines": { + "node": "*" + } + }, + "node_modules/crypto-browserify": { + "version": "3.12.0", + "resolved": "https://registry.npmmirror.com/crypto-browserify/-/crypto-browserify-3.12.0.tgz", + "integrity": "sha512-fz4spIh+znjO2VjL+IdhEpRJ3YN6sMzITSBijk6FK2UvTqruSQW+/cCZTSNsMiZNvUeq0CqurF+dAbyiGOY6Wg==", + "dependencies": { + "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" + }, + "engines": { + "node": "*" + } + }, + "node_modules/csp-html-webpack-plugin": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/csp-html-webpack-plugin/-/csp-html-webpack-plugin-4.0.0.tgz", + "integrity": "sha512-1YqQefNG0SrZisysThlly2bgs4Ab/W91xOM17S8wd+6vTo3E0OdL+y4IAR0MKpthRluNGzFB3QhPqdOhkXAExg==", + "dev": true, + "dependencies": { + "cheerio": "^1.0.0-rc.3", + "lodash": "^4.17.15", + "memory-fs": "^0.5.0" + }, + "peerDependencies": { + "html-webpack-plugin": "^2 || ^3 || ^4", + "webpack": "^2 || ^3 || ^4" + } + }, + "node_modules/csp-html-webpack-plugin/node_modules/memory-fs": { + "version": "0.5.0", + "resolved": "https://registry.npmmirror.com/memory-fs/-/memory-fs-0.5.0.tgz", + "integrity": "sha512-jA0rdU5KoQMC0e6ppoNRtpp6vjFq6+NY7r8hywnC7V+1Xj/MtHwGIbB1QaK/dunyjWteJzmkpd7ooeWg10T7GA==", + "dev": true, + "dependencies": { + "errno": "^0.1.3", + "readable-stream": "^2.0.1" + }, + "engines": { + "node": ">=4.3.0 <5.0.0 || >=5.10" + } + }, + "node_modules/css-color-names": { + "version": "0.0.4", + "resolved": "https://registry.npmmirror.com/css-color-names/-/css-color-names-0.0.4.tgz", + "integrity": "sha512-zj5D7X1U2h2zsXOAM8EyUREBnnts6H+Jm+d1M2DbiQQcUtnqgQsMrdo8JW9R80YFUmIdBZeMu5wvYM7hcgWP/Q==", + "dev": true, + "engines": { + "node": "*" + } + }, + "node_modules/css-declaration-sorter": { + "version": "4.0.1", + "resolved": "https://registry.npmmirror.com/css-declaration-sorter/-/css-declaration-sorter-4.0.1.tgz", + "integrity": "sha512-BcxQSKTSEEQUftYpBVnsH4SF05NTuBokb19/sBt6asXGKZ/6VP7PLG1CBCkFDYOnhXhPh0jMhO6xZ71oYHXHBA==", + "dev": true, + "dependencies": { + "postcss": "^7.0.1", + "timsort": "^0.3.0" + }, + "engines": { + "node": ">4" + } + }, + "node_modules/css-loader": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/css-loader/-/css-loader-1.0.1.tgz", + "integrity": "sha512-+ZHAZm/yqvJ2kDtPne3uX0C+Vr3Zn5jFn2N4HywtS5ujwvsVkyg0VArEXpl3BgczDA8anieki1FIzhchX4yrDw==", + "dev": true, + "dependencies": { + "babel-code-frame": "^6.26.0", + "css-selector-tokenizer": "^0.7.0", + "icss-utils": "^2.1.0", + "loader-utils": "^1.0.2", + "lodash": "^4.17.11", + "postcss": "^6.0.23", + "postcss-modules-extract-imports": "^1.2.0", + "postcss-modules-local-by-default": "^1.2.0", + "postcss-modules-scope": "^1.1.0", + "postcss-modules-values": "^1.3.0", + "postcss-value-parser": "^3.3.0", + "source-list-map": "^2.0.0" + }, + "engines": { + "node": ">= 6.9.0 <7.0.0 || >= 8.9.0" + }, + "peerDependencies": { + "webpack": "^4.0.0" + } + }, + "node_modules/css-loader/node_modules/postcss": { + "version": "6.0.23", + "resolved": "https://registry.npmmirror.com/postcss/-/postcss-6.0.23.tgz", + "integrity": "sha512-soOk1h6J3VMTZtVeVpv15/Hpdl2cBLX3CAw4TAbkpTJiNPk9YP/zWcD1ND+xEtvyuuvKzbxliTOIyvkSeSJ6ag==", + "dev": true, + "dependencies": { + "chalk": "^2.4.1", + "source-map": "^0.6.1", + "supports-color": "^5.4.0" + }, + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/css-loader/node_modules/postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmmirror.com/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + }, + "node_modules/css-loader/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmmirror.com/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/css-select": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/css-select/-/css-select-2.1.0.tgz", + "integrity": "sha512-Dqk7LQKpwLoH3VovzZnkzegqNSuAziQyNZUcrdDM401iY+R5NkGBXGmtO05/yaXQziALuPogeG0b7UAgjnTJTQ==", + "dependencies": { + "boolbase": "^1.0.0", + "css-what": "^3.2.1", + "domutils": "^1.7.0", + "nth-check": "^1.0.2" + } + }, + "node_modules/css-select-base-adapter": { + "version": "0.1.1", + "resolved": "https://registry.npmmirror.com/css-select-base-adapter/-/css-select-base-adapter-0.1.1.tgz", + "integrity": "sha512-jQVeeRG70QI08vSTwf1jHxp74JoZsr2XSgETae8/xC8ovSnL2WF87GTLO86Sbwdt2lK4Umg4HnnwMO4YF3Ce7w==" + }, + "node_modules/css-selector-tokenizer": { + "version": "0.7.3", + "resolved": "https://registry.npmmirror.com/css-selector-tokenizer/-/css-selector-tokenizer-0.7.3.tgz", + "integrity": "sha512-jWQv3oCEL5kMErj4wRnK/OPoBi0D+P1FR2cDCKYPaMeD2eW3/mttav8HT4hT1CKopiJI/psEULjkClhvJo4Lvg==", + "dev": true, + "dependencies": { + "cssesc": "^3.0.0", + "fastparse": "^1.1.2" + } + }, + "node_modules/css-tree": { + "version": "1.0.0-alpha.33", + "resolved": "https://registry.npmmirror.com/css-tree/-/css-tree-1.0.0-alpha.33.tgz", + "integrity": "sha512-SPt57bh5nQnpsTBsx/IXbO14sRc9xXu5MtMAVuo0BaQQmyf0NupNPPSoMaqiAF5tDFafYsTkfeH4Q/HCKXkg4w==", + "dependencies": { + "mdn-data": "2.0.4", + "source-map": "^0.5.3" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/css-what": { + "version": "3.4.2", + "resolved": "https://registry.npmmirror.com/css-what/-/css-what-3.4.2.tgz", + "integrity": "sha512-ACUm3L0/jiZTqfzRM3Hi9Q8eZqd6IK37mMWPLz9PJxkLWllYeRf+EHUSHYEtFop2Eqytaq1FizFVh7XfBnXCDQ==", + "engines": { + "node": ">= 6" + } + }, + "node_modules/cssesc": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/cssesc/-/cssesc-3.0.0.tgz", + "integrity": "sha512-/Tb/JcjK111nNScGob5MNtsntNM1aCNUDipB/TkwZFhyDrrE47SOx/18wF2bbjgc3ZzCSKW1T5nt5EbFoAz/Vg==", + "dev": true, + "bin": { + "cssesc": "bin/cssesc" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/cssfilter": { + "version": "0.0.10", + "resolved": "https://registry.npmmirror.com/cssfilter/-/cssfilter-0.0.10.tgz", + "integrity": "sha512-FAaLDaplstoRsDR8XGYH51znUN0UY7nMc6Z9/fvE8EXGwvJE9hu7W2vHwx1+bd6gCYnln9nLbzxFTrcO9YQDZw==" + }, + "node_modules/cssnano": { + "version": "4.1.11", + "resolved": "https://registry.npmmirror.com/cssnano/-/cssnano-4.1.11.tgz", + "integrity": "sha512-6gZm2htn7xIPJOHY824ERgj8cNPgPxyCSnkXc4v7YvNW+TdVfzgngHcEhy/8D11kUWRUMbke+tC+AUcUsnMz2g==", + "dev": true, + "dependencies": { + "cosmiconfig": "^5.0.0", + "cssnano-preset-default": "^4.0.8", + "is-resolvable": "^1.0.0", + "postcss": "^7.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/cssnano-preset-default": { + "version": "4.0.8", + "resolved": "https://registry.npmmirror.com/cssnano-preset-default/-/cssnano-preset-default-4.0.8.tgz", + "integrity": "sha512-LdAyHuq+VRyeVREFmuxUZR1TXjQm8QQU/ktoo/x7bz+SdOge1YKc5eMN6pRW7YWBmyq59CqYba1dJ5cUukEjLQ==", + "dev": true, + "dependencies": { + "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.3", + "postcss-unique-selectors": "^4.0.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/cssnano-util-get-arguments": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/cssnano-util-get-arguments/-/cssnano-util-get-arguments-4.0.0.tgz", + "integrity": "sha512-6RIcwmV3/cBMG8Aj5gucQRsJb4vv4I4rn6YjPbVWd5+Pn/fuG+YseGvXGk00XLkoZkaj31QOD7vMUpNPC4FIuw==", + "dev": true, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/cssnano-util-get-match": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/cssnano-util-get-match/-/cssnano-util-get-match-4.0.0.tgz", + "integrity": "sha512-JPMZ1TSMRUPVIqEalIBNoBtAYbi8okvcFns4O0YIhcdGebeYZK7dMyHJiQ6GqNBA9kE0Hym4Aqym5rPdsV/4Cw==", + "dev": true, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/cssnano-util-raw-cache": { + "version": "4.0.1", + "resolved": "https://registry.npmmirror.com/cssnano-util-raw-cache/-/cssnano-util-raw-cache-4.0.1.tgz", + "integrity": "sha512-qLuYtWK2b2Dy55I8ZX3ky1Z16WYsx544Q0UWViebptpwn/xDBmog2TLg4f+DBMg1rJ6JDWtn96WHbOKDWt1WQA==", + "dev": true, + "dependencies": { + "postcss": "^7.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/cssnano-util-same-parent": { + "version": "4.0.1", + "resolved": "https://registry.npmmirror.com/cssnano-util-same-parent/-/cssnano-util-same-parent-4.0.1.tgz", + "integrity": "sha512-WcKx5OY+KoSIAxBW6UBBRay1U6vkYheCdjyVNDm85zt5K9mHoGOfsOsqIszfAqrQQFIIKgjh2+FDgIj/zsl21Q==", + "dev": true, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/csso": { + "version": "3.5.1", + "resolved": "https://registry.npmmirror.com/csso/-/csso-3.5.1.tgz", + "integrity": "sha512-vrqULLffYU1Q2tLdJvaCYbONStnfkfimRxXNaGjxMldI0C7JPBC4rB1RyjhfdZ4m1frm8pM9uRPKH3d2knZ8gg==", + "dependencies": { + "css-tree": "1.0.0-alpha.29" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/csso/node_modules/css-tree": { + "version": "1.0.0-alpha.29", + "resolved": "https://registry.npmmirror.com/css-tree/-/css-tree-1.0.0-alpha.29.tgz", + "integrity": "sha512-sRNb1XydwkW9IOci6iB2xmy8IGCj6r/fr+JWitvJ2JxQRPzN3T4AGGVWCMlVmVwM1gtgALJRmGIlWv5ppnGGkg==", + "dependencies": { + "mdn-data": "~1.1.0", + "source-map": "^0.5.3" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/csso/node_modules/mdn-data": { + "version": "1.1.4", + "resolved": "https://registry.npmmirror.com/mdn-data/-/mdn-data-1.1.4.tgz", + "integrity": "sha512-FSYbp3lyKjyj3E7fMl6rYvUdX0FBXaluGqlFoYESWQlyUTq8R+wp0rkFxoYFqZlHCvsUXGjyJmLQSnXToYhOSA==" + }, + "node_modules/csstype": { + "version": "3.0.11", + "resolved": "https://registry.npmmirror.com/csstype/-/csstype-3.0.11.tgz", + "integrity": "sha512-sa6P2wJ+CAbgyy4KFssIb/JNMLxFvKF1pCYCSXS8ZMuqZnMsrxqI2E5sPyoTpxoPU/gVZMzr2zjOfg8GIZOMsw==" + }, + "node_modules/current-script-polyfill": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/current-script-polyfill/-/current-script-polyfill-1.0.0.tgz", + "integrity": "sha512-qv8s+G47V6Hq+g2kRE5th+ASzzrL7b6l+tap1DHKK25ZQJv3yIFhH96XaQ7NGL+zRW3t/RDbweJf/dJDe5Z5KA==", + "dev": true + }, + "node_modules/cyclist": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/cyclist/-/cyclist-1.0.1.tgz", + "integrity": "sha512-NJGVKPS81XejHcLhaLJS7plab0fK3slPh11mESeeDq2W4ZI5kUKK/LRRdVDvjJseojbPB7ZwjnyOybg3Igea/A==" + }, + "node_modules/d3": { + "version": "3.5.17", + "resolved": "https://registry.npmmirror.com/d3/-/d3-3.5.17.tgz", + "integrity": "sha512-yFk/2idb8OHPKkbAL8QaOaqENNoMhIaSHZerk3oQsECwkObkCpJyjYwCe+OHiq6UEdhe1m8ZGARRRO3ljFjlKg==" + }, + "node_modules/d3-dispatch": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/d3-dispatch/-/d3-dispatch-2.0.0.tgz", + "integrity": "sha512-S/m2VsXI7gAti2pBoLClFFTMOO1HTtT0j99AuXLoGFKO6deHDdnv6ZGTxSTTUTgO1zVcv82fCOtDjYK4EECmWA==" + }, + "node_modules/d3-force": { + "version": "2.1.1", + "resolved": "https://registry.npmmirror.com/d3-force/-/d3-force-2.1.1.tgz", + "integrity": "sha512-nAuHEzBqMvpFVMf9OX75d00OxvOXdxY+xECIXjW6Gv8BRrXu6gAWbv/9XKrvfJ5i5DCokDW7RYE50LRoK092ew==", + "dependencies": { + "d3-dispatch": "1 - 2", + "d3-quadtree": "1 - 2", + "d3-timer": "1 - 2" + } + }, + "node_modules/d3-quadtree": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/d3-quadtree/-/d3-quadtree-2.0.0.tgz", + "integrity": "sha512-b0Ed2t1UUalJpc3qXzKi+cPGxeXRr4KU9YSlocN74aTzp6R/Ud43t79yLLqxHRWZfsvWXmbDWPpoENK1K539xw==" + }, + "node_modules/d3-timer": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/d3-timer/-/d3-timer-2.0.0.tgz", + "integrity": "sha512-TO4VLh0/420Y/9dO3+f9abDEFYeCUr2WZRlxJvbp4HPTQcSylXNiL6yZa9FIUvV1yRiFufl1bszTCLDqv9PWNA==" + }, + "node_modules/dagre": { + "version": "0.8.5", + "resolved": "https://registry.npmmirror.com/dagre/-/dagre-0.8.5.tgz", + "integrity": "sha512-/aTqmnRta7x7MCCpExk7HQL2O4owCT2h8NT//9I1OQ9vt29Pa0BzSAkR5lwFUcQ7491yVi/3CXU9jQ5o0Mn2Sw==", + "dependencies": { + "graphlib": "^2.1.8", + "lodash": "^4.17.15" + } + }, + "node_modules/dargs": { + "version": "7.0.0", + "resolved": "https://registry.npmmirror.com/dargs/-/dargs-7.0.0.tgz", + "integrity": "sha512-2iy1EkLdlBzQGvbweYRFxmFath8+K7+AKB0TlhHWkNuH+TmovaMH/Wp7V7R4u7f4SnX3OgLsU9t1NI9ioDnUpg==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/dashdash": { + "version": "1.14.1", + "resolved": "https://registry.npmmirror.com/dashdash/-/dashdash-1.14.1.tgz", + "integrity": "sha512-jRFi8UDGo6j+odZiEpjazZaWqEal3w/basFjQHQEwVtZJGDpxbH1MeYluwCS8Xq5wmLJooDlMgvVarmWfGM44g==", + "dev": true, + "dependencies": { + "assert-plus": "^1.0.0" + }, + "engines": { + "node": ">=0.10" + } + }, + "node_modules/date-fns": { + "version": "1.30.1", + "resolved": "https://registry.npmmirror.com/date-fns/-/date-fns-1.30.1.tgz", + "integrity": "sha512-hBSVCvSmWC+QypYObzwGOd9wqdDpOt+0wl0KbU+R+uuZBS1jN8VsD1ss3irQDknRj5NvxiTF6oj/nDRnN/UQNw==", + "dev": true + }, + "node_modules/dateformat": { + "version": "3.0.3", + "resolved": "https://registry.npmmirror.com/dateformat/-/dateformat-3.0.3.tgz", + "integrity": "sha512-jyCETtSl3VMZMWeRo7iY1FL19ges1t55hMo5yaam4Jrsm5EPL89UQkoQRyiI+Yf4k8r2ZpdngkV8hr1lIdjb3Q==", + "dev": true, + "engines": { + "node": "*" + } + }, + "node_modules/dayjs": { + "version": "1.10.7", + "resolved": "https://registry.npmmirror.com/dayjs/-/dayjs-1.10.7.tgz", + "integrity": "sha512-P6twpd70BcPK34K26uJ1KT3wlhpuOAPoMwJzpsIWUxHZ7wpmbdZL/hQqBDfz7hGurYSa5PhzdhDHtt319hL3ig==" + }, + "node_modules/de-indent": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/de-indent/-/de-indent-1.0.2.tgz", + "integrity": "sha512-e/1zu3xH5MQryN2zdVaF0OrdNLUbvWxzMbi+iNA6Bky7l1RoP8a2fIbRocyHclXt/arDrrR6lL3TqFD9pMQTsg==", + "dev": true + }, + "node_modules/debug": { + "version": "2.6.9", + "resolved": "https://registry.npmmirror.com/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dependencies": { + "ms": "2.0.0" + } + }, + "node_modules/debuglog": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/debuglog/-/debuglog-1.0.1.tgz", + "integrity": "sha512-syBZ+rnAK3EgMsH2aYEOLUW7mZSY9Gb+0wUMCFsZvcmiz+HigA0LOcq/HoQqVuGG+EKykunc7QG2bzrponfaSw==", + "dev": true, + "engines": { + "node": "*" + } + }, + "node_modules/decamelize": { + "version": "1.2.0", + "resolved": "https://registry.npmmirror.com/decamelize/-/decamelize-1.2.0.tgz", + "integrity": "sha512-z2S+W9X73hAUUki+N+9Za2lBlun89zigOyGrsax+KUQ6wKW4ZoWpEYBkGhQjwAjjDCkWxhY0VKEhk8wzY7F5cA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/decamelize-keys": { + "version": "1.1.0", + "resolved": "https://registry.npmmirror.com/decamelize-keys/-/decamelize-keys-1.1.0.tgz", + "integrity": "sha512-ocLWuYzRPoS9bfiSdDd3cxvrzovVMZnRDVEzAs+hWIVXGDbHxWMECij2OBuyB/An0FFW/nLuq6Kv1i/YC5Qfzg==", + "dev": true, + "dependencies": { + "decamelize": "^1.1.0", + "map-obj": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/decamelize-keys/node_modules/map-obj": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/map-obj/-/map-obj-1.0.1.tgz", + "integrity": "sha512-7N/q3lyZ+LVCp7PzuxrJr4KMbBE2hW7BT7YNia330OFxIf4d3r5zVpicP2650l7CPN6RM9zOJRl3NGpqSiw3Eg==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/decode-uri-component": { + "version": "0.2.0", + "resolved": "https://registry.npmmirror.com/decode-uri-component/-/decode-uri-component-0.2.0.tgz", + "integrity": "sha512-hjf+xovcEn31w/EUYdTXQh/8smFL/dzYjohQGEIgjyNavaJfBY2p5F527Bo1VPATxv0VYTUC2bOcXvqFwk78Og==", + "engines": { + "node": ">=0.10" + } + }, + "node_modules/dedent": { + "version": "0.7.0", + "resolved": "https://registry.npmmirror.com/dedent/-/dedent-0.7.0.tgz", + "integrity": "sha512-Q6fKUPqnAHAyhiUgFU7BUzLiv0kd8saH9al7tnu5Q/okj6dnupxyTgFIBjVzJATdfIAm9NAsvXNzjaKa+bxVyA==", + "dev": true + }, + "node_modules/deep-equal": { + "version": "1.1.1", + "resolved": "https://registry.npmmirror.com/deep-equal/-/deep-equal-1.1.1.tgz", + "integrity": "sha512-yd9c5AdiqVcR+JjcwUQb9DkhJc8ngNr0MahEBGvDiJw8puWab2yZlh+nkasOnZP+EGTAP6rRp2JzJhJZzvNF8g==", + "dev": true, + "dependencies": { + "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" + } + }, + "node_modules/deep-is": { + "version": "0.1.4", + "resolved": "https://registry.npmmirror.com/deep-is/-/deep-is-0.1.4.tgz", + "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==", + "dev": true + }, + "node_modules/deepmerge": { + "version": "2.2.1", + "resolved": "https://registry.npmmirror.com/deepmerge/-/deepmerge-2.2.1.tgz", + "integrity": "sha512-R9hc1Xa/NOBi9WRVUWg19rl1UB7Tt4kuPd+thNJgFZoxXsTz7ncaPaeIm+40oSGuP33DfMb4sZt1QIGiJzC4EA==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/default-gateway": { + "version": "5.0.5", + "resolved": "https://registry.npmmirror.com/default-gateway/-/default-gateway-5.0.5.tgz", + "integrity": "sha512-z2RnruVmj8hVMmAnEJMTIJNijhKCDiGjbLP+BHJFOT7ld3Bo5qcIBpVYDniqhbMIIf+jZDlkP2MkPXiQy/DBLA==", + "dev": true, + "dependencies": { + "execa": "^3.3.0" + }, + "engines": { + "node": "^8.12.0 || >=9.7.0" + } + }, + "node_modules/default-gateway/node_modules/cross-spawn": { + "version": "7.0.3", + "resolved": "https://registry.npmmirror.com/cross-spawn/-/cross-spawn-7.0.3.tgz", + "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", + "dev": true, + "dependencies": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/default-gateway/node_modules/execa": { + "version": "3.4.0", + "resolved": "https://registry.npmmirror.com/execa/-/execa-3.4.0.tgz", + "integrity": "sha512-r9vdGQk4bmCuK1yKQu1KTwcT2zwfWdbdaXfCtAh+5nU/4fSX+JAb7vZGvI5naJrQlvONrEB20jeruESI69530g==", + "dev": true, + "dependencies": { + "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", + "p-finally": "^2.0.0", + "signal-exit": "^3.0.2", + "strip-final-newline": "^2.0.0" + }, + "engines": { + "node": "^8.12.0 || >=9.7.0" + } + }, + "node_modules/default-gateway/node_modules/get-stream": { + "version": "5.2.0", + "resolved": "https://registry.npmmirror.com/get-stream/-/get-stream-5.2.0.tgz", + "integrity": "sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA==", + "dev": true, + "dependencies": { + "pump": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/default-gateway/node_modules/is-stream": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/is-stream/-/is-stream-2.0.1.tgz", + "integrity": "sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/default-gateway/node_modules/mimic-fn": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/mimic-fn/-/mimic-fn-2.1.0.tgz", + "integrity": "sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/default-gateway/node_modules/npm-run-path": { + "version": "4.0.1", + "resolved": "https://registry.npmmirror.com/npm-run-path/-/npm-run-path-4.0.1.tgz", + "integrity": "sha512-S48WzZW777zhNIrn7gxOlISNAqi9ZC/uQFnRdbeIHhZhCA6UqpkOT8T1G7BvfdgP4Er8gF4sUbaS0i7QvIfCWw==", + "dev": true, + "dependencies": { + "path-key": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/default-gateway/node_modules/onetime": { + "version": "5.1.2", + "resolved": "https://registry.npmmirror.com/onetime/-/onetime-5.1.2.tgz", + "integrity": "sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==", + "dev": true, + "dependencies": { + "mimic-fn": "^2.1.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/default-gateway/node_modules/p-finally": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/p-finally/-/p-finally-2.0.1.tgz", + "integrity": "sha512-vpm09aKwq6H9phqRQzecoDpD8TmVyGw70qmWlyq5onxY7tqyTTFVvxMykxQSQKILBSFlbXpypIw2T1Ml7+DDtw==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/default-gateway/node_modules/path-key": { + "version": "3.1.1", + "resolved": "https://registry.npmmirror.com/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/default-gateway/node_modules/shebang-command": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/shebang-command/-/shebang-command-2.0.0.tgz", + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", + "dev": true, + "dependencies": { + "shebang-regex": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/default-gateway/node_modules/shebang-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/shebang-regex/-/shebang-regex-3.0.0.tgz", + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/default-gateway/node_modules/which": { + "version": "2.0.2", + "resolved": "https://registry.npmmirror.com/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "node-which": "bin/node-which" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/defaults": { + "version": "1.0.3", + "resolved": "https://registry.npmmirror.com/defaults/-/defaults-1.0.3.tgz", + "integrity": "sha512-s82itHOnYrN0Ib8r+z7laQz3sdE+4FP3d9Q7VLO7U+KRT+CR0GsWuyHxzdAY82I7cXv0G/twrqomTJLOssO5HA==", + "dev": true, + "dependencies": { + "clone": "^1.0.2" + } + }, + "node_modules/define-properties": { + "version": "1.1.4", + "resolved": "https://registry.npmmirror.com/define-properties/-/define-properties-1.1.4.tgz", + "integrity": "sha512-uckOqKcfaVvtBdsVkdPv3XjveQJsNQqmhXgRi8uhvWWuPYZCNlzT8qAyblUgNoXdHdjMTzAqeGjAoli8f+bzPA==", + "dependencies": { + "has-property-descriptors": "^1.0.0", + "object-keys": "^1.1.1" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/define-property": { + "version": "2.0.2", + "resolved": "https://registry.npmmirror.com/define-property/-/define-property-2.0.2.tgz", + "integrity": "sha512-jwK2UV4cnPpbcG7+VRARKTZPUWowwXA8bzH5NP6ud0oeAxyYPuGZUAC7hMugpCdz4BeSZl2Dl9k66CHJ/46ZYQ==", + "dependencies": { + "is-descriptor": "^1.0.2", + "isobject": "^3.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/define-property/node_modules/is-accessor-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", + "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", + "dependencies": { + "kind-of": "^6.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/define-property/node_modules/is-data-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", + "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", + "dependencies": { + "kind-of": "^6.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/define-property/node_modules/is-descriptor": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/is-descriptor/-/is-descriptor-1.0.2.tgz", + "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", + "dependencies": { + "is-accessor-descriptor": "^1.0.0", + "is-data-descriptor": "^1.0.0", + "kind-of": "^6.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/del": { + "version": "4.1.1", + "resolved": "https://registry.npmmirror.com/del/-/del-4.1.1.tgz", + "integrity": "sha512-QwGuEUouP2kVwQenAsOof5Fv8K9t3D8Ca8NxcXKrIpEHjTXK5J2nXLdP+ALI1cgv8wj7KuwBhTwBkOZSJKM5XQ==", + "dev": true, + "dependencies": { + "@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" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/del/node_modules/globby": { + "version": "6.1.0", + "resolved": "https://registry.npmmirror.com/globby/-/globby-6.1.0.tgz", + "integrity": "sha512-KVbFv2TQtbzCoxAnfD6JcHZTYCzyliEaaeM/gH8qQdkKr5s0OP9scEgvdcngyk7AVdY6YVW/TJHd+lQ/Df3Daw==", + "dev": true, + "dependencies": { + "array-union": "^1.0.1", + "glob": "^7.0.3", + "object-assign": "^4.0.1", + "pify": "^2.0.0", + "pinkie-promise": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/del/node_modules/globby/node_modules/pify": { + "version": "2.3.0", + "resolved": "https://registry.npmmirror.com/pify/-/pify-2.3.0.tgz", + "integrity": "sha512-udgsAY+fTnvv7kI7aaxbqwWNb0AHiB0qBO89PZKPkoTmGOgdbrHDKD+0B2X4uTfJ/FT1R09r9gTsjUjNJotuog==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/delayed-stream": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/delayed-stream/-/delayed-stream-1.0.0.tgz", + "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==", + "dev": true, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/delegate": { + "version": "3.2.0", + "resolved": "https://registry.npmmirror.com/delegate/-/delegate-3.2.0.tgz", + "integrity": "sha512-IofjkYBZaZivn0V8nnsMJGBr4jVLxHDheKSW88PyxS5QC4Vo9ZbZVvhzlSxY87fVq3STR6r+4cGepyHkcWOQSw==" + }, + "node_modules/delegates": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/delegates/-/delegates-1.0.0.tgz", + "integrity": "sha512-bd2L678uiWATM6m5Z1VzNCErI3jiGzt6HGY8OVICs40JQq/HALfbyNJmp0UDakEY4pMMaN0Ly5om/B1VI/+xfQ==", + "dev": true + }, + "node_modules/depd": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/depd/-/depd-2.0.0.tgz", + "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==", + "dev": true, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/deprecation": { + "version": "2.3.1", + "resolved": "https://registry.npmmirror.com/deprecation/-/deprecation-2.3.1.tgz", + "integrity": "sha512-xmHIy4F3scKVwMsQ4WnVaS8bHOx0DmVwRywosKhaILI0ywMDWPtBSku2HNxRvF7jtwDRsoEwYQSfbxj8b7RlJQ==", + "dev": true + }, + "node_modules/des.js": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/des.js/-/des.js-1.0.1.tgz", + "integrity": "sha512-Q0I4pfFrv2VPd34/vfLrFOoRmlYj3OV50i7fskps1jZWK1kApMWWT9G6RRUeYedLcBDIhnSDaUvJMb3AhUlaEA==", + "dependencies": { + "inherits": "^2.0.1", + "minimalistic-assert": "^1.0.0" + } + }, + "node_modules/destroy": { + "version": "1.2.0", + "resolved": "https://registry.npmmirror.com/destroy/-/destroy-1.2.0.tgz", + "integrity": "sha512-2sJGJTaXIIaR1w4iJSNoN0hnMY7Gpc/n8D4qSCJw8QqFWXf7cuAgnEHxBpweaVcPevC2l3KpjYCx3NypQQgaJg==", + "dev": true, + "engines": { + "node": ">= 0.8", + "npm": "1.2.8000 || >= 1.4.16" + } + }, + "node_modules/detect-indent": { + "version": "5.0.0", + "resolved": "https://registry.npmmirror.com/detect-indent/-/detect-indent-5.0.0.tgz", + "integrity": "sha512-rlpvsxUtM0PQvy9iZe640/IWwWYyBsTApREbA1pHOpmOUIl9MkP/U4z7vTtg4Oaojvqhxt7sdufnT0EzGaR31g==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/detect-node": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/detect-node/-/detect-node-2.1.0.tgz", + "integrity": "sha512-T0NIuQpnTvFDATNuHN5roPwSBG83rFsuO+MXXH9/3N1eFbn4wcPjttvjMLEPWJ0RGUYgQE7cGgS3tNxbqCGM7g==", + "dev": true + }, + "node_modules/dexie": { + "version": "2.0.4", + "resolved": "https://registry.npmmirror.com/dexie/-/dexie-2.0.4.tgz", + "integrity": "sha512-aQ/s1U2wHxwBKRrt2Z/mwFNHMQWhESerFsMYzE+5P5OsIe5o1kgpFMWkzKTtkvkyyEni6mWr/T4HUJuY9xIHLA==", + "engines": { + "node": ">=6.0" + } + }, + "node_modules/dezalgo": { + "version": "1.0.4", + "resolved": "https://registry.npmmirror.com/dezalgo/-/dezalgo-1.0.4.tgz", + "integrity": "sha512-rXSP0bf+5n0Qonsb+SVVfNfIsimO4HEtmnIpPHY8Q1UCzKlQrDMfdobr8nJOOsRgWCyMRqeSBQzmWUMq7zvVig==", + "dev": true, + "dependencies": { + "asap": "^2.0.0", + "wrappy": "1" + } + }, + "node_modules/diff-match-patch": { + "version": "1.0.5", + "resolved": "https://registry.npmmirror.com/diff-match-patch/-/diff-match-patch-1.0.5.tgz", + "integrity": "sha512-IayShXAgj/QMXgB0IWmKx+rOPuGMhqm5w6jvFxmVenXKIzRqTAAsbBPT3kWQeGANj3jGgvcvv4yK6SxqYmikgw==" + }, + "node_modules/diffie-hellman": { + "version": "5.0.3", + "resolved": "https://registry.npmmirror.com/diffie-hellman/-/diffie-hellman-5.0.3.tgz", + "integrity": "sha512-kqag/Nl+f3GwyK25fhUMYj81BUOrZ9IuJsjIcDE5icNM9FJHAVm3VcUDxdLPoQtTuUylWm6ZIknYJwwaPxsUzg==", + "dependencies": { + "bn.js": "^4.1.0", + "miller-rabin": "^4.0.0", + "randombytes": "^2.0.0" + } + }, + "node_modules/diffie-hellman/node_modules/bn.js": { + "version": "4.12.0", + "resolved": "https://registry.npmmirror.com/bn.js/-/bn.js-4.12.0.tgz", + "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==" + }, + "node_modules/dir-glob": { + "version": "2.2.2", + "resolved": "https://registry.npmmirror.com/dir-glob/-/dir-glob-2.2.2.tgz", + "integrity": "sha512-f9LBi5QWzIW3I6e//uxZoLBlUt9kcp66qo0sSCxL6YZKc75R1c4MFCoe/LaZiBGmgujvQdxc5Bn3QhfyvK5Hsw==", + "dev": true, + "dependencies": { + "path-type": "^3.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/dns-equal": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/dns-equal/-/dns-equal-1.0.0.tgz", + "integrity": "sha512-z+paD6YUQsk+AbGCEM4PrOXSss5gd66QfcVBFTKR/HpFL9jCqikS94HYwKww6fQyO7IxrIIyUu+g0Ka9tUS2Cg==", + "dev": true + }, + "node_modules/dns-packet": { + "version": "1.3.4", + "resolved": "https://registry.npmmirror.com/dns-packet/-/dns-packet-1.3.4.tgz", + "integrity": "sha512-BQ6F4vycLXBvdrJZ6S3gZewt6rcrks9KBgM9vrhW+knGRqc8uEdT7fuCwloc7nny5xNoMJ17HGH0R/6fpo8ECA==", + "dev": true, + "dependencies": { + "ip": "^1.1.0", + "safe-buffer": "^5.0.1" + } + }, + "node_modules/dns-txt": { + "version": "2.0.2", + "resolved": "https://registry.npmmirror.com/dns-txt/-/dns-txt-2.0.2.tgz", + "integrity": "sha512-Ix5PrWjphuSoUXV/Zv5gaFHjnaJtb02F2+Si3Ht9dyJ87+Z/lMmy+dpNHtTGraNK958ndXq2i+GLkWsWHcKaBQ==", + "dev": true, + "dependencies": { + "buffer-indexof": "^1.0.0" + } + }, + "node_modules/doctrine": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/doctrine/-/doctrine-2.1.0.tgz", + "integrity": "sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw==", + "dev": true, + "dependencies": { + "esutils": "^2.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/dom-converter": { + "version": "0.2.0", + "resolved": "https://registry.npmmirror.com/dom-converter/-/dom-converter-0.2.0.tgz", + "integrity": "sha512-gd3ypIPfOMr9h5jIKq8E3sHOTCjeirnl0WK5ZdS1AW0Odt0b1PaWaHdJ4Qk4klv+YB9aJBS7mESXjFoDQPu6DA==", + "dev": true, + "dependencies": { + "utila": "~0.4" + } + }, + "node_modules/dom-serializer": { + "version": "0.2.2", + "resolved": "https://registry.npmmirror.com/dom-serializer/-/dom-serializer-0.2.2.tgz", + "integrity": "sha512-2/xPb3ORsQ42nHYiSunXkDjPLBaEj/xTwUO4B7XCZQTRk7EBtTOPaygh10YAAh2OI1Qrp6NWfpAhzswj0ydt9g==", + "dependencies": { + "domelementtype": "^2.0.1", + "entities": "^2.0.0" + } + }, + "node_modules/dom-serializer/node_modules/domelementtype": { + "version": "2.3.0", + "resolved": "https://registry.npmmirror.com/domelementtype/-/domelementtype-2.3.0.tgz", + "integrity": "sha512-OLETBj6w0OsagBwdXnPdN0cnMfF9opN69co+7ZrbfPGrdpPVNBUj02spi6B1N7wChLQiPn4CSH/zJvXw56gmHw==" + }, + "node_modules/dom-to-image": { + "version": "2.6.0", + "resolved": "https://registry.npmmirror.com/dom-to-image/-/dom-to-image-2.6.0.tgz", + "integrity": "sha512-Dt0QdaHmLpjURjU7Tnu3AgYSF2LuOmksSGsUcE6ItvJoCWTBEmiMXcqBdNSAm9+QbbwD7JMoVsuuKX6ZVQv1qA==" + }, + "node_modules/domain-browser": { + "version": "1.2.0", + "resolved": "https://registry.npmmirror.com/domain-browser/-/domain-browser-1.2.0.tgz", + "integrity": "sha512-jnjyiM6eRyZl2H+W8Q/zLMA481hzi0eszAaBUzIVnmYVDBbnLxVNnfu1HgEBvCbL+71FrxMl3E6lpKH7Ge3OXA==", + "engines": { + "node": ">=0.4", + "npm": ">=1.2" + } + }, + "node_modules/domelementtype": { + "version": "1.3.1", + "resolved": "https://registry.npmmirror.com/domelementtype/-/domelementtype-1.3.1.tgz", + "integrity": "sha512-BSKB+TSpMpFI/HOxCNr1O8aMOTZ8hT3pM3GQ0w/mWRmkhEDSFJkkyzz4XQsBV44BChwGkrDfMyjVD0eA2aFV3w==" + }, + "node_modules/domhandler": { + "version": "4.3.1", + "resolved": "https://registry.npmmirror.com/domhandler/-/domhandler-4.3.1.tgz", + "integrity": "sha512-GrwoxYN+uWlzO8uhUXRl0P+kHE4GtVPfYzVLcUxPL7KNdHKj66vvlhiweIHqYYXWlw+T8iLMp42Lm67ghw4WMQ==", + "dev": true, + "dependencies": { + "domelementtype": "^2.2.0" + }, + "engines": { + "node": ">= 4" + } + }, + "node_modules/domhandler/node_modules/domelementtype": { + "version": "2.3.0", + "resolved": "https://registry.npmmirror.com/domelementtype/-/domelementtype-2.3.0.tgz", + "integrity": "sha512-OLETBj6w0OsagBwdXnPdN0cnMfF9opN69co+7ZrbfPGrdpPVNBUj02spi6B1N7wChLQiPn4CSH/zJvXw56gmHw==", + "dev": true + }, + "node_modules/domready": { + "version": "1.0.8", + "resolved": "https://registry.npmmirror.com/domready/-/domready-1.0.8.tgz", + "integrity": "sha512-uIzsOJUNk+AdGE9a6VDeessoMCzF8RrZvJCX/W8QtyfgdR6Uofn/MvRonih3OtCO79b2VDzDOymuiABrQ4z3XA==", + "dev": true + }, + "node_modules/domutils": { + "version": "1.7.0", + "resolved": "https://registry.npmmirror.com/domutils/-/domutils-1.7.0.tgz", + "integrity": "sha512-Lgd2XcJ/NjEw+7tFvfKxOzCYKZsdct5lczQ2ZaQY8Djz7pfAD3Gbp8ySJWtreII/vDlMVmxwa6pHmdxIYgttDg==", + "dependencies": { + "dom-serializer": "0", + "domelementtype": "1" + } + }, + "node_modules/dot-prop": { + "version": "5.3.0", + "resolved": "https://registry.npmmirror.com/dot-prop/-/dot-prop-5.3.0.tgz", + "integrity": "sha512-QM8q3zDe58hqUqjraQOmzZ1LIH9SWQJTlEKCH4kJ2oQvLZk7RbQXvtDM2XEq3fwkV9CCvvH4LA0AV+ogFsBM2Q==", + "dev": true, + "dependencies": { + "is-obj": "^2.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/dotenv": { + "version": "7.0.0", + "resolved": "https://registry.npmmirror.com/dotenv/-/dotenv-7.0.0.tgz", + "integrity": "sha512-M3NhsLbV1i6HuGzBUH8vXrtxOk+tWmzWKDMbAVSUp3Zsjm7ywFeuwrUXhmhQyRK1q5B5GGy7hcXPbj3bnfZg2g==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/dotenv-expand": { + "version": "5.1.0", + "resolved": "https://registry.npmmirror.com/dotenv-expand/-/dotenv-expand-5.1.0.tgz", + "integrity": "sha512-YXQl1DSa4/PQyRfgrv6aoNjhasp/p4qs9FjJ4q4cQk+8m4r6k4ZSiEyytKG8f8W9gi8WsQtIObNmKd+tMzNTmA==", + "dev": true + }, + "node_modules/dt-sql-parser": { + "version": "1.2.1", + "resolved": "https://registry.npmmirror.com/dt-sql-parser/-/dt-sql-parser-1.2.1.tgz", + "integrity": "sha512-bp6Rtm8N3m2Z3tavtKGo8J5ffaPQdPodYV2xGi9f6IsWbHzOkpNBxHlORcWtnSLtA/JvZcvImQxgbkdaGiv5cw==" + }, + "node_modules/duplexer": { + "version": "0.1.2", + "resolved": "https://registry.npmmirror.com/duplexer/-/duplexer-0.1.2.tgz", + "integrity": "sha512-jtD6YG370ZCIi/9GTaJKQxWTZD045+4R4hTk/x1UyoqadyJ9x9CgSi1RlVDQF8U2sxLLSnFkCaMihqljHIWgMg==", + "dev": true + }, + "node_modules/duplexify": { + "version": "3.7.1", + "resolved": "https://registry.npmmirror.com/duplexify/-/duplexify-3.7.1.tgz", + "integrity": "sha512-07z8uv2wMyS51kKhD1KsdXJg5WQ6t93RneqRxUHnskXVtlYYkLqM0gqStQZ3pj073g687jPCHrqNfCzawLYh5g==", + "dependencies": { + "end-of-stream": "^1.0.0", + "inherits": "^2.0.1", + "readable-stream": "^2.0.0", + "stream-shift": "^1.0.0" + } + }, + "node_modules/easings-css": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/easings-css/-/easings-css-1.0.0.tgz", + "integrity": "sha512-7Uq7NdazNfVtr0RNmPAys8it0zKCuaqxJStYKEl72D3j4gbvXhhaM7iWNbqhA4C94ygCye6VuyhzBRQC4szeBg==" + }, + "node_modules/easy-stack": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/easy-stack/-/easy-stack-1.0.1.tgz", + "integrity": "sha512-wK2sCs4feiiJeFXn3zvY0p41mdU5VUgbgs1rNsc/y5ngFUijdWd+iIN8eoyuZHKB8xN6BL4PdWmzqFmxNg6V2w==", + "dev": true, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/ecc-jsbn": { + "version": "0.1.2", + "resolved": "https://registry.npmmirror.com/ecc-jsbn/-/ecc-jsbn-0.1.2.tgz", + "integrity": "sha512-eh9O+hwRHNbG4BLTjEl3nw044CkGm5X6LoaCf7LPp7UU8Qrt47JYNi6nPX8xjW97TKGKm1ouctg0QSpZe9qrnw==", + "dev": true, + "dependencies": { + "jsbn": "~0.1.0", + "safer-buffer": "^2.1.0" + } + }, + "node_modules/echarts": { + "version": "4.9.0", + "resolved": "https://registry.npmmirror.com/echarts/-/echarts-4.9.0.tgz", + "integrity": "sha512-+ugizgtJ+KmsJyyDPxaw2Br5FqzuBnyOWwcxPKO6y0gc5caYcfnEUIlNStx02necw8jmKmTafmpHhGo4XDtEIA==", + "dependencies": { + "zrender": "4.3.2" + } + }, + "node_modules/ee-first": { + "version": "1.1.1", + "resolved": "https://registry.npmmirror.com/ee-first/-/ee-first-1.1.1.tgz", + "integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==", + "dev": true + }, + "node_modules/ejs": { + "version": "2.7.4", + "resolved": "https://registry.npmmirror.com/ejs/-/ejs-2.7.4.tgz", + "integrity": "sha512-7vmuyh5+kuUyJKePhQfRQBhXV5Ce+RnaeeQArKu1EAMpL3WbgMt5WG6uQZpEVvYSSsxMXRKOewtDk9RaTKXRlA==", + "dev": true, + "hasInstallScript": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/electron-to-chromium": { + "version": "1.4.121", + "resolved": "https://registry.npmmirror.com/electron-to-chromium/-/electron-to-chromium-1.4.121.tgz", + "integrity": "sha512-N7OXhMr1p2oa9EkOhmHpmOm43DHzs55dep2FF6M7y6px5QJBheqEE3nwwZ+xJowlff+AEmMOdg3ARYGB+0kzbA==", + "dev": true + }, + "node_modules/elegant-spinner": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/elegant-spinner/-/elegant-spinner-1.0.1.tgz", + "integrity": "sha512-B+ZM+RXvRqQaAmkMlO/oSe5nMUOaUnyfGYCEHoR8wrXsZR2mA0XVibsxV1bvTwxdRWah1PkQqso2EzhILGHtEQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/element-resize-detector": { + "version": "1.2.4", + "resolved": "https://registry.npmmirror.com/element-resize-detector/-/element-resize-detector-1.2.4.tgz", + "integrity": "sha512-Fl5Ftk6WwXE0wqCgNoseKWndjzZlDCwuPTcoVZfCP9R3EHQF8qUtr3YUPNETegRBOKqQKPW3n4kiIWngGi8tKg==", + "dependencies": { + "batch-processor": "1.0.0" + } + }, + "node_modules/elliptic": { + "version": "6.5.4", + "resolved": "https://registry.npmmirror.com/elliptic/-/elliptic-6.5.4.tgz", + "integrity": "sha512-iLhC6ULemrljPZb+QutR5TQGB+pdW6KGD5RSegS+8sorOZT+rdQFbsQFJgvN3eRqNALqJer4oQ16YvJHlU8hzQ==", + "dependencies": { + "bn.js": "^4.11.9", + "brorand": "^1.1.0", + "hash.js": "^1.0.0", + "hmac-drbg": "^1.0.1", + "inherits": "^2.0.4", + "minimalistic-assert": "^1.0.1", + "minimalistic-crypto-utils": "^1.0.1" + } + }, + "node_modules/elliptic/node_modules/bn.js": { + "version": "4.12.0", + "resolved": "https://registry.npmmirror.com/bn.js/-/bn.js-4.12.0.tgz", + "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==" + }, + "node_modules/emoji-regex": { + "version": "8.0.0", + "resolved": "https://registry.npmmirror.com/emoji-regex/-/emoji-regex-8.0.0.tgz", + "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", + "dev": true + }, + "node_modules/emojis-list": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/emojis-list/-/emojis-list-3.0.0.tgz", + "integrity": "sha512-/kyM18EfinwXZbno9FyUGeFh87KC8HRQBQGildHZbEuRyWFOmv1U10o9BBp8XVZDVNNuQKyIGIu5ZYAAXJ0V2Q==", + "engines": { + "node": ">= 4" + } + }, + "node_modules/encodeurl": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/encodeurl/-/encodeurl-1.0.2.tgz", + "integrity": "sha512-TPJXq8JqFaVYm2CWmPvnP2Iyo4ZSM7/QKcSmuMLDObfpH5fi7RUGmd/rTDf+rut/saiDiQEeVTNgAmJEdAOx0w==", + "dev": true, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/encoding": { + "version": "0.1.13", + "resolved": "https://registry.npmmirror.com/encoding/-/encoding-0.1.13.tgz", + "integrity": "sha512-ETBauow1T35Y/WZMkio9jiM0Z5xjHHmJ4XmjZOq1l/dXz3lr2sRn87nJy20RupqSh1F2m3HHPSp8ShIPQJrJ3A==", + "dev": true, + "optional": true, + "dependencies": { + "iconv-lite": "^0.6.2" + } + }, + "node_modules/encoding/node_modules/iconv-lite": { + "version": "0.6.3", + "resolved": "https://registry.npmmirror.com/iconv-lite/-/iconv-lite-0.6.3.tgz", + "integrity": "sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==", + "dev": true, + "optional": true, + "dependencies": { + "safer-buffer": ">= 2.1.2 < 3.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/end-of-stream": { + "version": "1.4.4", + "resolved": "https://registry.npmmirror.com/end-of-stream/-/end-of-stream-1.4.4.tgz", + "integrity": "sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==", + "dependencies": { + "once": "^1.4.0" + } + }, + "node_modules/enhanced-resolve": { + "version": "4.5.0", + "resolved": "https://registry.npmmirror.com/enhanced-resolve/-/enhanced-resolve-4.5.0.tgz", + "integrity": "sha512-Nv9m36S/vxpsI+Hc4/ZGRs0n9mXqSWGGq49zxb/cJfPAQMbUtttJAlNPS4AQzaBdw/pKskw5bMbekT/Y7W/Wlg==", + "dependencies": { + "graceful-fs": "^4.1.2", + "memory-fs": "^0.5.0", + "tapable": "^1.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/enhanced-resolve/node_modules/memory-fs": { + "version": "0.5.0", + "resolved": "https://registry.npmmirror.com/memory-fs/-/memory-fs-0.5.0.tgz", + "integrity": "sha512-jA0rdU5KoQMC0e6ppoNRtpp6vjFq6+NY7r8hywnC7V+1Xj/MtHwGIbB1QaK/dunyjWteJzmkpd7ooeWg10T7GA==", + "dependencies": { + "errno": "^0.1.3", + "readable-stream": "^2.0.1" + }, + "engines": { + "node": ">=4.3.0 <5.0.0 || >=5.10" + } + }, + "node_modules/entities": { + "version": "2.2.0", + "resolved": "https://registry.npmmirror.com/entities/-/entities-2.2.0.tgz", + "integrity": "sha512-p92if5Nz619I0w+akJrLZH0MX0Pb5DX39XOwQTtXSdQQOaYH03S1uIQp4mhOZtAXrxq4ViO67YTiLBo2638o9A==" + }, + "node_modules/env-paths": { + "version": "2.2.1", + "resolved": "https://registry.npmmirror.com/env-paths/-/env-paths-2.2.1.tgz", + "integrity": "sha512-+h1lkLKhZMTYjog1VEpJNG7NZJWcuc2DDk/qsqSTRRCOXiLjeQ1d1/udrUGhqMxUgAlwKNZ0cf2uqan5GLuS2A==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/envinfo": { + "version": "7.8.1", + "resolved": "https://registry.npmmirror.com/envinfo/-/envinfo-7.8.1.tgz", + "integrity": "sha512-/o+BXHmB7ocbHEAs6F2EnG0ogybVVUdkRunTT2glZU9XAaGmhqskrvKwqXuDfNjEO0LZKWdejEEpnq8aM0tOaw==", + "dev": true, + "bin": { + "envinfo": "dist/cli.js" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/err-code": { + "version": "2.0.3", + "resolved": "https://registry.npmmirror.com/err-code/-/err-code-2.0.3.tgz", + "integrity": "sha512-2bmlRpNKBxT/CRmPOlyISQpNj+qSeYvcym/uT0Jx2bMOlKLtSy1ZmLuVxSEKKyor/N5yhvp/ZiG1oE3DEYMSFA==", + "dev": true + }, + "node_modules/errno": { + "version": "0.1.8", + "resolved": "https://registry.npmmirror.com/errno/-/errno-0.1.8.tgz", + "integrity": "sha512-dJ6oBr5SQ1VSd9qkk7ByRgb/1SH4JZjCHSW/mr63/QcXO9zLVxvJ6Oy13nio03rxpSnVDDjFor75SjVeZWPW/A==", + "dependencies": { + "prr": "~1.0.1" + }, + "bin": { + "errno": "cli.js" + } + }, + "node_modules/error-ex": { + "version": "1.3.2", + "resolved": "https://registry.npmmirror.com/error-ex/-/error-ex-1.3.2.tgz", + "integrity": "sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==", + "dev": true, + "dependencies": { + "is-arrayish": "^0.2.1" + } + }, + "node_modules/error-stack-parser": { + "version": "2.0.7", + "resolved": "https://registry.npmmirror.com/error-stack-parser/-/error-stack-parser-2.0.7.tgz", + "integrity": "sha512-chLOW0ZGRf4s8raLrDxa5sdkvPec5YdvwbFnqJme4rk0rFajP8mPtrDL1+I+CwrQDCjswDA5sREX7jYQDQs9vA==", + "dev": true, + "dependencies": { + "stackframe": "^1.1.1" + } + }, + "node_modules/es-abstract": { + "version": "1.19.5", + "resolved": "https://registry.npmmirror.com/es-abstract/-/es-abstract-1.19.5.tgz", + "integrity": "sha512-Aa2G2+Rd3b6kxEUKTF4TaW67czBLyAv3z7VOhYRU50YBx+bbsYZ9xQP4lMNazePuFlybXI0V4MruPos7qUo5fA==", + "dependencies": { + "call-bind": "^1.0.2", + "es-to-primitive": "^1.2.1", + "function-bind": "^1.1.1", + "get-intrinsic": "^1.1.1", + "get-symbol-description": "^1.0.0", + "has": "^1.0.3", + "has-symbols": "^1.0.3", + "internal-slot": "^1.0.3", + "is-callable": "^1.2.4", + "is-negative-zero": "^2.0.2", + "is-regex": "^1.1.4", + "is-shared-array-buffer": "^1.0.2", + "is-string": "^1.0.7", + "is-weakref": "^1.0.2", + "object-inspect": "^1.12.0", + "object-keys": "^1.1.1", + "object.assign": "^4.1.2", + "string.prototype.trimend": "^1.0.4", + "string.prototype.trimstart": "^1.0.4", + "unbox-primitive": "^1.0.1" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/es-shim-unscopables": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/es-shim-unscopables/-/es-shim-unscopables-1.0.0.tgz", + "integrity": "sha512-Jm6GPcCdC30eMLbZ2x8z2WuRwAws3zTBBKuusffYVUrNj/GVSUAZ+xKMaUpfNDR5IbyNA5LJbaecoUVbmUcB1w==", + "dev": true, + "dependencies": { + "has": "^1.0.3" + } + }, + "node_modules/es-to-primitive": { + "version": "1.2.1", + "resolved": "https://registry.npmmirror.com/es-to-primitive/-/es-to-primitive-1.2.1.tgz", + "integrity": "sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA==", + "dependencies": { + "is-callable": "^1.1.4", + "is-date-object": "^1.0.1", + "is-symbol": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/escalade": { + "version": "3.1.1", + "resolved": "https://registry.npmmirror.com/escalade/-/escalade-3.1.1.tgz", + "integrity": "sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/escape-html": { + "version": "1.0.3", + "resolved": "https://registry.npmmirror.com/escape-html/-/escape-html-1.0.3.tgz", + "integrity": "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow==", + "dev": true + }, + "node_modules/escape-string-regexp": { + "version": "1.0.5", + "resolved": "https://registry.npmmirror.com/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", + "integrity": "sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==", + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/eslint": { + "version": "6.8.0", + "resolved": "https://registry.npmmirror.com/eslint/-/eslint-6.8.0.tgz", + "integrity": "sha512-K+Iayyo2LtyYhDSYwz5D5QdWw0hCacNzyq1Y821Xna2xSJj7cijoLLYmLxTQgcgZ9mC61nryMy9S7GRbYpI5Ig==", + "dev": true, + "dependencies": { + "@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" + }, + "bin": { + "eslint": "bin/eslint.js" + }, + "engines": { + "node": "^8.10.0 || ^10.13.0 || >=11.10.1" + } + }, + "node_modules/eslint-config-standard": { + "version": "12.0.0", + "resolved": "https://registry.npmmirror.com/eslint-config-standard/-/eslint-config-standard-12.0.0.tgz", + "integrity": "sha512-COUz8FnXhqFitYj4DTqHzidjIL/t4mumGZto5c7DrBpvWoie+Sn3P4sLEzUGeYhRElWuFEf8K1S1EfvD1vixCQ==", + "dev": true, + "peerDependencies": { + "eslint": ">=5.0.0", + "eslint-plugin-import": ">=2.13.0", + "eslint-plugin-node": ">=7.0.0", + "eslint-plugin-promise": ">=4.0.0", + "eslint-plugin-standard": ">=4.0.0" + } + }, + "node_modules/eslint-import-resolver-node": { + "version": "0.3.6", + "resolved": "https://registry.npmmirror.com/eslint-import-resolver-node/-/eslint-import-resolver-node-0.3.6.tgz", + "integrity": "sha512-0En0w03NRVMn9Uiyn8YRPDKvWjxCWkslUEhGNTdGx15RvPJYQ+lbOlqrlNI2vEAs4pDYK4f/HN2TbDmk5TP0iw==", + "dev": true, + "dependencies": { + "debug": "^3.2.7", + "resolve": "^1.20.0" + } + }, + "node_modules/eslint-import-resolver-node/node_modules/debug": { + "version": "3.2.7", + "resolved": "https://registry.npmmirror.com/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dev": true, + "dependencies": { + "ms": "^2.1.1" + } + }, + "node_modules/eslint-import-resolver-node/node_modules/ms": { + "version": "2.1.3", + "resolved": "https://registry.npmmirror.com/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", + "dev": true + }, + "node_modules/eslint-loader": { + "version": "2.2.1", + "resolved": "https://registry.npmmirror.com/eslint-loader/-/eslint-loader-2.2.1.tgz", + "integrity": "sha512-RLgV9hoCVsMLvOxCuNjdqOrUqIj9oJg8hF44vzJaYqsAHuY9G2YAeN3joQ9nxP0p5Th9iFSIpKo+SD8KISxXRg==", + "deprecated": "This loader has been deprecated. Please use eslint-webpack-plugin", + "dev": true, + "dependencies": { + "loader-fs-cache": "^1.0.0", + "loader-utils": "^1.0.2", + "object-assign": "^4.0.1", + "object-hash": "^1.1.4", + "rimraf": "^2.6.1" + }, + "peerDependencies": { + "eslint": ">=1.6.0 <7.0.0", + "webpack": ">=2.0.0 <5.0.0" + } + }, + "node_modules/eslint-module-utils": { + "version": "2.7.3", + "resolved": "https://registry.npmmirror.com/eslint-module-utils/-/eslint-module-utils-2.7.3.tgz", + "integrity": "sha512-088JEC7O3lDZM9xGe0RerkOMd0EjFl+Yvd1jPWIkMT5u3H9+HC34mWWPnqPrN13gieT9pBOO+Qt07Nb/6TresQ==", + "dev": true, + "dependencies": { + "debug": "^3.2.7", + "find-up": "^2.1.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/eslint-module-utils/node_modules/debug": { + "version": "3.2.7", + "resolved": "https://registry.npmmirror.com/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dev": true, + "dependencies": { + "ms": "^2.1.1" + } + }, + "node_modules/eslint-module-utils/node_modules/find-up": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/find-up/-/find-up-2.1.0.tgz", + "integrity": "sha512-NWzkk0jSJtTt08+FBFMvXoeZnOJD+jTtsRmBYbAIzJdX6l7dLgR7CTubCM5/eDdPUBvLCeVasP1brfVR/9/EZQ==", + "dev": true, + "dependencies": { + "locate-path": "^2.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/eslint-module-utils/node_modules/locate-path": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/locate-path/-/locate-path-2.0.0.tgz", + "integrity": "sha512-NCI2kiDkyR7VeEKm27Kda/iQHyKJe1Bu0FlTbYp3CqJu+9IFe9bLyAjMxf5ZDDbEg+iMPzB5zYyUTSm8wVTKmA==", + "dev": true, + "dependencies": { + "p-locate": "^2.0.0", + "path-exists": "^3.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/eslint-module-utils/node_modules/ms": { + "version": "2.1.3", + "resolved": "https://registry.npmmirror.com/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", + "dev": true + }, + "node_modules/eslint-module-utils/node_modules/p-limit": { + "version": "1.3.0", + "resolved": "https://registry.npmmirror.com/p-limit/-/p-limit-1.3.0.tgz", + "integrity": "sha512-vvcXsLAJ9Dr5rQOPk7toZQZJApBl2K4J6dANSsEuh6QI41JYcsS/qhTGa9ErIUUgK3WNQoJYvylxvjqmiqEA9Q==", + "dev": true, + "dependencies": { + "p-try": "^1.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/eslint-module-utils/node_modules/p-locate": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/p-locate/-/p-locate-2.0.0.tgz", + "integrity": "sha512-nQja7m7gSKuewoVRen45CtVfODR3crN3goVQ0DDZ9N3yHxgpkuBhZqsaiotSQRrADUrne346peY7kT3TSACykg==", + "dev": true, + "dependencies": { + "p-limit": "^1.1.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/eslint-module-utils/node_modules/p-try": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/p-try/-/p-try-1.0.0.tgz", + "integrity": "sha512-U1etNYuMJoIz3ZXSrrySFjsXQTWOx2/jdi86L+2pRvph/qMKL6sbcCYdH23fqsbm8TH2Gn0OybpT4eSFlCVHww==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/eslint-plugin-es": { + "version": "1.4.1", + "resolved": "https://registry.npmmirror.com/eslint-plugin-es/-/eslint-plugin-es-1.4.1.tgz", + "integrity": "sha512-5fa/gR2yR3NxQf+UXkeLeP8FBBl6tSgdrAz1+cF84v1FMM4twGwQoqTnn+QxFLcPOrF4pdKEJKDB/q9GoyJrCA==", + "dev": true, + "dependencies": { + "eslint-utils": "^1.4.2", + "regexpp": "^2.0.1" + }, + "engines": { + "node": ">=6.5.0" + }, + "peerDependencies": { + "eslint": ">=4.19.1" + } + }, + "node_modules/eslint-plugin-es/node_modules/regexpp": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/regexpp/-/regexpp-2.0.1.tgz", + "integrity": "sha512-lv0M6+TkDVniA3aD1Eg0DVpfU/booSu7Eev3TDO/mZKHBfVjgCGTV4t4buppESEYDtkArYFOxTJWv6S5C+iaNw==", + "dev": true, + "engines": { + "node": ">=6.5.0" + } + }, + "node_modules/eslint-plugin-import": { + "version": "2.26.0", + "resolved": "https://registry.npmmirror.com/eslint-plugin-import/-/eslint-plugin-import-2.26.0.tgz", + "integrity": "sha512-hYfi3FXaM8WPLf4S1cikh/r4IxnO6zrhZbEGz2b660EJRbuxgpDS5gkCuYgGWg2xxh2rBuIr4Pvhve/7c31koA==", + "dev": true, + "dependencies": { + "array-includes": "^3.1.4", + "array.prototype.flat": "^1.2.5", + "debug": "^2.6.9", + "doctrine": "^2.1.0", + "eslint-import-resolver-node": "^0.3.6", + "eslint-module-utils": "^2.7.3", + "has": "^1.0.3", + "is-core-module": "^2.8.1", + "is-glob": "^4.0.3", + "minimatch": "^3.1.2", + "object.values": "^1.1.5", + "resolve": "^1.22.0", + "tsconfig-paths": "^3.14.1" + }, + "engines": { + "node": ">=4" + }, + "peerDependencies": { + "eslint": "^2 || ^3 || ^4 || ^5 || ^6 || ^7.2.0 || ^8" + } + }, + "node_modules/eslint-plugin-node": { + "version": "8.0.1", + "resolved": "https://registry.npmmirror.com/eslint-plugin-node/-/eslint-plugin-node-8.0.1.tgz", + "integrity": "sha512-ZjOjbjEi6jd82rIpFSgagv4CHWzG9xsQAVp1ZPlhRnnYxcTgENUVBvhYmkQ7GvT1QFijUSo69RaiOJKhMu6i8w==", + "dev": true, + "dependencies": { + "eslint-plugin-es": "^1.3.1", + "eslint-utils": "^1.3.1", + "ignore": "^5.0.2", + "minimatch": "^3.0.4", + "resolve": "^1.8.1", + "semver": "^5.5.0" + }, + "engines": { + "node": ">=6" + }, + "peerDependencies": { + "eslint": ">=4.19.1" + } + }, + "node_modules/eslint-plugin-node/node_modules/ignore": { + "version": "5.2.0", + "resolved": "https://registry.npmmirror.com/ignore/-/ignore-5.2.0.tgz", + "integrity": "sha512-CmxgYGiEPCLhfLnpPp1MoRmifwEIOgjcHXxOBjv7mY96c+eWScsOP9c112ZyLdWHi0FxHjI+4uVhKYp/gcdRmQ==", + "dev": true, + "engines": { + "node": ">= 4" + } + }, + "node_modules/eslint-plugin-promise": { + "version": "4.3.1", + "resolved": "https://registry.npmmirror.com/eslint-plugin-promise/-/eslint-plugin-promise-4.3.1.tgz", + "integrity": "sha512-bY2sGqyptzFBDLh/GMbAxfdJC+b0f23ME63FOE4+Jao0oZ3E1LEwFtWJX/1pGMJLiTtrSSern2CRM/g+dfc0eQ==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/eslint-plugin-standard": { + "version": "4.1.0", + "resolved": "https://registry.npmmirror.com/eslint-plugin-standard/-/eslint-plugin-standard-4.1.0.tgz", + "integrity": "sha512-ZL7+QRixjTR6/528YNGyDotyffm5OQst/sGxKDwGb9Uqs4In5Egi4+jbobhqJoyoCM6/7v/1A5fhQ7ScMtDjaQ==", + "dev": true, + "peerDependencies": { + "eslint": ">=5.0.0" + } + }, + "node_modules/eslint-plugin-vue": { + "version": "6.2.2", + "resolved": "https://registry.npmmirror.com/eslint-plugin-vue/-/eslint-plugin-vue-6.2.2.tgz", + "integrity": "sha512-Nhc+oVAHm0uz/PkJAWscwIT4ijTrK5fqNqz9QB1D35SbbuMG1uB6Yr5AJpvPSWg+WOw7nYNswerYh0kOk64gqQ==", + "dev": true, + "dependencies": { + "natural-compare": "^1.4.0", + "semver": "^5.6.0", + "vue-eslint-parser": "^7.0.0" + }, + "engines": { + "node": ">=8.10" + }, + "peerDependencies": { + "eslint": "^5.0.0 || ^6.0.0" + } + }, + "node_modules/eslint-plugin-vue/node_modules/acorn": { + "version": "7.4.1", + "resolved": "https://registry.npmmirror.com/acorn/-/acorn-7.4.1.tgz", + "integrity": "sha512-nQyp0o1/mNdbTO1PO6kHkwSrmgZ0MT/jCCpNiwbUjGoRN4dlBhqJtoQuCnEOKzgTVwg0ZWiCoQy6SxMebQVh8A==", + "dev": true, + "bin": { + "acorn": "bin/acorn" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/eslint-plugin-vue/node_modules/acorn-jsx": { + "version": "5.3.2", + "resolved": "https://registry.npmmirror.com/acorn-jsx/-/acorn-jsx-5.3.2.tgz", + "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", + "dev": true, + "peerDependencies": { + "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0" + } + }, + "node_modules/eslint-plugin-vue/node_modules/debug": { + "version": "4.3.4", + "resolved": "https://registry.npmmirror.com/debug/-/debug-4.3.4.tgz", + "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", + "dev": true, + "dependencies": { + "ms": "2.1.2" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/eslint-plugin-vue/node_modules/eslint-scope": { + "version": "5.1.1", + "resolved": "https://registry.npmmirror.com/eslint-scope/-/eslint-scope-5.1.1.tgz", + "integrity": "sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw==", + "dev": true, + "dependencies": { + "esrecurse": "^4.3.0", + "estraverse": "^4.1.1" + }, + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/eslint-plugin-vue/node_modules/espree": { + "version": "6.2.1", + "resolved": "https://registry.npmmirror.com/espree/-/espree-6.2.1.tgz", + "integrity": "sha512-ysCxRQY3WaXJz9tdbWOwuWr5Y/XrPTGX9Kiz3yoUXwW0VZ4w30HTkQLaGx/+ttFjF8i+ACbArnB4ce68a9m5hw==", + "dev": true, + "dependencies": { + "acorn": "^7.1.1", + "acorn-jsx": "^5.2.0", + "eslint-visitor-keys": "^1.1.0" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/eslint-plugin-vue/node_modules/lodash": { + "version": "4.17.21", + "resolved": "https://registry.npmmirror.com/lodash/-/lodash-4.17.21.tgz", + "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==", + "dev": true + }, + "node_modules/eslint-plugin-vue/node_modules/ms": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", + "dev": true + }, + "node_modules/eslint-plugin-vue/node_modules/vue-eslint-parser": { + "version": "7.11.0", + "resolved": "https://registry.npmmirror.com/vue-eslint-parser/-/vue-eslint-parser-7.11.0.tgz", + "integrity": "sha512-qh3VhDLeh773wjgNTl7ss0VejY9bMMa0GoDG2fQVyDzRFdiU3L7fw74tWZDHNQXdZqxO3EveQroa9ct39D2nqg==", + "dev": true, + "dependencies": { + "debug": "^4.1.1", + "eslint-scope": "^5.1.1", + "eslint-visitor-keys": "^1.1.0", + "espree": "^6.2.1", + "esquery": "^1.4.0", + "lodash": "^4.17.21", + "semver": "^6.3.0" + }, + "engines": { + "node": ">=8.10" + }, + "peerDependencies": { + "eslint": ">=5.0.0" + } + }, + "node_modules/eslint-plugin-vue/node_modules/vue-eslint-parser/node_modules/semver": { + "version": "6.3.0", + "resolved": "https://registry.npmmirror.com/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/eslint-scope": { + "version": "4.0.3", + "resolved": "https://registry.npmmirror.com/eslint-scope/-/eslint-scope-4.0.3.tgz", + "integrity": "sha512-p7VutNr1O/QrxysMo3E45FjYDTeXBy0iTltPFNSqKAIfjDSXC+4dj+qfyuD8bfAXrW/y6lW3O76VaYNPKfpKrg==", + "dependencies": { + "esrecurse": "^4.1.0", + "estraverse": "^4.1.1" + }, + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/eslint-utils": { + "version": "1.4.3", + "resolved": "https://registry.npmmirror.com/eslint-utils/-/eslint-utils-1.4.3.tgz", + "integrity": "sha512-fbBN5W2xdY45KulGXmLHZ3c3FHfVYmKg0IrAKGOkT/464PQsx2UeIzfz1RmEci+KLm1bBaAzZAh8+/E+XAeZ8Q==", + "dev": true, + "dependencies": { + "eslint-visitor-keys": "^1.1.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/eslint-visitor-keys": { + "version": "1.3.0", + "resolved": "https://registry.npmmirror.com/eslint-visitor-keys/-/eslint-visitor-keys-1.3.0.tgz", + "integrity": "sha512-6J72N8UNa462wa/KFODt/PJ3IU60SDpC3QXC1Hjc1BXXpfL2C9R5+AU7jhe0F6GREqVMh4Juu+NY7xn+6dipUQ==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/eslint/node_modules/acorn": { + "version": "7.4.1", + "resolved": "https://registry.npmmirror.com/acorn/-/acorn-7.4.1.tgz", + "integrity": "sha512-nQyp0o1/mNdbTO1PO6kHkwSrmgZ0MT/jCCpNiwbUjGoRN4dlBhqJtoQuCnEOKzgTVwg0ZWiCoQy6SxMebQVh8A==", + "dev": true, + "bin": { + "acorn": "bin/acorn" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/eslint/node_modules/acorn-jsx": { + "version": "5.3.2", + "resolved": "https://registry.npmmirror.com/acorn-jsx/-/acorn-jsx-5.3.2.tgz", + "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", + "dev": true, + "peerDependencies": { + "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0" + } + }, + "node_modules/eslint/node_modules/ansi-escapes": { + "version": "4.3.2", + "resolved": "https://registry.npmmirror.com/ansi-escapes/-/ansi-escapes-4.3.2.tgz", + "integrity": "sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ==", + "dev": true, + "dependencies": { + "type-fest": "^0.21.3" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/eslint/node_modules/ansi-escapes/node_modules/type-fest": { + "version": "0.21.3", + "resolved": "https://registry.npmmirror.com/type-fest/-/type-fest-0.21.3.tgz", + "integrity": "sha512-t0rzBq87m3fVcduHDUFhKmyyX+9eo6WQjZvf51Ea/M0Q7+T374Jp1aUiyUl0GKxp8M/OETVHSDvmkyPgvX+X2w==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/eslint/node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmmirror.com/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/eslint/node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmmirror.com/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/eslint/node_modules/chardet": { + "version": "0.7.0", + "resolved": "https://registry.npmmirror.com/chardet/-/chardet-0.7.0.tgz", + "integrity": "sha512-mT8iDcrh03qDGRRmoA2hmBJnxpllMR+0/0qlzjqZES6NdiWDcZkCNAk4rPFZ9Q85r27unkiNNg8ZOiwZXBHwcA==", + "dev": true + }, + "node_modules/eslint/node_modules/cli-cursor": { + "version": "3.1.0", + "resolved": "https://registry.npmmirror.com/cli-cursor/-/cli-cursor-3.1.0.tgz", + "integrity": "sha512-I/zHAwsKf9FqGoXM4WWRACob9+SNukZTd94DWF57E4toouRulbCxcUh6RKUEOQlYTHJnzkPMySvPNaaSLNfLZw==", + "dev": true, + "dependencies": { + "restore-cursor": "^3.1.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/eslint/node_modules/cli-width": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/cli-width/-/cli-width-3.0.0.tgz", + "integrity": "sha512-FxqpkPPwu1HjuN93Omfm4h8uIanXofW0RxVEW3k5RKx+mJJYSthzNhp32Kzxxy3YAEZ/Dc/EWN1vZRY0+kOhbw==", + "dev": true, + "engines": { + "node": ">= 10" + } + }, + "node_modules/eslint/node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/eslint/node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmmirror.com/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "node_modules/eslint/node_modules/debug": { + "version": "4.3.4", + "resolved": "https://registry.npmmirror.com/debug/-/debug-4.3.4.tgz", + "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", + "dev": true, + "dependencies": { + "ms": "2.1.2" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/eslint/node_modules/doctrine": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/doctrine/-/doctrine-3.0.0.tgz", + "integrity": "sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==", + "dev": true, + "dependencies": { + "esutils": "^2.0.2" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/eslint/node_modules/eslint-scope": { + "version": "5.1.1", + "resolved": "https://registry.npmmirror.com/eslint-scope/-/eslint-scope-5.1.1.tgz", + "integrity": "sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw==", + "dev": true, + "dependencies": { + "esrecurse": "^4.3.0", + "estraverse": "^4.1.1" + }, + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/eslint/node_modules/espree": { + "version": "6.2.1", + "resolved": "https://registry.npmmirror.com/espree/-/espree-6.2.1.tgz", + "integrity": "sha512-ysCxRQY3WaXJz9tdbWOwuWr5Y/XrPTGX9Kiz3yoUXwW0VZ4w30HTkQLaGx/+ttFjF8i+ACbArnB4ce68a9m5hw==", + "dev": true, + "dependencies": { + "acorn": "^7.1.1", + "acorn-jsx": "^5.2.0", + "eslint-visitor-keys": "^1.1.0" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/eslint/node_modules/external-editor": { + "version": "3.1.0", + "resolved": "https://registry.npmmirror.com/external-editor/-/external-editor-3.1.0.tgz", + "integrity": "sha512-hMQ4CX1p1izmuLYyZqLMO/qGNw10wSv9QDCPfzXfyFrOaCSSoRfqE1Kf1s5an66J5JZC62NewG+mK49jOCtQew==", + "dev": true, + "dependencies": { + "chardet": "^0.7.0", + "iconv-lite": "^0.4.24", + "tmp": "^0.0.33" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/eslint/node_modules/figures": { + "version": "3.2.0", + "resolved": "https://registry.npmmirror.com/figures/-/figures-3.2.0.tgz", + "integrity": "sha512-yaduQFRKLXYOGgEn6AZau90j3ggSOyiqXU0F9JZfeXYhNa+Jk4X+s45A2zg5jns87GAFa34BBm2kXw4XpNcbdg==", + "dev": true, + "dependencies": { + "escape-string-regexp": "^1.0.5" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/eslint/node_modules/file-entry-cache": { + "version": "5.0.1", + "resolved": "https://registry.npmmirror.com/file-entry-cache/-/file-entry-cache-5.0.1.tgz", + "integrity": "sha512-bCg29ictuBaKUwwArK4ouCaqDgLZcysCFLmM/Yn/FDoqndh/9vNuQfXRDvTuXKLxfD/JtZQGKFT8MGcJBK644g==", + "dev": true, + "dependencies": { + "flat-cache": "^2.0.1" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/eslint/node_modules/flat-cache": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/flat-cache/-/flat-cache-2.0.1.tgz", + "integrity": "sha512-LoQe6yDuUMDzQAEH8sgmh4Md6oZnc/7PjtwjNFSzveXqSHt6ka9fPBuso7IGf9Rz4uqnSnWiFH2B/zj24a5ReA==", + "dev": true, + "dependencies": { + "flatted": "^2.0.0", + "rimraf": "2.6.3", + "write": "1.0.3" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/eslint/node_modules/globals": { + "version": "12.4.0", + "resolved": "https://registry.npmmirror.com/globals/-/globals-12.4.0.tgz", + "integrity": "sha512-BWICuzzDvDoH54NHKCseDanAhE3CeDorgDL5MT6LMXXj2WCnd9UC2szdk4AWLfjdgNBCXLUanXYcpBBKOSWGwg==", + "dev": true, + "dependencies": { + "type-fest": "^0.8.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/eslint/node_modules/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/eslint/node_modules/ignore": { + "version": "4.0.6", + "resolved": "https://registry.npmmirror.com/ignore/-/ignore-4.0.6.tgz", + "integrity": "sha512-cyFDKrqc/YdcWFniJhzI42+AzS+gNwmUzOSFcRCQYwySuBBBy/KjuxWLZ/FHEH6Moq1NizMOBWyTcv8O4OZIMg==", + "dev": true, + "engines": { + "node": ">= 4" + } + }, + "node_modules/eslint/node_modules/import-fresh": { + "version": "3.3.0", + "resolved": "https://registry.npmmirror.com/import-fresh/-/import-fresh-3.3.0.tgz", + "integrity": "sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==", + "dev": true, + "dependencies": { + "parent-module": "^1.0.0", + "resolve-from": "^4.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/eslint/node_modules/inquirer": { + "version": "7.3.3", + "resolved": "https://registry.npmmirror.com/inquirer/-/inquirer-7.3.3.tgz", + "integrity": "sha512-JG3eIAj5V9CwcGvuOmoo6LB9kbAYT8HXffUl6memuszlwDC/qvFAJw49XJ5NROSFNPxp3iQg1GqkFhaY/CR0IA==", + "dev": true, + "dependencies": { + "ansi-escapes": "^4.2.1", + "chalk": "^4.1.0", + "cli-cursor": "^3.1.0", + "cli-width": "^3.0.0", + "external-editor": "^3.0.3", + "figures": "^3.0.0", + "lodash": "^4.17.19", + "mute-stream": "0.0.8", + "run-async": "^2.4.0", + "rxjs": "^6.6.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0", + "through": "^2.3.6" + }, + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/eslint/node_modules/inquirer/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmmirror.com/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/eslint/node_modules/inquirer/node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmmirror.com/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/eslint/node_modules/is-fullwidth-code-point": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", + "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/eslint/node_modules/mimic-fn": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/mimic-fn/-/mimic-fn-2.1.0.tgz", + "integrity": "sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/eslint/node_modules/ms": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", + "dev": true + }, + "node_modules/eslint/node_modules/mute-stream": { + "version": "0.0.8", + "resolved": "https://registry.npmmirror.com/mute-stream/-/mute-stream-0.0.8.tgz", + "integrity": "sha512-nnbWWOkoWyUsTjKrhgD0dcz22mdkSnpYqbEjIm2nhwhuxlSkpywJmBo8h0ZqJdkp73mb90SssHkN4rsRaBAfAA==", + "dev": true + }, + "node_modules/eslint/node_modules/onetime": { + "version": "5.1.2", + "resolved": "https://registry.npmmirror.com/onetime/-/onetime-5.1.2.tgz", + "integrity": "sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==", + "dev": true, + "dependencies": { + "mimic-fn": "^2.1.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/eslint/node_modules/regexpp": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/regexpp/-/regexpp-2.0.1.tgz", + "integrity": "sha512-lv0M6+TkDVniA3aD1Eg0DVpfU/booSu7Eev3TDO/mZKHBfVjgCGTV4t4buppESEYDtkArYFOxTJWv6S5C+iaNw==", + "dev": true, + "engines": { + "node": ">=6.5.0" + } + }, + "node_modules/eslint/node_modules/resolve-from": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/resolve-from/-/resolve-from-4.0.0.tgz", + "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/eslint/node_modules/restore-cursor": { + "version": "3.1.0", + "resolved": "https://registry.npmmirror.com/restore-cursor/-/restore-cursor-3.1.0.tgz", + "integrity": "sha512-l+sSefzHpj5qimhFSE5a8nufZYAM3sBSVMAPtYkmC+4EH2anSGaEMXSD0izRQbu9nfyQ9y5JrVmp7E8oZrUjvA==", + "dev": true, + "dependencies": { + "onetime": "^5.1.0", + "signal-exit": "^3.0.2" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/eslint/node_modules/rimraf": { + "version": "2.6.3", + "resolved": "https://registry.npmmirror.com/rimraf/-/rimraf-2.6.3.tgz", + "integrity": "sha512-mwqeW5XsA2qAejG46gYdENaxXjx9onRNCfn7L0duuP4hCuTIi/QO7PDK07KJfp1d+izWPrzEJDcSqBa0OZQriA==", + "dev": true, + "dependencies": { + "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" + } + }, + "node_modules/eslint/node_modules/semver": { + "version": "6.3.0", + "resolved": "https://registry.npmmirror.com/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/eslint/node_modules/slice-ansi": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/slice-ansi/-/slice-ansi-2.1.0.tgz", + "integrity": "sha512-Qu+VC3EwYLldKa1fCxuuvULvSJOKEgk9pi8dZeCVK7TqBfUNTH4sFkk4joj8afVSfAYgJoSOetjx9QWOJ5mYoQ==", + "dev": true, + "dependencies": { + "ansi-styles": "^3.2.0", + "astral-regex": "^1.0.0", + "is-fullwidth-code-point": "^2.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/eslint/node_modules/slice-ansi/node_modules/ansi-styles": { + "version": "3.2.1", + "resolved": "https://registry.npmmirror.com/ansi-styles/-/ansi-styles-3.2.1.tgz", + "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", + "dev": true, + "dependencies": { + "color-convert": "^1.9.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/eslint/node_modules/slice-ansi/node_modules/color-convert": { + "version": "1.9.3", + "resolved": "https://registry.npmmirror.com/color-convert/-/color-convert-1.9.3.tgz", + "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", + "dev": true, + "dependencies": { + "color-name": "1.1.3" + } + }, + "node_modules/eslint/node_modules/slice-ansi/node_modules/color-name": { + "version": "1.1.3", + "resolved": "https://registry.npmmirror.com/color-name/-/color-name-1.1.3.tgz", + "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==", + "dev": true + }, + "node_modules/eslint/node_modules/slice-ansi/node_modules/is-fullwidth-code-point": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", + "integrity": "sha512-VHskAKYM8RfSFXwee5t5cbN5PZeq1Wrh6qd5bkyiXIf6UQcN6w/A0eXM9r6t8d+GYOh+o6ZhiEnb88LN/Y8m2w==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/eslint/node_modules/string-width": { + "version": "4.2.3", + "resolved": "https://registry.npmmirror.com/string-width/-/string-width-4.2.3.tgz", + "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", + "dev": true, + "dependencies": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/eslint/node_modules/string-width/node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmmirror.com/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/eslint/node_modules/strip-json-comments": { + "version": "3.1.1", + "resolved": "https://registry.npmmirror.com/strip-json-comments/-/strip-json-comments-3.1.1.tgz", + "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/eslint/node_modules/supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmmirror.com/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/eslint/node_modules/table": { + "version": "5.4.6", + "resolved": "https://registry.npmmirror.com/table/-/table-5.4.6.tgz", + "integrity": "sha512-wmEc8m4fjnob4gt5riFRtTu/6+4rSe12TpAELNSqHMfF3IqnA+CH37USM6/YR3qRZv7e56kAEAtd6nKZaxe0Ug==", + "dev": true, + "dependencies": { + "ajv": "^6.10.2", + "lodash": "^4.17.14", + "slice-ansi": "^2.1.0", + "string-width": "^3.0.0" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/eslint/node_modules/table/node_modules/emoji-regex": { + "version": "7.0.3", + "resolved": "https://registry.npmmirror.com/emoji-regex/-/emoji-regex-7.0.3.tgz", + "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", + "dev": true + }, + "node_modules/eslint/node_modules/table/node_modules/is-fullwidth-code-point": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", + "integrity": "sha512-VHskAKYM8RfSFXwee5t5cbN5PZeq1Wrh6qd5bkyiXIf6UQcN6w/A0eXM9r6t8d+GYOh+o6ZhiEnb88LN/Y8m2w==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/eslint/node_modules/table/node_modules/string-width": { + "version": "3.1.0", + "resolved": "https://registry.npmmirror.com/string-width/-/string-width-3.1.0.tgz", + "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", + "dev": true, + "dependencies": { + "emoji-regex": "^7.0.1", + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^5.1.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/eslint/node_modules/type-fest": { + "version": "0.8.1", + "resolved": "https://registry.npmmirror.com/type-fest/-/type-fest-0.8.1.tgz", + "integrity": "sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/eslint/node_modules/write": { + "version": "1.0.3", + "resolved": "https://registry.npmmirror.com/write/-/write-1.0.3.tgz", + "integrity": "sha512-/lg70HAjtkUgWPVZhZcm+T4hkL8Zbtp1nFNOn3lRrxnlv50SRBv7cR7RqR+GMsd3hUXy9hWBo4CHTbFTcOYwig==", + "dev": true, + "dependencies": { + "mkdirp": "^0.5.1" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/espree": { + "version": "3.5.4", + "resolved": "https://registry.npmmirror.com/espree/-/espree-3.5.4.tgz", + "integrity": "sha512-yAcIQxtmMiB/jL32dzEp2enBeidsB7xWPLNiw3IIkpVds1P+h7qF9YwJq1yUNzp2OKXgAprs4F61ih66UsoD1A==", + "dev": true, + "optional": true, + "dependencies": { + "acorn": "^5.5.0", + "acorn-jsx": "^3.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/espree/node_modules/acorn": { + "version": "5.7.4", + "resolved": "https://registry.npmmirror.com/acorn/-/acorn-5.7.4.tgz", + "integrity": "sha512-1D++VG7BhrtvQpNbBzovKNc1FLGGEE/oGe7b9xJm/RFHMBeUaUGpluV9RLjZa47YFdPcDAenEYuq9pQPcMdLJg==", + "dev": true, + "optional": true, + "bin": { + "acorn": "bin/acorn" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/esprima": { + "version": "4.0.1", + "resolved": "https://registry.npmmirror.com/esprima/-/esprima-4.0.1.tgz", + "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", + "bin": { + "esparse": "bin/esparse.js", + "esvalidate": "bin/esvalidate.js" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/esquery": { + "version": "1.4.0", + "resolved": "https://registry.npmmirror.com/esquery/-/esquery-1.4.0.tgz", + "integrity": "sha512-cCDispWt5vHHtwMY2YrAQ4ibFkAL8RbH5YGBnZBc90MolvvfkkQcJro/aZiAQUlQ3qgrYS6D6v8Gc5G5CQsc9w==", + "dev": true, + "dependencies": { + "estraverse": "^5.1.0" + }, + "engines": { + "node": ">=0.10" + } + }, + "node_modules/esquery/node_modules/estraverse": { + "version": "5.3.0", + "resolved": "https://registry.npmmirror.com/estraverse/-/estraverse-5.3.0.tgz", + "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", + "dev": true, + "engines": { + "node": ">=4.0" + } + }, + "node_modules/esrecurse": { + "version": "4.3.0", + "resolved": "https://registry.npmmirror.com/esrecurse/-/esrecurse-4.3.0.tgz", + "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==", + "dependencies": { + "estraverse": "^5.2.0" + }, + "engines": { + "node": ">=4.0" + } + }, + "node_modules/esrecurse/node_modules/estraverse": { + "version": "5.3.0", + "resolved": "https://registry.npmmirror.com/estraverse/-/estraverse-5.3.0.tgz", + "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", + "engines": { + "node": ">=4.0" + } + }, + "node_modules/estraverse": { + "version": "4.3.0", + "resolved": "https://registry.npmmirror.com/estraverse/-/estraverse-4.3.0.tgz", + "integrity": "sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==", + "engines": { + "node": ">=4.0" + } + }, + "node_modules/esutils": { + "version": "2.0.3", + "resolved": "https://registry.npmmirror.com/esutils/-/esutils-2.0.3.tgz", + "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/etag": { + "version": "1.8.1", + "resolved": "https://registry.npmmirror.com/etag/-/etag-1.8.1.tgz", + "integrity": "sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg==", + "dev": true, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/event-pubsub": { + "version": "4.3.0", + "resolved": "https://registry.npmmirror.com/event-pubsub/-/event-pubsub-4.3.0.tgz", + "integrity": "sha512-z7IyloorXvKbFx9Bpie2+vMJKKx1fH1EN5yiTfp8CiLOTptSYy1g8H4yDpGlEdshL1PBiFtBHepF2cNsqeEeFQ==", + "dev": true, + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/eventemitter3": { + "version": "4.0.7", + "resolved": "https://registry.npmmirror.com/eventemitter3/-/eventemitter3-4.0.7.tgz", + "integrity": "sha512-8guHBZCwKnFhYdHr2ysuRWErTwhoN2X8XELRlrRwpmfeY2jjuUN4taQMsULKUVo1K4DvZl+0pgfyoysHxvmvEw==" + }, + "node_modules/events": { + "version": "3.3.0", + "resolved": "https://registry.npmmirror.com/events/-/events-3.3.0.tgz", + "integrity": "sha512-mQw+2fkQbALzQ7V0MY0IqdnXNOeTtP4r0lN9z7AAawCXgqea7bDii20AYrIBrFd/Hx0M2Ocz6S111CaFkUcb0Q==", + "engines": { + "node": ">=0.8.x" + } + }, + "node_modules/eventsource": { + "version": "1.1.0", + "resolved": "https://registry.npmmirror.com/eventsource/-/eventsource-1.1.0.tgz", + "integrity": "sha512-VSJjT5oCNrFvCS6igjzPAt5hBzQ2qPBFIbJ03zLI9SE0mxwZpMw6BfJrbFHm1a141AavMEB8JHmBhWAd66PfCg==", + "dev": true, + "dependencies": { + "original": "^1.0.0" + }, + "engines": { + "node": ">=0.12.0" + } + }, + "node_modules/evp_bytestokey": { + "version": "1.0.3", + "resolved": "https://registry.npmmirror.com/evp_bytestokey/-/evp_bytestokey-1.0.3.tgz", + "integrity": "sha512-/f2Go4TognH/KvCISP7OUsHn85hT9nUkxxA9BEWxFn+Oj9o8ZNLm/40hdlgSLyuOimsrTKLUMEorQexp/aPQeA==", + "dependencies": { + "md5.js": "^1.3.4", + "safe-buffer": "^5.1.1" + } + }, + "node_modules/execa": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/execa/-/execa-1.0.0.tgz", + "integrity": "sha512-adbxcyWV46qiHyvSp50TKt05tB4tK3HcmF7/nxfAdhnox83seTDbwnaqKO4sXRy7roHAIFqJP/Rw/AuEbX61LA==", + "dev": true, + "dependencies": { + "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" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/expand-brackets": { + "version": "2.1.4", + "resolved": "https://registry.npmmirror.com/expand-brackets/-/expand-brackets-2.1.4.tgz", + "integrity": "sha512-w/ozOKR9Obk3qoWeY/WDi6MFta9AoMR+zud60mdnbniMcBxRuFJyDt2LdX/14A1UABeqk+Uk+LDfUpvoGKppZA==", + "dependencies": { + "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" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/expand-brackets/node_modules/define-property": { + "version": "0.2.5", + "resolved": "https://registry.npmmirror.com/define-property/-/define-property-0.2.5.tgz", + "integrity": "sha512-Rr7ADjQZenceVOAKop6ALkkRAmH1A4Gx9hV/7ZujPUN2rkATqFO0JZLZInbAjpZYoJ1gUx8MRMQVkYemcbMSTA==", + "dependencies": { + "is-descriptor": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/expand-brackets/node_modules/extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha512-zCnTtlxNoAiDc3gqY2aYAWFx7XWWiasuF2K8Me5WbN8otHKTUKBwjPtNpRs/rbUZm7KxWAaNj7P1a/p52GbVug==", + "dependencies": { + "is-extendable": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/expand-range": { + "version": "1.8.2", + "resolved": "https://registry.npmmirror.com/expand-range/-/expand-range-1.8.2.tgz", + "integrity": "sha512-AFASGfIlnIbkKPQwX1yHaDjFvh/1gyKJODme52V6IORh69uEYgZp0o9C+qsIGNVEiuuhQU0CSSl++Rlegg1qvA==", + "dev": true, + "dependencies": { + "fill-range": "^2.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/expand-range/node_modules/fill-range": { + "version": "2.2.4", + "resolved": "https://registry.npmmirror.com/fill-range/-/fill-range-2.2.4.tgz", + "integrity": "sha512-cnrcCbj01+j2gTG921VZPnHbjmdAf8oQV/iGeV2kZxGSyfYjjTyY79ErsK1WJWMpw6DaApEX72binqJE+/d+5Q==", + "dev": true, + "dependencies": { + "is-number": "^2.1.0", + "isobject": "^2.0.0", + "randomatic": "^3.0.0", + "repeat-element": "^1.1.2", + "repeat-string": "^1.5.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/expand-range/node_modules/is-number": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/is-number/-/is-number-2.1.0.tgz", + "integrity": "sha512-QUzH43Gfb9+5yckcrSA0VBDwEtDUchrk4F6tfJZQuNzDJbEDB9cZNzSfXGQ1jqmdDY/kl41lUOWM9syA8z8jlg==", + "dev": true, + "dependencies": { + "kind-of": "^3.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/expand-range/node_modules/isobject": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/isobject/-/isobject-2.1.0.tgz", + "integrity": "sha512-+OUdGJlgjOBZDfxnDjYYG6zp487z0JGNQq3cYQYg5f5hKR+syHMsaztzGeml/4kGG55CSpKSpWTY+jYGgsHLgA==", + "dev": true, + "dependencies": { + "isarray": "1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/expand-range/node_modules/kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmmirror.com/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha512-NOW9QQXMoZGg/oqnVNoNTTIFEIid1627WCffUBJEdMxYApq7mNE7CpzucIPc+ZQg25Phej7IJSmX3hO+oblOtQ==", + "dev": true, + "dependencies": { + "is-buffer": "^1.1.5" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/express": { + "version": "4.18.0", + "resolved": "https://registry.npmmirror.com/express/-/express-4.18.0.tgz", + "integrity": "sha512-EJEXxiTQJS3lIPrU1AE2vRuT7X7E+0KBbpm5GSoK524yl0K8X+er8zS2P14E64eqsVNoWbMCT7MpmQ+ErAhgRg==", + "dev": true, + "dependencies": { + "accepts": "~1.3.8", + "array-flatten": "1.1.1", + "body-parser": "1.20.0", + "content-disposition": "0.5.4", + "content-type": "~1.0.4", + "cookie": "0.5.0", + "cookie-signature": "1.0.6", + "debug": "2.6.9", + "depd": "2.0.0", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "etag": "~1.8.1", + "finalhandler": "1.2.0", + "fresh": "0.5.2", + "http-errors": "2.0.0", + "merge-descriptors": "1.0.1", + "methods": "~1.1.2", + "on-finished": "2.4.1", + "parseurl": "~1.3.3", + "path-to-regexp": "0.1.7", + "proxy-addr": "~2.0.7", + "qs": "6.10.3", + "range-parser": "~1.2.1", + "safe-buffer": "5.2.1", + "send": "0.18.0", + "serve-static": "1.15.0", + "setprototypeof": "1.2.0", + "statuses": "2.0.1", + "type-is": "~1.6.18", + "utils-merge": "1.0.1", + "vary": "~1.1.2" + }, + "engines": { + "node": ">= 0.10.0" + } + }, + "node_modules/express/node_modules/qs": { + "version": "6.10.3", + "resolved": "https://registry.npmmirror.com/qs/-/qs-6.10.3.tgz", + "integrity": "sha512-wr7M2E0OFRfIfJZjKGieI8lBKb7fRCH4Fv5KNPEs7gJ8jadvotdsS08PzOKR7opXhZ/Xkjtt3WF9g38drmyRqQ==", + "dev": true, + "dependencies": { + "side-channel": "^1.0.4" + }, + "engines": { + "node": ">=0.6" + } + }, + "node_modules/express/node_modules/safe-buffer": { + "version": "5.2.1", + "resolved": "https://registry.npmmirror.com/safe-buffer/-/safe-buffer-5.2.1.tgz", + "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", + "dev": true + }, + "node_modules/extend": { + "version": "3.0.2", + "resolved": "https://registry.npmmirror.com/extend/-/extend-3.0.2.tgz", + "integrity": "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g==", + "dev": true + }, + "node_modules/extend-shallow": { + "version": "3.0.2", + "resolved": "https://registry.npmmirror.com/extend-shallow/-/extend-shallow-3.0.2.tgz", + "integrity": "sha512-BwY5b5Ql4+qZoefgMj2NUmx+tehVTH/Kf4k1ZEtOHNFcm2wSxMRo992l6X3TIgni2eZVTZ85xMOjF31fwZAj6Q==", + "dependencies": { + "assign-symbols": "^1.0.0", + "is-extendable": "^1.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/extend-shallow/node_modules/is-extendable": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/is-extendable/-/is-extendable-1.0.1.tgz", + "integrity": "sha512-arnXMxT1hhoKo9k1LZdmlNyJdDDfy2v0fXjFlmok4+i8ul/6WlbVge9bhM74OpNPQPMGUToDtz+KXa1PneJxOA==", + "dependencies": { + "is-plain-object": "^2.0.4" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/external-editor": { + "version": "2.2.0", + "resolved": "https://registry.npmmirror.com/external-editor/-/external-editor-2.2.0.tgz", + "integrity": "sha512-bSn6gvGxKt+b7+6TKEv1ZycHleA7aHhRHyAqJyp5pbUFuYYNIzpZnQDk7AsYckyWdEnTeAnay0aCy2aV6iTk9A==", + "dev": true, + "optional": true, + "dependencies": { + "chardet": "^0.4.0", + "iconv-lite": "^0.4.17", + "tmp": "^0.0.33" + }, + "engines": { + "node": ">=0.12" + } + }, + "node_modules/extglob": { + "version": "2.0.4", + "resolved": "https://registry.npmmirror.com/extglob/-/extglob-2.0.4.tgz", + "integrity": "sha512-Nmb6QXkELsuBr24CJSkilo6UHHgbekK5UiZgfE6UHD3Eb27YC6oD+bhcT+tJ6cl8dmsgdQxnWlcry8ksBIBLpw==", + "dependencies": { + "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" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/extglob/node_modules/define-property": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/define-property/-/define-property-1.0.0.tgz", + "integrity": "sha512-cZTYKFWspt9jZsMscWo8sc/5lbPC9Q0N5nBLgb+Yd915iL3udB1uFgS3B8YCx66UVHq018DAVFoee7x+gxggeA==", + "dependencies": { + "is-descriptor": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/extglob/node_modules/extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha512-zCnTtlxNoAiDc3gqY2aYAWFx7XWWiasuF2K8Me5WbN8otHKTUKBwjPtNpRs/rbUZm7KxWAaNj7P1a/p52GbVug==", + "dependencies": { + "is-extendable": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/extglob/node_modules/is-accessor-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", + "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", + "dependencies": { + "kind-of": "^6.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/extglob/node_modules/is-data-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", + "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", + "dependencies": { + "kind-of": "^6.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/extglob/node_modules/is-descriptor": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/is-descriptor/-/is-descriptor-1.0.2.tgz", + "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", + "dependencies": { + "is-accessor-descriptor": "^1.0.0", + "is-data-descriptor": "^1.0.0", + "kind-of": "^6.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/extsprintf": { + "version": "1.3.0", + "resolved": "https://registry.npmmirror.com/extsprintf/-/extsprintf-1.3.0.tgz", + "integrity": "sha512-11Ndz7Nv+mvAC1j0ktTa7fAb0vLyGGX+rMHNBYQviQDGU0Hw7lhctJANqbPhu9nV9/izT/IntTgZ7Im/9LJs9g==", + "dev": true, + "engines": [ + "node >=0.6.0" + ] + }, + "node_modules/fast-deep-equal": { + "version": "3.1.3", + "resolved": "https://registry.npmmirror.com/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", + "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==" + }, + "node_modules/fast-glob": { + "version": "2.2.7", + "resolved": "https://registry.npmmirror.com/fast-glob/-/fast-glob-2.2.7.tgz", + "integrity": "sha512-g1KuQwHOZAmOZMuBtHdxDtju+T2RT8jgCC9aANsbpdiDDTSnjgfuVsIBNKbUeJI3oKMRExcfNDtJl4OhbffMsw==", + "dev": true, + "dependencies": { + "@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" + }, + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/fast-glob/node_modules/glob-parent": { + "version": "3.1.0", + "resolved": "https://registry.npmmirror.com/glob-parent/-/glob-parent-3.1.0.tgz", + "integrity": "sha512-E8Ak/2+dZY6fnzlR7+ueWvhsH1SjHr4jjss4YS/h4py44jY9MhK/VFdaZJAWDz6BbL21KeteKxFSFpq8OS5gVA==", + "dev": true, + "dependencies": { + "is-glob": "^3.1.0", + "path-dirname": "^1.0.0" + } + }, + "node_modules/fast-glob/node_modules/glob-parent/node_modules/is-glob": { + "version": "3.1.0", + "resolved": "https://registry.npmmirror.com/is-glob/-/is-glob-3.1.0.tgz", + "integrity": "sha512-UFpDDrPgM6qpnFNI+rh/p3bUaq9hKLZN8bMUWzxmcnZVS3omf4IPK+BrewlnWjO1WmUsMYuSjKh4UJuV4+Lqmw==", + "dev": true, + "dependencies": { + "is-extglob": "^2.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/fast-json-stable-stringify": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", + "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==" + }, + "node_modules/fast-levenshtein": { + "version": "2.0.6", + "resolved": "https://registry.npmmirror.com/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", + "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==", + "dev": true + }, + "node_modules/fastparse": { + "version": "1.1.2", + "resolved": "https://registry.npmmirror.com/fastparse/-/fastparse-1.1.2.tgz", + "integrity": "sha512-483XLLxTVIwWK3QTrMGRqUfUpoOs/0hbQrl2oz4J0pAcm3A3bu84wxTFqGqkJzewCLdME38xJLJAxBABfQT8sQ==", + "dev": true + }, + "node_modules/fastq": { + "version": "1.13.0", + "resolved": "https://registry.npmmirror.com/fastq/-/fastq-1.13.0.tgz", + "integrity": "sha512-YpkpUnK8od0o1hmeSc7UUs/eB/vIPWJYjKck2QKIzAf71Vm1AAQ3EbuZB3g2JIy+pg+ERD0vqI79KyZiB2e2Nw==", + "dev": true, + "dependencies": { + "reusify": "^1.0.4" + } + }, + "node_modules/faye-websocket": { + "version": "0.11.4", + "resolved": "https://registry.npmmirror.com/faye-websocket/-/faye-websocket-0.11.4.tgz", + "integrity": "sha512-CzbClwlXAuiRQAlUyfqPgvPoNKTckTPGfwZV4ZdAhVcP2lh9KUxJg2b5GkE7XbjKQ3YJnQ9z6D9ntLAlB+tP8g==", + "dev": true, + "dependencies": { + "websocket-driver": ">=0.5.1" + }, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/figgy-pudding": { + "version": "3.5.2", + "resolved": "https://registry.npmmirror.com/figgy-pudding/-/figgy-pudding-3.5.2.tgz", + "integrity": "sha512-0btnI/H8f2pavGMN8w40mlSKOfTK2SVJmBfBeVIj3kNw0swwgzyRq0d5TJVOwodFmtvpPeWPN/MCcfuWF0Ezbw==" + }, + "node_modules/figures": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/figures/-/figures-2.0.0.tgz", + "integrity": "sha512-Oa2M9atig69ZkfwiApY8F2Yy+tzMbazyvqv21R0NsSC8floSOC09BbT1ITWAdoMGQvJ/aZnR1KMwdx9tvHnTNA==", + "dev": true, + "dependencies": { + "escape-string-regexp": "^1.0.5" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/file-entry-cache": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/file-entry-cache/-/file-entry-cache-2.0.0.tgz", + "integrity": "sha512-uXP/zGzxxFvFfcZGgBIwotm+Tdc55ddPAzF7iHshP4YGaXMww7rSF9peD9D1sui5ebONg5UobsZv+FfgEpGv/w==", + "dev": true, + "optional": true, + "dependencies": { + "flat-cache": "^1.2.1", + "object-assign": "^4.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/file-loader": { + "version": "3.0.1", + "resolved": "https://registry.npmmirror.com/file-loader/-/file-loader-3.0.1.tgz", + "integrity": "sha512-4sNIOXgtH/9WZq4NvlfU3Opn5ynUsqBwSLyM+I7UOwdGigTBYfVVQEwe/msZNX/j4pCJTIM14Fsw66Svo1oVrw==", + "dev": true, + "dependencies": { + "loader-utils": "^1.0.2", + "schema-utils": "^1.0.0" + }, + "engines": { + "node": ">= 6.9.0" + }, + "peerDependencies": { + "webpack": "^4.0.0" + } + }, + "node_modules/file-uri-to-path": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/file-uri-to-path/-/file-uri-to-path-1.0.0.tgz", + "integrity": "sha512-0Zt+s3L7Vf1biwWZ29aARiVYLx7iMGnEUl9x33fbB/j3jR81u/O2LbqK+Bm1CDSNDKVtJ/YjwY7TUd5SkeLQLw==", + "optional": true + }, + "node_modules/filemanager-webpack-plugin": { + "version": "2.0.5", + "resolved": "https://registry.npmmirror.com/filemanager-webpack-plugin/-/filemanager-webpack-plugin-2.0.5.tgz", + "integrity": "sha512-Yj5XIdKI2AN2r66uZc4MZ/n18SMqe2KKlkAqHHMW1OwveDs2Vc5129CpbFcI73rq/rjqso+2HsxieS7u5sx6XA==", + "dev": true, + "dependencies": { + "archiver": "^3.0.0", + "cpx": "^1.5.0", + "fs-extra": "^7.0.0", + "make-dir": "^1.1.0", + "mv": "^2.1.1", + "rimraf": "^2.6.2" + } + }, + "node_modules/filemanager-webpack-plugin/node_modules/make-dir": { + "version": "1.3.0", + "resolved": "https://registry.npmmirror.com/make-dir/-/make-dir-1.3.0.tgz", + "integrity": "sha512-2w31R7SJtieJJnQtGc7RVL2StM2vGYVfqUOvUDxH6bC6aJTxPxTF0GnIgCyu7tjockiUWAYQRbxa7vKn34s5sQ==", + "dev": true, + "dependencies": { + "pify": "^3.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/filemanager-webpack-plugin/node_modules/pify": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/pify/-/pify-3.0.0.tgz", + "integrity": "sha512-C3FsVNH1udSEX48gGX1xfvwTWfsYWj5U+8/uK15BGzIGrKoUpghX8hWZwa/OFnakBiiVNmBvemTJR5mcy7iPcg==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/filename-regex": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/filename-regex/-/filename-regex-2.0.1.tgz", + "integrity": "sha512-BTCqyBaWBTsauvnHiE8i562+EdJj+oUpkqWp2R1iCoR8f6oo8STRu3of7WJJ0TqWtxN50a5YFpzYK4Jj9esYfQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/filesize": { + "version": "3.6.1", + "resolved": "https://registry.npmmirror.com/filesize/-/filesize-3.6.1.tgz", + "integrity": "sha512-7KjR1vv6qnicaPMi1iiTcI85CyYwRO/PSFCu6SvqL8jN2Wjt/NIYQTFtFs7fSDCYOstUkEWIQGFUg5YZQfjlcg==", + "dev": true, + "engines": { + "node": ">= 0.4.0" + } + }, + "node_modules/fill-range": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/fill-range/-/fill-range-4.0.0.tgz", + "integrity": "sha512-VcpLTWqWDiTerugjj8e3+esbg+skS3M9e54UuR3iCeIDMXCLTsAH8hTSzDQU/X6/6t3eYkOKoZSef2PlU6U1XQ==", + "dependencies": { + "extend-shallow": "^2.0.1", + "is-number": "^3.0.0", + "repeat-string": "^1.6.1", + "to-regex-range": "^2.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/fill-range/node_modules/extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha512-zCnTtlxNoAiDc3gqY2aYAWFx7XWWiasuF2K8Me5WbN8otHKTUKBwjPtNpRs/rbUZm7KxWAaNj7P1a/p52GbVug==", + "dependencies": { + "is-extendable": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/filter-obj": { + "version": "1.1.0", + "resolved": "https://registry.npmmirror.com/filter-obj/-/filter-obj-1.1.0.tgz", + "integrity": "sha512-8rXg1ZnX7xzy2NGDVkBVaAy+lSlPNwad13BtgSlLuxfIslyt5Vg64U7tFcCt4WS1R0hvtnQybT/IyCkGZ3DpXQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/finalhandler": { + "version": "1.2.0", + "resolved": "https://registry.npmmirror.com/finalhandler/-/finalhandler-1.2.0.tgz", + "integrity": "sha512-5uXcUVftlQMFnWC9qu/svkWv3GTd2PfUhK/3PLkYNAe7FbqJMt3515HaxE6eRL74GdsriiwujiawdaB1BpEISg==", + "dev": true, + "dependencies": { + "debug": "2.6.9", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "on-finished": "2.4.1", + "parseurl": "~1.3.3", + "statuses": "2.0.1", + "unpipe": "~1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/find-babel-config": { + "version": "1.2.0", + "resolved": "https://registry.npmmirror.com/find-babel-config/-/find-babel-config-1.2.0.tgz", + "integrity": "sha512-jB2CHJeqy6a820ssiqwrKMeyC6nNdmrcgkKWJWmpoxpE8RKciYJXCcXRq1h2AzCo5I5BJeN2tkGEO3hLTuePRA==", + "dev": true, + "dependencies": { + "json5": "^0.5.1", + "path-exists": "^3.0.0" + }, + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/find-babel-config/node_modules/json5": { + "version": "0.5.1", + "resolved": "https://registry.npmmirror.com/json5/-/json5-0.5.1.tgz", + "integrity": "sha512-4xrs1aW+6N5DalkqSVA8fxh458CXvR99WU8WLKmq4v8eWAL86Xo3BVqyd3SkA9wEVjCMqyvvRRkshAdOnBp5rw==", + "dev": true, + "bin": { + "json5": "lib/cli.js" + } + }, + "node_modules/find-cache-dir": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/find-cache-dir/-/find-cache-dir-2.1.0.tgz", + "integrity": "sha512-Tq6PixE0w/VMFfCgbONnkiQIVol/JJL7nRMi20fqzA4NRs9AfeqMGeRdPi3wIhYkxjeBaWh2rxwapn5Tu3IqOQ==", + "dependencies": { + "commondir": "^1.0.1", + "make-dir": "^2.0.0", + "pkg-dir": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/find-index": { + "version": "0.1.1", + "resolved": "https://registry.npmmirror.com/find-index/-/find-index-0.1.1.tgz", + "integrity": "sha512-uJ5vWrfBKMcE6y2Z8834dwEZj9mNGxYa3t3I53OwFeuZ8D9oc2E5zcsrkuhX6h4iYrjhiv0T3szQmxlAV9uxDg==", + "dev": true + }, + "node_modules/find-up": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/find-up/-/find-up-3.0.0.tgz", + "integrity": "sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg==", + "dependencies": { + "locate-path": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/find-yarn-workspace-root": { + "version": "1.2.1", + "resolved": "https://registry.npmmirror.com/find-yarn-workspace-root/-/find-yarn-workspace-root-1.2.1.tgz", + "integrity": "sha512-dVtfb0WuQG+8Ag2uWkbG79hOUzEsRrhBzgfn86g2sJPkzmcpGdghbNTfUKGTxymFrY/tLIodDzLoW9nOJ4FY8Q==", + "dev": true, + "dependencies": { + "fs-extra": "^4.0.3", + "micromatch": "^3.1.4" + } + }, + "node_modules/find-yarn-workspace-root/node_modules/fs-extra": { + "version": "4.0.3", + "resolved": "https://registry.npmmirror.com/fs-extra/-/fs-extra-4.0.3.tgz", + "integrity": "sha512-q6rbdDd1o2mAnQreO7YADIxf/Whx4AHBiRf6d+/cVT8h44ss+lHgxf1FemcqDnQt9X3ct4McHr+JMGlYSsK7Cg==", + "dev": true, + "dependencies": { + "graceful-fs": "^4.1.2", + "jsonfile": "^4.0.0", + "universalify": "^0.1.0" + } + }, + "node_modules/flat-cache": { + "version": "1.3.4", + "resolved": "https://registry.npmmirror.com/flat-cache/-/flat-cache-1.3.4.tgz", + "integrity": "sha512-VwyB3Lkgacfik2vhqR4uv2rvebqmDvFu4jlN/C1RzWoJEo8I7z4Q404oiqYCkq41mni8EzQnm95emU9seckwtg==", + "dev": true, + "optional": true, + "dependencies": { + "circular-json": "^0.3.1", + "graceful-fs": "^4.1.2", + "rimraf": "~2.6.2", + "write": "^0.2.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/flat-cache/node_modules/rimraf": { + "version": "2.6.3", + "resolved": "https://registry.npmmirror.com/rimraf/-/rimraf-2.6.3.tgz", + "integrity": "sha512-mwqeW5XsA2qAejG46gYdENaxXjx9onRNCfn7L0duuP4hCuTIi/QO7PDK07KJfp1d+izWPrzEJDcSqBa0OZQriA==", + "dev": true, + "optional": true, + "dependencies": { + "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" + } + }, + "node_modules/flatted": { + "version": "2.0.2", + "resolved": "https://registry.npmmirror.com/flatted/-/flatted-2.0.2.tgz", + "integrity": "sha512-r5wGx7YeOwNWNlCA0wQ86zKyDLMQr+/RB8xy74M4hTphfmjlijTSSXGuH8rnvKZnfT9i+75zmd8jcKdMR4O6jA==", + "dev": true + }, + "node_modules/flush-write-stream": { + "version": "1.1.1", + "resolved": "https://registry.npmmirror.com/flush-write-stream/-/flush-write-stream-1.1.1.tgz", + "integrity": "sha512-3Z4XhFZ3992uIq0XOqb9AreonueSYphE6oYbpt5+3u06JWklbsPkNv3ZKkP9Bz/r+1MWCaMoSQ28P85+1Yc77w==", + "dependencies": { + "inherits": "^2.0.3", + "readable-stream": "^2.3.6" + } + }, + "node_modules/fn-name": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/fn-name/-/fn-name-2.0.1.tgz", + "integrity": "sha512-oIDB1rXf3BUnn00bh2jVM0byuqr94rBh6g7ZfdKcbmp1we2GQtPzKdloyvBXHs+q3fvxB8EqX5ecFba3RwCSjA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/follow-redirects": { + "version": "1.14.9", + "resolved": "https://registry.npmmirror.com/follow-redirects/-/follow-redirects-1.14.9.tgz", + "integrity": "sha512-MQDfihBQYMcyy5dhRDJUHcw7lb2Pv/TuE6xP1vyraLukNDHKbDxDNaOE3NbCAdKQApno+GPRyo1YAp89yCjK4w==", + "engines": { + "node": ">=4.0" + }, + "peerDependenciesMeta": { + "debug": { + "optional": true + } + } + }, + "node_modules/for-in": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/for-in/-/for-in-1.0.2.tgz", + "integrity": "sha512-7EwmXrOjyL+ChxMhmG5lnW9MPt1aIeZEwKhQzoBUdTV0N3zuwWDZYVJatDvZ2OyzPUvdIAZDsCetk3coyMfcnQ==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/for-own": { + "version": "0.1.5", + "resolved": "https://registry.npmmirror.com/for-own/-/for-own-0.1.5.tgz", + "integrity": "sha512-SKmowqGTJoPzLO1T0BBJpkfp3EMacCMOuH40hOUbrbzElVktk4DioXVM99QkLCyKoiuOmyjgcWMpVz2xjE7LZw==", + "dev": true, + "dependencies": { + "for-in": "^1.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/forever-agent": { + "version": "0.6.1", + "resolved": "https://registry.npmmirror.com/forever-agent/-/forever-agent-0.6.1.tgz", + "integrity": "sha512-j0KLYPhm6zeac4lz3oJ3o65qvgQCcPubiyotZrXqEaG4hNagNYO8qdlUrX5vwqv9ohqeT/Z3j6+yW067yWWdUw==", + "dev": true, + "engines": { + "node": "*" + } + }, + "node_modules/form-data": { + "version": "2.3.3", + "resolved": "https://registry.npmmirror.com/form-data/-/form-data-2.3.3.tgz", + "integrity": "sha512-1lLKB2Mu3aGP1Q/2eCOx0fNbRMe7XdwktwOruhfqqd0rIJWwN4Dh+E3hrPSlDCXnSR7UtZ1N38rVXm+6+MEhJQ==", + "dev": true, + "dependencies": { + "asynckit": "^0.4.0", + "combined-stream": "^1.0.6", + "mime-types": "^2.1.12" + }, + "engines": { + "node": ">= 0.12" + } + }, + "node_modules/forwarded": { + "version": "0.2.0", + "resolved": "https://registry.npmmirror.com/forwarded/-/forwarded-0.2.0.tgz", + "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==", + "dev": true, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/fragment-cache": { + "version": "0.2.1", + "resolved": "https://registry.npmmirror.com/fragment-cache/-/fragment-cache-0.2.1.tgz", + "integrity": "sha512-GMBAbW9antB8iZRHLoGw0b3HANt57diZYFO/HL1JGIC1MjKrdmhxvrJbupnVvpys0zsz7yBApXdQyfepKly2kA==", + "dependencies": { + "map-cache": "^0.2.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/fresh": { + "version": "0.5.2", + "resolved": "https://registry.npmmirror.com/fresh/-/fresh-0.5.2.tgz", + "integrity": "sha512-zJ2mQYM18rEFOudeV4GShTGIQ7RbzA7ozbU9I/XBpm7kqgMywgmylMwXHxZJmkVoYkna9d2pVXVXPdYTP9ej8Q==", + "dev": true, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/from2": { + "version": "2.3.0", + "resolved": "https://registry.npmmirror.com/from2/-/from2-2.3.0.tgz", + "integrity": "sha512-OMcX/4IC/uqEPVgGeyfN22LJk6AZrMkRZHxcHBMBvHScDGgwTm2GT2Wkgtocyd3JfZffjj2kYUDXXII0Fk9W0g==", + "dependencies": { + "inherits": "^2.0.1", + "readable-stream": "^2.0.0" + } + }, + "node_modules/fs-constants": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/fs-constants/-/fs-constants-1.0.0.tgz", + "integrity": "sha512-y6OAwoSIf7FyjMIv94u+b5rdheZEjzR63GTyZJm5qh4Bi+2YgwLCcI/fPFZkL5PSixOt6ZNKm+w+Hfp/Bciwow==", + "dev": true + }, + "node_modules/fs-extra": { + "version": "7.0.1", + "resolved": "https://registry.npmmirror.com/fs-extra/-/fs-extra-7.0.1.tgz", + "integrity": "sha512-YJDaCJZEnBmcbw13fvdAM9AwNOJwOzrE4pqMqBq5nFiEqXUqHwlK4B+3pUw6JNvfSPtX05xFHtYy/1ni01eGCw==", + "dev": true, + "dependencies": { + "graceful-fs": "^4.1.2", + "jsonfile": "^4.0.0", + "universalify": "^0.1.0" + }, + "engines": { + "node": ">=6 <7 || >=8" + } + }, + "node_modules/fs-minipass": { + "version": "1.2.7", + "resolved": "https://registry.npmmirror.com/fs-minipass/-/fs-minipass-1.2.7.tgz", + "integrity": "sha512-GWSSJGFy4e9GUeCcbIkED+bgAoFyj7XF1mV8rma3QW4NIqX9Kyx79N/PF61H5udOV3aY1IaMLs6pGbH71nlCTA==", + "dev": true, + "dependencies": { + "minipass": "^2.6.0" + } + }, + "node_modules/fs-write-stream-atomic": { + "version": "1.0.10", + "resolved": "https://registry.npmmirror.com/fs-write-stream-atomic/-/fs-write-stream-atomic-1.0.10.tgz", + "integrity": "sha512-gehEzmPn2nAwr39eay+x3X34Ra+M2QlVUTLhkXPjWdeO8RF9kszk116avgBJM3ZyNHgHXBNx+VmPaFC36k0PzA==", + "dependencies": { + "graceful-fs": "^4.1.2", + "iferr": "^0.1.5", + "imurmurhash": "^0.1.4", + "readable-stream": "1 || 2" + } + }, + "node_modules/fs.realpath": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/fs.realpath/-/fs.realpath-1.0.0.tgz", + "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==" + }, + "node_modules/fsevents": { + "version": "2.3.2", + "resolved": "https://registry.npmmirror.com/fsevents/-/fsevents-2.3.2.tgz", + "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", + "hasInstallScript": true, + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": "^8.16.0 || ^10.6.0 || >=11.0.0" + } + }, + "node_modules/function-bind": { + "version": "1.1.1", + "resolved": "https://registry.npmmirror.com/function-bind/-/function-bind-1.1.1.tgz", + "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==" + }, + "node_modules/functional-red-black-tree": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/functional-red-black-tree/-/functional-red-black-tree-1.0.1.tgz", + "integrity": "sha512-dsKNQNdj6xA3T+QlADDA7mOSlX0qiMINjn0cgr+eGHGsbSHzTabcIogz2+p/iqP1Xs6EP/sS2SbqH+brGTbq0g==", + "dev": true + }, + "node_modules/functions-have-names": { + "version": "1.2.3", + "resolved": "https://registry.npmmirror.com/functions-have-names/-/functions-have-names-1.2.3.tgz", + "integrity": "sha512-xckBUXyTIqT97tq2x2AMb+g163b5JFysYk0x4qxNFwbfQkmNZoiRHb6sPzI9/QV33WeuvVYBUIiD4NzNIyqaRQ==", + "dev": true + }, + "node_modules/fuzzysearch": { + "version": "1.0.3", + "resolved": "https://registry.npmmirror.com/fuzzysearch/-/fuzzysearch-1.0.3.tgz", + "integrity": "sha512-s+kNWQuI3mo9OALw0HJ6YGmMbLqEufCh2nX/zzV5CrICQ/y4AwPxM+6TIiF9ItFCHXFCyM/BfCCmN57NTIJuPg==" + }, + "node_modules/g-status": { + "version": "2.0.2", + "resolved": "https://registry.npmmirror.com/g-status/-/g-status-2.0.2.tgz", + "integrity": "sha512-kQoE9qH+T1AHKgSSD0Hkv98bobE90ILQcXAF4wvGgsr7uFqNvwmh8j+Lq3l0RVt3E3HjSbv2B9biEGcEtpHLCA==", + "dev": true, + "dependencies": { + "arrify": "^1.0.1", + "matcher": "^1.0.0", + "simple-git": "^1.85.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/g-status/node_modules/arrify": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/arrify/-/arrify-1.0.1.tgz", + "integrity": "sha512-3CYzex9M9FGQjCGMGyi6/31c8GJbgb0qGyrx5HWxPd0aCwh4cB2YjMb2Xf9UuoogrMrlO9cTqnB5rI5GHZTcUA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/gauge": { + "version": "2.7.4", + "resolved": "https://registry.npmmirror.com/gauge/-/gauge-2.7.4.tgz", + "integrity": "sha512-14x4kjc6lkD3ltw589k0NrPD6cCNTD6CWoVUNpB85+DrtONoZn+Rug6xZU5RvSC4+TZPxA5AnBibQYAvZn41Hg==", + "dev": true, + "dependencies": { + "aproba": "^1.0.3", + "console-control-strings": "^1.0.0", + "has-unicode": "^2.0.0", + "object-assign": "^4.1.0", + "signal-exit": "^3.0.0", + "string-width": "^1.0.1", + "strip-ansi": "^3.0.1", + "wide-align": "^1.1.0" + } + }, + "node_modules/gauge/node_modules/ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmmirror.com/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha512-TIGnTpdo+E3+pCyAluZvtED5p5wCqLdezCyhPZzKPcxvFplEt4i+W7OONCKgeZFT3+y5NZZfOOS/Bdcanm1MYA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/gauge/node_modules/is-fullwidth-code-point": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz", + "integrity": "sha512-1pqUqRjkhPJ9miNq9SwMfdvi6lBJcd6eFxvfaivQhaH3SgisfiuudvFntdKOmxuee/77l+FPjKrQjWvmPjWrRw==", + "dev": true, + "dependencies": { + "number-is-nan": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/gauge/node_modules/string-width": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/string-width/-/string-width-1.0.2.tgz", + "integrity": "sha512-0XsVpQLnVCXHJfyEs8tC0zpTVIr5PKKsQtkT29IwupnPTjtPmQ3xT/4yCREF9hYkV/3M3kzcUTSAZT6a6h81tw==", + "dev": true, + "dependencies": { + "code-point-at": "^1.0.0", + "is-fullwidth-code-point": "^1.0.0", + "strip-ansi": "^3.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/gauge/node_modules/strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmmirror.com/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha512-VhumSSbBqDTP8p2ZLKj40UjBCV4+v8bUSEpUb4KjRgWk9pbqGF4REFj6KEagidb2f/M6AzC0EmFyDNGaw9OCzg==", + "dev": true, + "dependencies": { + "ansi-regex": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/gaze": { + "version": "1.1.3", + "resolved": "https://registry.npmmirror.com/gaze/-/gaze-1.1.3.tgz", + "integrity": "sha512-BRdNm8hbWzFzWHERTrejLqwHDfS4GibPoq5wjTPIoJHoBtKGPg3xAFfxmM+9ztbXelxcf2hwQcaz1PtmFeue8g==", + "dev": true, + "dependencies": { + "globule": "^1.0.0" + }, + "engines": { + "node": ">= 4.0.0" + } + }, + "node_modules/gensync": { + "version": "1.0.0-beta.2", + "resolved": "https://registry.npmmirror.com/gensync/-/gensync-1.0.0-beta.2.tgz", + "integrity": "sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==", + "dev": true, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/get-caller-file": { + "version": "2.0.5", + "resolved": "https://registry.npmmirror.com/get-caller-file/-/get-caller-file-2.0.5.tgz", + "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", + "dev": true, + "engines": { + "node": "6.* || 8.* || >= 10.*" + } + }, + "node_modules/get-intrinsic": { + "version": "1.1.1", + "resolved": "https://registry.npmmirror.com/get-intrinsic/-/get-intrinsic-1.1.1.tgz", + "integrity": "sha512-kWZrnVM42QCiEA2Ig1bG8zjoIMOgxWwYCEeNdwY6Tv/cOSeGpcoX4pXHfKUxNKVoArnrEr2e9srnAxxGIraS9Q==", + "dependencies": { + "function-bind": "^1.1.1", + "has": "^1.0.3", + "has-symbols": "^1.0.1" + } + }, + "node_modules/get-own-enumerable-property-symbols": { + "version": "3.0.2", + "resolved": "https://registry.npmmirror.com/get-own-enumerable-property-symbols/-/get-own-enumerable-property-symbols-3.0.2.tgz", + "integrity": "sha512-I0UBV/XOz1XkIJHEUDMZAbzCThU/H8DxmSfmdGcKPnVhu2VfFqr34jr9777IyaTYvxjedWhqVIilEDsCdP5G6g==", + "dev": true + }, + "node_modules/get-pkg-repo": { + "version": "4.2.1", + "resolved": "https://registry.npmmirror.com/get-pkg-repo/-/get-pkg-repo-4.2.1.tgz", + "integrity": "sha512-2+QbHjFRfGB74v/pYWjd5OhU3TDIC2Gv/YKUTk/tCvAz0pkn/Mz6P3uByuBimLOcPvN2jYdScl3xGFSrx0jEcA==", + "dev": true, + "dependencies": { + "@hutson/parse-repository-url": "^3.0.0", + "hosted-git-info": "^4.0.0", + "through2": "^2.0.0", + "yargs": "^16.2.0" + }, + "bin": { + "get-pkg-repo": "src/cli.js" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/get-pkg-repo/node_modules/hosted-git-info": { + "version": "4.1.0", + "resolved": "https://registry.npmmirror.com/hosted-git-info/-/hosted-git-info-4.1.0.tgz", + "integrity": "sha512-kyCuEOWjJqZuDbRHzL8V93NzQhwIB71oFWSyzVo+KPZI+pnQPPxucdkrOZvkLRnrf5URsQM+IJ09Dw29cRALIA==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/get-pkg-repo/node_modules/lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/get-pkg-repo/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/get-port": { + "version": "5.1.1", + "resolved": "https://registry.npmmirror.com/get-port/-/get-port-5.1.1.tgz", + "integrity": "sha512-g/Q1aTSDOxFpchXC4i8ZWvxA1lnPqx/JHqcpIw0/LX9T8x/GBbi6YnlN5nhaKIFkT8oFsscUKgDJYxfwfS6QsQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/get-stdin": { + "version": "6.0.0", + "resolved": "https://registry.npmmirror.com/get-stdin/-/get-stdin-6.0.0.tgz", + "integrity": "sha512-jp4tHawyV7+fkkSKyvjuLZswblUtz+SQKzSWnBbii16BuZksJlU1wuBYXY75r+duh/llF1ur6oNwi+2ZzjKZ7g==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/get-stream": { + "version": "4.1.0", + "resolved": "https://registry.npmmirror.com/get-stream/-/get-stream-4.1.0.tgz", + "integrity": "sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w==", + "dev": true, + "dependencies": { + "pump": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/get-symbol-description": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/get-symbol-description/-/get-symbol-description-1.0.0.tgz", + "integrity": "sha512-2EmdH1YvIQiZpltCNgkuiUnyukzxM/R6NDJX31Ke3BG1Nq5b0S2PhX59UKi9vZpPDQVdqn+1IcaAwnzTT5vCjw==", + "dependencies": { + "call-bind": "^1.0.2", + "get-intrinsic": "^1.1.1" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/get-value": { + "version": "2.0.6", + "resolved": "https://registry.npmmirror.com/get-value/-/get-value-2.0.6.tgz", + "integrity": "sha512-Ln0UQDlxH1BapMu3GPtf7CuYNwRZf2gwCuPqbyG6pB8WfmFpzqcy4xtAaAMUhnNqjMKTiCPZG2oMT3YSx8U2NA==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/getpass": { + "version": "0.1.7", + "resolved": "https://registry.npmmirror.com/getpass/-/getpass-0.1.7.tgz", + "integrity": "sha512-0fzj9JxOLfJ+XGLhR8ze3unN0KZCgZwiSSDz168VERjK8Wl8kVSdcu2kspd4s4wtAa1y/qrVRiAA0WclVsu0ng==", + "dev": true, + "dependencies": { + "assert-plus": "^1.0.0" + } + }, + "node_modules/git-raw-commits": { + "version": "2.0.11", + "resolved": "https://registry.npmmirror.com/git-raw-commits/-/git-raw-commits-2.0.11.tgz", + "integrity": "sha512-VnctFhw+xfj8Va1xtfEqCUD2XDrbAPSJx+hSrE5K7fGdjZruW7XV+QOrN7LF/RJyvspRiD2I0asWsxFp0ya26A==", + "dev": true, + "dependencies": { + "dargs": "^7.0.0", + "lodash": "^4.17.15", + "meow": "^8.0.0", + "split2": "^3.0.0", + "through2": "^4.0.0" + }, + "bin": { + "git-raw-commits": "cli.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/git-raw-commits/node_modules/readable-stream": { + "version": "3.6.0", + "resolved": "https://registry.npmmirror.com/readable-stream/-/readable-stream-3.6.0.tgz", + "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", + "dev": true, + "dependencies": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/git-raw-commits/node_modules/through2": { + "version": "4.0.2", + "resolved": "https://registry.npmmirror.com/through2/-/through2-4.0.2.tgz", + "integrity": "sha512-iOqSav00cVxEEICeD7TjLB1sueEL+81Wpzp2bY17uZjZN0pWZPuo4suZ/61VujxmqSGFfgOcNuTZ85QJwNZQpw==", + "dev": true, + "dependencies": { + "readable-stream": "3" + } + }, + "node_modules/git-remote-origin-url": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/git-remote-origin-url/-/git-remote-origin-url-2.0.0.tgz", + "integrity": "sha512-eU+GGrZgccNJcsDH5LkXR3PB9M958hxc7sbA8DFJjrv9j4L2P/eZfKhM+QD6wyzpiv+b1BpK0XrYCxkovtjSLw==", + "dev": true, + "dependencies": { + "gitconfiglocal": "^1.0.0", + "pify": "^2.3.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/git-remote-origin-url/node_modules/pify": { + "version": "2.3.0", + "resolved": "https://registry.npmmirror.com/pify/-/pify-2.3.0.tgz", + "integrity": "sha512-udgsAY+fTnvv7kI7aaxbqwWNb0AHiB0qBO89PZKPkoTmGOgdbrHDKD+0B2X4uTfJ/FT1R09r9gTsjUjNJotuog==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/git-semver-tags": { + "version": "4.1.1", + "resolved": "https://registry.npmmirror.com/git-semver-tags/-/git-semver-tags-4.1.1.tgz", + "integrity": "sha512-OWyMt5zBe7xFs8vglMmhM9lRQzCWL3WjHtxNNfJTMngGym7pC1kh8sP6jevfydJ6LP3ZvGxfb6ABYgPUM0mtsA==", + "dev": true, + "dependencies": { + "meow": "^8.0.0", + "semver": "^6.0.0" + }, + "bin": { + "git-semver-tags": "cli.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/git-semver-tags/node_modules/semver": { + "version": "6.3.0", + "resolved": "https://registry.npmmirror.com/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/git-up": { + "version": "4.0.5", + "resolved": "https://registry.npmmirror.com/git-up/-/git-up-4.0.5.tgz", + "integrity": "sha512-YUvVDg/vX3d0syBsk/CKUTib0srcQME0JyHkL5BaYdwLsiCslPWmDSi8PUMo9pXYjrryMcmsCoCgsTpSCJEQaA==", + "dev": true, + "dependencies": { + "is-ssh": "^1.3.0", + "parse-url": "^6.0.0" + } + }, + "node_modules/git-url-parse": { + "version": "11.6.0", + "resolved": "https://registry.npmmirror.com/git-url-parse/-/git-url-parse-11.6.0.tgz", + "integrity": "sha512-WWUxvJs5HsyHL6L08wOusa/IXYtMuCAhrMmnTjQPpBU0TTHyDhnOATNH3xNQz7YOQUsqIIPTGr4xiVti1Hsk5g==", + "dev": true, + "dependencies": { + "git-up": "^4.0.0" + } + }, + "node_modules/gitconfiglocal": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/gitconfiglocal/-/gitconfiglocal-1.0.0.tgz", + "integrity": "sha512-spLUXeTAVHxDtKsJc8FkFVgFtMdEN9qPGpL23VfSHx4fP4+Ds097IXLvymbnDH8FnmxX5Nr9bPw3A+AQ6mWEaQ==", + "dev": true, + "dependencies": { + "ini": "^1.3.2" + } + }, + "node_modules/gl-matrix": { + "version": "3.4.3", + "resolved": "https://registry.npmmirror.com/gl-matrix/-/gl-matrix-3.4.3.tgz", + "integrity": "sha512-wcCp8vu8FT22BnvKVPjXa/ICBWRq/zjFfdofZy1WSpQZpphblv12/bOQLBC1rMM7SGOFS9ltVmKOHil5+Ml7gA==" + }, + "node_modules/glob": { + "version": "7.2.0", + "resolved": "https://registry.npmmirror.com/glob/-/glob-7.2.0.tgz", + "integrity": "sha512-lmLf6gtyrPq8tTjSmrO94wBeQbFR3HbLHbuyD69wuyQkImp2hWqMGB47OX65FBkPffO641IP9jWa1z4ivqG26Q==", + "dependencies": { + "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" + }, + "engines": { + "node": "*" + } + }, + "node_modules/glob-base": { + "version": "0.3.0", + "resolved": "https://registry.npmmirror.com/glob-base/-/glob-base-0.3.0.tgz", + "integrity": "sha512-ab1S1g1EbO7YzauaJLkgLp7DZVAqj9M/dvKlTt8DkXA2tiOIcSMrlVI2J1RZyB5iJVccEscjGn+kpOG9788MHA==", + "dev": true, + "dependencies": { + "glob-parent": "^2.0.0", + "is-glob": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/glob-base/node_modules/glob-parent": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/glob-parent/-/glob-parent-2.0.0.tgz", + "integrity": "sha512-JDYOvfxio/t42HKdxkAYaCiBN7oYiuxykOxKxdaUW5Qn0zaYN3gRQWolrwdnf0shM9/EP0ebuuTmyoXNr1cC5w==", + "dev": true, + "dependencies": { + "is-glob": "^2.0.0" + } + }, + "node_modules/glob-base/node_modules/is-extglob": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/is-extglob/-/is-extglob-1.0.0.tgz", + "integrity": "sha512-7Q+VbVafe6x2T+Tu6NcOf6sRklazEPmBoB3IWk3WdGZM2iGUwU/Oe3Wtq5lSEkDTTlpp8yx+5t4pzO/i9Ty1ww==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/glob-base/node_modules/is-glob": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/is-glob/-/is-glob-2.0.1.tgz", + "integrity": "sha512-a1dBeB19NXsf/E0+FHqkagizel/LQw2DjSQpvQrj3zT+jYPpaUCryPnrQajXKFLCMuf4I6FhRpaGtw4lPrG6Eg==", + "dev": true, + "dependencies": { + "is-extglob": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/glob-parent": { + "version": "5.1.2", + "resolved": "https://registry.npmmirror.com/glob-parent/-/glob-parent-5.1.2.tgz", + "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", + "devOptional": true, + "dependencies": { + "is-glob": "^4.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/glob-to-regexp": { + "version": "0.3.0", + "resolved": "https://registry.npmmirror.com/glob-to-regexp/-/glob-to-regexp-0.3.0.tgz", + "integrity": "sha512-Iozmtbqv0noj0uDDqoL0zNq0VBEfK2YFoMAZoxJe4cwphvLR+JskfF30QhXHOR4m3KrE6NLRYw+U9MRXvifyig==" + }, + "node_modules/glob2base": { + "version": "0.0.12", + "resolved": "https://registry.npmmirror.com/glob2base/-/glob2base-0.0.12.tgz", + "integrity": "sha512-ZyqlgowMbfj2NPjxaZZ/EtsXlOch28FRXgMd64vqZWk1bT9+wvSRLYD1om9M7QfQru51zJPAT17qXm4/zd+9QA==", + "dev": true, + "dependencies": { + "find-index": "^0.1.1" + }, + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/globals": { + "version": "11.12.0", + "resolved": "https://registry.npmmirror.com/globals/-/globals-11.12.0.tgz", + "integrity": "sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/globby": { + "version": "9.2.0", + "resolved": "https://registry.npmmirror.com/globby/-/globby-9.2.0.tgz", + "integrity": "sha512-ollPHROa5mcxDEkwg6bPt3QbEf4pDQSNtd6JPL1YvOvAo/7/0VAm9TccUeoTmarjPw4pfUthSCqcyfNB1I3ZSg==", + "dev": true, + "dependencies": { + "@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" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/globby/node_modules/ignore": { + "version": "4.0.6", + "resolved": "https://registry.npmmirror.com/ignore/-/ignore-4.0.6.tgz", + "integrity": "sha512-cyFDKrqc/YdcWFniJhzI42+AzS+gNwmUzOSFcRCQYwySuBBBy/KjuxWLZ/FHEH6Moq1NizMOBWyTcv8O4OZIMg==", + "dev": true, + "engines": { + "node": ">= 4" + } + }, + "node_modules/globule": { + "version": "1.3.3", + "resolved": "https://registry.npmmirror.com/globule/-/globule-1.3.3.tgz", + "integrity": "sha512-mb1aYtDbIjTu4ShMB85m3UzjX9BVKe9WCzsnfMSZk+K5GpIbBOexgg4PPCt5eHDEG5/ZQAUX2Kct02zfiPLsKg==", + "dev": true, + "dependencies": { + "glob": "~7.1.1", + "lodash": "~4.17.10", + "minimatch": "~3.0.2" + }, + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/globule/node_modules/glob": { + "version": "7.1.7", + "resolved": "https://registry.npmmirror.com/glob/-/glob-7.1.7.tgz", + "integrity": "sha512-OvD9ENzPLbegENnYP5UUfJIirTg4+XwMWGaQfQTY0JenxNvvIKP3U3/tAQSPIu/lHxXYSZmpXlUHeqAIdKzBLQ==", + "dev": true, + "dependencies": { + "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" + }, + "engines": { + "node": "*" + } + }, + "node_modules/globule/node_modules/minimatch": { + "version": "3.0.8", + "resolved": "https://registry.npmmirror.com/minimatch/-/minimatch-3.0.8.tgz", + "integrity": "sha512-6FsRAQsxQ61mw+qP1ZzbL9Bc78x2p5OqNgNpnoAFLTrX8n5Kxph0CsnhmKKNXTWjXqU5L0pGPR7hYk+XWZr60Q==", + "dev": true, + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/good-listener": { + "version": "1.2.2", + "resolved": "https://registry.npmmirror.com/good-listener/-/good-listener-1.2.2.tgz", + "integrity": "sha512-goW1b+d9q/HIwbVYZzZ6SsTr4IgE+WA44A0GmPIQstuOrgsFcT7VEJ48nmr9GaRtNu0XTKacFLGnBPAM6Afouw==", + "dependencies": { + "delegate": "^3.1.2" + } + }, + "node_modules/graceful-fs": { + "version": "4.2.10", + "resolved": "https://registry.npmmirror.com/graceful-fs/-/graceful-fs-4.2.10.tgz", + "integrity": "sha512-9ByhssR2fPVsNZj478qUUbKfmL0+t5BDVyjShtyZZLiK7ZDAArFFfopyOTj0M05wE2tJPisA4iTnnXl2YoPvOA==" + }, + "node_modules/graphlib": { + "version": "2.1.8", + "resolved": "https://registry.npmmirror.com/graphlib/-/graphlib-2.1.8.tgz", + "integrity": "sha512-jcLLfkpoVGmH7/InMC/1hIvOPSUh38oJtGhvrOFGzioE1DZ+0YW16RgmOJhHiuWTvGiJQ9Z1Ik43JvkRPRvE+A==", + "dependencies": { + "lodash": "^4.17.15" + } + }, + "node_modules/gzip-size": { + "version": "5.1.1", + "resolved": "https://registry.npmmirror.com/gzip-size/-/gzip-size-5.1.1.tgz", + "integrity": "sha512-FNHi6mmoHvs1mxZAds4PpdCS6QG8B4C1krxJsMutgxl5t3+GlRTzzI3NEkifXx2pVsOvJdOGSmIgDhQ55FwdPA==", + "dev": true, + "dependencies": { + "duplexer": "^0.1.1", + "pify": "^4.0.1" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/handle-thing": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/handle-thing/-/handle-thing-2.0.1.tgz", + "integrity": "sha512-9Qn4yBxelxoh2Ow62nP+Ka/kMnOXRi8BXnRaUwezLNhqelnN49xKz4F/dPP8OYLxLxq6JDtZb2i9XznUQbNPTg==", + "dev": true + }, + "node_modules/handlebars": { + "version": "4.7.7", + "resolved": "https://registry.npmmirror.com/handlebars/-/handlebars-4.7.7.tgz", + "integrity": "sha512-aAcXm5OAfE/8IXkcZvCepKU3VzW1/39Fb5ZuqMtgI/hT8X2YgoMvBY5dLhq/cpOvw7Lk1nK/UF71aLG/ZnVYRA==", + "dev": true, + "dependencies": { + "minimist": "^1.2.5", + "neo-async": "^2.6.0", + "source-map": "^0.6.1", + "wordwrap": "^1.0.0" + }, + "bin": { + "handlebars": "bin/handlebars" + }, + "engines": { + "node": ">=0.4.7" + }, + "optionalDependencies": { + "uglify-js": "^3.1.4" + } + }, + "node_modules/handlebars/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmmirror.com/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/har-schema": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/har-schema/-/har-schema-2.0.0.tgz", + "integrity": "sha512-Oqluz6zhGX8cyRaTQlFMPw80bSJVG2x/cFb8ZPhUILGgHka9SsokCCOQgpveePerqidZOrT14ipqfJb7ILcW5Q==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/har-validator": { + "version": "5.1.5", + "resolved": "https://registry.npmmirror.com/har-validator/-/har-validator-5.1.5.tgz", + "integrity": "sha512-nmT2T0lljbxdQZfspsno9hgrG3Uir6Ks5afism62poxqBM6sDnMEuPmzTq8XN0OEwqKLLdh1jQI3qyE66Nzb3w==", + "deprecated": "this library is no longer supported", + "dev": true, + "dependencies": { + "ajv": "^6.12.3", + "har-schema": "^2.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/hard-rejection": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/hard-rejection/-/hard-rejection-2.1.0.tgz", + "integrity": "sha512-VIZB+ibDhx7ObhAe7OVtoEbuP4h/MuOTHJ+J8h/eBXotJYl0fBgR72xDFCKgIh22OJZIOVNxBMWuhAr10r8HdA==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/has": { + "version": "1.0.3", + "resolved": "https://registry.npmmirror.com/has/-/has-1.0.3.tgz", + "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", + "dependencies": { + "function-bind": "^1.1.1" + }, + "engines": { + "node": ">= 0.4.0" + } + }, + "node_modules/has-ansi": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/has-ansi/-/has-ansi-2.0.0.tgz", + "integrity": "sha512-C8vBJ8DwUCx19vhm7urhTuUsr4/IyP6l4VzNQDv+ryHQObW3TTTp9yB68WpYgRe2bbaGuZ/se74IqFeVnMnLZg==", + "dev": true, + "dependencies": { + "ansi-regex": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/has-ansi/node_modules/ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmmirror.com/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha512-TIGnTpdo+E3+pCyAluZvtED5p5wCqLdezCyhPZzKPcxvFplEt4i+W7OONCKgeZFT3+y5NZZfOOS/Bdcanm1MYA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/has-bigints": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/has-bigints/-/has-bigints-1.0.2.tgz", + "integrity": "sha512-tSvCKtBr9lkF0Ex0aQiP9N+OpV4zi2r/Nee5VkRDbaqv35RLYMzbwQfFSZZH0kR+Rd6302UJZ2p/bJCEoR3VoQ==" + }, + "node_modules/has-flag": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/has-flag/-/has-flag-3.0.0.tgz", + "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==", + "engines": { + "node": ">=4" + } + }, + "node_modules/has-property-descriptors": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/has-property-descriptors/-/has-property-descriptors-1.0.0.tgz", + "integrity": "sha512-62DVLZGoiEBDHQyqG4w9xCuZ7eJEwNmJRWw2VY84Oedb7WFcA27fiEVe8oUQx9hAUJ4ekurquucTGwsyO1XGdQ==", + "dependencies": { + "get-intrinsic": "^1.1.1" + } + }, + "node_modules/has-symbols": { + "version": "1.0.3", + "resolved": "https://registry.npmmirror.com/has-symbols/-/has-symbols-1.0.3.tgz", + "integrity": "sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/has-tostringtag": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/has-tostringtag/-/has-tostringtag-1.0.0.tgz", + "integrity": "sha512-kFjcSNhnlGV1kyoGk7OXKSawH5JOb/LzUc5w9B02hOTO0dfFRjbHQKvg1d6cf3HbeUmtU9VbbV3qzZ2Teh97WQ==", + "dependencies": { + "has-symbols": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/has-unicode": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/has-unicode/-/has-unicode-2.0.1.tgz", + "integrity": "sha512-8Rf9Y83NBReMnx0gFzA8JImQACstCYWUplepDa9xprwwtmgEZUF0h/i5xSA625zB/I37EtrswSST6OXxwaaIJQ==", + "dev": true + }, + "node_modules/has-value": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/has-value/-/has-value-1.0.0.tgz", + "integrity": "sha512-IBXk4GTsLYdQ7Rvt+GRBrFSVEkmuOUy4re0Xjd9kJSUQpnTrWR4/y9RpfexN9vkAPMFuQoeWKwqzPozRTlasGw==", + "dependencies": { + "get-value": "^2.0.6", + "has-values": "^1.0.0", + "isobject": "^3.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/has-values": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/has-values/-/has-values-1.0.0.tgz", + "integrity": "sha512-ODYZC64uqzmtfGMEAX/FvZiRyWLpAC3vYnNunURUnkGVTS+mI0smVsWaPydRBsE3g+ok7h960jChO8mFcWlHaQ==", + "dependencies": { + "is-number": "^3.0.0", + "kind-of": "^4.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/has-values/node_modules/kind-of": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/kind-of/-/kind-of-4.0.0.tgz", + "integrity": "sha512-24XsCxmEbRwEDbz/qz3stgin8TTzZ1ESR56OMCN0ujYg+vRutNSiOj9bHH9u85DKgXguraugV5sFuvbD4FW/hw==", + "dependencies": { + "is-buffer": "^1.1.5" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/hash-base": { + "version": "3.1.0", + "resolved": "https://registry.npmmirror.com/hash-base/-/hash-base-3.1.0.tgz", + "integrity": "sha512-1nmYp/rhMDiE7AYkDw+lLwlAzz0AntGIe51F3RfFfEqyQ3feY2eI/NcwC6umIQVOASPMsWJLJScWKSSvzL9IVA==", + "dependencies": { + "inherits": "^2.0.4", + "readable-stream": "^3.6.0", + "safe-buffer": "^5.2.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/hash-base/node_modules/readable-stream": { + "version": "3.6.0", + "resolved": "https://registry.npmmirror.com/readable-stream/-/readable-stream-3.6.0.tgz", + "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", + "dependencies": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/hash-base/node_modules/safe-buffer": { + "version": "5.2.1", + "resolved": "https://registry.npmmirror.com/safe-buffer/-/safe-buffer-5.2.1.tgz", + "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==" + }, + "node_modules/hash-sum": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/hash-sum/-/hash-sum-1.0.2.tgz", + "integrity": "sha512-fUs4B4L+mlt8/XAtSOGMUO1TXmAelItBPtJG7CyHJfYTdDjwisntGO2JQz7oUsatOY9o68+57eziUVNw/mRHmA==", + "dev": true + }, + "node_modules/hash.js": { + "version": "1.1.7", + "resolved": "https://registry.npmmirror.com/hash.js/-/hash.js-1.1.7.tgz", + "integrity": "sha512-taOaskGt4z4SOANNseOviYDvjEJinIkRgmp7LbKP2YTTmVxWBl87s/uzK9r+44BclBSp2X7K1hqeNfz9JbBeXA==", + "dependencies": { + "inherits": "^2.0.3", + "minimalistic-assert": "^1.0.1" + } + }, + "node_modules/he": { + "version": "1.2.0", + "resolved": "https://registry.npmmirror.com/he/-/he-1.2.0.tgz", + "integrity": "sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==", + "dev": true, + "bin": { + "he": "bin/he" + } + }, + "node_modules/hex-color-regex": { + "version": "1.1.0", + "resolved": "https://registry.npmmirror.com/hex-color-regex/-/hex-color-regex-1.1.0.tgz", + "integrity": "sha512-l9sfDFsuqtOqKDsQdqrMRk0U85RZc0RtOR9yPI7mRVOa4FsR/BVnZ0shmQRM96Ji99kYZP/7hn1cedc1+ApsTQ==", + "dev": true + }, + "node_modules/highlight.js": { + "version": "9.18.3", + "resolved": "https://registry.npmmirror.com/highlight.js/-/highlight.js-9.18.3.tgz", + "integrity": "sha512-zBZAmhSupHIl5sITeMqIJnYCDfAEc3Gdkqj65wC1lpI468MMQeeQkhcIAvk+RylAkxrCcI9xy9piHiXeQ1BdzQ==", + "deprecated": "Version no longer supported. Upgrade to @latest", + "engines": { + "node": "*" + } + }, + "node_modules/hmac-drbg": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/hmac-drbg/-/hmac-drbg-1.0.1.tgz", + "integrity": "sha512-Tti3gMqLdZfhOQY1Mzf/AanLiqh1WTiJgEj26ZuYQ9fbkLomzGchCws4FyrSd4VkpBfiNhaE1On+lOz894jvXg==", + "dependencies": { + "hash.js": "^1.0.3", + "minimalistic-assert": "^1.0.0", + "minimalistic-crypto-utils": "^1.0.1" + } + }, + "node_modules/hoopy": { + "version": "0.1.4", + "resolved": "https://registry.npmmirror.com/hoopy/-/hoopy-0.1.4.tgz", + "integrity": "sha512-HRcs+2mr52W0K+x8RzcLzuPPmVIKMSv97RGHy0Ea9y/mpcaK+xTrjICA04KAHi4GRzxliNqNJEFYWHghy3rSfQ==", + "dev": true, + "engines": { + "node": ">= 6.0.0" + } + }, + "node_modules/hosted-git-info": { + "version": "2.8.9", + "resolved": "https://registry.npmmirror.com/hosted-git-info/-/hosted-git-info-2.8.9.tgz", + "integrity": "sha512-mxIDAb9Lsm6DoOJ7xH+5+X4y1LU/4Hi50L9C5sIswK3JzULS4bwk1FvjdBgvYR4bzT4tuUQiC15FE2f5HbLvYw==", + "dev": true + }, + "node_modules/hpack.js": { + "version": "2.1.6", + "resolved": "https://registry.npmmirror.com/hpack.js/-/hpack.js-2.1.6.tgz", + "integrity": "sha512-zJxVehUdMGIKsRaNt7apO2Gqp0BdqW5yaiGHXXmbpvxgBYVZnAql+BJb4RO5ad2MgpbZKn5G6nMnegrH1FcNYQ==", + "dev": true, + "dependencies": { + "inherits": "^2.0.1", + "obuf": "^1.0.0", + "readable-stream": "^2.0.1", + "wbuf": "^1.1.0" + } + }, + "node_modules/hsl-regex": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/hsl-regex/-/hsl-regex-1.0.0.tgz", + "integrity": "sha512-M5ezZw4LzXbBKMruP+BNANf0k+19hDQMgpzBIYnya//Al+fjNct9Wf3b1WedLqdEs2hKBvxq/jh+DsHJLj0F9A==", + "dev": true + }, + "node_modules/hsla-regex": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/hsla-regex/-/hsla-regex-1.0.0.tgz", + "integrity": "sha512-7Wn5GMLuHBjZCb2bTmnDOycho0p/7UVaAeqXZGbHrBCl6Yd/xDhQJAXe6Ga9AXJH2I5zY1dEdYw2u1UptnSBJA==", + "dev": true + }, + "node_modules/html-entities": { + "version": "1.4.0", + "resolved": "https://registry.npmmirror.com/html-entities/-/html-entities-1.4.0.tgz", + "integrity": "sha512-8nxjcBcd8wovbeKx7h3wTji4e6+rhaVuPNpMqwWgnHh+N9ToqsCs6XztWRBPQ+UtzsoMAdKZtUENoVzU/EMtZA==", + "dev": true + }, + "node_modules/html-minifier": { + "version": "3.5.21", + "resolved": "https://registry.npmmirror.com/html-minifier/-/html-minifier-3.5.21.tgz", + "integrity": "sha512-LKUKwuJDhxNa3uf/LPR/KVjm/l3rBqtYeCOAekvG8F1vItxMUpueGd94i/asDDr8/1u7InxzFA5EeGjhhG5mMA==", + "dev": true, + "dependencies": { + "camel-case": "3.0.x", + "clean-css": "4.2.x", + "commander": "2.17.x", + "he": "1.2.x", + "param-case": "2.1.x", + "relateurl": "0.2.x", + "uglify-js": "3.4.x" + }, + "bin": { + "html-minifier": "cli.js" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/html-minifier/node_modules/commander": { + "version": "2.17.1", + "resolved": "https://registry.npmmirror.com/commander/-/commander-2.17.1.tgz", + "integrity": "sha512-wPMUt6FnH2yzG95SA6mzjQOEKUU3aLaDEmzs1ti+1E9h+CsrZghRlqEM/EJ4KscsQVG8uNN4uVreUeT8+drlgg==", + "dev": true + }, + "node_modules/html-tags": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/html-tags/-/html-tags-2.0.0.tgz", + "integrity": "sha512-+Il6N8cCo2wB/Vd3gqy/8TZhTD3QvcVeQLCnZiGkGCH3JP28IgGAY41giccp2W4R3jfyJPAP318FQTa1yU7K7g==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/html-webpack-plugin": { + "version": "3.2.0", + "resolved": "https://registry.npmmirror.com/html-webpack-plugin/-/html-webpack-plugin-3.2.0.tgz", + "integrity": "sha512-Br4ifmjQojUP4EmHnRBoUIYcZ9J7M4bTMcm7u6xoIAIuq2Nte4TzXX0533owvkQKQD1WeMTTTyD4Ni4QKxS0Bg==", + "deprecated": "3.x is no longer supported", + "dev": true, + "dependencies": { + "html-minifier": "^3.2.3", + "loader-utils": "^0.2.16", + "lodash": "^4.17.3", + "pretty-error": "^2.0.2", + "tapable": "^1.0.0", + "toposort": "^1.0.0", + "util.promisify": "1.0.0" + }, + "engines": { + "node": ">=6.9" + }, + "peerDependencies": { + "webpack": "^1.0.0 || ^2.0.0 || ^3.0.0 || ^4.0.0" + } + }, + "node_modules/html-webpack-plugin/node_modules/big.js": { + "version": "3.2.0", + "resolved": "https://registry.npmmirror.com/big.js/-/big.js-3.2.0.tgz", + "integrity": "sha512-+hN/Zh2D08Mx65pZ/4g5bsmNiZUuChDiQfTUQ7qJr4/kuopCr88xZsAXv6mBoZEsUI4OuGHlX59qE94K2mMW8Q==", + "dev": true, + "engines": { + "node": "*" + } + }, + "node_modules/html-webpack-plugin/node_modules/emojis-list": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/emojis-list/-/emojis-list-2.1.0.tgz", + "integrity": "sha512-knHEZMgs8BB+MInokmNTg/OyPlAddghe1YBgNwJBc5zsJi/uyIcXoSDsL/W9ymOsBoBGdPIHXYJ9+qKFwRwDng==", + "dev": true, + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/html-webpack-plugin/node_modules/json5": { + "version": "0.5.1", + "resolved": "https://registry.npmmirror.com/json5/-/json5-0.5.1.tgz", + "integrity": "sha512-4xrs1aW+6N5DalkqSVA8fxh458CXvR99WU8WLKmq4v8eWAL86Xo3BVqyd3SkA9wEVjCMqyvvRRkshAdOnBp5rw==", + "dev": true, + "bin": { + "json5": "lib/cli.js" + } + }, + "node_modules/html-webpack-plugin/node_modules/loader-utils": { + "version": "0.2.17", + "resolved": "https://registry.npmmirror.com/loader-utils/-/loader-utils-0.2.17.tgz", + "integrity": "sha512-tiv66G0SmiOx+pLWMtGEkfSEejxvb6N6uRrQjfWJIT79W9GMpgKeCAmm9aVBKtd4WEgntciI8CsGqjpDoCWJug==", + "dev": true, + "dependencies": { + "big.js": "^3.1.3", + "emojis-list": "^2.0.0", + "json5": "^0.5.0", + "object-assign": "^4.0.1" + } + }, + "node_modules/html-webpack-plugin/node_modules/util.promisify": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/util.promisify/-/util.promisify-1.0.0.tgz", + "integrity": "sha512-i+6qA2MPhvoKLuxnJNpXAGhg7HphQOSUq2LKMZD0m15EiskXUkMvKdF4Uui0WYeCUGea+o2cw/ZuwehtfsrNkA==", + "dev": true, + "dependencies": { + "define-properties": "^1.1.2", + "object.getownpropertydescriptors": "^2.0.3" + } + }, + "node_modules/htmlparser2": { + "version": "6.1.0", + "resolved": "https://registry.npmmirror.com/htmlparser2/-/htmlparser2-6.1.0.tgz", + "integrity": "sha512-gyyPk6rgonLFEDGoeRgQNaEUvdJ4ktTmmUh/h2t7s+M8oPpIPxgNACWa+6ESR57kXstwqPiCut0V8NRpcwgU7A==", + "dev": true, + "dependencies": { + "domelementtype": "^2.0.1", + "domhandler": "^4.0.0", + "domutils": "^2.5.2", + "entities": "^2.0.0" + } + }, + "node_modules/htmlparser2/node_modules/dom-serializer": { + "version": "1.4.1", + "resolved": "https://registry.npmmirror.com/dom-serializer/-/dom-serializer-1.4.1.tgz", + "integrity": "sha512-VHwB3KfrcOOkelEG2ZOfxqLZdfkil8PtJi4P8N2MMXucZq2yLp75ClViUlOVwyoHEDjYU433Aq+5zWP61+RGag==", + "dev": true, + "dependencies": { + "domelementtype": "^2.0.1", + "domhandler": "^4.2.0", + "entities": "^2.0.0" + } + }, + "node_modules/htmlparser2/node_modules/domelementtype": { + "version": "2.3.0", + "resolved": "https://registry.npmmirror.com/domelementtype/-/domelementtype-2.3.0.tgz", + "integrity": "sha512-OLETBj6w0OsagBwdXnPdN0cnMfF9opN69co+7ZrbfPGrdpPVNBUj02spi6B1N7wChLQiPn4CSH/zJvXw56gmHw==", + "dev": true + }, + "node_modules/htmlparser2/node_modules/domutils": { + "version": "2.8.0", + "resolved": "https://registry.npmmirror.com/domutils/-/domutils-2.8.0.tgz", + "integrity": "sha512-w96Cjofp72M5IIhpjgobBimYEfoPjx1Vx0BSX9P30WBdZW2WIKU0T1Bd0kz2eNZ9ikjKgHbEyKx8BB6H1L3h3A==", + "dev": true, + "dependencies": { + "dom-serializer": "^1.0.1", + "domelementtype": "^2.2.0", + "domhandler": "^4.2.0" + } + }, + "node_modules/http-cache-semantics": { + "version": "4.1.0", + "resolved": "https://registry.npmmirror.com/http-cache-semantics/-/http-cache-semantics-4.1.0.tgz", + "integrity": "sha512-carPklcUh7ROWRK7Cv27RPtdhYhUsela/ue5/jKzjegVvXDqM2ILE9Q2BGn9JZJh1g87cp56su/FgQSzcWS8cQ==", + "dev": true + }, + "node_modules/http-deceiver": { + "version": "1.2.7", + "resolved": "https://registry.npmmirror.com/http-deceiver/-/http-deceiver-1.2.7.tgz", + "integrity": "sha512-LmpOGxTfbpgtGVxJrj5k7asXHCgNZp5nLfp+hWc8QQRqtb7fUy6kRY3BO1h9ddF6yIPYUARgxGOwB42DnxIaNw==", + "dev": true + }, + "node_modules/http-errors": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/http-errors/-/http-errors-2.0.0.tgz", + "integrity": "sha512-FtwrG/euBzaEjYeRqOgly7G0qviiXoJWnvEH2Z1plBdXgbyjv34pHTSb9zoeHMyDy33+DWy5Wt9Wo+TURtOYSQ==", + "dev": true, + "dependencies": { + "depd": "2.0.0", + "inherits": "2.0.4", + "setprototypeof": "1.2.0", + "statuses": "2.0.1", + "toidentifier": "1.0.1" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/http-parser-js": { + "version": "0.5.6", + "resolved": "https://registry.npmmirror.com/http-parser-js/-/http-parser-js-0.5.6.tgz", + "integrity": "sha512-vDlkRPDJn93swjcjqMSaGSPABbIarsr1TLAui/gLDXzV5VsJNdXNzMYDyNBLQkjWQCJ1uizu8T2oDMhmGt0PRA==", + "dev": true + }, + "node_modules/http-proxy": { + "version": "1.18.1", + "resolved": "https://registry.npmmirror.com/http-proxy/-/http-proxy-1.18.1.tgz", + "integrity": "sha512-7mz/721AbnJwIVbnaSv1Cz3Am0ZLT/UBwkC92VlxhXv/k/BBQfM2fXElQNC27BVGr0uwUpplYPQM9LnaBMR5NQ==", + "dev": true, + "dependencies": { + "eventemitter3": "^4.0.0", + "follow-redirects": "^1.0.0", + "requires-port": "^1.0.0" + }, + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/http-proxy-agent": { + "version": "4.0.1", + "resolved": "https://registry.npmmirror.com/http-proxy-agent/-/http-proxy-agent-4.0.1.tgz", + "integrity": "sha512-k0zdNgqWTGA6aeIRVpvfVob4fL52dTfaehylg0Y4UvSySvOq/Y+BOyPrgpUrA7HylqvU8vIZGsRuXmspskV0Tg==", + "dev": true, + "dependencies": { + "@tootallnate/once": "1", + "agent-base": "6", + "debug": "4" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/http-proxy-agent/node_modules/debug": { + "version": "4.3.4", + "resolved": "https://registry.npmmirror.com/debug/-/debug-4.3.4.tgz", + "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", + "dev": true, + "dependencies": { + "ms": "2.1.2" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/http-proxy-agent/node_modules/ms": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", + "dev": true + }, + "node_modules/http-proxy-middleware": { + "version": "0.19.1", + "resolved": "https://registry.npmmirror.com/http-proxy-middleware/-/http-proxy-middleware-0.19.1.tgz", + "integrity": "sha512-yHYTgWMQO8VvwNS22eLLloAkvungsKdKTLO8AJlftYIKNfJr3GK3zK0ZCfzDDGUBttdGc8xFy1mCitvNKQtC3Q==", + "dev": true, + "dependencies": { + "http-proxy": "^1.17.0", + "is-glob": "^4.0.0", + "lodash": "^4.17.11", + "micromatch": "^3.1.10" + }, + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/http-signature": { + "version": "1.2.0", + "resolved": "https://registry.npmmirror.com/http-signature/-/http-signature-1.2.0.tgz", + "integrity": "sha512-CAbnr6Rz4CYQkLYUtSNXxQPUH2gK8f3iWexVlsnMeD+GjlsQ0Xsy1cOX+mN3dtxYomRy21CiOzU8Uhw6OwncEQ==", + "dev": true, + "dependencies": { + "assert-plus": "^1.0.0", + "jsprim": "^1.2.2", + "sshpk": "^1.7.0" + }, + "engines": { + "node": ">=0.8", + "npm": ">=1.3.7" + } + }, + "node_modules/https-browserify": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/https-browserify/-/https-browserify-1.0.0.tgz", + "integrity": "sha512-J+FkSdyD+0mA0N+81tMotaRMfSL9SGi+xpD3T6YApKsc3bGSXJlfXri3VyFOeYkfLRQisDk1W+jIFFKBeUBbBg==" + }, + "node_modules/https-proxy-agent": { + "version": "5.0.1", + "resolved": "https://registry.npmmirror.com/https-proxy-agent/-/https-proxy-agent-5.0.1.tgz", + "integrity": "sha512-dFcAjpTQFgoLMzC2VwU+C/CbS7uRL0lWmxDITmqm7C+7F0Odmj6s9l6alZc6AELXhrnggM2CeWSXHGOdX2YtwA==", + "dev": true, + "dependencies": { + "agent-base": "6", + "debug": "4" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/https-proxy-agent/node_modules/debug": { + "version": "4.3.4", + "resolved": "https://registry.npmmirror.com/debug/-/debug-4.3.4.tgz", + "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", + "dev": true, + "dependencies": { + "ms": "2.1.2" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/https-proxy-agent/node_modules/ms": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", + "dev": true + }, + "node_modules/human-signals": { + "version": "1.1.1", + "resolved": "https://registry.npmmirror.com/human-signals/-/human-signals-1.1.1.tgz", + "integrity": "sha512-SEQu7vl8KjNL2eoGBLF3+wAjpsNfA9XMlXAYj/3EdaNfAlxKthD1xjEQfGOUhllCGGJVNY34bRr6lPINhNjyZw==", + "dev": true, + "engines": { + "node": ">=8.12.0" + } + }, + "node_modules/humanize-ms": { + "version": "1.2.1", + "resolved": "https://registry.npmmirror.com/humanize-ms/-/humanize-ms-1.2.1.tgz", + "integrity": "sha512-Fl70vYtsAFb/C06PTS9dZBo7ihau+Tu/DNCk/OyHhea07S+aeMWpFFkUaXRa8fI+ScZbEI8dfSxwY7gxZ9SAVQ==", + "dev": true, + "dependencies": { + "ms": "^2.0.0" + } + }, + "node_modules/husky": { + "version": "1.3.1", + "resolved": "https://registry.npmmirror.com/husky/-/husky-1.3.1.tgz", + "integrity": "sha512-86U6sVVVf4b5NYSZ0yvv88dRgBSSXXmHaiq5pP4KDj5JVzdwKgBjEtUPOm8hcoytezFwbU+7gotXNhpHdystlg==", + "dev": true, + "hasInstallScript": true, + "dependencies": { + "cosmiconfig": "^5.0.7", + "execa": "^1.0.0", + "find-up": "^3.0.0", + "get-stdin": "^6.0.0", + "is-ci": "^2.0.0", + "pkg-dir": "^3.0.0", + "please-upgrade-node": "^3.1.1", + "read-pkg": "^4.0.1", + "run-node": "^1.0.0", + "slash": "^2.0.0" + }, + "bin": { + "husky-upgrade": "lib/upgrader/bin.js" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/husky/node_modules/ci-info": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/ci-info/-/ci-info-2.0.0.tgz", + "integrity": "sha512-5tK7EtrZ0N+OLFMthtqOj4fI2Jeb88C4CAZPu25LDVUgXJ0A3Js4PMGqrn0JU1W0Mh1/Z8wZzYPxqUrXeBboCQ==", + "dev": true + }, + "node_modules/husky/node_modules/is-ci": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/is-ci/-/is-ci-2.0.0.tgz", + "integrity": "sha512-YfJT7rkpQB0updsdHLGWrvhBJfcfzNNawYDNIyQXJz0IViGf75O8EBPKSdvw2rF+LGCsX4FZ8tcr3b19LcZq4w==", + "dev": true, + "dependencies": { + "ci-info": "^2.0.0" + }, + "bin": { + "is-ci": "bin.js" + } + }, + "node_modules/husky/node_modules/pify": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/pify/-/pify-3.0.0.tgz", + "integrity": "sha512-C3FsVNH1udSEX48gGX1xfvwTWfsYWj5U+8/uK15BGzIGrKoUpghX8hWZwa/OFnakBiiVNmBvemTJR5mcy7iPcg==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/husky/node_modules/read-pkg": { + "version": "4.0.1", + "resolved": "https://registry.npmmirror.com/read-pkg/-/read-pkg-4.0.1.tgz", + "integrity": "sha512-+UBirHHDm5J+3WDmLBZYSklRYg82nMlz+enn+GMZ22nSR2f4bzxmhso6rzQW/3mT2PVzpzDTiYIZahk8UmZ44w==", + "dev": true, + "dependencies": { + "normalize-package-data": "^2.3.2", + "parse-json": "^4.0.0", + "pify": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/iconv-lite": { + "version": "0.4.24", + "resolved": "https://registry.npmmirror.com/iconv-lite/-/iconv-lite-0.4.24.tgz", + "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", + "dev": true, + "dependencies": { + "safer-buffer": ">= 2.1.2 < 3" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/icss-replace-symbols": { + "version": "1.1.0", + "resolved": "https://registry.npmmirror.com/icss-replace-symbols/-/icss-replace-symbols-1.1.0.tgz", + "integrity": "sha512-chIaY3Vh2mh2Q3RGXttaDIzeiPvaVXJ+C4DAh/w3c37SKZ/U6PGMmuicR2EQQp9bKG8zLMCl7I+PtIoOOPp8Gg==", + "dev": true + }, + "node_modules/icss-utils": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/icss-utils/-/icss-utils-2.1.0.tgz", + "integrity": "sha512-bsVoyn/1V4R1kYYjLcWLedozAM4FClZUdjE9nIr8uWY7xs78y9DATgwz2wGU7M+7z55KenmmTkN2DVJ7bqzjAA==", + "dev": true, + "dependencies": { + "postcss": "^6.0.1" + } + }, + "node_modules/icss-utils/node_modules/postcss": { + "version": "6.0.23", + "resolved": "https://registry.npmmirror.com/postcss/-/postcss-6.0.23.tgz", + "integrity": "sha512-soOk1h6J3VMTZtVeVpv15/Hpdl2cBLX3CAw4TAbkpTJiNPk9YP/zWcD1ND+xEtvyuuvKzbxliTOIyvkSeSJ6ag==", + "dev": true, + "dependencies": { + "chalk": "^2.4.1", + "source-map": "^0.6.1", + "supports-color": "^5.4.0" + }, + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/icss-utils/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmmirror.com/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/ieee754": { + "version": "1.2.1", + "resolved": "https://registry.npmmirror.com/ieee754/-/ieee754-1.2.1.tgz", + "integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==" + }, + "node_modules/iferr": { + "version": "0.1.5", + "resolved": "https://registry.npmmirror.com/iferr/-/iferr-0.1.5.tgz", + "integrity": "sha512-DUNFN5j7Tln0D+TxzloUjKB+CtVu6myn0JEFak6dG18mNt9YkQ6lzGCdafwofISZ1lLF3xRHJ98VKy9ynkcFaA==" + }, + "node_modules/ignore": { + "version": "3.3.10", + "resolved": "https://registry.npmmirror.com/ignore/-/ignore-3.3.10.tgz", + "integrity": "sha512-Pgs951kaMm5GXP7MOvxERINe3gsaVjUWFm+UZPSq9xYriQAksyhg0csnS0KXSNRD5NmNdapXEpjxG49+AKh/ug==", + "dev": true + }, + "node_modules/ignore-walk": { + "version": "3.0.4", + "resolved": "https://registry.npmmirror.com/ignore-walk/-/ignore-walk-3.0.4.tgz", + "integrity": "sha512-PY6Ii8o1jMRA1z4F2hRkH/xN59ox43DavKvD3oDpfurRlOJyAHpifIwpbdv1n4jt4ov0jSpw3kQ4GhJnpBL6WQ==", + "dev": true, + "dependencies": { + "minimatch": "^3.0.4" + } + }, + "node_modules/image-size": { + "version": "0.5.5", + "resolved": "https://registry.npmmirror.com/image-size/-/image-size-0.5.5.tgz", + "integrity": "sha512-6TDAlDPZxUFCv+fuOkIoXT/V/f3Qbq8e37p+YOiYrUv3v9cc3/6x78VdfPgFVaB9dZYeLUfKgHRebpkm/oP2VQ==", + "dev": true, + "bin": { + "image-size": "bin/image-size.js" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/import-cwd": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/import-cwd/-/import-cwd-2.1.0.tgz", + "integrity": "sha512-Ew5AZzJQFqrOV5BTW3EIoHAnoie1LojZLXKcCQ/yTRyVZosBhK1x1ViYjHGf5pAFOq8ZyChZp6m/fSN7pJyZtg==", + "dev": true, + "dependencies": { + "import-from": "^2.1.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/import-fresh": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/import-fresh/-/import-fresh-2.0.0.tgz", + "integrity": "sha512-eZ5H8rcgYazHbKC3PG4ClHNykCSxtAhxSSEM+2mb+7evD2CKF5V7c0dNum7AdpDh0ZdICwZY9sRSn8f+KH96sg==", + "dev": true, + "dependencies": { + "caller-path": "^2.0.0", + "resolve-from": "^3.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/import-fresh/node_modules/caller-path": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/caller-path/-/caller-path-2.0.0.tgz", + "integrity": "sha512-MCL3sf6nCSXOwCTzvPKhN18TU7AHTvdtam8DAogxcrJ8Rjfbbg7Lgng64H9Iy+vUV6VGFClN/TyxBkAebLRR4A==", + "dev": true, + "dependencies": { + "caller-callsite": "^2.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/import-fresh/node_modules/resolve-from": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/resolve-from/-/resolve-from-3.0.0.tgz", + "integrity": "sha512-GnlH6vxLymXJNMBo7XP1fJIzBFbdYt49CuTwmB/6N53t+kMPRMFKz783LlQ4tv28XoQfMWinAJX6WCGf2IlaIw==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/import-from": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/import-from/-/import-from-2.1.0.tgz", + "integrity": "sha512-0vdnLL2wSGnhlRmzHJAg5JHjt1l2vYhzJ7tNLGbeVg0fse56tpGaH0uzH+r9Slej+BSXXEHvBKDEnVSLLE9/+w==", + "dev": true, + "dependencies": { + "resolve-from": "^3.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/import-from/node_modules/resolve-from": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/resolve-from/-/resolve-from-3.0.0.tgz", + "integrity": "sha512-GnlH6vxLymXJNMBo7XP1fJIzBFbdYt49CuTwmB/6N53t+kMPRMFKz783LlQ4tv28XoQfMWinAJX6WCGf2IlaIw==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/import-local": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/import-local/-/import-local-2.0.0.tgz", + "integrity": "sha512-b6s04m3O+s3CGSbqDIyP4R6aAwAeYlVq9+WUWep6iHa8ETRf9yei1U48C5MmfJmV9AiLYYBKPMq/W+/WRpQmCQ==", + "dev": true, + "dependencies": { + "pkg-dir": "^3.0.0", + "resolve-cwd": "^2.0.0" + }, + "bin": { + "import-local-fixture": "fixtures/cli.js" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/imurmurhash": { + "version": "0.1.4", + "resolved": "https://registry.npmmirror.com/imurmurhash/-/imurmurhash-0.1.4.tgz", + "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==", + "engines": { + "node": ">=0.8.19" + } + }, + "node_modules/indent-string": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/indent-string/-/indent-string-4.0.0.tgz", + "integrity": "sha512-EdDDZu4A2OyIK7Lr/2zG+w5jmbuk1DVBnEwREQvBzspBJkCEbRa8GxU1lghYcaGJCnRWibjDXlq779X1/y5xwg==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/indexes-of": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/indexes-of/-/indexes-of-1.0.1.tgz", + "integrity": "sha512-bup+4tap3Hympa+JBJUG7XuOsdNQ6fxt0MHyXMKuLBKn0OqsTfvUxkUrroEX1+B2VsSHvCjiIcZVxRtYa4nllA==", + "dev": true + }, + "node_modules/infer-owner": { + "version": "1.0.4", + "resolved": "https://registry.npmmirror.com/infer-owner/-/infer-owner-1.0.4.tgz", + "integrity": "sha512-IClj+Xz94+d7irH5qRyfJonOdfTzuDaifE6ZPWfx0N0+/ATZCbuTPq2prFl526urkQd90WyUKIh1DfBQ2hMz9A==" + }, + "node_modules/inflight": { + "version": "1.0.6", + "resolved": "https://registry.npmmirror.com/inflight/-/inflight-1.0.6.tgz", + "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==", + "dependencies": { + "once": "^1.3.0", + "wrappy": "1" + } + }, + "node_modules/inherits": { + "version": "2.0.4", + "resolved": "https://registry.npmmirror.com/inherits/-/inherits-2.0.4.tgz", + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" + }, + "node_modules/ini": { + "version": "1.3.8", + "resolved": "https://registry.npmmirror.com/ini/-/ini-1.3.8.tgz", + "integrity": "sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==", + "dev": true + }, + "node_modules/init-package-json": { + "version": "2.0.5", + "resolved": "https://registry.npmmirror.com/init-package-json/-/init-package-json-2.0.5.tgz", + "integrity": "sha512-u1uGAtEFu3VA6HNl/yUWw57jmKEMx8SKOxHhxjGnOFUiIlFnohKDFg4ZrPpv9wWqk44nDxGJAtqjdQFm+9XXQA==", + "dev": true, + "dependencies": { + "npm-package-arg": "^8.1.5", + "promzard": "^0.3.0", + "read": "~1.0.1", + "read-package-json": "^4.1.1", + "semver": "^7.3.5", + "validate-npm-package-license": "^3.0.4", + "validate-npm-package-name": "^3.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/init-package-json/node_modules/hosted-git-info": { + "version": "4.1.0", + "resolved": "https://registry.npmmirror.com/hosted-git-info/-/hosted-git-info-4.1.0.tgz", + "integrity": "sha512-kyCuEOWjJqZuDbRHzL8V93NzQhwIB71oFWSyzVo+KPZI+pnQPPxucdkrOZvkLRnrf5URsQM+IJ09Dw29cRALIA==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/init-package-json/node_modules/lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/init-package-json/node_modules/normalize-package-data": { + "version": "3.0.3", + "resolved": "https://registry.npmmirror.com/normalize-package-data/-/normalize-package-data-3.0.3.tgz", + "integrity": "sha512-p2W1sgqij3zMMyRC067Dg16bfzVH+w7hyegmpIvZ4JNjqtGOVAIvLmjBx3yP7YTe9vKJgkoNOPjwQGogDoMXFA==", + "dev": true, + "dependencies": { + "hosted-git-info": "^4.0.1", + "is-core-module": "^2.5.0", + "semver": "^7.3.4", + "validate-npm-package-license": "^3.0.1" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/init-package-json/node_modules/read-package-json": { + "version": "4.1.2", + "resolved": "https://registry.npmmirror.com/read-package-json/-/read-package-json-4.1.2.tgz", + "integrity": "sha512-Dqer4pqzamDE2O4M55xp1qZMuLPqi4ldk2ya648FOMHRjwMzFhuxVrG04wd0c38IsvkVdr3vgHI6z+QTPdAjrQ==", + "dev": true, + "dependencies": { + "glob": "^7.1.1", + "json-parse-even-better-errors": "^2.3.0", + "normalize-package-data": "^3.0.0", + "npm-normalize-package-bin": "^1.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/init-package-json/node_modules/semver": { + "version": "7.3.7", + "resolved": "https://registry.npmmirror.com/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/init-package-json/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/inquirer": { + "version": "3.3.0", + "resolved": "https://registry.npmmirror.com/inquirer/-/inquirer-3.3.0.tgz", + "integrity": "sha512-h+xtnyk4EwKvFWHrUYsWErEVR+igKtLdchu+o0Z1RL7VU/jVMFbYir2bp6bAj8efFNxWqHX0dIss6fJQ+/+qeQ==", + "dev": true, + "optional": true, + "dependencies": { + "ansi-escapes": "^3.0.0", + "chalk": "^2.0.0", + "cli-cursor": "^2.1.0", + "cli-width": "^2.0.0", + "external-editor": "^2.0.4", + "figures": "^2.0.0", + "lodash": "^4.3.0", + "mute-stream": "0.0.7", + "run-async": "^2.2.0", + "rx-lite": "^4.0.8", + "rx-lite-aggregates": "^4.0.8", + "string-width": "^2.1.0", + "strip-ansi": "^4.0.0", + "through": "^2.3.6" + } + }, + "node_modules/inquirer/node_modules/ansi-regex": { + "version": "3.0.1", + "resolved": "https://registry.npmmirror.com/ansi-regex/-/ansi-regex-3.0.1.tgz", + "integrity": "sha512-+O9Jct8wf++lXxxFc4hc8LsjaSq0HFzzL7cVsw8pRDIPdjKD2mT4ytDZlLuSBZ4cLKZFXIrMGO7DbQCtMJJMKw==", + "dev": true, + "optional": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/inquirer/node_modules/strip-ansi": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/strip-ansi/-/strip-ansi-4.0.0.tgz", + "integrity": "sha512-4XaJ2zQdCzROZDivEVIDPkcQn8LMFSa8kj8Gxb/Lnwzv9A8VctNZ+lfivC/sV3ivW8ElJTERXZoPBRrZKkNKow==", + "dev": true, + "optional": true, + "dependencies": { + "ansi-regex": "^3.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/internal-ip": { + "version": "4.3.0", + "resolved": "https://registry.npmmirror.com/internal-ip/-/internal-ip-4.3.0.tgz", + "integrity": "sha512-S1zBo1D6zcsyuC6PMmY5+55YMILQ9av8lotMx447Bq6SAgo/sDK6y6uUKmuYhW7eacnIhFfsPmCNYdDzsnnDCg==", + "dev": true, + "dependencies": { + "default-gateway": "^4.2.0", + "ipaddr.js": "^1.9.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/internal-ip/node_modules/default-gateway": { + "version": "4.2.0", + "resolved": "https://registry.npmmirror.com/default-gateway/-/default-gateway-4.2.0.tgz", + "integrity": "sha512-h6sMrVB1VMWVrW13mSc6ia/DwYYw5MN6+exNu1OaJeFac5aSAvwM7lZ0NVfTABuSkQelr4h5oebg3KB1XPdjgA==", + "dev": true, + "dependencies": { + "execa": "^1.0.0", + "ip-regex": "^2.1.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/internal-slot": { + "version": "1.0.3", + "resolved": "https://registry.npmmirror.com/internal-slot/-/internal-slot-1.0.3.tgz", + "integrity": "sha512-O0DB1JC/sPyZl7cIo78n5dR7eUSwwpYPiXRhTzNxZVAMUuB8vlnRFyLxdrVToks6XPLVnFfbzaVd5WLjhgg+vA==", + "dependencies": { + "get-intrinsic": "^1.1.0", + "has": "^1.0.3", + "side-channel": "^1.0.4" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/invariant": { + "version": "2.2.4", + "resolved": "https://registry.npmmirror.com/invariant/-/invariant-2.2.4.tgz", + "integrity": "sha512-phJfQVBuaJM5raOpJjSfkiD6BpbCE4Ns//LaXl6wGYtUBY83nWS6Rf9tXm2e8VaK60JEjYldbPif/A2B1C2gNA==", + "dev": true, + "dependencies": { + "loose-envify": "^1.0.0" + } + }, + "node_modules/ip": { + "version": "1.1.5", + "resolved": "https://registry.npmmirror.com/ip/-/ip-1.1.5.tgz", + "integrity": "sha512-rBtCAQAJm8A110nbwn6YdveUnuZH3WrC36IwkRXxDnq53JvXA2NVQvB7IHyKomxK1MJ4VDNw3UtFDdXQ+AvLYA==", + "dev": true + }, + "node_modules/ip-regex": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/ip-regex/-/ip-regex-2.1.0.tgz", + "integrity": "sha512-58yWmlHpp7VYfcdTwMTvwMmqx/Elfxjd9RXTDyMsbL7lLWmhMylLEqiYVLKuLzOZqVgiWXD9MfR62Vv89VRxkw==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/ipaddr.js": { + "version": "1.9.1", + "resolved": "https://registry.npmmirror.com/ipaddr.js/-/ipaddr.js-1.9.1.tgz", + "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==", + "dev": true, + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/is-absolute-url": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/is-absolute-url/-/is-absolute-url-2.1.0.tgz", + "integrity": "sha512-vOx7VprsKyllwjSkLV79NIhpyLfr3jAp7VaTCMXOJHu4m0Ew1CZ2fcjASwmV1jI3BWuWHB013M48eyeldk9gYg==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-accessor-descriptor": { + "version": "0.1.6", + "resolved": "https://registry.npmmirror.com/is-accessor-descriptor/-/is-accessor-descriptor-0.1.6.tgz", + "integrity": "sha512-e1BM1qnDbMRG3ll2U9dSK0UMHuWOs3pY3AtcFsmvwPtKL3MML/Q86i+GilLfvqEs4GW+ExB91tQ3Ig9noDIZ+A==", + "dependencies": { + "kind-of": "^3.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-accessor-descriptor/node_modules/kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmmirror.com/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha512-NOW9QQXMoZGg/oqnVNoNTTIFEIid1627WCffUBJEdMxYApq7mNE7CpzucIPc+ZQg25Phej7IJSmX3hO+oblOtQ==", + "dependencies": { + "is-buffer": "^1.1.5" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-any-array": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/is-any-array/-/is-any-array-2.0.0.tgz", + "integrity": "sha512-WdPV58rT3aOWXvvyuBydnCq4S2BM1Yz8shKxlEpk/6x+GX202XRvXOycEFtNgnHVLoc46hpexPFx8Pz1/sMS0w==" + }, + "node_modules/is-arguments": { + "version": "1.1.1", + "resolved": "https://registry.npmmirror.com/is-arguments/-/is-arguments-1.1.1.tgz", + "integrity": "sha512-8Q7EARjzEnKpt/PCD7e1cgUS0a6X8u5tdSiMqXhojOdoV9TsMsiO+9VLC5vAmO8N7/GmXn7yjR8qnA6bVAEzfA==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.2", + "has-tostringtag": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/is-arrayish": { + "version": "0.2.1", + "resolved": "https://registry.npmmirror.com/is-arrayish/-/is-arrayish-0.2.1.tgz", + "integrity": "sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg==", + "dev": true + }, + "node_modules/is-bigint": { + "version": "1.0.4", + "resolved": "https://registry.npmmirror.com/is-bigint/-/is-bigint-1.0.4.tgz", + "integrity": "sha512-zB9CruMamjym81i2JZ3UMn54PKGsQzsJeo6xvN3HJJ4CAsQNB6iRutp2To77OfCNuoxspsIhzaPoO1zyCEhFOg==", + "dependencies": { + "has-bigints": "^1.0.1" + } + }, + "node_modules/is-binary-path": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/is-binary-path/-/is-binary-path-2.1.0.tgz", + "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", + "optional": true, + "dependencies": { + "binary-extensions": "^2.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/is-boolean-object": { + "version": "1.1.2", + "resolved": "https://registry.npmmirror.com/is-boolean-object/-/is-boolean-object-1.1.2.tgz", + "integrity": "sha512-gDYaKHJmnj4aWxyj6YHyXVpdQawtVLHU5cb+eztPGczf6cjuTdwve5ZIEfgXqH4e57An1D1AKf8CZ3kYrQRqYA==", + "dependencies": { + "call-bind": "^1.0.2", + "has-tostringtag": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/is-buffer": { + "version": "1.1.6", + "resolved": "https://registry.npmmirror.com/is-buffer/-/is-buffer-1.1.6.tgz", + "integrity": "sha512-NcdALwpXkTm5Zvvbk7owOUSvVvBKDgKP5/ewfXEznmQFfs4ZRmanOeKBTjRVjka3QFoN6XJ+9F3USqfHqTaU5w==" + }, + "node_modules/is-callable": { + "version": "1.2.4", + "resolved": "https://registry.npmmirror.com/is-callable/-/is-callable-1.2.4.tgz", + "integrity": "sha512-nsuwtxZfMX67Oryl9LCQ+upnC0Z0BgpwntpS89m1H/TLF0zNfzfLMV/9Wa/6MZsj0acpEjAO0KF1xT6ZdLl95w==", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/is-ci": { + "version": "1.2.1", + "resolved": "https://registry.npmmirror.com/is-ci/-/is-ci-1.2.1.tgz", + "integrity": "sha512-s6tfsaQaQi3JNciBH6shVqEDvhGut0SUXr31ag8Pd8BBbVVlcGfWhpPmEOoM6RJ5TFhbypvf5yyRw/VXW1IiWg==", + "dev": true, + "dependencies": { + "ci-info": "^1.5.0" + }, + "bin": { + "is-ci": "bin.js" + } + }, + "node_modules/is-color-stop": { + "version": "1.1.0", + "resolved": "https://registry.npmmirror.com/is-color-stop/-/is-color-stop-1.1.0.tgz", + "integrity": "sha512-H1U8Vz0cfXNujrJzEcvvwMDW9Ra+biSYA3ThdQvAnMLJkEHQXn6bWzLkxHtVYJ+Sdbx0b6finn3jZiaVe7MAHA==", + "dev": true, + "dependencies": { + "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" + } + }, + "node_modules/is-core-module": { + "version": "2.9.0", + "resolved": "https://registry.npmmirror.com/is-core-module/-/is-core-module-2.9.0.tgz", + "integrity": "sha512-+5FPy5PnwmO3lvfMb0AsoPaBG+5KHUI0wYFXOtYPnVVVspTFUuMZNfNaNVRt3FZadstu2c8x23vykRW/NBoU6A==", + "dev": true, + "dependencies": { + "has": "^1.0.3" + } + }, + "node_modules/is-data-descriptor": { + "version": "0.1.4", + "resolved": "https://registry.npmmirror.com/is-data-descriptor/-/is-data-descriptor-0.1.4.tgz", + "integrity": "sha512-+w9D5ulSoBNlmw9OHn3U2v51SyoCd0he+bB3xMl62oijhrspxowjU+AIcDY0N3iEJbUEkB15IlMASQsxYigvXg==", + "dependencies": { + "kind-of": "^3.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-data-descriptor/node_modules/kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmmirror.com/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha512-NOW9QQXMoZGg/oqnVNoNTTIFEIid1627WCffUBJEdMxYApq7mNE7CpzucIPc+ZQg25Phej7IJSmX3hO+oblOtQ==", + "dependencies": { + "is-buffer": "^1.1.5" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-date-object": { + "version": "1.0.5", + "resolved": "https://registry.npmmirror.com/is-date-object/-/is-date-object-1.0.5.tgz", + "integrity": "sha512-9YQaSxsAiSwcvS33MBk3wTCVnWK+HhF8VZR2jRxehM16QcVOdHqPn4VPHmRK4lSr38n9JriurInLcP90xsYNfQ==", + "dependencies": { + "has-tostringtag": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/is-descriptor": { + "version": "0.1.6", + "resolved": "https://registry.npmmirror.com/is-descriptor/-/is-descriptor-0.1.6.tgz", + "integrity": "sha512-avDYr0SB3DwO9zsMov0gKCESFYqCnE4hq/4z3TdUlukEy5t9C0YRq7HLrsN52NAcqXKaepeCD0n+B0arnVG3Hg==", + "dependencies": { + "is-accessor-descriptor": "^0.1.6", + "is-data-descriptor": "^0.1.4", + "kind-of": "^5.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-descriptor/node_modules/kind-of": { + "version": "5.1.0", + "resolved": "https://registry.npmmirror.com/kind-of/-/kind-of-5.1.0.tgz", + "integrity": "sha512-NGEErnH6F2vUuXDh+OlbcKW7/wOcfdRHaZ7VWtqCztfHri/++YKmP51OdWeGPuqCOba6kk2OTe5d02VmTB80Pw==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-directory": { + "version": "0.3.1", + "resolved": "https://registry.npmmirror.com/is-directory/-/is-directory-0.3.1.tgz", + "integrity": "sha512-yVChGzahRFvbkscn2MlwGismPO12i9+znNruC5gVEntG3qu0xQMzsGg/JFbrsqDOHtHFPci+V5aP5T9I+yeKqw==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-docker": { + "version": "2.2.1", + "resolved": "https://registry.npmmirror.com/is-docker/-/is-docker-2.2.1.tgz", + "integrity": "sha512-F+i2BKsFrH66iaUFc0woD8sLy8getkwTwtOBjvs56Cx4CgJDeKQeqfz8wAYiSb8JOprWhHH5p77PbmYCvvUuXQ==", + "dev": true, + "bin": { + "is-docker": "cli.js" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/is-dotfile": { + "version": "1.0.3", + "resolved": "https://registry.npmmirror.com/is-dotfile/-/is-dotfile-1.0.3.tgz", + "integrity": "sha512-9YclgOGtN/f8zx0Pr4FQYMdibBiTaH3sn52vjYip4ZSf6C4/6RfTEZ+MR4GvKhCxdPh21Bg42/WL55f6KSnKpg==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-equal-shallow": { + "version": "0.1.3", + "resolved": "https://registry.npmmirror.com/is-equal-shallow/-/is-equal-shallow-0.1.3.tgz", + "integrity": "sha512-0EygVC5qPvIyb+gSz7zdD5/AAoS6Qrx1e//6N4yv4oNm30kqvdmG66oZFWVlQHUWe5OjP08FuTw2IdT0EOTcYA==", + "dev": true, + "dependencies": { + "is-primitive": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-extendable": { + "version": "0.1.1", + "resolved": "https://registry.npmmirror.com/is-extendable/-/is-extendable-0.1.1.tgz", + "integrity": "sha512-5BMULNob1vgFX6EjQw5izWDxrecWK9AM72rugNr0TFldMOi0fj6Jk+zeKIt0xGj4cEfQIJth4w3OKWOJ4f+AFw==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmmirror.com/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", + "devOptional": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-fullwidth-code-point": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", + "integrity": "sha512-VHskAKYM8RfSFXwee5t5cbN5PZeq1Wrh6qd5bkyiXIf6UQcN6w/A0eXM9r6t8d+GYOh+o6ZhiEnb88LN/Y8m2w==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/is-glob": { + "version": "4.0.3", + "resolved": "https://registry.npmmirror.com/is-glob/-/is-glob-4.0.3.tgz", + "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", + "devOptional": true, + "dependencies": { + "is-extglob": "^2.1.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-lambda": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/is-lambda/-/is-lambda-1.0.1.tgz", + "integrity": "sha512-z7CMFGNrENq5iFB9Bqo64Xk6Y9sg+epq1myIcdHaGnbMTYOxvzsEtdYqQUylB7LxfkvgrrjP32T6Ywciio9UIQ==", + "dev": true + }, + "node_modules/is-negative-zero": { + "version": "2.0.2", + "resolved": "https://registry.npmmirror.com/is-negative-zero/-/is-negative-zero-2.0.2.tgz", + "integrity": "sha512-dqJvarLawXsFbNDeJW7zAz8ItJ9cd28YufuuFzh0G8pNHjJMnY08Dv7sYX2uF5UpQOwieAeOExEYAWWfu7ZZUA==", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/is-number": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/is-number/-/is-number-3.0.0.tgz", + "integrity": "sha512-4cboCqIpliH+mAvFNegjZQ4kgKc3ZUhQVr3HvWbSh5q3WH2v82ct+T2Y1hdU5Gdtorx/cLifQjqCbL7bpznLTg==", + "dependencies": { + "kind-of": "^3.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-number-object": { + "version": "1.0.7", + "resolved": "https://registry.npmmirror.com/is-number-object/-/is-number-object-1.0.7.tgz", + "integrity": "sha512-k1U0IRzLMo7ZlYIfzRu23Oh6MiIFasgpb9X76eqfFZAqwH44UI4KTBvBYIZ1dSL9ZzChTB9ShHfLkR4pdW5krQ==", + "dependencies": { + "has-tostringtag": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/is-number/node_modules/kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmmirror.com/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha512-NOW9QQXMoZGg/oqnVNoNTTIFEIid1627WCffUBJEdMxYApq7mNE7CpzucIPc+ZQg25Phej7IJSmX3hO+oblOtQ==", + "dependencies": { + "is-buffer": "^1.1.5" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-obj": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/is-obj/-/is-obj-2.0.0.tgz", + "integrity": "sha512-drqDG3cbczxxEJRoOXcOjtdp1J/lyp1mNn0xaznRs8+muBhgQcrnbspox5X5fOw0HnMnbfDzvnEMEtqDEJEo8w==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/is-observable": { + "version": "1.1.0", + "resolved": "https://registry.npmmirror.com/is-observable/-/is-observable-1.1.0.tgz", + "integrity": "sha512-NqCa4Sa2d+u7BWc6CukaObG3Fh+CU9bvixbpcXYhy2VvYS7vVGIdAgnIS5Ks3A/cqk4rebLJ9s8zBstT2aKnIA==", + "dev": true, + "dependencies": { + "symbol-observable": "^1.1.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/is-path-cwd": { + "version": "2.2.0", + "resolved": "https://registry.npmmirror.com/is-path-cwd/-/is-path-cwd-2.2.0.tgz", + "integrity": "sha512-w942bTcih8fdJPJmQHFzkS76NEP8Kzzvmw92cXsazb8intwLqPibPPdXf4ANdKV3rYMuuQYGIWtvz9JilB3NFQ==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/is-path-in-cwd": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/is-path-in-cwd/-/is-path-in-cwd-2.1.0.tgz", + "integrity": "sha512-rNocXHgipO+rvnP6dk3zI20RpOtrAM/kzbB258Uw5BWr3TpXi861yzjo16Dn4hUox07iw5AyeMLHWsujkjzvRQ==", + "dev": true, + "dependencies": { + "is-path-inside": "^2.1.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/is-path-inside": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/is-path-inside/-/is-path-inside-2.1.0.tgz", + "integrity": "sha512-wiyhTzfDWsvwAW53OBWF5zuvaOGlZ6PwYxAbPVDhpm+gM09xKQGjBq/8uYN12aDvMxnAnq3dxTyoSoRNmg5YFg==", + "dev": true, + "dependencies": { + "path-is-inside": "^1.0.2" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/is-plain-obj": { + "version": "1.1.0", + "resolved": "https://registry.npmmirror.com/is-plain-obj/-/is-plain-obj-1.1.0.tgz", + "integrity": "sha512-yvkRyxmFKEOQ4pNXCmJG5AEQNlXJS5LaONXo5/cLdTZdWvsZ1ioJEonLGAosKlMWE8lwUy/bJzMjcw8az73+Fg==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-plain-object": { + "version": "2.0.4", + "resolved": "https://registry.npmmirror.com/is-plain-object/-/is-plain-object-2.0.4.tgz", + "integrity": "sha512-h5PpgXkWitc38BBMYawTYMWJHFZJVnBquFE57xFpjB8pJFiF6gZ+bU+WyI/yqXiFR5mdLsgYNaPe8uao6Uv9Og==", + "dependencies": { + "isobject": "^3.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-posix-bracket": { + "version": "0.1.1", + "resolved": "https://registry.npmmirror.com/is-posix-bracket/-/is-posix-bracket-0.1.1.tgz", + "integrity": "sha512-Yu68oeXJ7LeWNmZ3Zov/xg/oDBnBK2RNxwYY1ilNJX+tKKZqgPK+qOn/Gs9jEu66KDY9Netf5XLKNGzas/vPfQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-primitive": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/is-primitive/-/is-primitive-2.0.0.tgz", + "integrity": "sha512-N3w1tFaRfk3UrPfqeRyD+GYDASU3W5VinKhlORy8EWVf/sIdDL9GAcew85XmktCfH+ngG7SRXEVDoO18WMdB/Q==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-promise": { + "version": "2.2.2", + "resolved": "https://registry.npmmirror.com/is-promise/-/is-promise-2.2.2.tgz", + "integrity": "sha512-+lP4/6lKUBfQjZ2pdxThZvLUAafmZb8OAxFb8XXtiQmS35INgr85hdOGoEs124ez1FCnZJt6jau/T+alh58QFQ==" + }, + "node_modules/is-regex": { + "version": "1.1.4", + "resolved": "https://registry.npmmirror.com/is-regex/-/is-regex-1.1.4.tgz", + "integrity": "sha512-kvRdxDsxZjhzUX07ZnLydzS1TU/TJlTUHHY4YLL87e37oUA49DfkLqgy+VjFocowy29cKvcSiu+kIv728jTTVg==", + "dependencies": { + "call-bind": "^1.0.2", + "has-tostringtag": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/is-regexp": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/is-regexp/-/is-regexp-1.0.0.tgz", + "integrity": "sha512-7zjFAPO4/gwyQAAgRRmqeEeyIICSdmCqa3tsVHMdBzaXXRiqopZL4Cyghg/XulGWrtABTpbnYYzzIRffLkP4oA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-resolvable": { + "version": "1.1.0", + "resolved": "https://registry.npmmirror.com/is-resolvable/-/is-resolvable-1.1.0.tgz", + "integrity": "sha512-qgDYXFSR5WvEfuS5dMj6oTMEbrrSaM0CrFk2Yiq/gXnBvD9pMa2jGXxyhGLfvhZpuMZe18CJpFxAt3CRs42NMg==", + "dev": true + }, + "node_modules/is-shared-array-buffer": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/is-shared-array-buffer/-/is-shared-array-buffer-1.0.2.tgz", + "integrity": "sha512-sqN2UDu1/0y6uvXyStCOzyhAjCSlHceFoMKJW8W9EU9cvic/QdsZ0kEU93HEy3IUEFZIiH/3w+AH/UQbPHNdhA==", + "dependencies": { + "call-bind": "^1.0.2" + } + }, + "node_modules/is-ssh": { + "version": "1.3.3", + "resolved": "https://registry.npmmirror.com/is-ssh/-/is-ssh-1.3.3.tgz", + "integrity": "sha512-NKzJmQzJfEEma3w5cJNcUMxoXfDjz0Zj0eyCalHn2E6VOwlzjZo0yuO2fcBSf8zhFuVCL/82/r5gRcoi6aEPVQ==", + "dev": true, + "dependencies": { + "protocols": "^1.1.0" + } + }, + "node_modules/is-stream": { + "version": "1.1.0", + "resolved": "https://registry.npmmirror.com/is-stream/-/is-stream-1.1.0.tgz", + "integrity": "sha512-uQPm8kcs47jx38atAcWTVxyltQYoPT68y9aWYdV6yWXSyW8mzSat0TL6CiWdZeCdF3KrAvpVtnHbTv4RN+rqdQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-string": { + "version": "1.0.7", + "resolved": "https://registry.npmmirror.com/is-string/-/is-string-1.0.7.tgz", + "integrity": "sha512-tE2UXzivje6ofPW7l23cjDOMa09gb7xlAqG6jG5ej6uPV32TlWP3NKPigtaGeHNu9fohccRYvIiZMfOOnOYUtg==", + "dependencies": { + "has-tostringtag": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/is-symbol": { + "version": "1.0.4", + "resolved": "https://registry.npmmirror.com/is-symbol/-/is-symbol-1.0.4.tgz", + "integrity": "sha512-C/CPBqKWnvdcxqIARxyOh4v1UUEOCHpgDa0WYgpKDFMszcrPcffg5uhwSgPCLD2WWxmq6isisz87tzT01tuGhg==", + "dependencies": { + "has-symbols": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/is-text-path": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/is-text-path/-/is-text-path-1.0.1.tgz", + "integrity": "sha512-xFuJpne9oFz5qDaodwmmG08e3CawH/2ZV8Qqza1Ko7Sk8POWbkRdwIoAWVhqvq0XeUzANEhKo2n0IXUGBm7A/w==", + "dev": true, + "dependencies": { + "text-extensions": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-typedarray": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/is-typedarray/-/is-typedarray-1.0.0.tgz", + "integrity": "sha512-cyA56iCMHAh5CdzjJIa4aohJyeO1YbwLi3Jc35MmRU6poroFjIGZzUzupGiRPOjgHg9TLu43xbpwXk523fMxKA==", + "dev": true + }, + "node_modules/is-weakref": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/is-weakref/-/is-weakref-1.0.2.tgz", + "integrity": "sha512-qctsuLZmIQ0+vSSMfoVvyFe2+GSEvnmZ2ezTup1SBse9+twCCeial6EEi3Nc2KFcf6+qz2FBPnjXsk8xhKSaPQ==", + "dependencies": { + "call-bind": "^1.0.2" + } + }, + "node_modules/is-what": { + "version": "3.14.1", + "resolved": "https://registry.npmmirror.com/is-what/-/is-what-3.14.1.tgz", + "integrity": "sha512-sNxgpk9793nzSs7bA6JQJGeIuRBQhAaNGG77kzYQgMkrID+lS6SlK07K5LaptscDlSaIgH+GPFzf+d75FVxozA==", + "dev": true + }, + "node_modules/is-windows": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/is-windows/-/is-windows-1.0.2.tgz", + "integrity": "sha512-eXK1UInq2bPmjyX6e3VHIzMLobc4J94i4AWn+Hpq3OU5KkrRC96OAcR3PRJ/pGu6m8TRnBHP9dkXQVsT/COVIA==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-wsl": { + "version": "1.1.0", + "resolved": "https://registry.npmmirror.com/is-wsl/-/is-wsl-1.1.0.tgz", + "integrity": "sha512-gfygJYZ2gLTDlmbWMI0CE2MwnFzSN/2SZfkMlItC4K/JBlsWVDB0bO6XhqcY13YXE7iMcAJnzTCJjPiTeJJ0Mw==", + "engines": { + "node": ">=4" + } + }, + "node_modules/isarray": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/isarray/-/isarray-1.0.0.tgz", + "integrity": "sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ==" + }, + "node_modules/isexe": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", + "dev": true + }, + "node_modules/isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmmirror.com/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha512-WhB9zCku7EGTj/HQQRz5aUQEUeoQZH2bWcltRErOpymJ4boYE6wL9Tbr23krRPSZ+C5zqNSrSw+Cc7sZZ4b7vg==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/isstream": { + "version": "0.1.2", + "resolved": "https://registry.npmmirror.com/isstream/-/isstream-0.1.2.tgz", + "integrity": "sha512-Yljz7ffyPbrLpLngrMtZ7NduUgVvi6wG9RJ9IUcyCd59YQ911PBJphODUcbOVbqYfxe1wuYf/LJ8PauMRwsM/g==", + "dev": true + }, + "node_modules/iview": { + "version": "3.5.4", + "resolved": "https://registry.npmmirror.com/iview/-/iview-3.5.4.tgz", + "integrity": "sha512-CEDHdAXxpGciMV+m1jdMDs0UVqzk/AaFhCDtSGcKDLidXRMOG2TgCZhtEHhI4mf3lchKEUFO/BcXtHAfBNuy7Q==", + "dependencies": { + "async-validator": "^1.12.2", + "deepmerge": "^2.2.1", + "element-resize-detector": "^1.2.0", + "js-calendar": "^1.2.3", + "lodash.throttle": "^4.1.1", + "popper.js": "^1.14.6", + "tinycolor2": "^1.4.1", + "v-click-outside-x": "^4.0.19" + }, + "engines": { + "node": ">=8.9.1", + "npm": ">=5.5.1", + "yarn": ">=1.3.2" + }, + "peerDependencies": { + "vue": "^2.5.2" + } + }, + "node_modules/javascript-stringify": { + "version": "1.6.0", + "resolved": "https://registry.npmmirror.com/javascript-stringify/-/javascript-stringify-1.6.0.tgz", + "integrity": "sha512-fnjC0up+0SjEJtgmmG+teeel68kutkvzfctO/KxE3qJlbunkJYAshgH3boU++gSBHP8z5/r0ts0qRIrHf0RTQQ==", + "dev": true + }, + "node_modules/jquery": { + "version": "3.6.0", + "resolved": "https://registry.npmmirror.com/jquery/-/jquery-3.6.0.tgz", + "integrity": "sha512-JVzAR/AjBvVt2BmYhxRCSYysDsPcssdmTFnzyLEts9qNwmjmu4JTAMYubEfwVOSwpQ1I1sKKFcxhZCI2buerfw==" + }, + "node_modules/js-base64": { + "version": "2.6.4", + "resolved": "https://registry.npmmirror.com/js-base64/-/js-base64-2.6.4.tgz", + "integrity": "sha512-pZe//GGmwJndub7ZghVHz7vjb2LgC1m8B07Au3eYqeqv9emhESByMXxaEgkUkEqJe87oBbSniGYoQNIBklc7IQ==", + "dev": true + }, + "node_modules/js-calendar": { + "version": "1.2.3", + "resolved": "https://registry.npmmirror.com/js-calendar/-/js-calendar-1.2.3.tgz", + "integrity": "sha512-dAA1/Zbp4+c5E+ARCVTIuKepXsNLzSYfzvOimiYD4S5eeP9QuplSHLcdhfqFSwyM1o1u6ku6RRRCyaZ0YAjiBw==" + }, + "node_modules/js-levenshtein": { + "version": "1.1.6", + "resolved": "https://registry.npmmirror.com/js-levenshtein/-/js-levenshtein-1.1.6.tgz", + "integrity": "sha512-X2BB11YZtrRqY4EnQcLX5Rh373zbK4alC1FW7D7MBhL2gtcC17cTnr6DmfHZeS0s2rTHjUTMMHfG7gO8SSdw+g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/js-message": { + "version": "1.0.7", + "resolved": "https://registry.npmmirror.com/js-message/-/js-message-1.0.7.tgz", + "integrity": "sha512-efJLHhLjIyKRewNS9EGZ4UpI8NguuL6fKkhRxVuMmrGV2xN/0APGdQYwLFky5w9naebSZ0OwAGp0G6/2Cg90rA==", + "dev": true, + "engines": { + "node": ">=0.6.0" + } + }, + "node_modules/js-queue": { + "version": "2.0.2", + "resolved": "https://registry.npmmirror.com/js-queue/-/js-queue-2.0.2.tgz", + "integrity": "sha512-pbKLsbCfi7kriM3s1J4DDCo7jQkI58zPLHi0heXPzPlj0hjUsm+FesPUbE0DSbIVIK503A36aUBoCN7eMFedkA==", + "dev": true, + "dependencies": { + "easy-stack": "^1.0.1" + }, + "engines": { + "node": ">=1.0.0" + } + }, + "node_modules/js-tokens": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/js-tokens/-/js-tokens-4.0.0.tgz", + "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", + "dev": true + }, + "node_modules/js-yaml": { + "version": "3.14.1", + "resolved": "https://registry.npmmirror.com/js-yaml/-/js-yaml-3.14.1.tgz", + "integrity": "sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==", + "dependencies": { + "argparse": "^1.0.7", + "esprima": "^4.0.0" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, + "node_modules/jsbn": { + "version": "0.1.1", + "resolved": "https://registry.npmmirror.com/jsbn/-/jsbn-0.1.1.tgz", + "integrity": "sha512-UVU9dibq2JcFWxQPA6KCqj5O42VOmAY3zQUfEKxU0KpTGXwNoCjkX1e13eHNvw/xPynt6pU0rZ1htjWTNTSXsg==", + "dev": true + }, + "node_modules/jsencrypt": { + "version": "3.2.1", + "resolved": "https://registry.npmmirror.com/jsencrypt/-/jsencrypt-3.2.1.tgz", + "integrity": "sha512-k1sD5QV0KPn+D8uG9AdGzTQuamt82QZ3A3l6f7TRwMU6Oi2Vg0BsL+wZIQBONcraO1pc78ExMdvmBBJ8WhNYUA==" + }, + "node_modules/jsesc": { + "version": "2.5.2", + "resolved": "https://registry.npmmirror.com/jsesc/-/jsesc-2.5.2.tgz", + "integrity": "sha512-OYu7XEzjkCQ3C5Ps3QIZsQfNpqoJyZZA99wd9aWd05NCtC5pWOkShK2mkL6HXQR6/Cy2lbNdPlZBpuQHXE63gA==", + "dev": true, + "bin": { + "jsesc": "bin/jsesc" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/json-parse-better-errors": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/json-parse-better-errors/-/json-parse-better-errors-1.0.2.tgz", + "integrity": "sha512-mrqyZKfX5EhL7hvqcV6WG1yYjnjeuYDzDhhcAAUrq8Po85NBQBJP+ZDUT75qZQ98IkUoBqdkExkukOU7Ts2wrw==" + }, + "node_modules/json-parse-even-better-errors": { + "version": "2.3.1", + "resolved": "https://registry.npmmirror.com/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz", + "integrity": "sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==", + "dev": true + }, + "node_modules/json-schema": { + "version": "0.4.0", + "resolved": "https://registry.npmmirror.com/json-schema/-/json-schema-0.4.0.tgz", + "integrity": "sha512-es94M3nTIfsEPisRafak+HDLfHXnKBhV3vU5eqPcS3flIWqcxJWgXHXiey3YrpaNsanY5ei1VoYEbOzijuq9BA==", + "dev": true + }, + "node_modules/json-schema-traverse": { + "version": "0.4.1", + "resolved": "https://registry.npmmirror.com/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", + "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==" + }, + "node_modules/json-stable-stringify-without-jsonify": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", + "integrity": "sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==", + "dev": true + }, + "node_modules/json-stringify-safe": { + "version": "5.0.1", + "resolved": "https://registry.npmmirror.com/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz", + "integrity": "sha512-ZClg6AaYvamvYEE82d3Iyd3vSSIjQ+odgjaTzRuO3s7toCdFKczob2i0zCh7JE8kWn17yvAWhUVxvqGwUalsRA==", + "dev": true + }, + "node_modules/json5": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/json5/-/json5-1.0.1.tgz", + "integrity": "sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow==", + "dependencies": { + "minimist": "^1.2.0" + }, + "bin": { + "json5": "lib/cli.js" + } + }, + "node_modules/jsonfile": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/jsonfile/-/jsonfile-4.0.0.tgz", + "integrity": "sha512-m6F1R3z8jjlf2imQHS2Qez5sjKWQzbuuhuJ/FKYFRZvPE3PuHcSMVZzfsLhGVOkfd20obL5SWEBew5ShlquNxg==", + "dev": true, + "optionalDependencies": { + "graceful-fs": "^4.1.6" + } + }, + "node_modules/jsonparse": { + "version": "1.3.1", + "resolved": "https://registry.npmmirror.com/jsonparse/-/jsonparse-1.3.1.tgz", + "integrity": "sha512-POQXvpdL69+CluYsillJ7SUhKvytYjW9vG/GKpnf+xP8UWgYEM/RaMzHHofbALDiKbbP1W8UEYmgGl39WkPZsg==", + "dev": true, + "engines": [ + "node >= 0.2.0" + ] + }, + "node_modules/JSONStream": { + "version": "1.3.5", + "resolved": "https://registry.npmmirror.com/JSONStream/-/JSONStream-1.3.5.tgz", + "integrity": "sha512-E+iruNOY8VV9s4JEbe1aNEm6MiszPRr/UfcHMz0TQh1BXSxHK+ASV1R6W4HpjBhSeS+54PIsAMCBmwD06LLsqQ==", + "dev": true, + "dependencies": { + "jsonparse": "^1.2.0", + "through": ">=2.2.7 <3" + }, + "bin": { + "JSONStream": "bin.js" + }, + "engines": { + "node": "*" + } + }, + "node_modules/jsprim": { + "version": "1.4.2", + "resolved": "https://registry.npmmirror.com/jsprim/-/jsprim-1.4.2.tgz", + "integrity": "sha512-P2bSOMAc/ciLz6DzgjVlGJP9+BrJWu5UDGK70C2iweC5QBIeFf0ZXRvGjEj2uYgrY2MkAAhsSWHDWlFtEroZWw==", + "dev": true, + "dependencies": { + "assert-plus": "1.0.0", + "extsprintf": "1.3.0", + "json-schema": "0.4.0", + "verror": "1.10.0" + }, + "engines": { + "node": ">=0.6.0" + } + }, + "node_modules/killable": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/killable/-/killable-1.0.1.tgz", + "integrity": "sha512-LzqtLKlUwirEUyl/nicirVmNiPvYs7l5n8wOPP7fyJVpUPkvCnW/vuiXGpylGUlnPDnB7311rARzAt3Mhswpjg==", + "dev": true + }, + "node_modules/kind-of": { + "version": "6.0.3", + "resolved": "https://registry.npmmirror.com/kind-of/-/kind-of-6.0.3.tgz", + "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/klaw-sync": { + "version": "6.0.0", + "resolved": "https://registry.npmmirror.com/klaw-sync/-/klaw-sync-6.0.0.tgz", + "integrity": "sha512-nIeuVSzdCCs6TDPTqI8w1Yre34sSq7AkZ4B3sfOBbI2CgVSB4Du4aLQijFU2+lhAFCwt9+42Hel6lQNIv6AntQ==", + "dev": true, + "dependencies": { + "graceful-fs": "^4.1.11" + } + }, + "node_modules/klona": { + "version": "2.0.5", + "resolved": "https://registry.npmmirror.com/klona/-/klona-2.0.5.tgz", + "integrity": "sha512-pJiBpiXMbt7dkzXe8Ghj/u4FfXOOa98fPW+bihOJ4SjnoijweJrNThJfd3ifXpXhREjpoF2mZVH1GfS9LV3kHQ==", + "dev": true, + "engines": { + "node": ">= 8" + } + }, + "node_modules/launch-editor": { + "version": "2.3.0", + "resolved": "https://registry.npmmirror.com/launch-editor/-/launch-editor-2.3.0.tgz", + "integrity": "sha512-3QrsCXejlWYHjBPFXTyGNhPj4rrQdB+5+r5r3wArpLH201aR+nWUgw/zKKkTmilCfY/sv6u8qo98pNvtg8LUTA==", + "dev": true, + "dependencies": { + "picocolors": "^1.0.0", + "shell-quote": "^1.6.1" + } + }, + "node_modules/launch-editor-middleware": { + "version": "2.3.0", + "resolved": "https://registry.npmmirror.com/launch-editor-middleware/-/launch-editor-middleware-2.3.0.tgz", + "integrity": "sha512-GJR64trLdFFwCoL9DMn/d1SZX0OzTDPixu4mcfWTShQ4tIqCHCGvlg9fOEYQXyBlrSMQwylsJfUWncheShfV2w==", + "dev": true, + "dependencies": { + "launch-editor": "^2.3.0" + } + }, + "node_modules/lazystream": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/lazystream/-/lazystream-1.0.1.tgz", + "integrity": "sha512-b94GiNHQNy6JNTrt5w6zNyffMrNkXZb3KTkCZJb2V1xaEGCk093vkZ2jk3tpaeP33/OiXC+WvK9AxUebnf5nbw==", + "dev": true, + "dependencies": { + "readable-stream": "^2.0.5" + }, + "engines": { + "node": ">= 0.6.3" + } + }, + "node_modules/lerna": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/lerna/-/lerna-4.0.0.tgz", + "integrity": "sha512-DD/i1znurfOmNJb0OBw66NmNqiM8kF6uIrzrJ0wGE3VNdzeOhz9ziWLYiRaZDGGwgbcjOo6eIfcx9O5Qynz+kg==", + "dev": true, + "dependencies": { + "@lerna/add": "4.0.0", + "@lerna/bootstrap": "4.0.0", + "@lerna/changed": "4.0.0", + "@lerna/clean": "4.0.0", + "@lerna/cli": "4.0.0", + "@lerna/create": "4.0.0", + "@lerna/diff": "4.0.0", + "@lerna/exec": "4.0.0", + "@lerna/import": "4.0.0", + "@lerna/info": "4.0.0", + "@lerna/init": "4.0.0", + "@lerna/link": "4.0.0", + "@lerna/list": "4.0.0", + "@lerna/publish": "4.0.0", + "@lerna/run": "4.0.0", + "@lerna/version": "4.0.0", + "import-local": "^3.0.2", + "npmlog": "^4.1.2" + }, + "bin": { + "lerna": "cli.js" + }, + "engines": { + "node": ">= 10.18.0" + } + }, + "node_modules/lerna/node_modules/find-up": { + "version": "4.1.0", + "resolved": "https://registry.npmmirror.com/find-up/-/find-up-4.1.0.tgz", + "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", + "dev": true, + "dependencies": { + "locate-path": "^5.0.0", + "path-exists": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/lerna/node_modules/import-local": { + "version": "3.1.0", + "resolved": "https://registry.npmmirror.com/import-local/-/import-local-3.1.0.tgz", + "integrity": "sha512-ASB07uLtnDs1o6EHjKpX34BKYDSqnFerfTOJL2HvMqF70LnxpjkzDB8J44oT9pu4AMPkQwf8jl6szgvNd2tRIg==", + "dev": true, + "dependencies": { + "pkg-dir": "^4.2.0", + "resolve-cwd": "^3.0.0" + }, + "bin": { + "import-local-fixture": "fixtures/cli.js" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/lerna/node_modules/locate-path": { + "version": "5.0.0", + "resolved": "https://registry.npmmirror.com/locate-path/-/locate-path-5.0.0.tgz", + "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", + "dev": true, + "dependencies": { + "p-locate": "^4.1.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/lerna/node_modules/p-locate": { + "version": "4.1.0", + "resolved": "https://registry.npmmirror.com/p-locate/-/p-locate-4.1.0.tgz", + "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", + "dev": true, + "dependencies": { + "p-limit": "^2.2.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/lerna/node_modules/path-exists": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/path-exists/-/path-exists-4.0.0.tgz", + "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/lerna/node_modules/pkg-dir": { + "version": "4.2.0", + "resolved": "https://registry.npmmirror.com/pkg-dir/-/pkg-dir-4.2.0.tgz", + "integrity": "sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ==", + "dev": true, + "dependencies": { + "find-up": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/lerna/node_modules/resolve-cwd": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/resolve-cwd/-/resolve-cwd-3.0.0.tgz", + "integrity": "sha512-OrZaX2Mb+rJCpH/6CpSqt9xFVpN++x01XnN2ie9g6P5/3xelLAkXWVADpdz1IHD/KFfEXyE6V0U01OQ3UO2rEg==", + "dev": true, + "dependencies": { + "resolve-from": "^5.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/lerna/node_modules/resolve-from": { + "version": "5.0.0", + "resolved": "https://registry.npmmirror.com/resolve-from/-/resolve-from-5.0.0.tgz", + "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/less": { + "version": "3.13.1", + "resolved": "https://registry.npmmirror.com/less/-/less-3.13.1.tgz", + "integrity": "sha512-SwA1aQXGUvp+P5XdZslUOhhLnClSLIjWvJhmd+Vgib5BFIr9lMNlQwmwUNOjXThF/A0x+MCYYPeWEfeWiLRnTw==", + "dev": true, + "dependencies": { + "copy-anything": "^2.0.1", + "tslib": "^1.10.0" + }, + "bin": { + "lessc": "bin/lessc" + }, + "engines": { + "node": ">=6" + }, + "optionalDependencies": { + "errno": "^0.1.1", + "graceful-fs": "^4.1.2", + "image-size": "~0.5.0", + "make-dir": "^2.1.0", + "mime": "^1.4.1", + "native-request": "^1.0.5", + "source-map": "~0.6.0" + } + }, + "node_modules/less-loader": { + "version": "6.1.0", + "resolved": "https://registry.npmmirror.com/less-loader/-/less-loader-6.1.0.tgz", + "integrity": "sha512-/jLzOwLyqJ7Kt3xg5sHHkXtOyShWwFj410K9Si9WO+/h8rmYxxkSR0A3/hFEntWudE20zZnWMtpMYnLzqTVdUA==", + "dev": true, + "dependencies": { + "clone": "^2.1.2", + "less": "^3.11.1", + "loader-utils": "^2.0.0", + "schema-utils": "^2.6.6" + }, + "engines": { + "node": ">= 10.13.0" + }, + "peerDependencies": { + "webpack": "^4.0.0 || ^5.0.0" + } + }, + "node_modules/less-loader/node_modules/clone": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/clone/-/clone-2.1.2.tgz", + "integrity": "sha512-3Pe/CF1Nn94hyhIYpjtiLhdCoEoz0DqQ+988E9gmeEdQZlojxnOb74wctFyuwWQHzqyf9X7C7MG8juUpqBJT8w==", + "dev": true, + "engines": { + "node": ">=0.8" + } + }, + "node_modules/less-loader/node_modules/json5": { + "version": "2.2.1", + "resolved": "https://registry.npmmirror.com/json5/-/json5-2.2.1.tgz", + "integrity": "sha512-1hqLFMSrGHRHxav9q9gNjJ5EXznIxGVO09xQRrwplcS8qs28pZ8s8hupZAmqDwZUmVZ2Qb2jnyPOWcDH8m8dlA==", + "dev": true, + "bin": { + "json5": "lib/cli.js" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/less-loader/node_modules/loader-utils": { + "version": "2.0.2", + "resolved": "https://registry.npmmirror.com/loader-utils/-/loader-utils-2.0.2.tgz", + "integrity": "sha512-TM57VeHptv569d/GKh6TAYdzKblwDNiumOdkFnejjD0XwTH87K90w3O7AiJRqdQoXygvi1VQTJTLGhJl7WqA7A==", + "dev": true, + "dependencies": { + "big.js": "^5.2.2", + "emojis-list": "^3.0.0", + "json5": "^2.1.2" + }, + "engines": { + "node": ">=8.9.0" + } + }, + "node_modules/less-loader/node_modules/schema-utils": { + "version": "2.7.1", + "resolved": "https://registry.npmmirror.com/schema-utils/-/schema-utils-2.7.1.tgz", + "integrity": "sha512-SHiNtMOUGWBQJwzISiVYKu82GiV4QYGePp3odlY1tuKO7gPtphAT5R/py0fA6xtbgLL/RvtJZnU9b8s0F1q0Xg==", + "dev": true, + "dependencies": { + "@types/json-schema": "^7.0.5", + "ajv": "^6.12.4", + "ajv-keywords": "^3.5.2" + }, + "engines": { + "node": ">= 8.9.0" + } + }, + "node_modules/less/node_modules/mime": { + "version": "1.6.0", + "resolved": "https://registry.npmmirror.com/mime/-/mime-1.6.0.tgz", + "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==", + "dev": true, + "optional": true, + "bin": { + "mime": "cli.js" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/less/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmmirror.com/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "optional": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/less/node_modules/tslib": { + "version": "1.14.1", + "resolved": "https://registry.npmmirror.com/tslib/-/tslib-1.14.1.tgz", + "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==", + "dev": true + }, + "node_modules/levn": { + "version": "0.3.0", + "resolved": "https://registry.npmmirror.com/levn/-/levn-0.3.0.tgz", + "integrity": "sha512-0OO4y2iOHix2W6ujICbKIaEQXvFQHue65vUG3pb5EUomzPI90z9hsA1VsO/dbIIpC53J8gxM9Q4Oho0jrCM/yA==", + "dev": true, + "dependencies": { + "prelude-ls": "~1.1.2", + "type-check": "~0.3.2" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/libnpmaccess": { + "version": "4.0.3", + "resolved": "https://registry.npmmirror.com/libnpmaccess/-/libnpmaccess-4.0.3.tgz", + "integrity": "sha512-sPeTSNImksm8O2b6/pf3ikv4N567ERYEpeKRPSmqlNt1dTZbvgpJIzg5vAhXHpw2ISBsELFRelk0jEahj1c6nQ==", + "dev": true, + "dependencies": { + "aproba": "^2.0.0", + "minipass": "^3.1.1", + "npm-package-arg": "^8.1.2", + "npm-registry-fetch": "^11.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/libnpmaccess/node_modules/aproba": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/aproba/-/aproba-2.0.0.tgz", + "integrity": "sha512-lYe4Gx7QT+MKGbDsA+Z+he/Wtef0BiwDOlK/XkBrdfsh9J/jPPXbX0tE9x9cl27Tmu5gg3QUbUrQYa/y+KOHPQ==", + "dev": true + }, + "node_modules/libnpmaccess/node_modules/minipass": { + "version": "3.1.6", + "resolved": "https://registry.npmmirror.com/minipass/-/minipass-3.1.6.tgz", + "integrity": "sha512-rty5kpw9/z8SX9dmxblFA6edItUmwJgMeYDZRrwlIVN27i8gysGbznJwUggw2V/FVqFSDdWy040ZPS811DYAqQ==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/libnpmaccess/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/libnpmpublish": { + "version": "4.0.2", + "resolved": "https://registry.npmmirror.com/libnpmpublish/-/libnpmpublish-4.0.2.tgz", + "integrity": "sha512-+AD7A2zbVeGRCFI2aO//oUmapCwy7GHqPXFJh3qpToSRNU+tXKJ2YFUgjt04LPPAf2dlEH95s6EhIHM1J7bmOw==", + "dev": true, + "dependencies": { + "normalize-package-data": "^3.0.2", + "npm-package-arg": "^8.1.2", + "npm-registry-fetch": "^11.0.0", + "semver": "^7.1.3", + "ssri": "^8.0.1" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/libnpmpublish/node_modules/hosted-git-info": { + "version": "4.1.0", + "resolved": "https://registry.npmmirror.com/hosted-git-info/-/hosted-git-info-4.1.0.tgz", + "integrity": "sha512-kyCuEOWjJqZuDbRHzL8V93NzQhwIB71oFWSyzVo+KPZI+pnQPPxucdkrOZvkLRnrf5URsQM+IJ09Dw29cRALIA==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/libnpmpublish/node_modules/lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/libnpmpublish/node_modules/minipass": { + "version": "3.1.6", + "resolved": "https://registry.npmmirror.com/minipass/-/minipass-3.1.6.tgz", + "integrity": "sha512-rty5kpw9/z8SX9dmxblFA6edItUmwJgMeYDZRrwlIVN27i8gysGbznJwUggw2V/FVqFSDdWy040ZPS811DYAqQ==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/libnpmpublish/node_modules/normalize-package-data": { + "version": "3.0.3", + "resolved": "https://registry.npmmirror.com/normalize-package-data/-/normalize-package-data-3.0.3.tgz", + "integrity": "sha512-p2W1sgqij3zMMyRC067Dg16bfzVH+w7hyegmpIvZ4JNjqtGOVAIvLmjBx3yP7YTe9vKJgkoNOPjwQGogDoMXFA==", + "dev": true, + "dependencies": { + "hosted-git-info": "^4.0.1", + "is-core-module": "^2.5.0", + "semver": "^7.3.4", + "validate-npm-package-license": "^3.0.1" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/libnpmpublish/node_modules/semver": { + "version": "7.3.7", + "resolved": "https://registry.npmmirror.com/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/libnpmpublish/node_modules/ssri": { + "version": "8.0.1", + "resolved": "https://registry.npmmirror.com/ssri/-/ssri-8.0.1.tgz", + "integrity": "sha512-97qShzy1AiyxvPNIkLWoGua7xoQzzPjQ0HAH4B0rWKo7SZ6USuPcrUiAFrws0UH8RrbWmgq3LMTObhPIHbbBeQ==", + "dev": true, + "dependencies": { + "minipass": "^3.1.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/libnpmpublish/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/lines-and-columns": { + "version": "1.2.4", + "resolved": "https://registry.npmmirror.com/lines-and-columns/-/lines-and-columns-1.2.4.tgz", + "integrity": "sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==", + "dev": true + }, + "node_modules/lint-staged": { + "version": "8.2.1", + "resolved": "https://registry.npmmirror.com/lint-staged/-/lint-staged-8.2.1.tgz", + "integrity": "sha512-n0tDGR/rTCgQNwXnUf/eWIpPNddGWxC32ANTNYsj2k02iZb7Cz5ox2tytwBu+2r0zDXMEMKw7Y9OD/qsav561A==", + "dev": true, + "dependencies": { + "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" + }, + "bin": { + "lint-staged": "index.js" + } + }, + "node_modules/lint-staged/node_modules/debug": { + "version": "3.2.7", + "resolved": "https://registry.npmmirror.com/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dev": true, + "dependencies": { + "ms": "^2.1.1" + } + }, + "node_modules/lint-staged/node_modules/del": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/del/-/del-3.0.0.tgz", + "integrity": "sha512-7yjqSoVSlJzA4t/VUwazuEagGeANEKB3f/aNI//06pfKgwoCb7f6Q1gETN1sZzYaj6chTQ0AhIwDiPdfOjko4A==", + "dev": true, + "dependencies": { + "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" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/lint-staged/node_modules/globby": { + "version": "6.1.0", + "resolved": "https://registry.npmmirror.com/globby/-/globby-6.1.0.tgz", + "integrity": "sha512-KVbFv2TQtbzCoxAnfD6JcHZTYCzyliEaaeM/gH8qQdkKr5s0OP9scEgvdcngyk7AVdY6YVW/TJHd+lQ/Df3Daw==", + "dev": true, + "dependencies": { + "array-union": "^1.0.1", + "glob": "^7.0.3", + "object-assign": "^4.0.1", + "pify": "^2.0.0", + "pinkie-promise": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/lint-staged/node_modules/globby/node_modules/pify": { + "version": "2.3.0", + "resolved": "https://registry.npmmirror.com/pify/-/pify-2.3.0.tgz", + "integrity": "sha512-udgsAY+fTnvv7kI7aaxbqwWNb0AHiB0qBO89PZKPkoTmGOgdbrHDKD+0B2X4uTfJ/FT1R09r9gTsjUjNJotuog==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/lint-staged/node_modules/is-path-cwd": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/is-path-cwd/-/is-path-cwd-1.0.0.tgz", + "integrity": "sha512-cnS56eR9SPAscL77ik76ATVqoPARTqPIVkMDVxRaWH06zT+6+CzIroYRJ0VVvm0Z1zfAvxvz9i/D3Ppjaqt5Nw==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/lint-staged/node_modules/is-path-in-cwd": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/is-path-in-cwd/-/is-path-in-cwd-1.0.1.tgz", + "integrity": "sha512-FjV1RTW48E7CWM7eE/J2NJvAEEVektecDBVBE5Hh3nM1Jd0kvhHtX68Pr3xsDf857xt3Y4AkwVULK1Vku62aaQ==", + "dev": true, + "dependencies": { + "is-path-inside": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/lint-staged/node_modules/is-path-inside": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/is-path-inside/-/is-path-inside-1.0.1.tgz", + "integrity": "sha512-qhsCR/Esx4U4hg/9I19OVUAJkGWtjRYHMRgUMZE2TDdj+Ag+kttZanLupfddNyglzz50cUlmWzUaI37GDfNx/g==", + "dev": true, + "dependencies": { + "path-is-inside": "^1.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/lint-staged/node_modules/ms": { + "version": "2.1.3", + "resolved": "https://registry.npmmirror.com/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", + "dev": true + }, + "node_modules/lint-staged/node_modules/p-map": { + "version": "1.2.0", + "resolved": "https://registry.npmmirror.com/p-map/-/p-map-1.2.0.tgz", + "integrity": "sha512-r6zKACMNhjPJMTl8KcFH4li//gkrXWfbD6feV8l6doRHlzljFWGJ2AP6iKaCJXyZmAUMOPtvbW7EXkbWO/pLEA==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/lint-staged/node_modules/pify": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/pify/-/pify-3.0.0.tgz", + "integrity": "sha512-C3FsVNH1udSEX48gGX1xfvwTWfsYWj5U+8/uK15BGzIGrKoUpghX8hWZwa/OFnakBiiVNmBvemTJR5mcy7iPcg==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/listr": { + "version": "0.14.3", + "resolved": "https://registry.npmmirror.com/listr/-/listr-0.14.3.tgz", + "integrity": "sha512-RmAl7su35BFd/xoMamRjpIE4j3v+L28o8CT5YhAXQJm1fD+1l9ngXY8JAQRJ+tFK2i5njvi0iRUKV09vPwA0iA==", + "dev": true, + "dependencies": { + "@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" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/listr-silent-renderer": { + "version": "1.1.1", + "resolved": "https://registry.npmmirror.com/listr-silent-renderer/-/listr-silent-renderer-1.1.1.tgz", + "integrity": "sha512-L26cIFm7/oZeSNVhWB6faeorXhMg4HNlb/dS/7jHhr708jxlXrtrBWo4YUxZQkc6dGoxEAe6J/D3juTRBUzjtA==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/listr-update-renderer": { + "version": "0.5.0", + "resolved": "https://registry.npmmirror.com/listr-update-renderer/-/listr-update-renderer-0.5.0.tgz", + "integrity": "sha512-tKRsZpKz8GSGqoI/+caPmfrypiaq+OQCbd+CovEC24uk1h952lVj5sC7SqyFUm+OaJ5HN/a1YLt5cit2FMNsFA==", + "dev": true, + "dependencies": { + "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" + }, + "engines": { + "node": ">=6" + }, + "peerDependencies": { + "listr": "^0.14.2" + } + }, + "node_modules/listr-update-renderer/node_modules/ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmmirror.com/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha512-TIGnTpdo+E3+pCyAluZvtED5p5wCqLdezCyhPZzKPcxvFplEt4i+W7OONCKgeZFT3+y5NZZfOOS/Bdcanm1MYA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/listr-update-renderer/node_modules/ansi-styles": { + "version": "2.2.1", + "resolved": "https://registry.npmmirror.com/ansi-styles/-/ansi-styles-2.2.1.tgz", + "integrity": "sha512-kmCevFghRiWM7HB5zTPULl4r9bVFSWjz62MhqizDGUrq2NWuNMQyuv4tHHoKJHs69M/MF64lEcHdYIocrdWQYA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/listr-update-renderer/node_modules/chalk": { + "version": "1.1.3", + "resolved": "https://registry.npmmirror.com/chalk/-/chalk-1.1.3.tgz", + "integrity": "sha512-U3lRVLMSlsCfjqYPbLyVv11M9CPW4I728d6TCKMAOJueEeB9/8o+eSsMnxPJD+Q+K909sdESg7C+tIkoH6on1A==", + "dev": true, + "dependencies": { + "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" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/listr-update-renderer/node_modules/figures": { + "version": "1.7.0", + "resolved": "https://registry.npmmirror.com/figures/-/figures-1.7.0.tgz", + "integrity": "sha512-UxKlfCRuCBxSXU4C6t9scbDyWZ4VlaFFdojKtzJuSkuOBQ5CNFum+zZXFwHjo+CxBC1t6zlYPgHIgFjL8ggoEQ==", + "dev": true, + "dependencies": { + "escape-string-regexp": "^1.0.5", + "object-assign": "^4.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/listr-update-renderer/node_modules/indent-string": { + "version": "3.2.0", + "resolved": "https://registry.npmmirror.com/indent-string/-/indent-string-3.2.0.tgz", + "integrity": "sha512-BYqTHXTGUIvg7t1r4sJNKcbDZkL92nkXA8YtRpbjFHRHGDL/NtUeiBJMeE60kIFN/Mg8ESaWQvftaYMGJzQZCQ==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/listr-update-renderer/node_modules/log-symbols": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/log-symbols/-/log-symbols-1.0.2.tgz", + "integrity": "sha512-mmPrW0Fh2fxOzdBbFv4g1m6pR72haFLPJ2G5SJEELf1y+iaQrDG6cWCPjy54RHYbZAt7X+ls690Kw62AdWXBzQ==", + "dev": true, + "dependencies": { + "chalk": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/listr-update-renderer/node_modules/strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmmirror.com/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha512-VhumSSbBqDTP8p2ZLKj40UjBCV4+v8bUSEpUb4KjRgWk9pbqGF4REFj6KEagidb2f/M6AzC0EmFyDNGaw9OCzg==", + "dev": true, + "dependencies": { + "ansi-regex": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/listr-update-renderer/node_modules/supports-color": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/supports-color/-/supports-color-2.0.0.tgz", + "integrity": "sha512-KKNVtd6pCYgPIKU4cp2733HWYCpplQhddZLBUryaAHou723x+FRzQ5Df824Fj+IyyuiQTRoub4SnIFfIcrp70g==", + "dev": true, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/listr-verbose-renderer": { + "version": "0.5.0", + "resolved": "https://registry.npmmirror.com/listr-verbose-renderer/-/listr-verbose-renderer-0.5.0.tgz", + "integrity": "sha512-04PDPqSlsqIOaaaGZ+41vq5FejI9auqTInicFRndCBgE3bXG8D6W1I+mWhk+1nqbHmyhla/6BUrd5OSiHwKRXw==", + "dev": true, + "dependencies": { + "chalk": "^2.4.1", + "cli-cursor": "^2.1.0", + "date-fns": "^1.27.2", + "figures": "^2.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/load-json-file": { + "version": "6.2.0", + "resolved": "https://registry.npmmirror.com/load-json-file/-/load-json-file-6.2.0.tgz", + "integrity": "sha512-gUD/epcRms75Cw8RT1pUdHugZYM5ce64ucs2GEISABwkRsOQr0q2wm/MV2TKThycIe5e0ytRweW2RZxclogCdQ==", + "dev": true, + "dependencies": { + "graceful-fs": "^4.1.15", + "parse-json": "^5.0.0", + "strip-bom": "^4.0.0", + "type-fest": "^0.6.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/load-json-file/node_modules/parse-json": { + "version": "5.2.0", + "resolved": "https://registry.npmmirror.com/parse-json/-/parse-json-5.2.0.tgz", + "integrity": "sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==", + "dev": true, + "dependencies": { + "@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" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/load-json-file/node_modules/strip-bom": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/strip-bom/-/strip-bom-4.0.0.tgz", + "integrity": "sha512-3xurFv5tEgii33Zi8Jtp55wEIILR9eh34FAW00PZf+JnSsTmV/ioewSgQl97JHvgjoRGwPShsWm+IdrxB35d0w==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/loader-fs-cache": { + "version": "1.0.3", + "resolved": "https://registry.npmmirror.com/loader-fs-cache/-/loader-fs-cache-1.0.3.tgz", + "integrity": "sha512-ldcgZpjNJj71n+2Mf6yetz+c9bM4xpKtNds4LbqXzU/PTdeAX0g3ytnU1AJMEcTk2Lex4Smpe3Q/eCTsvUBxbA==", + "dev": true, + "dependencies": { + "find-cache-dir": "^0.1.1", + "mkdirp": "^0.5.1" + } + }, + "node_modules/loader-fs-cache/node_modules/find-cache-dir": { + "version": "0.1.1", + "resolved": "https://registry.npmmirror.com/find-cache-dir/-/find-cache-dir-0.1.1.tgz", + "integrity": "sha512-Z9XSBoNE7xQiV6MSgPuCfyMokH2K7JdpRkOYE1+mu3d4BFJtx3GW+f6Bo4q8IX6rlf5MYbLBKW0pjl2cWdkm2A==", + "dev": true, + "dependencies": { + "commondir": "^1.0.1", + "mkdirp": "^0.5.1", + "pkg-dir": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/loader-fs-cache/node_modules/find-up": { + "version": "1.1.2", + "resolved": "https://registry.npmmirror.com/find-up/-/find-up-1.1.2.tgz", + "integrity": "sha512-jvElSjyuo4EMQGoTwo1uJU5pQMwTW5lS1x05zzfJuTIyLR3zwO27LYrxNg+dlvKpGOuGy/MzBdXh80g0ve5+HA==", + "dev": true, + "dependencies": { + "path-exists": "^2.0.0", + "pinkie-promise": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/loader-fs-cache/node_modules/path-exists": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/path-exists/-/path-exists-2.1.0.tgz", + "integrity": "sha512-yTltuKuhtNeFJKa1PiRzfLAU5182q1y4Eb4XCJ3PBqyzEDkAZRzBrKKBct682ls9reBVHf9udYLN5Nd+K1B9BQ==", + "dev": true, + "dependencies": { + "pinkie-promise": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/loader-fs-cache/node_modules/pkg-dir": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/pkg-dir/-/pkg-dir-1.0.0.tgz", + "integrity": "sha512-c6pv3OE78mcZ92ckebVDqg0aWSoKhOTbwCV6qbCWMk546mAL9pZln0+QsN/yQ7fkucd4+yJPLrCBXNt8Ruk+Eg==", + "dev": true, + "dependencies": { + "find-up": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/loader-runner": { + "version": "2.4.0", + "resolved": "https://registry.npmmirror.com/loader-runner/-/loader-runner-2.4.0.tgz", + "integrity": "sha512-Jsmr89RcXGIwivFY21FcRrisYZfvLMTWx5kOLc+JTxtpBOG6xML0vzbc6SEQG2FO9/4Fc3wW4LVcB5DmGflaRw==", + "engines": { + "node": ">=4.3.0 <5.0.0 || >=5.10" + } + }, + "node_modules/loader-utils": { + "version": "1.4.0", + "resolved": "https://registry.npmmirror.com/loader-utils/-/loader-utils-1.4.0.tgz", + "integrity": "sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA==", + "dependencies": { + "big.js": "^5.2.2", + "emojis-list": "^3.0.0", + "json5": "^1.0.1" + }, + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/locate-path": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/locate-path/-/locate-path-3.0.0.tgz", + "integrity": "sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A==", + "dependencies": { + "p-locate": "^3.0.0", + "path-exists": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/lodash": { + "version": "4.17.20", + "resolved": "https://registry.npmmirror.com/lodash/-/lodash-4.17.20.tgz", + "integrity": "sha512-PlhdFcillOINfeV7Ni6oF1TAEayyZBoZ8bcshTHqOYJYlrqzRK5hagpagky5o4HfCzzd1TRkXPMFq6cKk9rGmA==" + }, + "node_modules/lodash._reinterpolate": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/lodash._reinterpolate/-/lodash._reinterpolate-3.0.0.tgz", + "integrity": "sha512-xYHt68QRoYGjeeM/XOE1uJtvXQAgvszfBhjV4yvsQH0u2i9I6cI6c6/eG4Hh3UAOVn0y/xAXwmTzEay49Q//HA==", + "dev": true + }, + "node_modules/lodash.debounce": { + "version": "4.0.8", + "resolved": "https://registry.npmmirror.com/lodash.debounce/-/lodash.debounce-4.0.8.tgz", + "integrity": "sha512-FT1yDzDYEoYWhnSGnpE/4Kj1fLZkDFyqRb7fNt6FdYOSxlUWAtp42Eh6Wb0rGIv/m9Bgo7x4GhQbm5Ys4SG5ow==", + "dev": true + }, + "node_modules/lodash.defaults": { + "version": "4.2.0", + "resolved": "https://registry.npmmirror.com/lodash.defaults/-/lodash.defaults-4.2.0.tgz", + "integrity": "sha512-qjxPLHd3r5DnsdGacqOMU6pb/avJzdh9tFX2ymgoZE27BmjXrNy/y4LoaiTeAb+O3gL8AfpJGtqfX/ae2leYYQ==", + "dev": true + }, + "node_modules/lodash.defaultsdeep": { + "version": "4.6.1", + "resolved": "https://registry.npmmirror.com/lodash.defaultsdeep/-/lodash.defaultsdeep-4.6.1.tgz", + "integrity": "sha512-3j8wdDzYuWO3lM3Reg03MuQR957t287Rpcxp1njpEa8oDrikb+FwGdW3n+FELh/A6qib6yPit0j/pv9G/yeAqA==", + "dev": true + }, + "node_modules/lodash.difference": { + "version": "4.5.0", + "resolved": "https://registry.npmmirror.com/lodash.difference/-/lodash.difference-4.5.0.tgz", + "integrity": "sha512-dS2j+W26TQ7taQBGN8Lbbq04ssV3emRw4NY58WErlTO29pIqS0HmoT5aJ9+TUQ1N3G+JOZSji4eugsWwGp9yPA==", + "dev": true + }, + "node_modules/lodash.flatten": { + "version": "4.4.0", + "resolved": "https://registry.npmmirror.com/lodash.flatten/-/lodash.flatten-4.4.0.tgz", + "integrity": "sha512-C5N2Z3DgnnKr0LOpv/hKCgKdb7ZZwafIrsesve6lmzvZIRZRGaZ/l6Q8+2W7NaT+ZwO3fFlSCzCzrDCFdJfZ4g==", + "dev": true + }, + "node_modules/lodash.ismatch": { + "version": "4.4.0", + "resolved": "https://registry.npmmirror.com/lodash.ismatch/-/lodash.ismatch-4.4.0.tgz", + "integrity": "sha512-fPMfXjGQEV9Xsq/8MTSgUf255gawYRbjwMyDbcvDhXgV7enSZA0hynz6vMPnpAb5iONEzBHBPsT+0zes5Z301g==", + "dev": true + }, + "node_modules/lodash.isplainobject": { + "version": "4.0.6", + "resolved": "https://registry.npmmirror.com/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz", + "integrity": "sha512-oSXzaWypCMHkPC3NvBEaPHf0KsA5mvPrOPgQWDsbg8n7orZ290M0BmC/jgRZ4vcJ6DTAhjrsSYgdsW/F+MFOBA==", + "dev": true + }, + "node_modules/lodash.kebabcase": { + "version": "4.1.1", + "resolved": "https://registry.npmmirror.com/lodash.kebabcase/-/lodash.kebabcase-4.1.1.tgz", + "integrity": "sha512-N8XRTIMMqqDgSy4VLKPnJ/+hpGZN+PHQiJnSenYqPaVV/NCqEogTnAdZLQiGKhxX+JCs8waWq2t1XHWKOmlY8g==", + "dev": true + }, + "node_modules/lodash.mapvalues": { + "version": "4.6.0", + "resolved": "https://registry.npmmirror.com/lodash.mapvalues/-/lodash.mapvalues-4.6.0.tgz", + "integrity": "sha512-JPFqXFeZQ7BfS00H58kClY7SPVeHertPE0lNuCyZ26/XlN8TvakYD7b9bGyNmXbT/D3BbtPAAmq90gPWqLkxlQ==", + "dev": true + }, + "node_modules/lodash.memoize": { + "version": "4.1.2", + "resolved": "https://registry.npmmirror.com/lodash.memoize/-/lodash.memoize-4.1.2.tgz", + "integrity": "sha512-t7j+NzmgnQzTAYXcsHYLgimltOV1MXHtlOWf6GjL9Kj8GK5FInw5JotxvbOs+IvV1/Dzo04/fCGfLVs7aXb4Ag==", + "dev": true + }, + "node_modules/lodash.template": { + "version": "4.5.0", + "resolved": "https://registry.npmmirror.com/lodash.template/-/lodash.template-4.5.0.tgz", + "integrity": "sha512-84vYFxIkmidUiFxidA/KjjH9pAycqW+h980j7Fuz5qxRtO9pgB7MDFTdys1N7A5mcucRiDyEq4fusljItR1T/A==", + "dev": true, + "dependencies": { + "lodash._reinterpolate": "^3.0.0", + "lodash.templatesettings": "^4.0.0" + } + }, + "node_modules/lodash.templatesettings": { + "version": "4.2.0", + "resolved": "https://registry.npmmirror.com/lodash.templatesettings/-/lodash.templatesettings-4.2.0.tgz", + "integrity": "sha512-stgLz+i3Aa9mZgnjr/O+v9ruKZsPsndy7qPZOchbqk2cnTU1ZaldKK+v7m54WoKIyxiuMZTKT2H81F8BeAc3ZQ==", + "dev": true, + "dependencies": { + "lodash._reinterpolate": "^3.0.0" + } + }, + "node_modules/lodash.throttle": { + "version": "4.1.1", + "resolved": "https://registry.npmmirror.com/lodash.throttle/-/lodash.throttle-4.1.1.tgz", + "integrity": "sha512-wIkUCfVKpVsWo3JSZlc+8MB5it+2AN5W8J7YVMST30UrvcQNZ1Okbj+rbVniijTWE6FGYy4XJq/rHkas8qJMLQ==" + }, + "node_modules/lodash.transform": { + "version": "4.6.0", + "resolved": "https://registry.npmmirror.com/lodash.transform/-/lodash.transform-4.6.0.tgz", + "integrity": "sha512-LO37ZnhmBVx0GvOU/caQuipEh4GN82TcWv3yHlebGDgOxbxiwwzW5Pcx2AcvpIv2WmvmSMoC492yQFNhy/l/UQ==", + "dev": true + }, + "node_modules/lodash.union": { + "version": "4.6.0", + "resolved": "https://registry.npmmirror.com/lodash.union/-/lodash.union-4.6.0.tgz", + "integrity": "sha512-c4pB2CdGrGdjMKYLA+XiRDO7Y0PRQbm/Gzg8qMj+QH+pFVAoTp5sBpO0odL3FjoPCGjK96p6qsP+yQoiLoOBcw==", + "dev": true + }, + "node_modules/lodash.uniq": { + "version": "4.5.0", + "resolved": "https://registry.npmmirror.com/lodash.uniq/-/lodash.uniq-4.5.0.tgz", + "integrity": "sha512-xfBaXQd9ryd9dlSDvnvI0lvxfLJlYAZzXomUYzLKtUeOQvOP5piqAWuGtrhWeqaXK9hhoM/iyJc5AV+XfsX3HQ==", + "dev": true + }, + "node_modules/log-symbols": { + "version": "2.2.0", + "resolved": "https://registry.npmmirror.com/log-symbols/-/log-symbols-2.2.0.tgz", + "integrity": "sha512-VeIAFslyIerEJLXHziedo2basKbMKtTw3vfn5IzG0XTjhAVEJyNHnL2p7vc+wBDSdQuUpNw3M2u6xb9QsAY5Eg==", + "dev": true, + "dependencies": { + "chalk": "^2.0.1" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/log-update": { + "version": "2.3.0", + "resolved": "https://registry.npmmirror.com/log-update/-/log-update-2.3.0.tgz", + "integrity": "sha512-vlP11XfFGyeNQlmEn9tJ66rEW1coA/79m5z6BCkudjbAGE83uhAcGYrBFwfs3AdLiLzGRusRPAbSPK9xZteCmg==", + "dev": true, + "dependencies": { + "ansi-escapes": "^3.0.0", + "cli-cursor": "^2.0.0", + "wrap-ansi": "^3.0.1" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/log-update/node_modules/ansi-regex": { + "version": "3.0.1", + "resolved": "https://registry.npmmirror.com/ansi-regex/-/ansi-regex-3.0.1.tgz", + "integrity": "sha512-+O9Jct8wf++lXxxFc4hc8LsjaSq0HFzzL7cVsw8pRDIPdjKD2mT4ytDZlLuSBZ4cLKZFXIrMGO7DbQCtMJJMKw==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/log-update/node_modules/strip-ansi": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/strip-ansi/-/strip-ansi-4.0.0.tgz", + "integrity": "sha512-4XaJ2zQdCzROZDivEVIDPkcQn8LMFSa8kj8Gxb/Lnwzv9A8VctNZ+lfivC/sV3ivW8ElJTERXZoPBRrZKkNKow==", + "dev": true, + "dependencies": { + "ansi-regex": "^3.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/log-update/node_modules/wrap-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmmirror.com/wrap-ansi/-/wrap-ansi-3.0.1.tgz", + "integrity": "sha512-iXR3tDXpbnTpzjKSylUJRkLuOrEC7hwEB221cgn6wtF8wpmz28puFXAEfPT5zrjM3wahygB//VuWEr1vTkDcNQ==", + "dev": true, + "dependencies": { + "string-width": "^2.1.1", + "strip-ansi": "^4.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/loglevel": { + "version": "1.8.0", + "resolved": "https://registry.npmmirror.com/loglevel/-/loglevel-1.8.0.tgz", + "integrity": "sha512-G6A/nJLRgWOuuwdNuA6koovfEV1YpqqAG4pRUlFaz3jj2QNZ8M4vBqnVA+HBTmU/AMNUtlOsMmSpF6NyOjztbA==", + "dev": true, + "engines": { + "node": ">= 0.6.0" + } + }, + "node_modules/loose-envify": { + "version": "1.4.0", + "resolved": "https://registry.npmmirror.com/loose-envify/-/loose-envify-1.4.0.tgz", + "integrity": "sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==", + "dev": true, + "dependencies": { + "js-tokens": "^3.0.0 || ^4.0.0" + }, + "bin": { + "loose-envify": "cli.js" + } + }, + "node_modules/lower-case": { + "version": "1.1.4", + "resolved": "https://registry.npmmirror.com/lower-case/-/lower-case-1.1.4.tgz", + "integrity": "sha512-2Fgx1Ycm599x+WGpIYwJOvsjmXFzTSc34IwDWALRA/8AopUKAVPwfJ+h5+f85BCp0PWmmJcWzEpxOpoXycMpdA==", + "dev": true + }, + "node_modules/lru-cache": { + "version": "5.1.1", + "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-5.1.1.tgz", + "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==", + "dependencies": { + "yallist": "^3.0.2" + } + }, + "node_modules/make-dir": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/make-dir/-/make-dir-2.1.0.tgz", + "integrity": "sha512-LS9X+dc8KLxXCb8dni79fLIIUA5VyZoyjSMCwTluaXA0o27cCK0bhXkpgw+sTXVpPy/lSO57ilRixqk0vDmtRA==", + "dependencies": { + "pify": "^4.0.1", + "semver": "^5.6.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/make-fetch-happen": { + "version": "9.1.0", + "resolved": "https://registry.npmmirror.com/make-fetch-happen/-/make-fetch-happen-9.1.0.tgz", + "integrity": "sha512-+zopwDy7DNknmwPQplem5lAZX/eCOzSvSNNcSKm5eVwTkOBzoktEfXsa9L23J/GIRhxRsaxzkPEhrJEpE2F4Gg==", + "dev": true, + "dependencies": { + "agentkeepalive": "^4.1.3", + "cacache": "^15.2.0", + "http-cache-semantics": "^4.1.0", + "http-proxy-agent": "^4.0.1", + "https-proxy-agent": "^5.0.0", + "is-lambda": "^1.0.1", + "lru-cache": "^6.0.0", + "minipass": "^3.1.3", + "minipass-collect": "^1.0.2", + "minipass-fetch": "^1.3.2", + "minipass-flush": "^1.0.5", + "minipass-pipeline": "^1.2.4", + "negotiator": "^0.6.2", + "promise-retry": "^2.0.1", + "socks-proxy-agent": "^6.0.0", + "ssri": "^8.0.0" + }, + "engines": { + "node": ">= 10" + } + }, + "node_modules/make-fetch-happen/node_modules/cacache": { + "version": "15.3.0", + "resolved": "https://registry.npmmirror.com/cacache/-/cacache-15.3.0.tgz", + "integrity": "sha512-VVdYzXEn+cnbXpFgWs5hTT7OScegHVmLhJIR8Ufqk3iFD6A6j5iSX1KuBTfNEv4tdJWE2PzA6IVFtcLC7fN9wQ==", + "dev": true, + "dependencies": { + "@npmcli/fs": "^1.0.0", + "@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.1", + "tar": "^6.0.2", + "unique-filename": "^1.1.1" + }, + "engines": { + "node": ">= 10" + } + }, + "node_modules/make-fetch-happen/node_modules/chownr": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/chownr/-/chownr-2.0.0.tgz", + "integrity": "sha512-bIomtDF5KGpdogkLd9VspvFzk9KfpyyGlS8YFVZl7TGPBHL5snIOnxeshwVgPteQ9b4Eydl+pVbIyE1DcvCWgQ==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/make-fetch-happen/node_modules/fs-minipass": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/fs-minipass/-/fs-minipass-2.1.0.tgz", + "integrity": "sha512-V/JgOLFCS+R6Vcq0slCuaeWEdNC3ouDlJMNIsacH2VtALiu9mV4LPrHc5cDl8k5aw6J8jwgWWpiTo5RYhmIzvg==", + "dev": true, + "dependencies": { + "minipass": "^3.0.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/make-fetch-happen/node_modules/lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/make-fetch-happen/node_modules/minipass": { + "version": "3.1.6", + "resolved": "https://registry.npmmirror.com/minipass/-/minipass-3.1.6.tgz", + "integrity": "sha512-rty5kpw9/z8SX9dmxblFA6edItUmwJgMeYDZRrwlIVN27i8gysGbznJwUggw2V/FVqFSDdWy040ZPS811DYAqQ==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/make-fetch-happen/node_modules/minizlib": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/minizlib/-/minizlib-2.1.2.tgz", + "integrity": "sha512-bAxsR8BVfj60DWXHE3u30oHzfl4G7khkSuPW+qvpd7jFRHm7dLxOjUk1EHACJ/hxLY8phGJ0YhYHZo7jil7Qdg==", + "dev": true, + "dependencies": { + "minipass": "^3.0.0", + "yallist": "^4.0.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/make-fetch-happen/node_modules/mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmmirror.com/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true, + "bin": { + "mkdirp": "bin/cmd.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/make-fetch-happen/node_modules/p-map": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/p-map/-/p-map-4.0.0.tgz", + "integrity": "sha512-/bjOqmgETBYB5BoEeGVea8dmvHb2m9GLy1E9W43yeyfP6QQCZGFNa+XRceJEuDB6zqr+gKpIAmlLebMpykw/MQ==", + "dev": true, + "dependencies": { + "aggregate-error": "^3.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/make-fetch-happen/node_modules/rimraf": { + "version": "3.0.2", + "resolved": "https://registry.npmmirror.com/rimraf/-/rimraf-3.0.2.tgz", + "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", + "dev": true, + "dependencies": { + "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" + } + }, + "node_modules/make-fetch-happen/node_modules/ssri": { + "version": "8.0.1", + "resolved": "https://registry.npmmirror.com/ssri/-/ssri-8.0.1.tgz", + "integrity": "sha512-97qShzy1AiyxvPNIkLWoGua7xoQzzPjQ0HAH4B0rWKo7SZ6USuPcrUiAFrws0UH8RrbWmgq3LMTObhPIHbbBeQ==", + "dev": true, + "dependencies": { + "minipass": "^3.1.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/make-fetch-happen/node_modules/tar": { + "version": "6.1.11", + "resolved": "https://registry.npmmirror.com/tar/-/tar-6.1.11.tgz", + "integrity": "sha512-an/KZQzQUkZCkuoAA64hM92X0Urb6VpRhAFllDzz44U2mcD5scmT3zBc4VgVpkugF580+DQn8eAFSyoQt0tznA==", + "dev": true, + "dependencies": { + "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" + }, + "engines": { + "node": ">= 10" + } + }, + "node_modules/make-fetch-happen/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/map-cache": { + "version": "0.2.2", + "resolved": "https://registry.npmmirror.com/map-cache/-/map-cache-0.2.2.tgz", + "integrity": "sha512-8y/eV9QQZCiyn1SprXSrCmqJN0yNRATe+PO8ztwqrvrbdRLA3eYJF0yaR0YayLWkMbsQSKWS9N2gPcGEc4UsZg==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/map-obj": { + "version": "4.3.0", + "resolved": "https://registry.npmmirror.com/map-obj/-/map-obj-4.3.0.tgz", + "integrity": "sha512-hdN1wVrZbb29eBGiGjJbeP8JbKjq1urkHJ/LIP/NY48MZ1QVXUsQBV1G1zvYFHn1XE06cwjBsOI2K3Ulnj1YXQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/map-visit": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/map-visit/-/map-visit-1.0.0.tgz", + "integrity": "sha512-4y7uGv8bd2WdM9vpQsiQNo41Ln1NvhvDRuVt0k2JZQ+ezN2uaQes7lZeZ+QQUHOLQAtDaBJ+7wCbi+ab/KFs+w==", + "dependencies": { + "object-visit": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/matcher": { + "version": "1.1.1", + "resolved": "https://registry.npmmirror.com/matcher/-/matcher-1.1.1.tgz", + "integrity": "sha512-+BmqxWIubKTRKNWx/ahnCkk3mG8m7OturVlqq6HiojGJTd5hVYbgZm6WzcYPCoB+KBT4Vd6R7WSRG2OADNaCjg==", + "dev": true, + "dependencies": { + "escape-string-regexp": "^1.0.4" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/material-colors": { + "version": "1.2.6", + "resolved": "https://registry.npmmirror.com/material-colors/-/material-colors-1.2.6.tgz", + "integrity": "sha512-6qE4B9deFBIa9YSpOc9O0Sgc43zTeVYbgDT5veRKSlB2+ZuHNoVVxA1L/ckMUayV9Ay9y7Z/SZCLcGteW9i7bg==" + }, + "node_modules/math-random": { + "version": "1.0.4", + "resolved": "https://registry.npmmirror.com/math-random/-/math-random-1.0.4.tgz", + "integrity": "sha512-rUxjysqif/BZQH2yhd5Aaq7vXMSx9NdEsQcyA07uEzIvxgI7zIr33gGsh+RU0/XjmQpCW7RsVof1vlkvQVCK5A==", + "dev": true + }, + "node_modules/mavon-editor": { + "version": "2.10.4", + "resolved": "https://registry.npmmirror.com/mavon-editor/-/mavon-editor-2.10.4.tgz", + "integrity": "sha512-CFsBLkgt/KZBDg+SJYe2fyYv4zClY149PiwpH0rDAiiP4ae1XNs0GC8nBsoTeipsHcebDLN1QMkt3bUsnMDjQw==", + "dependencies": { + "xss": "^1.0.6" + } + }, + "node_modules/md5": { + "version": "2.3.0", + "resolved": "https://registry.npmmirror.com/md5/-/md5-2.3.0.tgz", + "integrity": "sha512-T1GITYmFaKuO91vxyoQMFETst+O71VUPEU3ze5GNzDm0OWdP8v1ziTaAEPUr/3kLsY3Sftgz242A1SetQiDL7g==", + "dependencies": { + "charenc": "0.0.2", + "crypt": "0.0.2", + "is-buffer": "~1.1.6" + } + }, + "node_modules/md5.js": { + "version": "1.3.5", + "resolved": "https://registry.npmmirror.com/md5.js/-/md5.js-1.3.5.tgz", + "integrity": "sha512-xitP+WxNPcTTOgnTJcrhM0xvdPepipPSf3I8EIpGKeFLjt3PlJLIDG3u8EX53ZIubkb+5U2+3rELYpEhHhzdkg==", + "dependencies": { + "hash-base": "^3.0.0", + "inherits": "^2.0.1", + "safe-buffer": "^5.1.2" + } + }, + "node_modules/mdn-data": { + "version": "2.0.4", + "resolved": "https://registry.npmmirror.com/mdn-data/-/mdn-data-2.0.4.tgz", + "integrity": "sha512-iV3XNKw06j5Q7mi6h+9vbx23Tv7JkjEVgKHW4pimwyDGWm0OIQntJJ+u1C6mg6mK1EaTv42XQ7w76yuzH7M2cA==" + }, + "node_modules/media-typer": { + "version": "0.3.0", + "resolved": "https://registry.npmmirror.com/media-typer/-/media-typer-0.3.0.tgz", + "integrity": "sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ==", + "dev": true, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/memory-fs": { + "version": "0.4.1", + "resolved": "https://registry.npmmirror.com/memory-fs/-/memory-fs-0.4.1.tgz", + "integrity": "sha512-cda4JKCxReDXFXRqOHPQscuIYg1PvxbE2S2GP45rnwfEK+vZaXC8C1OFvdHIbgw0DLzowXGVoxLaAmlgRy14GQ==", + "dependencies": { + "errno": "^0.1.3", + "readable-stream": "^2.0.1" + } + }, + "node_modules/meow": { + "version": "8.1.2", + "resolved": "https://registry.npmmirror.com/meow/-/meow-8.1.2.tgz", + "integrity": "sha512-r85E3NdZ+mpYk1C6RjPFEMSE+s1iZMuHtsHAqY0DT3jZczl0diWUZ8g6oU7h0M9cD2EL+PzaYghhCLzR0ZNn5Q==", + "dev": true, + "dependencies": { + "@types/minimist": "^1.2.0", + "camelcase-keys": "^6.2.2", + "decamelize-keys": "^1.1.0", + "hard-rejection": "^2.1.0", + "minimist-options": "4.1.0", + "normalize-package-data": "^3.0.0", + "read-pkg-up": "^7.0.1", + "redent": "^3.0.0", + "trim-newlines": "^3.0.0", + "type-fest": "^0.18.0", + "yargs-parser": "^20.2.3" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/meow/node_modules/find-up": { + "version": "4.1.0", + "resolved": "https://registry.npmmirror.com/find-up/-/find-up-4.1.0.tgz", + "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", + "dev": true, + "dependencies": { + "locate-path": "^5.0.0", + "path-exists": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/meow/node_modules/hosted-git-info": { + "version": "4.1.0", + "resolved": "https://registry.npmmirror.com/hosted-git-info/-/hosted-git-info-4.1.0.tgz", + "integrity": "sha512-kyCuEOWjJqZuDbRHzL8V93NzQhwIB71oFWSyzVo+KPZI+pnQPPxucdkrOZvkLRnrf5URsQM+IJ09Dw29cRALIA==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/meow/node_modules/locate-path": { + "version": "5.0.0", + "resolved": "https://registry.npmmirror.com/locate-path/-/locate-path-5.0.0.tgz", + "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", + "dev": true, + "dependencies": { + "p-locate": "^4.1.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/meow/node_modules/lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/meow/node_modules/normalize-package-data": { + "version": "3.0.3", + "resolved": "https://registry.npmmirror.com/normalize-package-data/-/normalize-package-data-3.0.3.tgz", + "integrity": "sha512-p2W1sgqij3zMMyRC067Dg16bfzVH+w7hyegmpIvZ4JNjqtGOVAIvLmjBx3yP7YTe9vKJgkoNOPjwQGogDoMXFA==", + "dev": true, + "dependencies": { + "hosted-git-info": "^4.0.1", + "is-core-module": "^2.5.0", + "semver": "^7.3.4", + "validate-npm-package-license": "^3.0.1" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/meow/node_modules/p-locate": { + "version": "4.1.0", + "resolved": "https://registry.npmmirror.com/p-locate/-/p-locate-4.1.0.tgz", + "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", + "dev": true, + "dependencies": { + "p-limit": "^2.2.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/meow/node_modules/path-exists": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/path-exists/-/path-exists-4.0.0.tgz", + "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/meow/node_modules/read-pkg-up": { + "version": "7.0.1", + "resolved": "https://registry.npmmirror.com/read-pkg-up/-/read-pkg-up-7.0.1.tgz", + "integrity": "sha512-zK0TB7Xd6JpCLmlLmufqykGE+/TlOePD6qKClNW7hHDKFh/J7/7gCWGR7joEQEW1bKq3a3yUZSObOoWLFQ4ohg==", + "dev": true, + "dependencies": { + "find-up": "^4.1.0", + "read-pkg": "^5.2.0", + "type-fest": "^0.8.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/meow/node_modules/read-pkg-up/node_modules/type-fest": { + "version": "0.8.1", + "resolved": "https://registry.npmmirror.com/type-fest/-/type-fest-0.8.1.tgz", + "integrity": "sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/meow/node_modules/semver": { + "version": "7.3.7", + "resolved": "https://registry.npmmirror.com/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/meow/node_modules/type-fest": { + "version": "0.18.1", + "resolved": "https://registry.npmmirror.com/type-fest/-/type-fest-0.18.1.tgz", + "integrity": "sha512-OIAYXk8+ISY+qTOwkHtKqzAuxchoMiD9Udx+FSGQDuiRR+PJKJHc2NJAXlbhkGwTt/4/nKZxELY1w3ReWOL8mw==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/meow/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/merge-descriptors": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/merge-descriptors/-/merge-descriptors-1.0.1.tgz", + "integrity": "sha512-cCi6g3/Zr1iqQi6ySbseM1Xvooa98N0w31jzUYrXPX2xqObmFGHJ0tQ5u74H3mVh7wLouTseZyYIq39g8cNp1w==", + "dev": true + }, + "node_modules/merge-options": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/merge-options/-/merge-options-1.0.1.tgz", + "integrity": "sha512-iuPV41VWKWBIOpBsjoxjDZw8/GbSfZ2mk7N1453bwMrfzdrIk7EzBd+8UVR6rkw67th7xnk9Dytl3J+lHPdxvg==", + "dev": true, + "dependencies": { + "is-plain-obj": "^1.1" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/merge-source-map": { + "version": "1.1.0", + "resolved": "https://registry.npmmirror.com/merge-source-map/-/merge-source-map-1.1.0.tgz", + "integrity": "sha512-Qkcp7P2ygktpMPh2mCQZaf3jhN6D3Z/qVZHSdWvQ+2Ef5HgRAPBO57A77+ENm0CPx2+1Ce/MYKi3ymqdfuqibw==", + "dev": true, + "dependencies": { + "source-map": "^0.6.1" + } + }, + "node_modules/merge-source-map/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmmirror.com/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/merge-stream": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/merge-stream/-/merge-stream-2.0.0.tgz", + "integrity": "sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w==", + "dev": true + }, + "node_modules/merge2": { + "version": "1.4.1", + "resolved": "https://registry.npmmirror.com/merge2/-/merge2-1.4.1.tgz", + "integrity": "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==", + "dev": true, + "engines": { + "node": ">= 8" + } + }, + "node_modules/methods": { + "version": "1.1.2", + "resolved": "https://registry.npmmirror.com/methods/-/methods-1.1.2.tgz", + "integrity": "sha512-iclAHeNqNm68zFtnZ0e+1L2yUIdvzNoauKU4WBA3VvH/vPFieF7qfRlwUZU+DA9P9bPXIS90ulxoUoCH23sV2w==", + "dev": true, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/micromatch": { + "version": "3.1.10", + "resolved": "https://registry.npmmirror.com/micromatch/-/micromatch-3.1.10.tgz", + "integrity": "sha512-MWikgl9n9M3w+bpsY3He8L+w9eF9338xRl8IAO5viDizwSzziFEyUzo2xrrloB64ADbTf8uA8vRqqttDTOmccg==", + "dependencies": { + "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" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/miller-rabin": { + "version": "4.0.1", + "resolved": "https://registry.npmmirror.com/miller-rabin/-/miller-rabin-4.0.1.tgz", + "integrity": "sha512-115fLhvZVqWwHPbClyntxEVfVDfl9DLLTuJvq3g2O/Oxi8AiNouAHvDSzHS0viUJc+V5vm3eq91Xwqn9dp4jRA==", + "dependencies": { + "bn.js": "^4.0.0", + "brorand": "^1.0.1" + }, + "bin": { + "miller-rabin": "bin/miller-rabin" + } + }, + "node_modules/miller-rabin/node_modules/bn.js": { + "version": "4.12.0", + "resolved": "https://registry.npmmirror.com/bn.js/-/bn.js-4.12.0.tgz", + "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==" + }, + "node_modules/mime": { + "version": "2.6.0", + "resolved": "https://registry.npmmirror.com/mime/-/mime-2.6.0.tgz", + "integrity": "sha512-USPkMeET31rOMiarsBNIHZKLGgvKc/LrjofAnBlOttf5ajRvqiRA8QsenbcooctK6d6Ts6aqZXBA+XbkKthiQg==", + "dev": true, + "bin": { + "mime": "cli.js" + }, + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/mime-db": { + "version": "1.52.0", + "resolved": "https://registry.npmmirror.com/mime-db/-/mime-db-1.52.0.tgz", + "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", + "dev": true, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/mime-types": { + "version": "2.1.35", + "resolved": "https://registry.npmmirror.com/mime-types/-/mime-types-2.1.35.tgz", + "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", + "dev": true, + "dependencies": { + "mime-db": "1.52.0" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/mimic-fn": { + "version": "1.2.0", + "resolved": "https://registry.npmmirror.com/mimic-fn/-/mimic-fn-1.2.0.tgz", + "integrity": "sha512-jf84uxzwiuiIVKiOLpfYk7N46TSy8ubTonmneY9vrpHNAnp0QBt2BxWV9dO3/j+BoVAb+a5G6YDPW3M5HOdMWQ==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/min-indent": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/min-indent/-/min-indent-1.0.1.tgz", + "integrity": "sha512-I9jwMn07Sy/IwOj3zVkVik2JTvgpaykDZEigL6Rx6N9LbMywwUSMtxET+7lVoDLLd3O3IXwJwvuuns8UB/HeAg==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/mini-css-extract-plugin": { + "version": "0.8.2", + "resolved": "https://registry.npmmirror.com/mini-css-extract-plugin/-/mini-css-extract-plugin-0.8.2.tgz", + "integrity": "sha512-a3Y4of27Wz+mqK3qrcd3VhYz6cU0iW5x3Sgvqzbj+XmlrSizmvu8QQMl5oMYJjgHOC4iyt+w7l4umP+dQeW3bw==", + "dev": true, + "dependencies": { + "loader-utils": "^1.1.0", + "normalize-url": "1.9.1", + "schema-utils": "^1.0.0", + "webpack-sources": "^1.1.0" + }, + "engines": { + "node": ">= 6.9.0" + }, + "peerDependencies": { + "webpack": "^4.4.0" + } + }, + "node_modules/mini-css-extract-plugin/node_modules/normalize-url": { + "version": "1.9.1", + "resolved": "https://registry.npmmirror.com/normalize-url/-/normalize-url-1.9.1.tgz", + "integrity": "sha512-A48My/mtCklowHBlI8Fq2jFWK4tX4lJ5E6ytFsSOq1fzpvT0SQSgKhSg7lN5c2uYFOrUAOQp6zhhJnpp1eMloQ==", + "dev": true, + "dependencies": { + "object-assign": "^4.0.1", + "prepend-http": "^1.0.0", + "query-string": "^4.1.0", + "sort-keys": "^1.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/minimalistic-assert": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/minimalistic-assert/-/minimalistic-assert-1.0.1.tgz", + "integrity": "sha512-UtJcAD4yEaGtjPezWuO9wC4nwUnVH/8/Im3yEHQP4b67cXlD/Qr9hdITCU1xDbSEXg2XKNaP8jsReV7vQd00/A==" + }, + "node_modules/minimalistic-crypto-utils": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/minimalistic-crypto-utils/-/minimalistic-crypto-utils-1.0.1.tgz", + "integrity": "sha512-JIYlbt6g8i5jKfJ3xz7rF0LXmv2TkDxBLUkiBeZ7bAx4GnnNMr8xFpGnOxn6GhTEHx3SjRrZEoU+j04prX1ktg==" + }, + "node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmmirror.com/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/minimist": { + "version": "1.2.6", + "resolved": "https://registry.npmmirror.com/minimist/-/minimist-1.2.6.tgz", + "integrity": "sha512-Jsjnk4bw3YJqYzbdyBiNsPWHPfO++UGG749Cxs6peCu5Xg4nrena6OVxOYxrQTqww0Jmwt+Ref8rggumkTLz9Q==" + }, + "node_modules/minimist-options": { + "version": "4.1.0", + "resolved": "https://registry.npmmirror.com/minimist-options/-/minimist-options-4.1.0.tgz", + "integrity": "sha512-Q4r8ghd80yhO/0j1O3B2BjweX3fiHg9cdOwjJd2J76Q135c+NDxGCqdYKQ1SKBuFfgWbAUzBfvYjPUEeNgqN1A==", + "dev": true, + "dependencies": { + "arrify": "^1.0.1", + "is-plain-obj": "^1.1.0", + "kind-of": "^6.0.3" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/minimist-options/node_modules/arrify": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/arrify/-/arrify-1.0.1.tgz", + "integrity": "sha512-3CYzex9M9FGQjCGMGyi6/31c8GJbgb0qGyrx5HWxPd0aCwh4cB2YjMb2Xf9UuoogrMrlO9cTqnB5rI5GHZTcUA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/minipass": { + "version": "2.9.0", + "resolved": "https://registry.npmmirror.com/minipass/-/minipass-2.9.0.tgz", + "integrity": "sha512-wxfUjg9WebH+CUDX/CdbRlh5SmfZiy/hpkxaRI16Y9W56Pa75sWgd/rvFilSgrauD9NyFymP/+JFV3KwzIsJeg==", + "dev": true, + "dependencies": { + "safe-buffer": "^5.1.2", + "yallist": "^3.0.0" + } + }, + "node_modules/minipass-collect": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/minipass-collect/-/minipass-collect-1.0.2.tgz", + "integrity": "sha512-6T6lH0H8OG9kITm/Jm6tdooIbogG9e0tLgpY6mphXSm/A9u8Nq1ryBG+Qspiub9LjWlBPsPS3tWQ/Botq4FdxA==", + "dev": true, + "dependencies": { + "minipass": "^3.0.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/minipass-collect/node_modules/minipass": { + "version": "3.1.6", + "resolved": "https://registry.npmmirror.com/minipass/-/minipass-3.1.6.tgz", + "integrity": "sha512-rty5kpw9/z8SX9dmxblFA6edItUmwJgMeYDZRrwlIVN27i8gysGbznJwUggw2V/FVqFSDdWy040ZPS811DYAqQ==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/minipass-collect/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/minipass-fetch": { + "version": "1.4.1", + "resolved": "https://registry.npmmirror.com/minipass-fetch/-/minipass-fetch-1.4.1.tgz", + "integrity": "sha512-CGH1eblLq26Y15+Azk7ey4xh0J/XfJfrCox5LDJiKqI2Q2iwOLOKrlmIaODiSQS8d18jalF6y2K2ePUm0CmShw==", + "dev": true, + "dependencies": { + "minipass": "^3.1.0", + "minipass-sized": "^1.0.3", + "minizlib": "^2.0.0" + }, + "engines": { + "node": ">=8" + }, + "optionalDependencies": { + "encoding": "^0.1.12" + } + }, + "node_modules/minipass-fetch/node_modules/minipass": { + "version": "3.1.6", + "resolved": "https://registry.npmmirror.com/minipass/-/minipass-3.1.6.tgz", + "integrity": "sha512-rty5kpw9/z8SX9dmxblFA6edItUmwJgMeYDZRrwlIVN27i8gysGbznJwUggw2V/FVqFSDdWy040ZPS811DYAqQ==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/minipass-fetch/node_modules/minizlib": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/minizlib/-/minizlib-2.1.2.tgz", + "integrity": "sha512-bAxsR8BVfj60DWXHE3u30oHzfl4G7khkSuPW+qvpd7jFRHm7dLxOjUk1EHACJ/hxLY8phGJ0YhYHZo7jil7Qdg==", + "dev": true, + "dependencies": { + "minipass": "^3.0.0", + "yallist": "^4.0.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/minipass-fetch/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/minipass-flush": { + "version": "1.0.5", + "resolved": "https://registry.npmmirror.com/minipass-flush/-/minipass-flush-1.0.5.tgz", + "integrity": "sha512-JmQSYYpPUqX5Jyn1mXaRwOda1uQ8HP5KAT/oDSLCzt1BYRhQU0/hDtsB1ufZfEEzMZ9aAVmsBw8+FWsIXlClWw==", + "dev": true, + "dependencies": { + "minipass": "^3.0.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/minipass-flush/node_modules/minipass": { + "version": "3.1.6", + "resolved": "https://registry.npmmirror.com/minipass/-/minipass-3.1.6.tgz", + "integrity": "sha512-rty5kpw9/z8SX9dmxblFA6edItUmwJgMeYDZRrwlIVN27i8gysGbznJwUggw2V/FVqFSDdWy040ZPS811DYAqQ==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/minipass-flush/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/minipass-json-stream": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/minipass-json-stream/-/minipass-json-stream-1.0.1.tgz", + "integrity": "sha512-ODqY18UZt/I8k+b7rl2AENgbWE8IDYam+undIJONvigAz8KR5GWblsFTEfQs0WODsjbSXWlm+JHEv8Gr6Tfdbg==", + "dev": true, + "dependencies": { + "jsonparse": "^1.3.1", + "minipass": "^3.0.0" + } + }, + "node_modules/minipass-json-stream/node_modules/minipass": { + "version": "3.1.6", + "resolved": "https://registry.npmmirror.com/minipass/-/minipass-3.1.6.tgz", + "integrity": "sha512-rty5kpw9/z8SX9dmxblFA6edItUmwJgMeYDZRrwlIVN27i8gysGbznJwUggw2V/FVqFSDdWy040ZPS811DYAqQ==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/minipass-json-stream/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/minipass-pipeline": { + "version": "1.2.4", + "resolved": "https://registry.npmmirror.com/minipass-pipeline/-/minipass-pipeline-1.2.4.tgz", + "integrity": "sha512-xuIq7cIOt09RPRJ19gdi4b+RiNvDFYe5JH+ggNvBqGqpQXcru3PcRmOZuHBKWK1Txf9+cQ+HMVN4d6z46LZP7A==", + "dev": true, + "dependencies": { + "minipass": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/minipass-pipeline/node_modules/minipass": { + "version": "3.1.6", + "resolved": "https://registry.npmmirror.com/minipass/-/minipass-3.1.6.tgz", + "integrity": "sha512-rty5kpw9/z8SX9dmxblFA6edItUmwJgMeYDZRrwlIVN27i8gysGbznJwUggw2V/FVqFSDdWy040ZPS811DYAqQ==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/minipass-pipeline/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/minipass-sized": { + "version": "1.0.3", + "resolved": "https://registry.npmmirror.com/minipass-sized/-/minipass-sized-1.0.3.tgz", + "integrity": "sha512-MbkQQ2CTiBMlA2Dm/5cY+9SWFEN8pzzOXi6rlM5Xxq0Yqbda5ZQy9sU75a673FE9ZK0Zsbr6Y5iP6u9nktfg2g==", + "dev": true, + "dependencies": { + "minipass": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/minipass-sized/node_modules/minipass": { + "version": "3.1.6", + "resolved": "https://registry.npmmirror.com/minipass/-/minipass-3.1.6.tgz", + "integrity": "sha512-rty5kpw9/z8SX9dmxblFA6edItUmwJgMeYDZRrwlIVN27i8gysGbznJwUggw2V/FVqFSDdWy040ZPS811DYAqQ==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/minipass-sized/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/minizlib": { + "version": "1.3.3", + "resolved": "https://registry.npmmirror.com/minizlib/-/minizlib-1.3.3.tgz", + "integrity": "sha512-6ZYMOEnmVsdCeTJVE0W9ZD+pVnE8h9Hma/iOwwRDsdQoePpoX56/8B6z3P9VNwppJuBKNRuFDRNRqRWexT9G9Q==", + "dev": true, + "dependencies": { + "minipass": "^2.9.0" + } + }, + "node_modules/mississippi": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/mississippi/-/mississippi-3.0.0.tgz", + "integrity": "sha512-x471SsVjUtBRtcvd4BzKE9kFC+/2TeWgKCgw0bZcw1b9l2X3QX5vCWgF+KaZaYm87Ss//rHnWryupDrgLvmSkA==", + "dependencies": { + "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" + }, + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/mitt": { + "version": "1.1.2", + "resolved": "https://registry.npmmirror.com/mitt/-/mitt-1.1.2.tgz", + "integrity": "sha512-3btxP0O9iGADGWAkteQ8mzDtEspZqu4I32y4GZYCV5BrwtzdcRpF4dQgNdJadCrbBx7Lu6Sq9AVrerMHR0Hkmw==", + "dev": true + }, + "node_modules/mixin-deep": { + "version": "1.3.2", + "resolved": "https://registry.npmmirror.com/mixin-deep/-/mixin-deep-1.3.2.tgz", + "integrity": "sha512-WRoDn//mXBiJ1H40rqa3vH0toePwSsGb45iInWlTySa+Uu4k3tYUSxa2v1KqAiLtvlrSzaExqS1gtk96A9zvEA==", + "dependencies": { + "for-in": "^1.0.2", + "is-extendable": "^1.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/mixin-deep/node_modules/is-extendable": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/is-extendable/-/is-extendable-1.0.1.tgz", + "integrity": "sha512-arnXMxT1hhoKo9k1LZdmlNyJdDDfy2v0fXjFlmok4+i8ul/6WlbVge9bhM74OpNPQPMGUToDtz+KXa1PneJxOA==", + "dependencies": { + "is-plain-object": "^2.0.4" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/mkdirp": { + "version": "0.5.6", + "resolved": "https://registry.npmmirror.com/mkdirp/-/mkdirp-0.5.6.tgz", + "integrity": "sha512-FP+p8RB8OWpF3YZBCrP5gtADmtXApB5AMLn+vdyA+PyxCjrCs00mjyUozssO33cwDeT3wNGdLxJ5M//YqtHAJw==", + "dependencies": { + "minimist": "^1.2.6" + }, + "bin": { + "mkdirp": "bin/cmd.js" + } + }, + "node_modules/mkdirp-infer-owner": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/mkdirp-infer-owner/-/mkdirp-infer-owner-2.0.0.tgz", + "integrity": "sha512-sdqtiFt3lkOaYvTXSRIUjkIdPTcxgv5+fgqYE/5qgwdw12cOrAuzzgzvVExIkH/ul1oeHN3bCLOWSG3XOqbKKw==", + "dev": true, + "dependencies": { + "chownr": "^2.0.0", + "infer-owner": "^1.0.4", + "mkdirp": "^1.0.3" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/mkdirp-infer-owner/node_modules/chownr": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/chownr/-/chownr-2.0.0.tgz", + "integrity": "sha512-bIomtDF5KGpdogkLd9VspvFzk9KfpyyGlS8YFVZl7TGPBHL5snIOnxeshwVgPteQ9b4Eydl+pVbIyE1DcvCWgQ==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/mkdirp-infer-owner/node_modules/mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmmirror.com/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true, + "bin": { + "mkdirp": "bin/cmd.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/ml-array-max": { + "version": "1.2.4", + "resolved": "https://registry.npmmirror.com/ml-array-max/-/ml-array-max-1.2.4.tgz", + "integrity": "sha512-BlEeg80jI0tW6WaPyGxf5Sa4sqvcyY6lbSn5Vcv44lp1I2GR6AWojfUvLnGTNsIXrZ8uqWmo8VcG1WpkI2ONMQ==", + "dependencies": { + "is-any-array": "^2.0.0" + } + }, + "node_modules/ml-array-min": { + "version": "1.2.3", + "resolved": "https://registry.npmmirror.com/ml-array-min/-/ml-array-min-1.2.3.tgz", + "integrity": "sha512-VcZ5f3VZ1iihtrGvgfh/q0XlMobG6GQ8FsNyQXD3T+IlstDv85g8kfV0xUG1QPRO/t21aukaJowDzMTc7j5V6Q==", + "dependencies": { + "is-any-array": "^2.0.0" + } + }, + "node_modules/ml-array-rescale": { + "version": "1.3.7", + "resolved": "https://registry.npmmirror.com/ml-array-rescale/-/ml-array-rescale-1.3.7.tgz", + "integrity": "sha512-48NGChTouvEo9KBctDfHC3udWnQKNKEWN0ziELvY3KG25GR5cA8K8wNVzracsqSW1QEkAXjTNx+ycgAv06/1mQ==", + "dependencies": { + "is-any-array": "^2.0.0", + "ml-array-max": "^1.2.4", + "ml-array-min": "^1.2.3" + } + }, + "node_modules/ml-matrix": { + "version": "6.10.0", + "resolved": "https://registry.npmmirror.com/ml-matrix/-/ml-matrix-6.10.0.tgz", + "integrity": "sha512-wU+jacx1dcP1QArV1/Kv49Ah6y2fq+BiQl2BnNVBC+hoCW7KgBZ4YZrowPopeoY164TB6Kes5wMeDjY8ODHYDg==", + "dependencies": { + "is-any-array": "^2.0.0", + "ml-array-rescale": "^1.3.7" + } + }, + "node_modules/mockjs": { + "version": "1.1.0", + "resolved": "https://registry.npmmirror.com/mockjs/-/mockjs-1.1.0.tgz", + "integrity": "sha512-eQsKcWzIaZzEZ07NuEyO4Nw65g0hdWAyurVol1IPl1gahRwY+svqzfgfey8U8dahLwG44d6/RwEzuK52rSa/JQ==", + "dev": true, + "dependencies": { + "commander": "*" + }, + "bin": { + "random": "bin/random" + } + }, + "node_modules/modify-values": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/modify-values/-/modify-values-1.0.1.tgz", + "integrity": "sha512-xV2bxeN6F7oYjZWTe/YPAy6MN2M+sL4u/Rlm2AHCIVGfo2p1yGmBHQ6vHehl4bRTZBdHu3TSkWdYgkwpYzAGSw==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/moment": { + "version": "2.29.4", + "resolved": "https://registry.npmmirror.com/moment/-/moment-2.29.4.tgz", + "integrity": "sha512-5LC9SOxjSc2HF6vO2CyuTDNivEdoz2IvyJJGj6X8DJ0eFyfszE0QiEd+iXmBvUP3WHxSjFH/vIsA0EN00cgr8w==", + "engines": { + "node": "*" + } + }, + "node_modules/moment-timezone": { + "version": "0.5.34", + "resolved": "https://registry.npmmirror.com/moment-timezone/-/moment-timezone-0.5.34.tgz", + "integrity": "sha512-3zAEHh2hKUs3EXLESx/wsgw6IQdusOT8Bxm3D9UrHPQR7zlMmzwybC8zHEM1tQ4LJwP7fcxrWr8tuBg05fFCbg==", + "dependencies": { + "moment": ">= 2.9.0" + }, + "engines": { + "node": "*" + } + }, + "node_modules/monaco-editor": { + "version": "0.19.3", + "resolved": "https://registry.npmmirror.com/monaco-editor/-/monaco-editor-0.19.3.tgz", + "integrity": "sha512-2n1vJBVQF2Hhi7+r1mMeYsmlf18hjVb6E0v5SoMZyb4aeOmYPKun+CE3gYpiNA1KEvtSdaDHFBqH9d7Wd9vREg==" + }, + "node_modules/monaco-editor-webpack-plugin": { + "version": "1.8.2", + "resolved": "https://registry.npmmirror.com/monaco-editor-webpack-plugin/-/monaco-editor-webpack-plugin-1.8.2.tgz", + "integrity": "sha512-g9G7A/lxQtpPsYaZFBqm73dwVkOziGUXExIR6iW7ksZUaiMkpvdTiE9O8edgdJGo+XtCmjycmIKB1Lt8VKbSTQ==", + "dev": true, + "dependencies": { + "loader-utils": "^1.2.3" + }, + "peerDependencies": { + "monaco-editor": "^0.19.1", + "webpack": "^4.5.0" + } + }, + "node_modules/monaco-languageclient": { + "version": "0.13.0", + "resolved": "https://registry.npmmirror.com/monaco-languageclient/-/monaco-languageclient-0.13.0.tgz", + "integrity": "sha512-aCwd33dTitwV5QwY56rpYHwzEGXei8TZ+yvZcvP3gEMd6Mizr8m3pOuoknDi2SUfLuNAHS6+ulvLgZlNQB5awg==", + "workspaces": [ + ".", + "workspace-a", + "workspace-b" + ], + "dependencies": { + "glob-to-regexp": "^0.3.0", + "vscode-jsonrpc": "^5.0.0", + "vscode-languageclient": "^6.0.0", + "vscode-uri": "^2.1.1" + }, + "engines": { + "vscode": "^1.41.0" + } + }, + "node_modules/move-concurrently": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/move-concurrently/-/move-concurrently-1.0.1.tgz", + "integrity": "sha512-hdrFxZOycD/g6A6SoI2bB5NA/5NEqD0569+S47WZhPvm46sD50ZHdYaFmnua5lndde9rCHGjmfK7Z8BuCt/PcQ==", + "dependencies": { + "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" + } + }, + "node_modules/ms": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/ms/-/ms-2.0.0.tgz", + "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==" + }, + "node_modules/multicast-dns": { + "version": "6.2.3", + "resolved": "https://registry.npmmirror.com/multicast-dns/-/multicast-dns-6.2.3.tgz", + "integrity": "sha512-ji6J5enbMyGRHIAkAOu3WdV8nggqviKCEKtXcOqfphZZtQrmHKycfynJ2V7eVPUA4NhJ6V7Wf4TmGbTwKE9B6g==", + "dev": true, + "dependencies": { + "dns-packet": "^1.3.1", + "thunky": "^1.0.2" + }, + "bin": { + "multicast-dns": "cli.js" + } + }, + "node_modules/multicast-dns-service-types": { + "version": "1.1.0", + "resolved": "https://registry.npmmirror.com/multicast-dns-service-types/-/multicast-dns-service-types-1.1.0.tgz", + "integrity": "sha512-cnAsSVxIDsYt0v7HmC0hWZFwwXSh+E6PgCrREDuN/EsjgLwA5XRmlMHhSiDPrt6HxY1gTivEa/Zh7GtODoLevQ==", + "dev": true + }, + "node_modules/multimatch": { + "version": "5.0.0", + "resolved": "https://registry.npmmirror.com/multimatch/-/multimatch-5.0.0.tgz", + "integrity": "sha512-ypMKuglUrZUD99Tk2bUQ+xNQj43lPEfAeX2o9cTteAmShXy2VHDJpuwu1o0xqoKCt9jLVAvwyFKdLTPXKAfJyA==", + "dev": true, + "dependencies": { + "@types/minimatch": "^3.0.3", + "array-differ": "^3.0.0", + "array-union": "^2.1.0", + "arrify": "^2.0.1", + "minimatch": "^3.0.4" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/multimatch/node_modules/array-union": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/array-union/-/array-union-2.1.0.tgz", + "integrity": "sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/mute-stream": { + "version": "0.0.7", + "resolved": "https://registry.npmmirror.com/mute-stream/-/mute-stream-0.0.7.tgz", + "integrity": "sha512-r65nCZhrbXXb6dXOACihYApHw2Q6pV0M3V0PSxd74N0+D8nzAdEAITq2oAjA1jVnKI+tGvEBUpqiMh0+rW6zDQ==", + "dev": true + }, + "node_modules/mv": { + "version": "2.1.1", + "resolved": "https://registry.npmmirror.com/mv/-/mv-2.1.1.tgz", + "integrity": "sha512-at/ZndSy3xEGJ8i0ygALh8ru9qy7gWW1cmkaqBN29JmMlIvM//MEO9y1sk/avxuwnPcfhkejkLsuPxH81BrkSg==", + "dev": true, + "dependencies": { + "mkdirp": "~0.5.1", + "ncp": "~2.0.0", + "rimraf": "~2.4.0" + }, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/mv/node_modules/glob": { + "version": "6.0.4", + "resolved": "https://registry.npmmirror.com/glob/-/glob-6.0.4.tgz", + "integrity": "sha512-MKZeRNyYZAVVVG1oZeLaWie1uweH40m9AZwIwxyPbTSX4hHrVYSzLg0Ro5Z5R7XKkIX+Cc6oD1rqeDJnwsB8/A==", + "dev": true, + "dependencies": { + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "2 || 3", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + }, + "engines": { + "node": "*" + } + }, + "node_modules/mv/node_modules/rimraf": { + "version": "2.4.5", + "resolved": "https://registry.npmmirror.com/rimraf/-/rimraf-2.4.5.tgz", + "integrity": "sha512-J5xnxTyqaiw06JjMftq7L9ouA448dw/E7dKghkP9WpKNuwmARNNg+Gk8/u5ryb9N/Yo2+z3MCwuqFK/+qPOPfQ==", + "dev": true, + "dependencies": { + "glob": "^6.0.1" + }, + "bin": { + "rimraf": "bin.js" + } + }, + "node_modules/mz": { + "version": "2.7.0", + "resolved": "https://registry.npmmirror.com/mz/-/mz-2.7.0.tgz", + "integrity": "sha512-z81GNO7nnYMEhrGh9LeymoE4+Yr0Wn5McHIZMK5cfQCl+NDX08sCZgUc9/6MHni9IWuFLm1Z3HTCXu2z9fN62Q==", + "dev": true, + "dependencies": { + "any-promise": "^1.0.0", + "object-assign": "^4.0.1", + "thenify-all": "^1.0.0" + } + }, + "node_modules/nan": { + "version": "2.15.0", + "resolved": "https://registry.npmmirror.com/nan/-/nan-2.15.0.tgz", + "integrity": "sha512-8ZtvEnA2c5aYCZYd1cvgdnU6cqwixRoYg70xPLWUws5ORTa/lnw+u4amixRS/Ac5U5mQVgp9pnlSUnbNWFaWZQ==", + "devOptional": true + }, + "node_modules/nanomatch": { + "version": "1.2.13", + "resolved": "https://registry.npmmirror.com/nanomatch/-/nanomatch-1.2.13.tgz", + "integrity": "sha512-fpoe2T0RbHwBTBUOftAfBPaDEi06ufaUai0mE6Yn1kacc3SnTErfb/h+X94VXzI64rKFHYImXSvdwGGCmwOqCA==", + "dependencies": { + "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" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/native-request": { + "version": "1.1.0", + "resolved": "https://registry.npmmirror.com/native-request/-/native-request-1.1.0.tgz", + "integrity": "sha512-uZ5rQaeRn15XmpgE0xoPL8YWqcX90VtCFglYwAgkvKM5e8fog+vePLAhHxuuv/gRkrQxIeh5U3q9sMNUrENqWw==", + "dev": true, + "optional": true + }, + "node_modules/natural-compare": { + "version": "1.4.0", + "resolved": "https://registry.npmmirror.com/natural-compare/-/natural-compare-1.4.0.tgz", + "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==", + "dev": true + }, + "node_modules/ncp": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/ncp/-/ncp-2.0.0.tgz", + "integrity": "sha512-zIdGUrPRFTUELUvr3Gmc7KZ2Sw/h1PiVM0Af/oHB6zgnV1ikqSfRk+TOufi79aHYCW3NiOXmr1BP5nWbzojLaA==", + "dev": true, + "bin": { + "ncp": "bin/ncp" + } + }, + "node_modules/negotiator": { + "version": "0.6.3", + "resolved": "https://registry.npmmirror.com/negotiator/-/negotiator-0.6.3.tgz", + "integrity": "sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg==", + "dev": true, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/neo-async": { + "version": "2.6.2", + "resolved": "https://registry.npmmirror.com/neo-async/-/neo-async-2.6.2.tgz", + "integrity": "sha512-Yd3UES5mWCSqR+qNT93S3UoYUkqAZ9lLg8a7g9rimsWmYGK8cVToA4/sF3RrshdyV3sAGMXVUmpMYOw+dLpOuw==" + }, + "node_modules/nice-try": { + "version": "1.0.5", + "resolved": "https://registry.npmmirror.com/nice-try/-/nice-try-1.0.5.tgz", + "integrity": "sha512-1nh45deeb5olNY7eX82BkPO7SSxR5SSYJiPTrTdFUVYwAl8CKMA5N9PjTYkHiRjisVcxcQ1HXdLhx2qxxJzLNQ==", + "dev": true + }, + "node_modules/no-case": { + "version": "2.3.2", + "resolved": "https://registry.npmmirror.com/no-case/-/no-case-2.3.2.tgz", + "integrity": "sha512-rmTZ9kz+f3rCvK2TD1Ue/oZlns7OGoIWP4fc3llxxRXlOkHKoWPPWJOfFYpITabSow43QJbRIoHQXtt10VldyQ==", + "dev": true, + "dependencies": { + "lower-case": "^1.1.1" + } + }, + "node_modules/node-fetch": { + "version": "2.6.7", + "resolved": "https://registry.npmmirror.com/node-fetch/-/node-fetch-2.6.7.tgz", + "integrity": "sha512-ZjMPFEfVx5j+y2yF35Kzx5sF7kDzxuDj6ziH4FFbOp87zKDZNx8yExJIb05OGF4Nlt9IHFIMBkRl41VdvcNdbQ==", + "dev": true, + "dependencies": { + "whatwg-url": "^5.0.0" + }, + "engines": { + "node": "4.x || >=6.0.0" + }, + "peerDependencies": { + "encoding": "^0.1.0" + }, + "peerDependenciesMeta": { + "encoding": { + "optional": true + } + } + }, + "node_modules/node-fetch/node_modules/tr46": { + "version": "0.0.3", + "resolved": "https://registry.npmmirror.com/tr46/-/tr46-0.0.3.tgz", + "integrity": "sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw==", + "dev": true + }, + "node_modules/node-fetch/node_modules/webidl-conversions": { + "version": "3.0.1", + "resolved": "https://registry.npmmirror.com/webidl-conversions/-/webidl-conversions-3.0.1.tgz", + "integrity": "sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ==", + "dev": true + }, + "node_modules/node-fetch/node_modules/whatwg-url": { + "version": "5.0.0", + "resolved": "https://registry.npmmirror.com/whatwg-url/-/whatwg-url-5.0.0.tgz", + "integrity": "sha512-saE57nupxk6v3HY35+jzBwYa0rKSy0XR8JSxZPwgLr7ys0IBzhGviA1/TUGJLmSVqs8pb9AnvICXEuOHLprYTw==", + "dev": true, + "dependencies": { + "tr46": "~0.0.3", + "webidl-conversions": "^3.0.0" + } + }, + "node_modules/node-forge": { + "version": "0.10.0", + "resolved": "https://registry.npmmirror.com/node-forge/-/node-forge-0.10.0.tgz", + "integrity": "sha512-PPmu8eEeG9saEUvI97fm4OYxXVB6bFvyNTyiUOBichBpFG8A1Ljw3bY62+5oOjDEMHRnd0Y7HQ+x7uzxOzC6JA==", + "dev": true, + "engines": { + "node": ">= 6.0.0" + } + }, + "node_modules/node-gyp": { + "version": "5.1.1", + "resolved": "https://registry.npmmirror.com/node-gyp/-/node-gyp-5.1.1.tgz", + "integrity": "sha512-WH0WKGi+a4i4DUt2mHnvocex/xPLp9pYt5R6M2JdFB7pJ7Z34hveZ4nDTGTiLXCkitA9T8HFZjhinBCiVHYcWw==", + "dev": true, + "dependencies": { + "env-paths": "^2.2.0", + "glob": "^7.1.4", + "graceful-fs": "^4.2.2", + "mkdirp": "^0.5.1", + "nopt": "^4.0.1", + "npmlog": "^4.1.2", + "request": "^2.88.0", + "rimraf": "^2.6.3", + "semver": "^5.7.1", + "tar": "^4.4.12", + "which": "^1.3.1" + }, + "bin": { + "node-gyp": "bin/node-gyp.js" + }, + "engines": { + "node": ">= 6.0.0" + } + }, + "node_modules/node-ipc": { + "version": "9.2.1", + "resolved": "https://registry.npmmirror.com/node-ipc/-/node-ipc-9.2.1.tgz", + "integrity": "sha512-mJzaM6O3xHf9VT8BULvJSbdVbmHUKRNOH7zDDkCrA1/T+CVjq2WVIDfLt0azZRXpgArJtl3rtmEozrbXPZ9GaQ==", + "dev": true, + "dependencies": { + "event-pubsub": "4.3.0", + "js-message": "1.0.7", + "js-queue": "2.0.2" + }, + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/node-libs-browser": { + "version": "2.2.1", + "resolved": "https://registry.npmmirror.com/node-libs-browser/-/node-libs-browser-2.2.1.tgz", + "integrity": "sha512-h/zcD8H9kaDZ9ALUWwlBUDo6TKF8a7qBSCSEGfjTVIYeqsioSKaAX+BN7NgiMGp6iSIXZ3PxgCu8KS3b71YK5Q==", + "dependencies": { + "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" + } + }, + "node_modules/node-libs-browser/node_modules/punycode": { + "version": "1.4.1", + "resolved": "https://registry.npmmirror.com/punycode/-/punycode-1.4.1.tgz", + "integrity": "sha512-jmYNElW7yvO7TV33CjSmvSiE2yco3bV2czu/OzDKdMNVZQWfxCblURLhf+47syQRBntjfLdd/H0egrzIG+oaFQ==" + }, + "node_modules/node-releases": { + "version": "2.0.3", + "resolved": "https://registry.npmmirror.com/node-releases/-/node-releases-2.0.3.tgz", + "integrity": "sha512-maHFz6OLqYxz+VQyCAtA3PTX4UP/53pa05fyDNc9CwjvJ0yEh6+xBwKsgCxMNhS8taUKBFYxfuiaD9U/55iFaw==", + "dev": true + }, + "node_modules/node-sass": { + "version": "6.0.1", + "resolved": "https://registry.npmmirror.com/node-sass/-/node-sass-6.0.1.tgz", + "integrity": "sha512-f+Rbqt92Ful9gX0cGtdYwjTrWAaGURgaK5rZCWOgCNyGWusFYHhbqCCBoFBeat+HKETOU02AyTxNhJV0YZf2jQ==", + "dev": true, + "hasInstallScript": true, + "dependencies": { + "async-foreach": "^0.1.3", + "chalk": "^1.1.1", + "cross-spawn": "^7.0.3", + "gaze": "^1.0.0", + "get-stdin": "^4.0.1", + "glob": "^7.0.3", + "lodash": "^4.17.15", + "meow": "^9.0.0", + "nan": "^2.13.2", + "node-gyp": "^7.1.0", + "npmlog": "^4.0.0", + "request": "^2.88.0", + "sass-graph": "2.2.5", + "stdout-stream": "^1.4.0", + "true-case-path": "^1.0.2" + }, + "bin": { + "node-sass": "bin/node-sass" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/node-sass/node_modules/ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmmirror.com/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha512-TIGnTpdo+E3+pCyAluZvtED5p5wCqLdezCyhPZzKPcxvFplEt4i+W7OONCKgeZFT3+y5NZZfOOS/Bdcanm1MYA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/node-sass/node_modules/ansi-styles": { + "version": "2.2.1", + "resolved": "https://registry.npmmirror.com/ansi-styles/-/ansi-styles-2.2.1.tgz", + "integrity": "sha512-kmCevFghRiWM7HB5zTPULl4r9bVFSWjz62MhqizDGUrq2NWuNMQyuv4tHHoKJHs69M/MF64lEcHdYIocrdWQYA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/node-sass/node_modules/chalk": { + "version": "1.1.3", + "resolved": "https://registry.npmmirror.com/chalk/-/chalk-1.1.3.tgz", + "integrity": "sha512-U3lRVLMSlsCfjqYPbLyVv11M9CPW4I728d6TCKMAOJueEeB9/8o+eSsMnxPJD+Q+K909sdESg7C+tIkoH6on1A==", + "dev": true, + "dependencies": { + "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" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/node-sass/node_modules/chownr": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/chownr/-/chownr-2.0.0.tgz", + "integrity": "sha512-bIomtDF5KGpdogkLd9VspvFzk9KfpyyGlS8YFVZl7TGPBHL5snIOnxeshwVgPteQ9b4Eydl+pVbIyE1DcvCWgQ==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/node-sass/node_modules/cross-spawn": { + "version": "7.0.3", + "resolved": "https://registry.npmmirror.com/cross-spawn/-/cross-spawn-7.0.3.tgz", + "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", + "dev": true, + "dependencies": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/node-sass/node_modules/find-up": { + "version": "4.1.0", + "resolved": "https://registry.npmmirror.com/find-up/-/find-up-4.1.0.tgz", + "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", + "dev": true, + "dependencies": { + "locate-path": "^5.0.0", + "path-exists": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/node-sass/node_modules/fs-minipass": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/fs-minipass/-/fs-minipass-2.1.0.tgz", + "integrity": "sha512-V/JgOLFCS+R6Vcq0slCuaeWEdNC3ouDlJMNIsacH2VtALiu9mV4LPrHc5cDl8k5aw6J8jwgWWpiTo5RYhmIzvg==", + "dev": true, + "dependencies": { + "minipass": "^3.0.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/node-sass/node_modules/get-stdin": { + "version": "4.0.1", + "resolved": "https://registry.npmmirror.com/get-stdin/-/get-stdin-4.0.1.tgz", + "integrity": "sha512-F5aQMywwJ2n85s4hJPTT9RPxGmubonuB10MNYo17/xph174n2MIR33HRguhzVag10O/npM7SPk73LMZNP+FaWw==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/node-sass/node_modules/hosted-git-info": { + "version": "4.1.0", + "resolved": "https://registry.npmmirror.com/hosted-git-info/-/hosted-git-info-4.1.0.tgz", + "integrity": "sha512-kyCuEOWjJqZuDbRHzL8V93NzQhwIB71oFWSyzVo+KPZI+pnQPPxucdkrOZvkLRnrf5URsQM+IJ09Dw29cRALIA==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/node-sass/node_modules/locate-path": { + "version": "5.0.0", + "resolved": "https://registry.npmmirror.com/locate-path/-/locate-path-5.0.0.tgz", + "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", + "dev": true, + "dependencies": { + "p-locate": "^4.1.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/node-sass/node_modules/lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/node-sass/node_modules/meow": { + "version": "9.0.0", + "resolved": "https://registry.npmmirror.com/meow/-/meow-9.0.0.tgz", + "integrity": "sha512-+obSblOQmRhcyBt62furQqRAQpNyWXo8BuQ5bN7dG8wmwQ+vwHKp/rCFD4CrTP8CsDQD1sjoZ94K417XEUk8IQ==", + "dev": true, + "dependencies": { + "@types/minimist": "^1.2.0", + "camelcase-keys": "^6.2.2", + "decamelize": "^1.2.0", + "decamelize-keys": "^1.1.0", + "hard-rejection": "^2.1.0", + "minimist-options": "4.1.0", + "normalize-package-data": "^3.0.0", + "read-pkg-up": "^7.0.1", + "redent": "^3.0.0", + "trim-newlines": "^3.0.0", + "type-fest": "^0.18.0", + "yargs-parser": "^20.2.3" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/node-sass/node_modules/minipass": { + "version": "3.3.4", + "resolved": "https://registry.npmmirror.com/minipass/-/minipass-3.3.4.tgz", + "integrity": "sha512-I9WPbWHCGu8W+6k1ZiGpPu0GkoKBeorkfKNuAFBNS1HNFJvke82sxvI5bzcCNpWPorkOO5QQ+zomzzwRxejXiw==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/node-sass/node_modules/minizlib": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/minizlib/-/minizlib-2.1.2.tgz", + "integrity": "sha512-bAxsR8BVfj60DWXHE3u30oHzfl4G7khkSuPW+qvpd7jFRHm7dLxOjUk1EHACJ/hxLY8phGJ0YhYHZo7jil7Qdg==", + "dev": true, + "dependencies": { + "minipass": "^3.0.0", + "yallist": "^4.0.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/node-sass/node_modules/mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmmirror.com/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true, + "bin": { + "mkdirp": "bin/cmd.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/node-sass/node_modules/node-gyp": { + "version": "7.1.2", + "resolved": "https://registry.npmmirror.com/node-gyp/-/node-gyp-7.1.2.tgz", + "integrity": "sha512-CbpcIo7C3eMu3dL1c3d0xw449fHIGALIJsRP4DDPHpyiW8vcriNY7ubh9TE4zEKfSxscY7PjeFnshE7h75ynjQ==", + "dev": true, + "dependencies": { + "env-paths": "^2.2.0", + "glob": "^7.1.4", + "graceful-fs": "^4.2.3", + "nopt": "^5.0.0", + "npmlog": "^4.1.2", + "request": "^2.88.2", + "rimraf": "^3.0.2", + "semver": "^7.3.2", + "tar": "^6.0.2", + "which": "^2.0.2" + }, + "bin": { + "node-gyp": "bin/node-gyp.js" + }, + "engines": { + "node": ">= 10.12.0" + } + }, + "node_modules/node-sass/node_modules/nopt": { + "version": "5.0.0", + "resolved": "https://registry.npmmirror.com/nopt/-/nopt-5.0.0.tgz", + "integrity": "sha512-Tbj67rffqceeLpcRXrT7vKAN8CwfPeIBgM7E6iBkmKLV7bEMwpGgYLGv0jACUsECaa/vuxP0IjEont6umdMgtQ==", + "dev": true, + "dependencies": { + "abbrev": "1" + }, + "bin": { + "nopt": "bin/nopt.js" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/node-sass/node_modules/normalize-package-data": { + "version": "3.0.3", + "resolved": "https://registry.npmmirror.com/normalize-package-data/-/normalize-package-data-3.0.3.tgz", + "integrity": "sha512-p2W1sgqij3zMMyRC067Dg16bfzVH+w7hyegmpIvZ4JNjqtGOVAIvLmjBx3yP7YTe9vKJgkoNOPjwQGogDoMXFA==", + "dev": true, + "dependencies": { + "hosted-git-info": "^4.0.1", + "is-core-module": "^2.5.0", + "semver": "^7.3.4", + "validate-npm-package-license": "^3.0.1" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/node-sass/node_modules/p-locate": { + "version": "4.1.0", + "resolved": "https://registry.npmmirror.com/p-locate/-/p-locate-4.1.0.tgz", + "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", + "dev": true, + "dependencies": { + "p-limit": "^2.2.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/node-sass/node_modules/path-exists": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/path-exists/-/path-exists-4.0.0.tgz", + "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/node-sass/node_modules/path-key": { + "version": "3.1.1", + "resolved": "https://registry.npmmirror.com/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/node-sass/node_modules/read-pkg-up": { + "version": "7.0.1", + "resolved": "https://registry.npmmirror.com/read-pkg-up/-/read-pkg-up-7.0.1.tgz", + "integrity": "sha512-zK0TB7Xd6JpCLmlLmufqykGE+/TlOePD6qKClNW7hHDKFh/J7/7gCWGR7joEQEW1bKq3a3yUZSObOoWLFQ4ohg==", + "dev": true, + "dependencies": { + "find-up": "^4.1.0", + "read-pkg": "^5.2.0", + "type-fest": "^0.8.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/node-sass/node_modules/read-pkg-up/node_modules/type-fest": { + "version": "0.8.1", + "resolved": "https://registry.npmmirror.com/type-fest/-/type-fest-0.8.1.tgz", + "integrity": "sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/node-sass/node_modules/rimraf": { + "version": "3.0.2", + "resolved": "https://registry.npmmirror.com/rimraf/-/rimraf-3.0.2.tgz", + "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", + "dev": true, + "dependencies": { + "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" + } + }, + "node_modules/node-sass/node_modules/semver": { + "version": "7.3.8", + "resolved": "https://registry.npmmirror.com/semver/-/semver-7.3.8.tgz", + "integrity": "sha512-NB1ctGL5rlHrPJtFDVIVzTyQylMLu9N9VICA6HSFJo8MCGVTMW6gfpicwKmmK/dAjTOrqu5l63JJOpDSrAis3A==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/node-sass/node_modules/shebang-command": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/shebang-command/-/shebang-command-2.0.0.tgz", + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", + "dev": true, + "dependencies": { + "shebang-regex": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/node-sass/node_modules/shebang-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/shebang-regex/-/shebang-regex-3.0.0.tgz", + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/node-sass/node_modules/strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmmirror.com/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha512-VhumSSbBqDTP8p2ZLKj40UjBCV4+v8bUSEpUb4KjRgWk9pbqGF4REFj6KEagidb2f/M6AzC0EmFyDNGaw9OCzg==", + "dev": true, + "dependencies": { + "ansi-regex": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/node-sass/node_modules/supports-color": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/supports-color/-/supports-color-2.0.0.tgz", + "integrity": "sha512-KKNVtd6pCYgPIKU4cp2733HWYCpplQhddZLBUryaAHou723x+FRzQ5Df824Fj+IyyuiQTRoub4SnIFfIcrp70g==", + "dev": true, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/node-sass/node_modules/tar": { + "version": "6.1.12", + "resolved": "https://registry.npmmirror.com/tar/-/tar-6.1.12.tgz", + "integrity": "sha512-jU4TdemS31uABHd+Lt5WEYJuzn+TJTCBLljvIAHZOz6M9Os5pJ4dD+vRFLxPa/n3T0iEFzpi+0x1UfuDZYbRMw==", + "dev": true, + "dependencies": { + "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" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/node-sass/node_modules/type-fest": { + "version": "0.18.1", + "resolved": "https://registry.npmmirror.com/type-fest/-/type-fest-0.18.1.tgz", + "integrity": "sha512-OIAYXk8+ISY+qTOwkHtKqzAuxchoMiD9Udx+FSGQDuiRR+PJKJHc2NJAXlbhkGwTt/4/nKZxELY1w3ReWOL8mw==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/node-sass/node_modules/which": { + "version": "2.0.2", + "resolved": "https://registry.npmmirror.com/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "node-which": "bin/node-which" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/node-sass/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/nopt": { + "version": "4.0.3", + "resolved": "https://registry.npmmirror.com/nopt/-/nopt-4.0.3.tgz", + "integrity": "sha512-CvaGwVMztSMJLOeXPrez7fyfObdZqNUK1cPAEzLHrTybIua9pMdmmPR5YwtfNftIOMv3DPUhFaxsZMNTQO20Kg==", + "dev": true, + "dependencies": { + "abbrev": "1", + "osenv": "^0.1.4" + }, + "bin": { + "nopt": "bin/nopt.js" + } + }, + "node_modules/normalize-package-data": { + "version": "2.5.0", + "resolved": "https://registry.npmmirror.com/normalize-package-data/-/normalize-package-data-2.5.0.tgz", + "integrity": "sha512-/5CMN3T0R4XTj4DcGaexo+roZSdSFW/0AOOTROrjxzCG1wrWXEsGbRKevjlIL+ZDE4sZlJr5ED4YW0yqmkK+eA==", + "dev": true, + "dependencies": { + "hosted-git-info": "^2.1.4", + "resolve": "^1.10.0", + "semver": "2 || 3 || 4 || 5", + "validate-npm-package-license": "^3.0.1" + } + }, + "node_modules/normalize-path": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/normalize-path/-/normalize-path-3.0.0.tgz", + "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", + "devOptional": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/normalize-range": { + "version": "0.1.2", + "resolved": "https://registry.npmmirror.com/normalize-range/-/normalize-range-0.1.2.tgz", + "integrity": "sha512-bdok/XvKII3nUpklnV6P2hxtMNrCboOjAcyBuQnWEhO665FwrSNRxU+AqpsyvO6LgGYPspN+lu5CLtw4jPRKNA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/normalize-url": { + "version": "3.3.0", + "resolved": "https://registry.npmmirror.com/normalize-url/-/normalize-url-3.3.0.tgz", + "integrity": "sha512-U+JJi7duF1o+u2pynbp2zXDW2/PADgC30f0GsHZtRh+HOcXHnw137TrNlyxxRvWW5fjKd3bcLHPxofWuCjaeZg==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/npm-bundled": { + "version": "1.1.2", + "resolved": "https://registry.npmmirror.com/npm-bundled/-/npm-bundled-1.1.2.tgz", + "integrity": "sha512-x5DHup0SuyQcmL3s7Rx/YQ8sbw/Hzg0rj48eN0dV7hf5cmQq5PXIeioroH3raV1QC1yh3uTYuMThvEQF3iKgGQ==", + "dev": true, + "dependencies": { + "npm-normalize-package-bin": "^1.0.1" + } + }, + "node_modules/npm-install-checks": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/npm-install-checks/-/npm-install-checks-4.0.0.tgz", + "integrity": "sha512-09OmyDkNLYwqKPOnbI8exiOZU2GVVmQp7tgez2BPi5OZC8M82elDAps7sxC4l//uSUtotWqoEIDwjRvWH4qz8w==", + "dev": true, + "dependencies": { + "semver": "^7.1.1" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/npm-install-checks/node_modules/lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/npm-install-checks/node_modules/semver": { + "version": "7.3.7", + "resolved": "https://registry.npmmirror.com/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/npm-install-checks/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/npm-lifecycle": { + "version": "3.1.5", + "resolved": "https://registry.npmmirror.com/npm-lifecycle/-/npm-lifecycle-3.1.5.tgz", + "integrity": "sha512-lDLVkjfZmvmfvpvBzA4vzee9cn+Me4orq0QF8glbswJVEbIcSNWib7qGOffolysc3teCqbbPZZkzbr3GQZTL1g==", + "dev": true, + "dependencies": { + "byline": "^5.0.0", + "graceful-fs": "^4.1.15", + "node-gyp": "^5.0.2", + "resolve-from": "^4.0.0", + "slide": "^1.1.6", + "uid-number": "0.0.6", + "umask": "^1.1.0", + "which": "^1.3.1" + } + }, + "node_modules/npm-lifecycle/node_modules/resolve-from": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/resolve-from/-/resolve-from-4.0.0.tgz", + "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/npm-normalize-package-bin": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/npm-normalize-package-bin/-/npm-normalize-package-bin-1.0.1.tgz", + "integrity": "sha512-EPfafl6JL5/rU+ot6P3gRSCpPDW5VmIzX959Ob1+ySFUuuYHWHekXpwdUZcKP5C+DS4GEtdJluwBjnsNDl+fSA==", + "dev": true + }, + "node_modules/npm-package-arg": { + "version": "8.1.5", + "resolved": "https://registry.npmmirror.com/npm-package-arg/-/npm-package-arg-8.1.5.tgz", + "integrity": "sha512-LhgZrg0n0VgvzVdSm1oiZworPbTxYHUJCgtsJW8mGvlDpxTM1vSJc3m5QZeUkhAHIzbz3VCHd/R4osi1L1Tg/Q==", + "dev": true, + "dependencies": { + "hosted-git-info": "^4.0.1", + "semver": "^7.3.4", + "validate-npm-package-name": "^3.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/npm-package-arg/node_modules/hosted-git-info": { + "version": "4.1.0", + "resolved": "https://registry.npmmirror.com/hosted-git-info/-/hosted-git-info-4.1.0.tgz", + "integrity": "sha512-kyCuEOWjJqZuDbRHzL8V93NzQhwIB71oFWSyzVo+KPZI+pnQPPxucdkrOZvkLRnrf5URsQM+IJ09Dw29cRALIA==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/npm-package-arg/node_modules/lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/npm-package-arg/node_modules/semver": { + "version": "7.3.7", + "resolved": "https://registry.npmmirror.com/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/npm-package-arg/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/npm-packlist": { + "version": "2.2.2", + "resolved": "https://registry.npmmirror.com/npm-packlist/-/npm-packlist-2.2.2.tgz", + "integrity": "sha512-Jt01acDvJRhJGthnUJVF/w6gumWOZxO7IkpY/lsX9//zqQgnF7OJaxgQXcerd4uQOLu7W5bkb4mChL9mdfm+Zg==", + "dev": true, + "dependencies": { + "glob": "^7.1.6", + "ignore-walk": "^3.0.3", + "npm-bundled": "^1.1.1", + "npm-normalize-package-bin": "^1.0.1" + }, + "bin": { + "npm-packlist": "bin/index.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/npm-path": { + "version": "2.0.4", + "resolved": "https://registry.npmmirror.com/npm-path/-/npm-path-2.0.4.tgz", + "integrity": "sha512-IFsj0R9C7ZdR5cP+ET342q77uSRdtWOlWpih5eC+lu29tIDbNEgDbzgVJ5UFvYHWhxDZ5TFkJafFioO0pPQjCw==", + "dev": true, + "dependencies": { + "which": "^1.2.10" + }, + "bin": { + "npm-path": "bin/npm-path" + }, + "engines": { + "node": ">=0.8" + } + }, + "node_modules/npm-pick-manifest": { + "version": "6.1.1", + "resolved": "https://registry.npmmirror.com/npm-pick-manifest/-/npm-pick-manifest-6.1.1.tgz", + "integrity": "sha512-dBsdBtORT84S8V8UTad1WlUyKIY9iMsAmqxHbLdeEeBNMLQDlDWWra3wYUx9EBEIiG/YwAy0XyNHDd2goAsfuA==", + "dev": true, + "dependencies": { + "npm-install-checks": "^4.0.0", + "npm-normalize-package-bin": "^1.0.1", + "npm-package-arg": "^8.1.2", + "semver": "^7.3.4" + } + }, + "node_modules/npm-pick-manifest/node_modules/lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/npm-pick-manifest/node_modules/semver": { + "version": "7.3.7", + "resolved": "https://registry.npmmirror.com/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/npm-pick-manifest/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/npm-registry-fetch": { + "version": "11.0.0", + "resolved": "https://registry.npmmirror.com/npm-registry-fetch/-/npm-registry-fetch-11.0.0.tgz", + "integrity": "sha512-jmlgSxoDNuhAtxUIG6pVwwtz840i994dL14FoNVZisrmZW5kWd63IUTNv1m/hyRSGSqWjCUp/YZlS1BJyNp9XA==", + "dev": true, + "dependencies": { + "make-fetch-happen": "^9.0.1", + "minipass": "^3.1.3", + "minipass-fetch": "^1.3.0", + "minipass-json-stream": "^1.0.1", + "minizlib": "^2.0.0", + "npm-package-arg": "^8.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/npm-registry-fetch/node_modules/minipass": { + "version": "3.1.6", + "resolved": "https://registry.npmmirror.com/minipass/-/minipass-3.1.6.tgz", + "integrity": "sha512-rty5kpw9/z8SX9dmxblFA6edItUmwJgMeYDZRrwlIVN27i8gysGbznJwUggw2V/FVqFSDdWy040ZPS811DYAqQ==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/npm-registry-fetch/node_modules/minizlib": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/minizlib/-/minizlib-2.1.2.tgz", + "integrity": "sha512-bAxsR8BVfj60DWXHE3u30oHzfl4G7khkSuPW+qvpd7jFRHm7dLxOjUk1EHACJ/hxLY8phGJ0YhYHZo7jil7Qdg==", + "dev": true, + "dependencies": { + "minipass": "^3.0.0", + "yallist": "^4.0.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/npm-registry-fetch/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/npm-run-path": { + "version": "2.0.2", + "resolved": "https://registry.npmmirror.com/npm-run-path/-/npm-run-path-2.0.2.tgz", + "integrity": "sha512-lJxZYlT4DW/bRUtFh1MQIWqmLwQfAxnqWG4HhEdjMlkrJYnJn0Jrr2u3mgxqaWsdiBc76TYkTG/mhrnYTuzfHw==", + "dev": true, + "dependencies": { + "path-key": "^2.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/npm-which": { + "version": "3.0.1", + "resolved": "https://registry.npmmirror.com/npm-which/-/npm-which-3.0.1.tgz", + "integrity": "sha512-CM8vMpeFQ7MAPin0U3wzDhSGV0hMHNwHU0wjo402IVizPDrs45jSfSuoC+wThevY88LQti8VvaAnqYAeVy3I1A==", + "dev": true, + "dependencies": { + "commander": "^2.9.0", + "npm-path": "^2.0.2", + "which": "^1.2.10" + }, + "bin": { + "npm-which": "bin/npm-which.js" + }, + "engines": { + "node": ">=4.2.0" + } + }, + "node_modules/npmlog": { + "version": "4.1.2", + "resolved": "https://registry.npmmirror.com/npmlog/-/npmlog-4.1.2.tgz", + "integrity": "sha512-2uUqazuKlTaSI/dC8AzicUck7+IrEaOnN/e0jd3Xtt1KcGpwx30v50mL7oPyr/h9bL3E4aZccVwpwP+5W9Vjkg==", + "dev": true, + "dependencies": { + "are-we-there-yet": "~1.1.2", + "console-control-strings": "~1.1.0", + "gauge": "~2.7.3", + "set-blocking": "~2.0.0" + } + }, + "node_modules/nth-check": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/nth-check/-/nth-check-1.0.2.tgz", + "integrity": "sha512-WeBOdju8SnzPN5vTUJYxYUxLeXpCaVP5i5e0LF8fg7WORF2Wd7wFX/pk0tYZk7s8T+J7VLy0Da6J1+wCT0AtHg==", + "dependencies": { + "boolbase": "~1.0.0" + } + }, + "node_modules/num2fraction": { + "version": "1.2.2", + "resolved": "https://registry.npmmirror.com/num2fraction/-/num2fraction-1.2.2.tgz", + "integrity": "sha512-Y1wZESM7VUThYY+4W+X4ySH2maqcA+p7UR+w8VWNWVAd6lwuXXWz/w/Cz43J/dI2I+PS6wD5N+bJUF+gjWvIqg==", + "dev": true + }, + "node_modules/number-is-nan": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/number-is-nan/-/number-is-nan-1.0.1.tgz", + "integrity": "sha512-4jbtZXNAsfZbAHiiqjLPBiCl16dES1zI4Hpzzxw61Tk+loF+sBDBKx1ICKKKwIqQ7M0mFn1TmkN7euSncWgHiQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/oauth-sign": { + "version": "0.9.0", + "resolved": "https://registry.npmmirror.com/oauth-sign/-/oauth-sign-0.9.0.tgz", + "integrity": "sha512-fexhUFFPTGV8ybAtSIGbV6gOkSv8UtRbDBnAyLQw4QPKkgNlsH2ByPGtMUqdWkos6YCRmAqViwgZrJc/mRDzZQ==", + "dev": true, + "engines": { + "node": "*" + } + }, + "node_modules/object-assign": { + "version": "4.1.1", + "resolved": "https://registry.npmmirror.com/object-assign/-/object-assign-4.1.1.tgz", + "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/object-copy": { + "version": "0.1.0", + "resolved": "https://registry.npmmirror.com/object-copy/-/object-copy-0.1.0.tgz", + "integrity": "sha512-79LYn6VAb63zgtmAteVOWo9Vdj71ZVBy3Pbse+VqxDpEP83XuujMrGqHIwAXJ5I/aM0zU7dIyIAhifVTPrNItQ==", + "dependencies": { + "copy-descriptor": "^0.1.0", + "define-property": "^0.2.5", + "kind-of": "^3.0.3" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/object-copy/node_modules/define-property": { + "version": "0.2.5", + "resolved": "https://registry.npmmirror.com/define-property/-/define-property-0.2.5.tgz", + "integrity": "sha512-Rr7ADjQZenceVOAKop6ALkkRAmH1A4Gx9hV/7ZujPUN2rkATqFO0JZLZInbAjpZYoJ1gUx8MRMQVkYemcbMSTA==", + "dependencies": { + "is-descriptor": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/object-copy/node_modules/kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmmirror.com/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha512-NOW9QQXMoZGg/oqnVNoNTTIFEIid1627WCffUBJEdMxYApq7mNE7CpzucIPc+ZQg25Phej7IJSmX3hO+oblOtQ==", + "dependencies": { + "is-buffer": "^1.1.5" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/object-hash": { + "version": "1.3.1", + "resolved": "https://registry.npmmirror.com/object-hash/-/object-hash-1.3.1.tgz", + "integrity": "sha512-OSuu/pU4ENM9kmREg0BdNrUDIl1heYa4mBZacJc+vVWz4GtAwu7jO8s4AIt2aGRUTqxykpWzI3Oqnsm13tTMDA==", + "dev": true, + "engines": { + "node": ">= 0.10.0" + } + }, + "node_modules/object-inspect": { + "version": "1.12.0", + "resolved": "https://registry.npmmirror.com/object-inspect/-/object-inspect-1.12.0.tgz", + "integrity": "sha512-Ho2z80bVIvJloH+YzRmpZVQe87+qASmBUKZDWgx9cu+KDrX2ZDH/3tMy+gXbZETVGs2M8YdxObOh7XAtim9Y0g==" + }, + "node_modules/object-is": { + "version": "1.1.5", + "resolved": "https://registry.npmmirror.com/object-is/-/object-is-1.1.5.tgz", + "integrity": "sha512-3cyDsyHgtmi7I7DfSSI2LDp6SK2lwvtbg0p0R1e0RvTqF5ceGx+K2dfSjm1bKDMVCFEDAQvy+o8c6a7VujOddw==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.3" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/object-keys": { + "version": "1.1.1", + "resolved": "https://registry.npmmirror.com/object-keys/-/object-keys-1.1.1.tgz", + "integrity": "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/object-visit": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/object-visit/-/object-visit-1.0.1.tgz", + "integrity": "sha512-GBaMwwAVK9qbQN3Scdo0OyvgPW7l3lnaVMj84uTOZlswkX0KpF6fyDBJhtTthf7pymztoN36/KEr1DyhF96zEA==", + "dependencies": { + "isobject": "^3.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/object.assign": { + "version": "4.1.2", + "resolved": "https://registry.npmmirror.com/object.assign/-/object.assign-4.1.2.tgz", + "integrity": "sha512-ixT2L5THXsApyiUPYKmW+2EHpXXe5Ii3M+f4e+aJFAHao5amFRW6J0OO6c/LU8Be47utCx2GL89hxGB6XSmKuQ==", + "dependencies": { + "call-bind": "^1.0.0", + "define-properties": "^1.1.3", + "has-symbols": "^1.0.1", + "object-keys": "^1.1.1" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/object.getownpropertydescriptors": { + "version": "2.1.3", + "resolved": "https://registry.npmmirror.com/object.getownpropertydescriptors/-/object.getownpropertydescriptors-2.1.3.tgz", + "integrity": "sha512-VdDoCwvJI4QdC6ndjpqFmoL3/+HxffFBbcJzKi5hwLLqqx3mdbedRpfZDdK0SrOSauj8X4GzBvnDZl4vTN7dOw==", + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.3", + "es-abstract": "^1.19.1" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/object.omit": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/object.omit/-/object.omit-2.0.1.tgz", + "integrity": "sha512-UiAM5mhmIuKLsOvrL+B0U2d1hXHF3bFYWIuH1LMpuV2EJEHG1Ntz06PgLEHjm6VFd87NpH8rastvPoyv6UW2fA==", + "dev": true, + "dependencies": { + "for-own": "^0.1.4", + "is-extendable": "^0.1.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/object.pick": { + "version": "1.3.0", + "resolved": "https://registry.npmmirror.com/object.pick/-/object.pick-1.3.0.tgz", + "integrity": "sha512-tqa/UMy/CCoYmj+H5qc07qvSL9dqcs/WZENZ1JbtWBlATP+iVOe778gE6MSijnyCnORzDuX6hU+LA4SZ09YjFQ==", + "dependencies": { + "isobject": "^3.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/object.values": { + "version": "1.1.5", + "resolved": "https://registry.npmmirror.com/object.values/-/object.values-1.1.5.tgz", + "integrity": "sha512-QUZRW0ilQ3PnPpbNtgdNV1PDbEqLIiSFB3l+EnGtBQ/8SUTLj1PZwtQHABZtLgwpJZTSZhuGLOGk57Drx2IvYg==", + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.3", + "es-abstract": "^1.19.1" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/obuf": { + "version": "1.1.2", + "resolved": "https://registry.npmmirror.com/obuf/-/obuf-1.1.2.tgz", + "integrity": "sha512-PX1wu0AmAdPqOL1mWhqmlOd8kOIZQwGZw6rh7uby9fTc5lhaOWFLX3I6R1hrF9k3zUY40e6igsLGkDXK92LJNg==", + "dev": true + }, + "node_modules/on-finished": { + "version": "2.4.1", + "resolved": "https://registry.npmmirror.com/on-finished/-/on-finished-2.4.1.tgz", + "integrity": "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==", + "dev": true, + "dependencies": { + "ee-first": "1.1.1" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/on-headers": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/on-headers/-/on-headers-1.0.2.tgz", + "integrity": "sha512-pZAE+FJLoyITytdqK0U5s+FIpjN0JP3OzFi/u8Rx+EV5/W+JTWGXG8xFzevE7AjBfDqHv/8vL8qQsIhHnqRkrA==", + "dev": true, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/once": { + "version": "1.4.0", + "resolved": "https://registry.npmmirror.com/once/-/once-1.4.0.tgz", + "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", + "dependencies": { + "wrappy": "1" + } + }, + "node_modules/onetime": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/onetime/-/onetime-2.0.1.tgz", + "integrity": "sha512-oyyPpiMaKARvvcgip+JV+7zci5L8D1W9RZIz2l1o08AM3pfspitVWnPt3mzHcBPp12oYMTy0pqrFs/C+m3EwsQ==", + "dev": true, + "dependencies": { + "mimic-fn": "^1.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/open": { + "version": "6.4.0", + "resolved": "https://registry.npmmirror.com/open/-/open-6.4.0.tgz", + "integrity": "sha512-IFenVPgF70fSm1keSd2iDBIDIBZkroLeuffXq+wKTzTJlBpesFWojV9lb8mzOfaAzM1sr7HQHuO0vtV0zYekGg==", + "dev": true, + "dependencies": { + "is-wsl": "^1.1.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/opener": { + "version": "1.5.2", + "resolved": "https://registry.npmmirror.com/opener/-/opener-1.5.2.tgz", + "integrity": "sha512-ur5UIdyw5Y7yEj9wLzhqXiy6GZ3Mwx0yGI+5sMn2r0N0v3cKJvUmFH5yPP+WXh9e0xfyzyJX95D8l088DNFj7A==", + "dev": true, + "bin": { + "opener": "bin/opener-bin.js" + } + }, + "node_modules/opn": { + "version": "5.5.0", + "resolved": "https://registry.npmmirror.com/opn/-/opn-5.5.0.tgz", + "integrity": "sha512-PqHpggC9bLV0VeWcdKhkpxY+3JTzetLSqTCWL/z/tFIbI6G8JCjondXklT1JinczLz2Xib62sSp0T/gKT4KksA==", + "dev": true, + "dependencies": { + "is-wsl": "^1.1.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/optionator": { + "version": "0.8.3", + "resolved": "https://registry.npmmirror.com/optionator/-/optionator-0.8.3.tgz", + "integrity": "sha512-+IW9pACdk3XWmmTXG8m3upGUJst5XRGzxMRjXzAuJ1XnIFNvfhjjIuYkDvysnPQ7qzqVzLt78BCruntqRhWQbA==", + "dev": true, + "dependencies": { + "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" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/ora": { + "version": "3.4.0", + "resolved": "https://registry.npmmirror.com/ora/-/ora-3.4.0.tgz", + "integrity": "sha512-eNwHudNbO1folBP3JsZ19v9azXWtQZjICdr3Q0TDPIaeBQ3mXLrh54wM+er0+hSp+dWKf+Z8KM58CYzEyIYxYg==", + "dev": true, + "dependencies": { + "chalk": "^2.4.2", + "cli-cursor": "^2.1.0", + "cli-spinners": "^2.0.0", + "log-symbols": "^2.2.0", + "strip-ansi": "^5.2.0", + "wcwidth": "^1.0.1" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/original": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/original/-/original-1.0.2.tgz", + "integrity": "sha512-hyBVl6iqqUOJ8FqRe+l/gS8H+kKYjrEndd5Pm1MfBtsEKA038HkkdbAl/72EAXGyonD/PFsvmVG+EvcIpliMBg==", + "dev": true, + "dependencies": { + "url-parse": "^1.4.3" + } + }, + "node_modules/os-browserify": { + "version": "0.3.0", + "resolved": "https://registry.npmmirror.com/os-browserify/-/os-browserify-0.3.0.tgz", + "integrity": "sha512-gjcpUc3clBf9+210TRaDWbf+rZZZEshZ+DlXMRCeAjp0xhTrnQsKHypIy1J3d5hKdUzj69t708EHtU8P6bUn0A==" + }, + "node_modules/os-homedir": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/os-homedir/-/os-homedir-1.0.2.tgz", + "integrity": "sha512-B5JU3cabzk8c67mRRd3ECmROafjYMXbuzlwtqdM8IbS8ktlTix8aFGb2bAGKrSRIlnfKwovGUUr72JUPyOb6kQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/os-tmpdir": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/os-tmpdir/-/os-tmpdir-1.0.2.tgz", + "integrity": "sha512-D2FR03Vir7FIu45XBY20mTb+/ZSWB00sjU9jdQXt83gDrI4Ztz5Fs7/yy74g2N5SVQY4xY1qDr4rNddwYRVX0g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/osenv": { + "version": "0.1.5", + "resolved": "https://registry.npmmirror.com/osenv/-/osenv-0.1.5.tgz", + "integrity": "sha512-0CWcCECdMVc2Rw3U5w9ZjqX6ga6ubk1xDVKxtBQPK7wis/0F2r9T6k4ydGYhecl7YUBxBVxhL5oisPsNxAPe2g==", + "dev": true, + "dependencies": { + "os-homedir": "^1.0.0", + "os-tmpdir": "^1.0.0" + } + }, + "node_modules/p-finally": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/p-finally/-/p-finally-1.0.0.tgz", + "integrity": "sha512-LICb2p9CB7FS+0eR1oqWnHhp0FljGLZCWBE9aix0Uye9W8LTQPwMTYVGWQWIw9RdQiDg4+epXQODwIYJtSJaow==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/p-limit": { + "version": "2.3.0", + "resolved": "https://registry.npmmirror.com/p-limit/-/p-limit-2.3.0.tgz", + "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", + "dependencies": { + "p-try": "^2.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/p-locate": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/p-locate/-/p-locate-3.0.0.tgz", + "integrity": "sha512-x+12w/To+4GFfgJhBEpiDcLozRJGegY+Ei7/z0tSLkMmxGZNybVMSfWj9aJn8Z5Fc7dBUNJOOVgPv2H7IwulSQ==", + "dependencies": { + "p-limit": "^2.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/p-map": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/p-map/-/p-map-2.1.0.tgz", + "integrity": "sha512-y3b8Kpd8OAN444hxfBbFfj1FY/RjtTd8tzYwhUqNYXx0fXx2iX4maP4Qr6qhIKbQXI02wTLAda4fYUbDagTUFw==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/p-map-series": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/p-map-series/-/p-map-series-2.1.0.tgz", + "integrity": "sha512-RpYIIK1zXSNEOdwxcfe7FdvGcs7+y5n8rifMhMNWvaxRNMPINJHF5GDeuVxWqnfrcHPSCnp7Oo5yNXHId9Av2Q==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/p-pipe": { + "version": "3.1.0", + "resolved": "https://registry.npmmirror.com/p-pipe/-/p-pipe-3.1.0.tgz", + "integrity": "sha512-08pj8ATpzMR0Y80x50yJHn37NF6vjrqHutASaX5LiH5npS9XPvrUmscd9MF5R4fuYRHOxQR1FfMIlF7AzwoPqw==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/p-queue": { + "version": "6.6.2", + "resolved": "https://registry.npmmirror.com/p-queue/-/p-queue-6.6.2.tgz", + "integrity": "sha512-RwFpb72c/BhQLEXIZ5K2e+AhgNVmIejGlTgiB9MzZ0e93GRvqZ7uSi0dvRF7/XIXDeNkra2fNHBxTyPDGySpjQ==", + "dev": true, + "dependencies": { + "eventemitter3": "^4.0.4", + "p-timeout": "^3.2.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/p-reduce": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/p-reduce/-/p-reduce-2.1.0.tgz", + "integrity": "sha512-2USApvnsutq8uoxZBGbbWM0JIYLiEMJ9RlaN7fAzVNb9OZN0SHjjTTfIcb667XynS5Y1VhwDJVDa72TnPzAYWw==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/p-retry": { + "version": "3.0.1", + "resolved": "https://registry.npmmirror.com/p-retry/-/p-retry-3.0.1.tgz", + "integrity": "sha512-XE6G4+YTTkT2a0UWb2kjZe8xNwf8bIbnqpc/IS/idOBVhyves0mK5OJgeocjx7q5pvX/6m23xuzVPYT1uGM73w==", + "dev": true, + "dependencies": { + "retry": "^0.12.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/p-timeout": { + "version": "3.2.0", + "resolved": "https://registry.npmmirror.com/p-timeout/-/p-timeout-3.2.0.tgz", + "integrity": "sha512-rhIwUycgwwKcP9yTOOFK/AKsAopjjCakVqLHePO3CC6Mir1Z99xT+R63jZxAT5lFZLa2inS5h+ZS2GvR99/FBg==", + "dev": true, + "dependencies": { + "p-finally": "^1.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/p-try": { + "version": "2.2.0", + "resolved": "https://registry.npmmirror.com/p-try/-/p-try-2.2.0.tgz", + "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==", + "engines": { + "node": ">=6" + } + }, + "node_modules/p-waterfall": { + "version": "2.1.1", + "resolved": "https://registry.npmmirror.com/p-waterfall/-/p-waterfall-2.1.1.tgz", + "integrity": "sha512-RRTnDb2TBG/epPRI2yYXsimO0v3BXC8Yd3ogr1545IaqKK17VGhbWVeGGN+XfCm/08OK8635nH31c8bATkHuSw==", + "dev": true, + "dependencies": { + "p-reduce": "^2.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/pacote": { + "version": "11.3.5", + "resolved": "https://registry.npmmirror.com/pacote/-/pacote-11.3.5.tgz", + "integrity": "sha512-fT375Yczn4zi+6Hkk2TBe1x1sP8FgFsEIZ2/iWaXY2r/NkhDJfxbcn5paz1+RTFCyNf+dPnaoBDJoAxXSU8Bkg==", + "dev": true, + "dependencies": { + "@npmcli/git": "^2.1.0", + "@npmcli/installed-package-contents": "^1.0.6", + "@npmcli/promise-spawn": "^1.2.0", + "@npmcli/run-script": "^1.8.2", + "cacache": "^15.0.5", + "chownr": "^2.0.0", + "fs-minipass": "^2.1.0", + "infer-owner": "^1.0.4", + "minipass": "^3.1.3", + "mkdirp": "^1.0.3", + "npm-package-arg": "^8.0.1", + "npm-packlist": "^2.1.4", + "npm-pick-manifest": "^6.0.0", + "npm-registry-fetch": "^11.0.0", + "promise-retry": "^2.0.1", + "read-package-json-fast": "^2.0.1", + "rimraf": "^3.0.2", + "ssri": "^8.0.1", + "tar": "^6.1.0" + }, + "bin": { + "pacote": "lib/bin.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/pacote/node_modules/cacache": { + "version": "15.3.0", + "resolved": "https://registry.npmmirror.com/cacache/-/cacache-15.3.0.tgz", + "integrity": "sha512-VVdYzXEn+cnbXpFgWs5hTT7OScegHVmLhJIR8Ufqk3iFD6A6j5iSX1KuBTfNEv4tdJWE2PzA6IVFtcLC7fN9wQ==", + "dev": true, + "dependencies": { + "@npmcli/fs": "^1.0.0", + "@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.1", + "tar": "^6.0.2", + "unique-filename": "^1.1.1" + }, + "engines": { + "node": ">= 10" + } + }, + "node_modules/pacote/node_modules/chownr": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/chownr/-/chownr-2.0.0.tgz", + "integrity": "sha512-bIomtDF5KGpdogkLd9VspvFzk9KfpyyGlS8YFVZl7TGPBHL5snIOnxeshwVgPteQ9b4Eydl+pVbIyE1DcvCWgQ==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/pacote/node_modules/fs-minipass": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/fs-minipass/-/fs-minipass-2.1.0.tgz", + "integrity": "sha512-V/JgOLFCS+R6Vcq0slCuaeWEdNC3ouDlJMNIsacH2VtALiu9mV4LPrHc5cDl8k5aw6J8jwgWWpiTo5RYhmIzvg==", + "dev": true, + "dependencies": { + "minipass": "^3.0.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/pacote/node_modules/lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/pacote/node_modules/minipass": { + "version": "3.1.6", + "resolved": "https://registry.npmmirror.com/minipass/-/minipass-3.1.6.tgz", + "integrity": "sha512-rty5kpw9/z8SX9dmxblFA6edItUmwJgMeYDZRrwlIVN27i8gysGbznJwUggw2V/FVqFSDdWy040ZPS811DYAqQ==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/pacote/node_modules/minizlib": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/minizlib/-/minizlib-2.1.2.tgz", + "integrity": "sha512-bAxsR8BVfj60DWXHE3u30oHzfl4G7khkSuPW+qvpd7jFRHm7dLxOjUk1EHACJ/hxLY8phGJ0YhYHZo7jil7Qdg==", + "dev": true, + "dependencies": { + "minipass": "^3.0.0", + "yallist": "^4.0.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/pacote/node_modules/mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmmirror.com/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true, + "bin": { + "mkdirp": "bin/cmd.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/pacote/node_modules/p-map": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/p-map/-/p-map-4.0.0.tgz", + "integrity": "sha512-/bjOqmgETBYB5BoEeGVea8dmvHb2m9GLy1E9W43yeyfP6QQCZGFNa+XRceJEuDB6zqr+gKpIAmlLebMpykw/MQ==", + "dev": true, + "dependencies": { + "aggregate-error": "^3.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/pacote/node_modules/rimraf": { + "version": "3.0.2", + "resolved": "https://registry.npmmirror.com/rimraf/-/rimraf-3.0.2.tgz", + "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", + "dev": true, + "dependencies": { + "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" + } + }, + "node_modules/pacote/node_modules/ssri": { + "version": "8.0.1", + "resolved": "https://registry.npmmirror.com/ssri/-/ssri-8.0.1.tgz", + "integrity": "sha512-97qShzy1AiyxvPNIkLWoGua7xoQzzPjQ0HAH4B0rWKo7SZ6USuPcrUiAFrws0UH8RrbWmgq3LMTObhPIHbbBeQ==", + "dev": true, + "dependencies": { + "minipass": "^3.1.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/pacote/node_modules/tar": { + "version": "6.1.11", + "resolved": "https://registry.npmmirror.com/tar/-/tar-6.1.11.tgz", + "integrity": "sha512-an/KZQzQUkZCkuoAA64hM92X0Urb6VpRhAFllDzz44U2mcD5scmT3zBc4VgVpkugF580+DQn8eAFSyoQt0tznA==", + "dev": true, + "dependencies": { + "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" + }, + "engines": { + "node": ">= 10" + } + }, + "node_modules/pacote/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/pako": { + "version": "1.0.11", + "resolved": "https://registry.npmmirror.com/pako/-/pako-1.0.11.tgz", + "integrity": "sha512-4hLB8Py4zZce5s4yd9XzopqwVv/yGNhV1Bl8NTmCq1763HeK2+EwVTv+leGeL13Dnh2wfbqowVPXCIO0z4taYw==" + }, + "node_modules/parallel-transform": { + "version": "1.2.0", + "resolved": "https://registry.npmmirror.com/parallel-transform/-/parallel-transform-1.2.0.tgz", + "integrity": "sha512-P2vSmIu38uIlvdcU7fDkyrxj33gTUy/ABO5ZUbGowxNCopBq/OoD42bP4UmMrJoPyk4Uqf0mu3mtWBhHCZD8yg==", + "dependencies": { + "cyclist": "^1.0.1", + "inherits": "^2.0.3", + "readable-stream": "^2.1.5" + } + }, + "node_modules/param-case": { + "version": "2.1.1", + "resolved": "https://registry.npmmirror.com/param-case/-/param-case-2.1.1.tgz", + "integrity": "sha512-eQE845L6ot89sk2N8liD8HAuH4ca6Vvr7VWAWwt7+kvvG5aBcPmmphQ68JsEG2qa9n1TykS2DLeMt363AAH8/w==", + "dev": true, + "dependencies": { + "no-case": "^2.2.0" + } + }, + "node_modules/parent-module": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/parent-module/-/parent-module-1.0.1.tgz", + "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", + "dev": true, + "dependencies": { + "callsites": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/parent-module/node_modules/callsites": { + "version": "3.1.0", + "resolved": "https://registry.npmmirror.com/callsites/-/callsites-3.1.0.tgz", + "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/parse-asn1": { + "version": "5.1.6", + "resolved": "https://registry.npmmirror.com/parse-asn1/-/parse-asn1-5.1.6.tgz", + "integrity": "sha512-RnZRo1EPU6JBnra2vGHj0yhp6ebyjBZpmUCLHWiFhxlzvBCCpAuZ7elsBp1PVAbQN0/04VD/19rfzlBSwLstMw==", + "dependencies": { + "asn1.js": "^5.2.0", + "browserify-aes": "^1.0.0", + "evp_bytestokey": "^1.0.0", + "pbkdf2": "^3.0.3", + "safe-buffer": "^5.1.1" + } + }, + "node_modules/parse-glob": { + "version": "3.0.4", + "resolved": "https://registry.npmmirror.com/parse-glob/-/parse-glob-3.0.4.tgz", + "integrity": "sha512-FC5TeK0AwXzq3tUBFtH74naWkPQCEWs4K+xMxWZBlKDWu0bVHXGZa+KKqxKidd7xwhdZ19ZNuF2uO1M/r196HA==", + "dev": true, + "dependencies": { + "glob-base": "^0.3.0", + "is-dotfile": "^1.0.0", + "is-extglob": "^1.0.0", + "is-glob": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/parse-glob/node_modules/is-extglob": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/is-extglob/-/is-extglob-1.0.0.tgz", + "integrity": "sha512-7Q+VbVafe6x2T+Tu6NcOf6sRklazEPmBoB3IWk3WdGZM2iGUwU/Oe3Wtq5lSEkDTTlpp8yx+5t4pzO/i9Ty1ww==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/parse-glob/node_modules/is-glob": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/is-glob/-/is-glob-2.0.1.tgz", + "integrity": "sha512-a1dBeB19NXsf/E0+FHqkagizel/LQw2DjSQpvQrj3zT+jYPpaUCryPnrQajXKFLCMuf4I6FhRpaGtw4lPrG6Eg==", + "dev": true, + "dependencies": { + "is-extglob": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/parse-json": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/parse-json/-/parse-json-4.0.0.tgz", + "integrity": "sha512-aOIos8bujGN93/8Ox/jPLh7RwVnPEysynVFE+fQZyg6jKELEHwzgKdLRFHUgXJL6kylijVSBC4BvN9OmsB48Rw==", + "dev": true, + "dependencies": { + "error-ex": "^1.3.1", + "json-parse-better-errors": "^1.0.1" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/parse-path": { + "version": "4.0.3", + "resolved": "https://registry.npmmirror.com/parse-path/-/parse-path-4.0.3.tgz", + "integrity": "sha512-9Cepbp2asKnWTJ9x2kpw6Fe8y9JDbqwahGCTvklzd/cEq5C5JC59x2Xb0Kx+x0QZ8bvNquGO8/BWP0cwBHzSAA==", + "dev": true, + "dependencies": { + "is-ssh": "^1.3.0", + "protocols": "^1.4.0", + "qs": "^6.9.4", + "query-string": "^6.13.8" + } + }, + "node_modules/parse-path/node_modules/query-string": { + "version": "6.14.1", + "resolved": "https://registry.npmmirror.com/query-string/-/query-string-6.14.1.tgz", + "integrity": "sha512-XDxAeVmpfu1/6IjyT/gXHOl+S0vQ9owggJ30hhWKdHAsNPOcasn5o9BW0eejZqL2e4vMjhAxoW3jVHcD6mbcYw==", + "dev": true, + "dependencies": { + "decode-uri-component": "^0.2.0", + "filter-obj": "^1.1.0", + "split-on-first": "^1.0.0", + "strict-uri-encode": "^2.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/parse-path/node_modules/strict-uri-encode": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/strict-uri-encode/-/strict-uri-encode-2.0.0.tgz", + "integrity": "sha512-QwiXZgpRcKkhTj2Scnn++4PKtWsH0kpzZ62L2R6c/LUVYv7hVnZqcg2+sMuT6R7Jusu1vviK/MFsu6kNJfWlEQ==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/parse-url": { + "version": "6.0.0", + "resolved": "https://registry.npmmirror.com/parse-url/-/parse-url-6.0.0.tgz", + "integrity": "sha512-cYyojeX7yIIwuJzledIHeLUBVJ6COVLeT4eF+2P6aKVzwvgKQPndCBv3+yQ7pcWjqToYwaligxzSYNNmGoMAvw==", + "dev": true, + "dependencies": { + "is-ssh": "^1.3.0", + "normalize-url": "^6.1.0", + "parse-path": "^4.0.0", + "protocols": "^1.4.0" + } + }, + "node_modules/parse-url/node_modules/normalize-url": { + "version": "6.1.0", + "resolved": "https://registry.npmmirror.com/normalize-url/-/normalize-url-6.1.0.tgz", + "integrity": "sha512-DlL+XwOy3NxAQ8xuC0okPgK46iuVNAK01YN7RueYBqqFeGsBjV9XmCAzAdgt+667bCl5kPh9EqKKDwnaPG1I7A==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/parse5": { + "version": "5.1.1", + "resolved": "https://registry.npmmirror.com/parse5/-/parse5-5.1.1.tgz", + "integrity": "sha512-ugq4DFI0Ptb+WWjAdOK16+u/nHfiIrcE+sh8kZMaM0WllQKLI9rOUq6c2b7cwPkXdzfQESqvoqK6ug7U/Yyzug==", + "dev": true + }, + "node_modules/parse5-htmlparser2-tree-adapter": { + "version": "6.0.1", + "resolved": "https://registry.npmmirror.com/parse5-htmlparser2-tree-adapter/-/parse5-htmlparser2-tree-adapter-6.0.1.tgz", + "integrity": "sha512-qPuWvbLgvDGilKc5BoicRovlT4MtYT6JfJyBOMDsKoiT+GiuP5qyrPCnR9HcPECIJJmZh5jRndyNThnhhb/vlA==", + "dev": true, + "dependencies": { + "parse5": "^6.0.1" + } + }, + "node_modules/parse5-htmlparser2-tree-adapter/node_modules/parse5": { + "version": "6.0.1", + "resolved": "https://registry.npmmirror.com/parse5/-/parse5-6.0.1.tgz", + "integrity": "sha512-Ofn/CTFzRGTTxwpNEs9PP93gXShHcTq255nzRYSKe8AkVpZY7e1fpmTfOyoIvjP5HG7Z2ZM7VS9PPhQGW2pOpw==", + "dev": true + }, + "node_modules/parseurl": { + "version": "1.3.3", + "resolved": "https://registry.npmmirror.com/parseurl/-/parseurl-1.3.3.tgz", + "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==", + "dev": true, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/pascalcase": { + "version": "0.1.1", + "resolved": "https://registry.npmmirror.com/pascalcase/-/pascalcase-0.1.1.tgz", + "integrity": "sha512-XHXfu/yOQRy9vYOtUDVMN60OEJjW013GoObG1o+xwQTpB9eYJX/BjXMsdW13ZDPruFhYYn0AG22w0xgQMwl3Nw==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/patch-package": { + "version": "6.2.2", + "resolved": "https://registry.npmmirror.com/patch-package/-/patch-package-6.2.2.tgz", + "integrity": "sha512-YqScVYkVcClUY0v8fF0kWOjDYopzIM8e3bj/RU1DPeEF14+dCGm6UeOYm4jvCyxqIEQ5/eJzmbWfDWnUleFNMg==", + "dev": true, + "dependencies": { + "@yarnpkg/lockfile": "^1.1.0", + "chalk": "^2.4.2", + "cross-spawn": "^6.0.5", + "find-yarn-workspace-root": "^1.2.1", + "fs-extra": "^7.0.1", + "is-ci": "^2.0.0", + "klaw-sync": "^6.0.0", + "minimist": "^1.2.0", + "rimraf": "^2.6.3", + "semver": "^5.6.0", + "slash": "^2.0.0", + "tmp": "^0.0.33" + }, + "bin": { + "patch-package": "index.js" + }, + "engines": { + "npm": ">5" + } + }, + "node_modules/patch-package/node_modules/ci-info": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/ci-info/-/ci-info-2.0.0.tgz", + "integrity": "sha512-5tK7EtrZ0N+OLFMthtqOj4fI2Jeb88C4CAZPu25LDVUgXJ0A3Js4PMGqrn0JU1W0Mh1/Z8wZzYPxqUrXeBboCQ==", + "dev": true + }, + "node_modules/patch-package/node_modules/is-ci": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/is-ci/-/is-ci-2.0.0.tgz", + "integrity": "sha512-YfJT7rkpQB0updsdHLGWrvhBJfcfzNNawYDNIyQXJz0IViGf75O8EBPKSdvw2rF+LGCsX4FZ8tcr3b19LcZq4w==", + "dev": true, + "dependencies": { + "ci-info": "^2.0.0" + }, + "bin": { + "is-ci": "bin.js" + } + }, + "node_modules/path-browserify": { + "version": "0.0.1", + "resolved": "https://registry.npmmirror.com/path-browserify/-/path-browserify-0.0.1.tgz", + "integrity": "sha512-BapA40NHICOS+USX9SN4tyhq+A2RrN/Ws5F0Z5aMHDp98Fl86lX8Oti8B7uN93L4Ifv4fHOEA+pQw87gmMO/lQ==" + }, + "node_modules/path-dirname": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/path-dirname/-/path-dirname-1.0.2.tgz", + "integrity": "sha512-ALzNPpyNq9AqXMBjeymIjFDAkAFH06mHJH/cSBHAgU0s4vfpBn6b2nf8tiRLvagKD8RbTpq2FKTBg7cl9l3c7Q==", + "devOptional": true + }, + "node_modules/path-exists": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/path-exists/-/path-exists-3.0.0.tgz", + "integrity": "sha512-bpC7GYwiDYQ4wYLe+FA8lhRjhQCMcQGuSgGGqDkg/QerRWw9CmGRT0iSOVRSZJ29NMLZgIzqaljJ63oaL4NIJQ==", + "engines": { + "node": ">=4" + } + }, + "node_modules/path-is-absolute": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/path-is-absolute/-/path-is-absolute-1.0.1.tgz", + "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/path-is-inside": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/path-is-inside/-/path-is-inside-1.0.2.tgz", + "integrity": "sha512-DUWJr3+ULp4zXmol/SZkFf3JGsS9/SIv+Y3Rt93/UjPpDpklB5f1er4O3POIbUuUJ3FXgqte2Q7SrU6zAqwk8w==", + "dev": true + }, + "node_modules/path-key": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/path-key/-/path-key-2.0.1.tgz", + "integrity": "sha512-fEHGKCSmUSDPv4uoj8AlD+joPlq3peND+HRYyxFz4KPw4z926S/b8rIuFs2FYJg3BwsxJf6A9/3eIdLaYC+9Dw==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/path-parse": { + "version": "1.0.7", + "resolved": "https://registry.npmmirror.com/path-parse/-/path-parse-1.0.7.tgz", + "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==", + "dev": true + }, + "node_modules/path-to-regexp": { + "version": "0.1.7", + "resolved": "https://registry.npmmirror.com/path-to-regexp/-/path-to-regexp-0.1.7.tgz", + "integrity": "sha512-5DFkuoqlv1uYQKxy8omFBeJPQcdoE07Kv2sferDCrAq1ohOU+MSDswDIbnx3YAM60qIOnYa53wBhXW0EbMonrQ==", + "dev": true + }, + "node_modules/path-type": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/path-type/-/path-type-3.0.0.tgz", + "integrity": "sha512-T2ZUsdZFHgA3u4e5PfPbjd7HDDpxPnQb5jN0SrDsjNSuVXHJqtwTnWqG0B1jZrgmJ/7lj1EmVIByWt1gxGkWvg==", + "dev": true, + "dependencies": { + "pify": "^3.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/path-type/node_modules/pify": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/pify/-/pify-3.0.0.tgz", + "integrity": "sha512-C3FsVNH1udSEX48gGX1xfvwTWfsYWj5U+8/uK15BGzIGrKoUpghX8hWZwa/OFnakBiiVNmBvemTJR5mcy7iPcg==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/pbkdf2": { + "version": "3.1.2", + "resolved": "https://registry.npmmirror.com/pbkdf2/-/pbkdf2-3.1.2.tgz", + "integrity": "sha512-iuh7L6jA7JEGu2WxDwtQP1ddOpaJNC4KlDEFfdQajSGgGPNi4OyDc2R7QnbY2bR9QjBVGwgvTdNJZoE7RaxUMA==", + "dependencies": { + "create-hash": "^1.1.2", + "create-hmac": "^1.1.4", + "ripemd160": "^2.0.1", + "safe-buffer": "^5.0.1", + "sha.js": "^2.4.8" + }, + "engines": { + "node": ">=0.12" + } + }, + "node_modules/performance-now": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/performance-now/-/performance-now-2.1.0.tgz", + "integrity": "sha512-7EAHlyLHI56VEIdK57uwHdHKIaAGbnXPiw0yWbarQZOKaKpvUIgW0jWRVLiatnM+XXlSwsanIBH/hzGMJulMow==", + "dev": true + }, + "node_modules/picocolors": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/picocolors/-/picocolors-1.0.0.tgz", + "integrity": "sha512-1fygroTLlHu66zi26VoTDv8yRgm0Fccecssto+MhsZ0D/DGW2sm8E8AjW7NU5VVTRt5GxbeZ5qBuJr+HyLYkjQ==", + "dev": true + }, + "node_modules/picomatch": { + "version": "2.3.1", + "resolved": "https://registry.npmmirror.com/picomatch/-/picomatch-2.3.1.tgz", + "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", + "devOptional": true, + "engines": { + "node": ">=8.6" + } + }, + "node_modules/pify": { + "version": "4.0.1", + "resolved": "https://registry.npmmirror.com/pify/-/pify-4.0.1.tgz", + "integrity": "sha512-uB80kBFb/tfd68bVleG9T5GGsGPjJrLAUpR5PZIrhBnIaRTQRjqdJSsIKkOP6OAIFbj7GOrcudc5pNjZ+geV2g==", + "engines": { + "node": ">=6" + } + }, + "node_modules/pinkie": { + "version": "2.0.4", + "resolved": "https://registry.npmmirror.com/pinkie/-/pinkie-2.0.4.tgz", + "integrity": "sha512-MnUuEycAemtSaeFSjXKW/aroV7akBbY+Sv+RkyqFjgAe73F+MR0TBWKBRDkmfWq/HiFmdavfZ1G7h4SPZXaCSg==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/pinkie-promise": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/pinkie-promise/-/pinkie-promise-2.0.1.tgz", + "integrity": "sha512-0Gni6D4UcLTbv9c57DfxDGdr41XfgUjqWZu492f0cIGr16zDU06BWP/RAEvOuo7CQ0CNjHaLlM59YJJFm3NWlw==", + "dev": true, + "dependencies": { + "pinkie": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/pkg-dir": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/pkg-dir/-/pkg-dir-3.0.0.tgz", + "integrity": "sha512-/E57AYkoeQ25qkxMj5PBOVgF8Kiu/h7cYS30Z5+R7WaiCCBfLq58ZI/dSeaEKb9WVJV5n/03QwrN3IeWIFllvw==", + "dependencies": { + "find-up": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/pkg-up": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/pkg-up/-/pkg-up-2.0.0.tgz", + "integrity": "sha512-fjAPuiws93rm7mPUu21RdBnkeZNrbfCFCwfAhPWY+rR3zG0ubpe5cEReHOw5fIbfmsxEV/g2kSxGTATY3Bpnwg==", + "dev": true, + "dependencies": { + "find-up": "^2.1.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/pkg-up/node_modules/find-up": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/find-up/-/find-up-2.1.0.tgz", + "integrity": "sha512-NWzkk0jSJtTt08+FBFMvXoeZnOJD+jTtsRmBYbAIzJdX6l7dLgR7CTubCM5/eDdPUBvLCeVasP1brfVR/9/EZQ==", + "dev": true, + "dependencies": { + "locate-path": "^2.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/pkg-up/node_modules/locate-path": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/locate-path/-/locate-path-2.0.0.tgz", + "integrity": "sha512-NCI2kiDkyR7VeEKm27Kda/iQHyKJe1Bu0FlTbYp3CqJu+9IFe9bLyAjMxf5ZDDbEg+iMPzB5zYyUTSm8wVTKmA==", + "dev": true, + "dependencies": { + "p-locate": "^2.0.0", + "path-exists": "^3.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/pkg-up/node_modules/p-limit": { + "version": "1.3.0", + "resolved": "https://registry.npmmirror.com/p-limit/-/p-limit-1.3.0.tgz", + "integrity": "sha512-vvcXsLAJ9Dr5rQOPk7toZQZJApBl2K4J6dANSsEuh6QI41JYcsS/qhTGa9ErIUUgK3WNQoJYvylxvjqmiqEA9Q==", + "dev": true, + "dependencies": { + "p-try": "^1.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/pkg-up/node_modules/p-locate": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/p-locate/-/p-locate-2.0.0.tgz", + "integrity": "sha512-nQja7m7gSKuewoVRen45CtVfODR3crN3goVQ0DDZ9N3yHxgpkuBhZqsaiotSQRrADUrne346peY7kT3TSACykg==", + "dev": true, + "dependencies": { + "p-limit": "^1.1.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/pkg-up/node_modules/p-try": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/p-try/-/p-try-1.0.0.tgz", + "integrity": "sha512-U1etNYuMJoIz3ZXSrrySFjsXQTWOx2/jdi86L+2pRvph/qMKL6sbcCYdH23fqsbm8TH2Gn0OybpT4eSFlCVHww==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/please-upgrade-node": { + "version": "3.2.0", + "resolved": "https://registry.npmmirror.com/please-upgrade-node/-/please-upgrade-node-3.2.0.tgz", + "integrity": "sha512-gQR3WpIgNIKwBMVLkpMUeR3e1/E1y42bqDQZfql+kDeXd8COYfM8PQA4X6y7a8u9Ua9FHmsrrmirW2vHs45hWg==", + "dev": true, + "dependencies": { + "semver-compare": "^1.0.0" + } + }, + "node_modules/pluralize": { + "version": "7.0.0", + "resolved": "https://registry.npmmirror.com/pluralize/-/pluralize-7.0.0.tgz", + "integrity": "sha512-ARhBOdzS3e41FbkW/XWrTEtukqqLoK5+Z/4UeDaLuSW+39JPeFgs4gCGqsrJHVZX0fUrx//4OF0K1CUGwlIFow==", + "dev": true, + "optional": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/popper.js": { + "version": "1.16.1", + "resolved": "https://registry.npmmirror.com/popper.js/-/popper.js-1.16.1.tgz", + "integrity": "sha512-Wb4p1J4zyFTbM+u6WuO4XstYx4Ky9Cewe4DWrel7B0w6VVICvPwdOpotjzcf6eD8TsckVnIMNONQyPIUFOUbCQ==", + "deprecated": "You can find the new Popper v2 at @popperjs/core, this package is dedicated to the legacy v1" + }, + "node_modules/portfinder": { + "version": "1.0.28", + "resolved": "https://registry.npmmirror.com/portfinder/-/portfinder-1.0.28.tgz", + "integrity": "sha512-Se+2isanIcEqf2XMHjyUKskczxbPH7dQnlMjXX6+dybayyHvAf/TCgyMRlzf/B6QDhAEFOGes0pzRo3by4AbMA==", + "dev": true, + "dependencies": { + "async": "^2.6.2", + "debug": "^3.1.1", + "mkdirp": "^0.5.5" + }, + "engines": { + "node": ">= 0.12.0" + } + }, + "node_modules/portfinder/node_modules/debug": { + "version": "3.2.7", + "resolved": "https://registry.npmmirror.com/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dev": true, + "dependencies": { + "ms": "^2.1.1" + } + }, + "node_modules/portfinder/node_modules/ms": { + "version": "2.1.3", + "resolved": "https://registry.npmmirror.com/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", + "dev": true + }, + "node_modules/posix-character-classes": { + "version": "0.1.1", + "resolved": "https://registry.npmmirror.com/posix-character-classes/-/posix-character-classes-0.1.1.tgz", + "integrity": "sha512-xTgYBc3fuo7Yt7JbiuFxSYGToMoz8fLoE6TC9Wx1P/u+LfeThMOAqmuyECnlBaaJb+u1m9hHiXUEtwW4OzfUJg==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/postcss": { + "version": "7.0.39", + "resolved": "https://registry.npmmirror.com/postcss/-/postcss-7.0.39.tgz", + "integrity": "sha512-yioayjNbHn6z1/Bywyb2Y4s3yvDAeXGOyxqD+LnVOinq6Mdmd++SW2wUNVzavyyHxd6+DxzWGIuosg6P1Rj8uA==", + "dev": true, + "dependencies": { + "picocolors": "^0.2.1", + "source-map": "^0.6.1" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/postcss-calc": { + "version": "7.0.5", + "resolved": "https://registry.npmmirror.com/postcss-calc/-/postcss-calc-7.0.5.tgz", + "integrity": "sha512-1tKHutbGtLtEZF6PT4JSihCHfIVldU72mZ8SdZHIYriIZ9fh9k9aWSppaT8rHsyI3dX+KSR+W+Ix9BMY3AODrg==", + "dev": true, + "dependencies": { + "postcss": "^7.0.27", + "postcss-selector-parser": "^6.0.2", + "postcss-value-parser": "^4.0.2" + } + }, + "node_modules/postcss-colormin": { + "version": "4.0.3", + "resolved": "https://registry.npmmirror.com/postcss-colormin/-/postcss-colormin-4.0.3.tgz", + "integrity": "sha512-WyQFAdDZpExQh32j0U0feWisZ0dmOtPl44qYmJKkq9xFWY3p+4qnRzCHeNrkeRhwPHz9bQ3mo0/yVkaply0MNw==", + "dev": true, + "dependencies": { + "browserslist": "^4.0.0", + "color": "^3.0.0", + "has": "^1.0.0", + "postcss": "^7.0.0", + "postcss-value-parser": "^3.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/postcss-colormin/node_modules/postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmmirror.com/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + }, + "node_modules/postcss-convert-values": { + "version": "4.0.1", + "resolved": "https://registry.npmmirror.com/postcss-convert-values/-/postcss-convert-values-4.0.1.tgz", + "integrity": "sha512-Kisdo1y77KUC0Jmn0OXU/COOJbzM8cImvw1ZFsBgBgMgb1iL23Zs/LXRe3r+EZqM3vGYKdQ2YJVQ5VkJI+zEJQ==", + "dev": true, + "dependencies": { + "postcss": "^7.0.0", + "postcss-value-parser": "^3.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/postcss-convert-values/node_modules/postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmmirror.com/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + }, + "node_modules/postcss-discard-comments": { + "version": "4.0.2", + "resolved": "https://registry.npmmirror.com/postcss-discard-comments/-/postcss-discard-comments-4.0.2.tgz", + "integrity": "sha512-RJutN259iuRf3IW7GZyLM5Sw4GLTOH8FmsXBnv8Ab/Tc2k4SR4qbV4DNbyyY4+Sjo362SyDmW2DQ7lBSChrpkg==", + "dev": true, + "dependencies": { + "postcss": "^7.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/postcss-discard-duplicates": { + "version": "4.0.2", + "resolved": "https://registry.npmmirror.com/postcss-discard-duplicates/-/postcss-discard-duplicates-4.0.2.tgz", + "integrity": "sha512-ZNQfR1gPNAiXZhgENFfEglF93pciw0WxMkJeVmw8eF+JZBbMD7jp6C67GqJAXVZP2BWbOztKfbsdmMp/k8c6oQ==", + "dev": true, + "dependencies": { + "postcss": "^7.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/postcss-discard-empty": { + "version": "4.0.1", + "resolved": "https://registry.npmmirror.com/postcss-discard-empty/-/postcss-discard-empty-4.0.1.tgz", + "integrity": "sha512-B9miTzbznhDjTfjvipfHoqbWKwd0Mj+/fL5s1QOz06wufguil+Xheo4XpOnc4NqKYBCNqqEzgPv2aPBIJLox0w==", + "dev": true, + "dependencies": { + "postcss": "^7.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/postcss-discard-overridden": { + "version": "4.0.1", + "resolved": "https://registry.npmmirror.com/postcss-discard-overridden/-/postcss-discard-overridden-4.0.1.tgz", + "integrity": "sha512-IYY2bEDD7g1XM1IDEsUT4//iEYCxAmP5oDSFMVU/JVvT7gh+l4fmjciLqGgwjdWpQIdb0Che2VX00QObS5+cTg==", + "dev": true, + "dependencies": { + "postcss": "^7.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/postcss-load-config": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/postcss-load-config/-/postcss-load-config-2.1.2.tgz", + "integrity": "sha512-/rDeGV6vMUo3mwJZmeHfEDvwnTKKqQ0S7OHUi/kJvvtx3aWtyWG2/0ZWnzCt2keEclwN6Tf0DST2v9kITdOKYw==", + "dev": true, + "dependencies": { + "cosmiconfig": "^5.0.0", + "import-cwd": "^2.0.0" + }, + "engines": { + "node": ">= 4" + } + }, + "node_modules/postcss-loader": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/postcss-loader/-/postcss-loader-3.0.0.tgz", + "integrity": "sha512-cLWoDEY5OwHcAjDnkyRQzAXfs2jrKjXpO/HQFcc5b5u/r7aa471wdmChmwfnv7x2u840iat/wi0lQ5nbRgSkUA==", + "dev": true, + "dependencies": { + "loader-utils": "^1.1.0", + "postcss": "^7.0.0", + "postcss-load-config": "^2.0.0", + "schema-utils": "^1.0.0" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/postcss-merge-longhand": { + "version": "4.0.11", + "resolved": "https://registry.npmmirror.com/postcss-merge-longhand/-/postcss-merge-longhand-4.0.11.tgz", + "integrity": "sha512-alx/zmoeXvJjp7L4mxEMjh8lxVlDFX1gqWHzaaQewwMZiVhLo42TEClKaeHbRf6J7j82ZOdTJ808RtN0ZOZwvw==", + "dev": true, + "dependencies": { + "css-color-names": "0.0.4", + "postcss": "^7.0.0", + "postcss-value-parser": "^3.0.0", + "stylehacks": "^4.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/postcss-merge-longhand/node_modules/postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmmirror.com/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + }, + "node_modules/postcss-merge-rules": { + "version": "4.0.3", + "resolved": "https://registry.npmmirror.com/postcss-merge-rules/-/postcss-merge-rules-4.0.3.tgz", + "integrity": "sha512-U7e3r1SbvYzO0Jr3UT/zKBVgYYyhAz0aitvGIYOYK5CPmkNih+WDSsS5tvPrJ8YMQYlEMvsZIiqmn7HdFUaeEQ==", + "dev": true, + "dependencies": { + "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" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/postcss-merge-rules/node_modules/postcss-selector-parser": { + "version": "3.1.2", + "resolved": "https://registry.npmmirror.com/postcss-selector-parser/-/postcss-selector-parser-3.1.2.tgz", + "integrity": "sha512-h7fJ/5uWuRVyOtkO45pnt1Ih40CEleeyCHzipqAZO2e5H20g25Y48uYnFUiShvY4rZWNJ/Bib/KVPmanaCtOhA==", + "dev": true, + "dependencies": { + "dot-prop": "^5.2.0", + "indexes-of": "^1.0.1", + "uniq": "^1.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/postcss-minify-font-values": { + "version": "4.0.2", + "resolved": "https://registry.npmmirror.com/postcss-minify-font-values/-/postcss-minify-font-values-4.0.2.tgz", + "integrity": "sha512-j85oO6OnRU9zPf04+PZv1LYIYOprWm6IA6zkXkrJXyRveDEuQggG6tvoy8ir8ZwjLxLuGfNkCZEQG7zan+Hbtg==", + "dev": true, + "dependencies": { + "postcss": "^7.0.0", + "postcss-value-parser": "^3.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/postcss-minify-font-values/node_modules/postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmmirror.com/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + }, + "node_modules/postcss-minify-gradients": { + "version": "4.0.2", + "resolved": "https://registry.npmmirror.com/postcss-minify-gradients/-/postcss-minify-gradients-4.0.2.tgz", + "integrity": "sha512-qKPfwlONdcf/AndP1U8SJ/uzIJtowHlMaSioKzebAXSG4iJthlWC9iSWznQcX4f66gIWX44RSA841HTHj3wK+Q==", + "dev": true, + "dependencies": { + "cssnano-util-get-arguments": "^4.0.0", + "is-color-stop": "^1.0.0", + "postcss": "^7.0.0", + "postcss-value-parser": "^3.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/postcss-minify-gradients/node_modules/postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmmirror.com/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + }, + "node_modules/postcss-minify-params": { + "version": "4.0.2", + "resolved": "https://registry.npmmirror.com/postcss-minify-params/-/postcss-minify-params-4.0.2.tgz", + "integrity": "sha512-G7eWyzEx0xL4/wiBBJxJOz48zAKV2WG3iZOqVhPet/9geefm/Px5uo1fzlHu+DOjT+m0Mmiz3jkQzVHe6wxAWg==", + "dev": true, + "dependencies": { + "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" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/postcss-minify-params/node_modules/postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmmirror.com/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + }, + "node_modules/postcss-minify-selectors": { + "version": "4.0.2", + "resolved": "https://registry.npmmirror.com/postcss-minify-selectors/-/postcss-minify-selectors-4.0.2.tgz", + "integrity": "sha512-D5S1iViljXBj9kflQo4YutWnJmwm8VvIsU1GeXJGiG9j8CIg9zs4voPMdQDUmIxetUOh60VilsNzCiAFTOqu3g==", + "dev": true, + "dependencies": { + "alphanum-sort": "^1.0.0", + "has": "^1.0.0", + "postcss": "^7.0.0", + "postcss-selector-parser": "^3.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/postcss-minify-selectors/node_modules/postcss-selector-parser": { + "version": "3.1.2", + "resolved": "https://registry.npmmirror.com/postcss-selector-parser/-/postcss-selector-parser-3.1.2.tgz", + "integrity": "sha512-h7fJ/5uWuRVyOtkO45pnt1Ih40CEleeyCHzipqAZO2e5H20g25Y48uYnFUiShvY4rZWNJ/Bib/KVPmanaCtOhA==", + "dev": true, + "dependencies": { + "dot-prop": "^5.2.0", + "indexes-of": "^1.0.1", + "uniq": "^1.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/postcss-modules-extract-imports": { + "version": "1.2.1", + "resolved": "https://registry.npmmirror.com/postcss-modules-extract-imports/-/postcss-modules-extract-imports-1.2.1.tgz", + "integrity": "sha512-6jt9XZwUhwmRUhb/CkyJY020PYaPJsCyt3UjbaWo6XEbH/94Hmv6MP7fG2C5NDU/BcHzyGYxNtHvM+LTf9HrYw==", + "dev": true, + "dependencies": { + "postcss": "^6.0.1" + } + }, + "node_modules/postcss-modules-extract-imports/node_modules/postcss": { + "version": "6.0.23", + "resolved": "https://registry.npmmirror.com/postcss/-/postcss-6.0.23.tgz", + "integrity": "sha512-soOk1h6J3VMTZtVeVpv15/Hpdl2cBLX3CAw4TAbkpTJiNPk9YP/zWcD1ND+xEtvyuuvKzbxliTOIyvkSeSJ6ag==", + "dev": true, + "dependencies": { + "chalk": "^2.4.1", + "source-map": "^0.6.1", + "supports-color": "^5.4.0" + }, + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/postcss-modules-extract-imports/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmmirror.com/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/postcss-modules-local-by-default": { + "version": "1.2.0", + "resolved": "https://registry.npmmirror.com/postcss-modules-local-by-default/-/postcss-modules-local-by-default-1.2.0.tgz", + "integrity": "sha512-X4cquUPIaAd86raVrBwO8fwRfkIdbwFu7CTfEOjiZQHVQwlHRSkTgH5NLDmMm5+1hQO8u6dZ+TOOJDbay1hYpA==", + "dev": true, + "dependencies": { + "css-selector-tokenizer": "^0.7.0", + "postcss": "^6.0.1" + } + }, + "node_modules/postcss-modules-local-by-default/node_modules/postcss": { + "version": "6.0.23", + "resolved": "https://registry.npmmirror.com/postcss/-/postcss-6.0.23.tgz", + "integrity": "sha512-soOk1h6J3VMTZtVeVpv15/Hpdl2cBLX3CAw4TAbkpTJiNPk9YP/zWcD1ND+xEtvyuuvKzbxliTOIyvkSeSJ6ag==", + "dev": true, + "dependencies": { + "chalk": "^2.4.1", + "source-map": "^0.6.1", + "supports-color": "^5.4.0" + }, + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/postcss-modules-local-by-default/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmmirror.com/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/postcss-modules-scope": { + "version": "1.1.0", + "resolved": "https://registry.npmmirror.com/postcss-modules-scope/-/postcss-modules-scope-1.1.0.tgz", + "integrity": "sha512-LTYwnA4C1He1BKZXIx1CYiHixdSe9LWYVKadq9lK5aCCMkoOkFyZ7aigt+srfjlRplJY3gIol6KUNefdMQJdlw==", + "dev": true, + "dependencies": { + "css-selector-tokenizer": "^0.7.0", + "postcss": "^6.0.1" + } + }, + "node_modules/postcss-modules-scope/node_modules/postcss": { + "version": "6.0.23", + "resolved": "https://registry.npmmirror.com/postcss/-/postcss-6.0.23.tgz", + "integrity": "sha512-soOk1h6J3VMTZtVeVpv15/Hpdl2cBLX3CAw4TAbkpTJiNPk9YP/zWcD1ND+xEtvyuuvKzbxliTOIyvkSeSJ6ag==", + "dev": true, + "dependencies": { + "chalk": "^2.4.1", + "source-map": "^0.6.1", + "supports-color": "^5.4.0" + }, + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/postcss-modules-scope/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmmirror.com/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/postcss-modules-values": { + "version": "1.3.0", + "resolved": "https://registry.npmmirror.com/postcss-modules-values/-/postcss-modules-values-1.3.0.tgz", + "integrity": "sha512-i7IFaR9hlQ6/0UgFuqM6YWaCfA1Ej8WMg8A5DggnH1UGKJvTV/ugqq/KaULixzzOi3T/tF6ClBXcHGCzdd5unA==", + "dev": true, + "dependencies": { + "icss-replace-symbols": "^1.1.0", + "postcss": "^6.0.1" + } + }, + "node_modules/postcss-modules-values/node_modules/postcss": { + "version": "6.0.23", + "resolved": "https://registry.npmmirror.com/postcss/-/postcss-6.0.23.tgz", + "integrity": "sha512-soOk1h6J3VMTZtVeVpv15/Hpdl2cBLX3CAw4TAbkpTJiNPk9YP/zWcD1ND+xEtvyuuvKzbxliTOIyvkSeSJ6ag==", + "dev": true, + "dependencies": { + "chalk": "^2.4.1", + "source-map": "^0.6.1", + "supports-color": "^5.4.0" + }, + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/postcss-modules-values/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmmirror.com/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/postcss-normalize-charset": { + "version": "4.0.1", + "resolved": "https://registry.npmmirror.com/postcss-normalize-charset/-/postcss-normalize-charset-4.0.1.tgz", + "integrity": "sha512-gMXCrrlWh6G27U0hF3vNvR3w8I1s2wOBILvA87iNXaPvSNo5uZAMYsZG7XjCUf1eVxuPfyL4TJ7++SGZLc9A3g==", + "dev": true, + "dependencies": { + "postcss": "^7.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/postcss-normalize-display-values": { + "version": "4.0.2", + "resolved": "https://registry.npmmirror.com/postcss-normalize-display-values/-/postcss-normalize-display-values-4.0.2.tgz", + "integrity": "sha512-3F2jcsaMW7+VtRMAqf/3m4cPFhPD3EFRgNs18u+k3lTJJlVe7d0YPO+bnwqo2xg8YiRpDXJI2u8A0wqJxMsQuQ==", + "dev": true, + "dependencies": { + "cssnano-util-get-match": "^4.0.0", + "postcss": "^7.0.0", + "postcss-value-parser": "^3.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/postcss-normalize-display-values/node_modules/postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmmirror.com/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + }, + "node_modules/postcss-normalize-positions": { + "version": "4.0.2", + "resolved": "https://registry.npmmirror.com/postcss-normalize-positions/-/postcss-normalize-positions-4.0.2.tgz", + "integrity": "sha512-Dlf3/9AxpxE+NF1fJxYDeggi5WwV35MXGFnnoccP/9qDtFrTArZ0D0R+iKcg5WsUd8nUYMIl8yXDCtcrT8JrdA==", + "dev": true, + "dependencies": { + "cssnano-util-get-arguments": "^4.0.0", + "has": "^1.0.0", + "postcss": "^7.0.0", + "postcss-value-parser": "^3.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/postcss-normalize-positions/node_modules/postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmmirror.com/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + }, + "node_modules/postcss-normalize-repeat-style": { + "version": "4.0.2", + "resolved": "https://registry.npmmirror.com/postcss-normalize-repeat-style/-/postcss-normalize-repeat-style-4.0.2.tgz", + "integrity": "sha512-qvigdYYMpSuoFs3Is/f5nHdRLJN/ITA7huIoCyqqENJe9PvPmLhNLMu7QTjPdtnVf6OcYYO5SHonx4+fbJE1+Q==", + "dev": true, + "dependencies": { + "cssnano-util-get-arguments": "^4.0.0", + "cssnano-util-get-match": "^4.0.0", + "postcss": "^7.0.0", + "postcss-value-parser": "^3.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/postcss-normalize-repeat-style/node_modules/postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmmirror.com/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + }, + "node_modules/postcss-normalize-string": { + "version": "4.0.2", + "resolved": "https://registry.npmmirror.com/postcss-normalize-string/-/postcss-normalize-string-4.0.2.tgz", + "integrity": "sha512-RrERod97Dnwqq49WNz8qo66ps0swYZDSb6rM57kN2J+aoyEAJfZ6bMx0sx/F9TIEX0xthPGCmeyiam/jXif0eA==", + "dev": true, + "dependencies": { + "has": "^1.0.0", + "postcss": "^7.0.0", + "postcss-value-parser": "^3.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/postcss-normalize-string/node_modules/postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmmirror.com/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + }, + "node_modules/postcss-normalize-timing-functions": { + "version": "4.0.2", + "resolved": "https://registry.npmmirror.com/postcss-normalize-timing-functions/-/postcss-normalize-timing-functions-4.0.2.tgz", + "integrity": "sha512-acwJY95edP762e++00Ehq9L4sZCEcOPyaHwoaFOhIwWCDfik6YvqsYNxckee65JHLKzuNSSmAdxwD2Cud1Z54A==", + "dev": true, + "dependencies": { + "cssnano-util-get-match": "^4.0.0", + "postcss": "^7.0.0", + "postcss-value-parser": "^3.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/postcss-normalize-timing-functions/node_modules/postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmmirror.com/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + }, + "node_modules/postcss-normalize-unicode": { + "version": "4.0.1", + "resolved": "https://registry.npmmirror.com/postcss-normalize-unicode/-/postcss-normalize-unicode-4.0.1.tgz", + "integrity": "sha512-od18Uq2wCYn+vZ/qCOeutvHjB5jm57ToxRaMeNuf0nWVHaP9Hua56QyMF6fs/4FSUnVIw0CBPsU0K4LnBPwYwg==", + "dev": true, + "dependencies": { + "browserslist": "^4.0.0", + "postcss": "^7.0.0", + "postcss-value-parser": "^3.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/postcss-normalize-unicode/node_modules/postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmmirror.com/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + }, + "node_modules/postcss-normalize-url": { + "version": "4.0.1", + "resolved": "https://registry.npmmirror.com/postcss-normalize-url/-/postcss-normalize-url-4.0.1.tgz", + "integrity": "sha512-p5oVaF4+IHwu7VpMan/SSpmpYxcJMtkGppYf0VbdH5B6hN8YNmVyJLuY9FmLQTzY3fag5ESUUHDqM+heid0UVA==", + "dev": true, + "dependencies": { + "is-absolute-url": "^2.0.0", + "normalize-url": "^3.0.0", + "postcss": "^7.0.0", + "postcss-value-parser": "^3.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/postcss-normalize-url/node_modules/postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmmirror.com/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + }, + "node_modules/postcss-normalize-whitespace": { + "version": "4.0.2", + "resolved": "https://registry.npmmirror.com/postcss-normalize-whitespace/-/postcss-normalize-whitespace-4.0.2.tgz", + "integrity": "sha512-tO8QIgrsI3p95r8fyqKV+ufKlSHh9hMJqACqbv2XknufqEDhDvbguXGBBqxw9nsQoXWf0qOqppziKJKHMD4GtA==", + "dev": true, + "dependencies": { + "postcss": "^7.0.0", + "postcss-value-parser": "^3.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/postcss-normalize-whitespace/node_modules/postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmmirror.com/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + }, + "node_modules/postcss-ordered-values": { + "version": "4.1.2", + "resolved": "https://registry.npmmirror.com/postcss-ordered-values/-/postcss-ordered-values-4.1.2.tgz", + "integrity": "sha512-2fCObh5UanxvSxeXrtLtlwVThBvHn6MQcu4ksNT2tsaV2Fg76R2CV98W7wNSlX+5/pFwEyaDwKLLoEV7uRybAw==", + "dev": true, + "dependencies": { + "cssnano-util-get-arguments": "^4.0.0", + "postcss": "^7.0.0", + "postcss-value-parser": "^3.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/postcss-ordered-values/node_modules/postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmmirror.com/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + }, + "node_modules/postcss-prefix-selector": { + "version": "1.15.0", + "resolved": "https://registry.npmmirror.com/postcss-prefix-selector/-/postcss-prefix-selector-1.15.0.tgz", + "integrity": "sha512-9taaTPs6I4906QC03zBBt0LfTWAhrqEWlKSj0jRlxrg1yV+O91h0wcquu6krcA5L6aEv3QnCeG8B1vZ5WT4ecQ==", + "dev": true, + "peerDependencies": { + "postcss": ">4 <9" + } + }, + "node_modules/postcss-reduce-initial": { + "version": "4.0.3", + "resolved": "https://registry.npmmirror.com/postcss-reduce-initial/-/postcss-reduce-initial-4.0.3.tgz", + "integrity": "sha512-gKWmR5aUulSjbzOfD9AlJiHCGH6AEVLaM0AV+aSioxUDd16qXP1PCh8d1/BGVvpdWn8k/HiK7n6TjeoXN1F7DA==", + "dev": true, + "dependencies": { + "browserslist": "^4.0.0", + "caniuse-api": "^3.0.0", + "has": "^1.0.0", + "postcss": "^7.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/postcss-reduce-transforms": { + "version": "4.0.2", + "resolved": "https://registry.npmmirror.com/postcss-reduce-transforms/-/postcss-reduce-transforms-4.0.2.tgz", + "integrity": "sha512-EEVig1Q2QJ4ELpJXMZR8Vt5DQx8/mo+dGWSR7vWXqcob2gQLyQGsionYcGKATXvQzMPn6DSN1vTN7yFximdIAg==", + "dev": true, + "dependencies": { + "cssnano-util-get-match": "^4.0.0", + "has": "^1.0.0", + "postcss": "^7.0.0", + "postcss-value-parser": "^3.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/postcss-reduce-transforms/node_modules/postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmmirror.com/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + }, + "node_modules/postcss-selector-parser": { + "version": "6.0.10", + "resolved": "https://registry.npmmirror.com/postcss-selector-parser/-/postcss-selector-parser-6.0.10.tgz", + "integrity": "sha512-IQ7TZdoaqbT+LCpShg46jnZVlhWD2w6iQYAcYXfHARZ7X1t/UGhhceQDs5X0cGqKvYlHNOuv7Oa1xmb0oQuA3w==", + "dev": true, + "dependencies": { + "cssesc": "^3.0.0", + "util-deprecate": "^1.0.2" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/postcss-svgo": { + "version": "4.0.3", + "resolved": "https://registry.npmmirror.com/postcss-svgo/-/postcss-svgo-4.0.3.tgz", + "integrity": "sha512-NoRbrcMWTtUghzuKSoIm6XV+sJdvZ7GZSc3wdBN0W19FTtp2ko8NqLsgoh/m9CzNhU3KLPvQmjIwtaNFkaFTvw==", + "dev": true, + "dependencies": { + "postcss": "^7.0.0", + "postcss-value-parser": "^3.0.0", + "svgo": "^1.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/postcss-svgo/node_modules/postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmmirror.com/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", + "dev": true + }, + "node_modules/postcss-unique-selectors": { + "version": "4.0.1", + "resolved": "https://registry.npmmirror.com/postcss-unique-selectors/-/postcss-unique-selectors-4.0.1.tgz", + "integrity": "sha512-+JanVaryLo9QwZjKrmJgkI4Fn8SBgRO6WXQBJi7KiAVPlmxikB5Jzc4EvXMT2H0/m0RjrVVm9rGNhZddm/8Spg==", + "dev": true, + "dependencies": { + "alphanum-sort": "^1.0.0", + "postcss": "^7.0.0", + "uniqs": "^2.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/postcss-value-parser": { + "version": "4.2.0", + "resolved": "https://registry.npmmirror.com/postcss-value-parser/-/postcss-value-parser-4.2.0.tgz", + "integrity": "sha512-1NNCs6uurfkVbeXG4S8JFT9t19m45ICnif8zWLd5oPSZ50QnwMfK+H3jv408d4jw/7Bttv5axS5IiHoLaVNHeQ==", + "dev": true + }, + "node_modules/postcss/node_modules/picocolors": { + "version": "0.2.1", + "resolved": "https://registry.npmmirror.com/picocolors/-/picocolors-0.2.1.tgz", + "integrity": "sha512-cMlDqaLEqfSaW8Z7N5Jw+lyIW869EzT73/F5lhtY9cLGoVxSXznfgfXMO0Z5K0o0Q2TkTXq+0KFsdnSe3jDViA==", + "dev": true + }, + "node_modules/postcss/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmmirror.com/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/posthtml": { + "version": "0.9.2", + "resolved": "https://registry.npmmirror.com/posthtml/-/posthtml-0.9.2.tgz", + "integrity": "sha512-spBB5sgC4cv2YcW03f/IAUN1pgDJWNWD8FzkyY4mArLUMJW+KlQhlmUdKAHQuPfb00Jl5xIfImeOsf6YL8QK7Q==", + "dev": true, + "dependencies": { + "posthtml-parser": "^0.2.0", + "posthtml-render": "^1.0.5" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/posthtml-parser": { + "version": "0.2.1", + "resolved": "https://registry.npmmirror.com/posthtml-parser/-/posthtml-parser-0.2.1.tgz", + "integrity": "sha512-nPC53YMqJnc/+1x4fRYFfm81KV2V+G9NZY+hTohpYg64Ay7NemWWcV4UWuy/SgMupqQ3kJ88M/iRfZmSnxT+pw==", + "dev": true, + "dependencies": { + "htmlparser2": "^3.8.3", + "isobject": "^2.1.0" + } + }, + "node_modules/posthtml-parser/node_modules/domhandler": { + "version": "2.4.2", + "resolved": "https://registry.npmmirror.com/domhandler/-/domhandler-2.4.2.tgz", + "integrity": "sha512-JiK04h0Ht5u/80fdLMCEmV4zkNh2BcoMFBmZ/91WtYZ8qVXSKjiw7fXMgFPnHcSZgOo3XdinHvmnDUeMf5R4wA==", + "dev": true, + "dependencies": { + "domelementtype": "1" + } + }, + "node_modules/posthtml-parser/node_modules/entities": { + "version": "1.1.2", + "resolved": "https://registry.npmmirror.com/entities/-/entities-1.1.2.tgz", + "integrity": "sha512-f2LZMYl1Fzu7YSBKg+RoROelpOaNrcGmE9AZubeDfrCEia483oW4MI4VyFd5VNHIgQ/7qm1I0wUHK1eJnn2y2w==", + "dev": true + }, + "node_modules/posthtml-parser/node_modules/htmlparser2": { + "version": "3.10.1", + "resolved": "https://registry.npmmirror.com/htmlparser2/-/htmlparser2-3.10.1.tgz", + "integrity": "sha512-IgieNijUMbkDovyoKObU1DUhm1iwNYE/fuifEoEHfd1oZKZDaONBSkal7Y01shxsM49R4XaMdGez3WnF9UfiCQ==", + "dev": true, + "dependencies": { + "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" + } + }, + "node_modules/posthtml-parser/node_modules/isobject": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/isobject/-/isobject-2.1.0.tgz", + "integrity": "sha512-+OUdGJlgjOBZDfxnDjYYG6zp487z0JGNQq3cYQYg5f5hKR+syHMsaztzGeml/4kGG55CSpKSpWTY+jYGgsHLgA==", + "dev": true, + "dependencies": { + "isarray": "1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/posthtml-parser/node_modules/readable-stream": { + "version": "3.6.0", + "resolved": "https://registry.npmmirror.com/readable-stream/-/readable-stream-3.6.0.tgz", + "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", + "dev": true, + "dependencies": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/posthtml-rename-id": { + "version": "1.0.12", + "resolved": "https://registry.npmmirror.com/posthtml-rename-id/-/posthtml-rename-id-1.0.12.tgz", + "integrity": "sha512-UKXf9OF/no8WZo9edRzvuMenb6AD5hDLzIepJW+a4oJT+T/Lx7vfMYWT4aWlGNQh0WMhnUx1ipN9OkZ9q+ddEw==", + "dev": true, + "dependencies": { + "escape-string-regexp": "1.0.5" + } + }, + "node_modules/posthtml-render": { + "version": "1.4.0", + "resolved": "https://registry.npmmirror.com/posthtml-render/-/posthtml-render-1.4.0.tgz", + "integrity": "sha512-W1779iVHGfq0Fvh2PROhCe2QhB8mEErgqzo1wpIt36tCgChafP+hbXIhLDOM8ePJrZcFs0vkNEtdibEWVqChqw==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/posthtml-svg-mode": { + "version": "1.0.3", + "resolved": "https://registry.npmmirror.com/posthtml-svg-mode/-/posthtml-svg-mode-1.0.3.tgz", + "integrity": "sha512-hEqw9NHZ9YgJ2/0G7CECOeuLQKZi8HjWLkBaSVtOWjygQ9ZD8P7tqeowYs7WrFdKsWEKG7o+IlsPY8jrr0CJpQ==", + "dev": true, + "dependencies": { + "merge-options": "1.0.1", + "posthtml": "^0.9.2", + "posthtml-parser": "^0.2.1", + "posthtml-render": "^1.0.6" + } + }, + "node_modules/prelude-ls": { + "version": "1.1.2", + "resolved": "https://registry.npmmirror.com/prelude-ls/-/prelude-ls-1.1.2.tgz", + "integrity": "sha512-ESF23V4SKG6lVSGZgYNpbsiaAkdab6ZgOxe52p7+Kid3W3u3bxR4Vfd/o21dmN7jSt0IwgZ4v5MUd26FEtXE9w==", + "dev": true, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/prepend-http": { + "version": "1.0.4", + "resolved": "https://registry.npmmirror.com/prepend-http/-/prepend-http-1.0.4.tgz", + "integrity": "sha512-PhmXi5XmoyKw1Un4E+opM2KcsJInDvKyuOumcjjw3waw86ZNjHwVUOOWLc4bCzLdcKNaWBH9e99sbWzDQsVaYg==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/preserve": { + "version": "0.2.0", + "resolved": "https://registry.npmmirror.com/preserve/-/preserve-0.2.0.tgz", + "integrity": "sha512-s/46sYeylUfHNjI+sA/78FAHlmIuKqI9wNnzEOGehAlUUYeObv5C2mOinXBjyUyWmJ2SfcS2/ydApH4hTF4WXQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/prettier": { + "version": "2.6.2", + "resolved": "https://registry.npmmirror.com/prettier/-/prettier-2.6.2.tgz", + "integrity": "sha512-PkUpF+qoXTqhOeWL9fu7As8LXsIUZ1WYaJiY/a7McAQzxjk82OF0tibkFXVCDImZtWxbvojFjerkiLb0/q8mew==", + "dev": true, + "optional": true, + "bin": { + "prettier": "bin-prettier.js" + }, + "engines": { + "node": ">=10.13.0" + } + }, + "node_modules/pretty-error": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/pretty-error/-/pretty-error-2.1.2.tgz", + "integrity": "sha512-EY5oDzmsX5wvuynAByrmY0P0hcp+QpnAKbJng2A2MPjVKXCxrDSUkzghVJ4ZGPIv+JC4gX8fPUWscC0RtjsWGw==", + "dev": true, + "dependencies": { + "lodash": "^4.17.20", + "renderkid": "^2.0.4" + } + }, + "node_modules/process": { + "version": "0.11.10", + "resolved": "https://registry.npmmirror.com/process/-/process-0.11.10.tgz", + "integrity": "sha512-cdGef/drWFoydD1JsMzuFf8100nZl+GT+yacc2bEced5f9Rjk4z+WtFUTBu9PhOi9j/jfmBPu0mMEY4wIdAF8A==", + "engines": { + "node": ">= 0.6.0" + } + }, + "node_modules/process-nextick-args": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/process-nextick-args/-/process-nextick-args-2.0.1.tgz", + "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==" + }, + "node_modules/progress": { + "version": "2.0.3", + "resolved": "https://registry.npmmirror.com/progress/-/progress-2.0.3.tgz", + "integrity": "sha512-7PiHtLll5LdnKIMw100I+8xJXR5gW2QwWYkT6iJva0bXitZKa/XMrSbdmg3r2Xnaidz9Qumd0VPaMrZlF9V9sA==", + "dev": true, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/promise-inflight": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/promise-inflight/-/promise-inflight-1.0.1.tgz", + "integrity": "sha512-6zWPyEOFaQBJYcGMHBKTKJ3u6TBsnMFOIZSa6ce1e/ZrrsOlnHRHbabMjLiBYKp+n44X9eUI6VUPaukCXHuG4g==" + }, + "node_modules/promise-retry": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/promise-retry/-/promise-retry-2.0.1.tgz", + "integrity": "sha512-y+WKFlBR8BGXnsNlIHFGPZmyDf3DFMoLhaflAnyZgV6rG6xu+JwesTo2Q9R6XwYmtmwAFCkAk3e35jEdoeh/3g==", + "dev": true, + "dependencies": { + "err-code": "^2.0.2", + "retry": "^0.12.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/promzard": { + "version": "0.3.0", + "resolved": "https://registry.npmmirror.com/promzard/-/promzard-0.3.0.tgz", + "integrity": "sha512-JZeYqd7UAcHCwI+sTOeUDYkvEU+1bQ7iE0UT1MgB/tERkAPkesW46MrpIySzODi+owTjZtiF8Ay5j9m60KmMBw==", + "dev": true, + "dependencies": { + "read": "1" + } + }, + "node_modules/property-expr": { + "version": "1.5.1", + "resolved": "https://registry.npmmirror.com/property-expr/-/property-expr-1.5.1.tgz", + "integrity": "sha512-CGuc0VUTGthpJXL36ydB6jnbyOf/rAHFvmVrJlH+Rg0DqqLFQGAP6hIaxD/G0OAmBJPhXDHuEJigrp0e0wFV6g==", + "dev": true + }, + "node_modules/proto-list": { + "version": "1.2.4", + "resolved": "https://registry.npmmirror.com/proto-list/-/proto-list-1.2.4.tgz", + "integrity": "sha512-vtK/94akxsTMhe0/cbfpR+syPuszcuwhqVjJq26CuNDgFGj682oRBXOP5MJpv2r7JtE8MsiepGIqvvOTBwn2vA==", + "dev": true + }, + "node_modules/protocols": { + "version": "1.4.8", + "resolved": "https://registry.npmmirror.com/protocols/-/protocols-1.4.8.tgz", + "integrity": "sha512-IgjKyaUSjsROSO8/D49Ab7hP8mJgTYcqApOqdPhLoPxAplXmkp+zRvsrSQjFn5by0rhm4VH0GAUELIPpx7B1yg==", + "dev": true + }, + "node_modules/proxy-addr": { + "version": "2.0.7", + "resolved": "https://registry.npmmirror.com/proxy-addr/-/proxy-addr-2.0.7.tgz", + "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==", + "dev": true, + "dependencies": { + "forwarded": "0.2.0", + "ipaddr.js": "1.9.1" + }, + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/prr": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/prr/-/prr-1.0.1.tgz", + "integrity": "sha512-yPw4Sng1gWghHQWj0B3ZggWUm4qVbPwPFcRG8KyxiU7J2OHFSoEHKS+EZ3fv5l1t9CyCiop6l/ZYeWbrgoQejw==" + }, + "node_modules/pseudomap": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/pseudomap/-/pseudomap-1.0.2.tgz", + "integrity": "sha512-b/YwNhb8lk1Zz2+bXXpS/LK9OisiZZ1SNsSLxN1x2OXVEhW2Ckr/7mWE5vrC1ZTiJlD9g19jWszTmJsB+oEpFQ==", + "dev": true + }, + "node_modules/psl": { + "version": "1.8.0", + "resolved": "https://registry.npmmirror.com/psl/-/psl-1.8.0.tgz", + "integrity": "sha512-RIdOzyoavK+hA18OGGWDqUTsCLhtA7IcZ/6NCs4fFJaHBDab+pDDmDIByWFRQJq2Cd7r1OoQxBGKOaztq+hjIQ==", + "dev": true + }, + "node_modules/public-encrypt": { + "version": "4.0.3", + "resolved": "https://registry.npmmirror.com/public-encrypt/-/public-encrypt-4.0.3.tgz", + "integrity": "sha512-zVpa8oKZSz5bTMTFClc1fQOnyyEzpl5ozpi1B5YcvBrdohMjH2rfsBtyXcuNuwjsDIXmBYlF2N5FlJYhR29t8Q==", + "dependencies": { + "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" + } + }, + "node_modules/public-encrypt/node_modules/bn.js": { + "version": "4.12.0", + "resolved": "https://registry.npmmirror.com/bn.js/-/bn.js-4.12.0.tgz", + "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==" + }, + "node_modules/pump": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/pump/-/pump-3.0.0.tgz", + "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==", + "dependencies": { + "end-of-stream": "^1.1.0", + "once": "^1.3.1" + } + }, + "node_modules/pumpify": { + "version": "1.5.1", + "resolved": "https://registry.npmmirror.com/pumpify/-/pumpify-1.5.1.tgz", + "integrity": "sha512-oClZI37HvuUJJxSKKrC17bZ9Cu0ZYhEAGPsPUy9KlMUmv9dKX2o77RUmq7f3XjIxbwyGwYzbzQ1L2Ks8sIradQ==", + "dependencies": { + "duplexify": "^3.6.0", + "inherits": "^2.0.3", + "pump": "^2.0.0" + } + }, + "node_modules/pumpify/node_modules/pump": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/pump/-/pump-2.0.1.tgz", + "integrity": "sha512-ruPMNRkN3MHP1cWJc9OWr+T/xDP0jhXYCLfJcBuX54hhfIBnaQmAUMfDcG4DM5UMWByBbJY69QSphm3jtDKIkA==", + "dependencies": { + "end-of-stream": "^1.1.0", + "once": "^1.3.1" + } + }, + "node_modules/punycode": { + "version": "2.1.1", + "resolved": "https://registry.npmmirror.com/punycode/-/punycode-2.1.1.tgz", + "integrity": "sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A==", + "engines": { + "node": ">=6" + } + }, + "node_modules/q": { + "version": "1.5.1", + "resolved": "https://registry.npmmirror.com/q/-/q-1.5.1.tgz", + "integrity": "sha512-kV/CThkXo6xyFEZUugw/+pIOywXcDbFYgSct5cT3gqlbkBE1SJdwy6UQoZvodiWF/ckQLZyDE/Bu1M6gVu5lVw==", + "engines": { + "node": ">=0.6.0", + "teleport": ">=0.2.0" + } + }, + "node_modules/qs": { + "version": "6.9.4", + "resolved": "https://registry.npmmirror.com/qs/-/qs-6.9.4.tgz", + "integrity": "sha512-A1kFqHekCTM7cz0udomYUoYNWjBebHm/5wzU/XqrBRBNWectVH0QIiN+NEcZ0Dte5hvzHwbr8+XQmguPhJ6WdQ==", + "engines": { + "node": ">=0.6" + } + }, + "node_modules/query-string": { + "version": "4.3.4", + "resolved": "https://registry.npmmirror.com/query-string/-/query-string-4.3.4.tgz", + "integrity": "sha512-O2XLNDBIg1DnTOa+2XrIwSiXEV8h2KImXUnjhhn2+UsvZ+Es2uyd5CCRTNQlDGbzUQOW3aYCBx9rVA6dzsiY7Q==", + "dev": true, + "dependencies": { + "object-assign": "^4.1.0", + "strict-uri-encode": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/querystring": { + "version": "0.2.0", + "resolved": "https://registry.npmmirror.com/querystring/-/querystring-0.2.0.tgz", + "integrity": "sha512-X/xY82scca2tau62i9mDyU9K+I+djTMUsvwf7xnUX5GLvVzgJybOJf4Y6o9Zx3oJK/LSXg5tTZBjwzqVPaPO2g==", + "deprecated": "The querystring API is considered Legacy. new code should use the URLSearchParams API instead.", + "engines": { + "node": ">=0.4.x" + } + }, + "node_modules/querystring-es3": { + "version": "0.2.1", + "resolved": "https://registry.npmmirror.com/querystring-es3/-/querystring-es3-0.2.1.tgz", + "integrity": "sha512-773xhDQnZBMFobEiztv8LIl70ch5MSF/jUQVlhwFyBILqq96anmoctVIYz+ZRp0qbCKATTn6ev02M3r7Ga5vqA==", + "engines": { + "node": ">=0.4.x" + } + }, + "node_modules/querystringify": { + "version": "2.2.0", + "resolved": "https://registry.npmmirror.com/querystringify/-/querystringify-2.2.0.tgz", + "integrity": "sha512-FIqgj2EUvTa7R50u0rGsyTftzjYmv/a3hO345bZNrqabNqjtgiDMgmo4mkUjd+nzU5oF3dClKqFIPUKybUyqoQ==", + "dev": true + }, + "node_modules/queue-microtask": { + "version": "1.2.3", + "resolved": "https://registry.npmmirror.com/queue-microtask/-/queue-microtask-1.2.3.tgz", + "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==", + "dev": true + }, + "node_modules/quick-lru": { + "version": "4.0.1", + "resolved": "https://registry.npmmirror.com/quick-lru/-/quick-lru-4.0.1.tgz", + "integrity": "sha512-ARhCpm70fzdcvNQfPoy49IaanKkTlRWF2JMzqhcJbhSFRZv7nPTvZJdcY7301IPmvW+/p0RgIWnQDLJxifsQ7g==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/randomatic": { + "version": "3.1.1", + "resolved": "https://registry.npmmirror.com/randomatic/-/randomatic-3.1.1.tgz", + "integrity": "sha512-TuDE5KxZ0J461RVjrJZCJc+J+zCkTb1MbH9AQUq68sMhOMcy9jLcb3BrZKgp9q9Ncltdg4QVqWrH02W2EFFVYw==", + "dev": true, + "dependencies": { + "is-number": "^4.0.0", + "kind-of": "^6.0.0", + "math-random": "^1.0.1" + }, + "engines": { + "node": ">= 0.10.0" + } + }, + "node_modules/randomatic/node_modules/is-number": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/is-number/-/is-number-4.0.0.tgz", + "integrity": "sha512-rSklcAIlf1OmFdyAqbnWTLVelsQ58uvZ66S/ZyawjWqIviTWCjg2PzVGw8WUA+nNuPTqb4wgA+NszrJ+08LlgQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/randombytes": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/randombytes/-/randombytes-2.1.0.tgz", + "integrity": "sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==", + "dependencies": { + "safe-buffer": "^5.1.0" + } + }, + "node_modules/randomfill": { + "version": "1.0.4", + "resolved": "https://registry.npmmirror.com/randomfill/-/randomfill-1.0.4.tgz", + "integrity": "sha512-87lcbR8+MhcWcUiQ+9e+Rwx8MyR2P7qnt15ynUlbm3TU/fjbgz4GsvfSUDTemtCCtVCqb4ZcEFlyPNTh9bBTLw==", + "dependencies": { + "randombytes": "^2.0.5", + "safe-buffer": "^5.1.0" + } + }, + "node_modules/range-parser": { + "version": "1.2.1", + "resolved": "https://registry.npmmirror.com/range-parser/-/range-parser-1.2.1.tgz", + "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==", + "dev": true, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/raw-body": { + "version": "2.5.1", + "resolved": "https://registry.npmmirror.com/raw-body/-/raw-body-2.5.1.tgz", + "integrity": "sha512-qqJBtEyVgS0ZmPGdCFPWJ3FreoqvG4MVQln/kCgF7Olq95IbOp0/BWyMwbdtn4VTvkM8Y7khCQ2Xgk/tcrCXig==", + "dev": true, + "dependencies": { + "bytes": "3.1.2", + "http-errors": "2.0.0", + "iconv-lite": "0.4.24", + "unpipe": "1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/read": { + "version": "1.0.7", + "resolved": "https://registry.npmmirror.com/read/-/read-1.0.7.tgz", + "integrity": "sha512-rSOKNYUmaxy0om1BNjMN4ezNT6VKK+2xF4GBhc81mkH7L60i6dp8qPYrkndNLT3QPphoII3maL9PVC9XmhHwVQ==", + "dev": true, + "dependencies": { + "mute-stream": "~0.0.4" + }, + "engines": { + "node": ">=0.8" + } + }, + "node_modules/read-cmd-shim": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/read-cmd-shim/-/read-cmd-shim-2.0.0.tgz", + "integrity": "sha512-HJpV9bQpkl6KwjxlJcBoqu9Ba0PQg8TqSNIOrulGt54a0uup0HtevreFHzYzkm0lpnleRdNBzXznKrgxglEHQw==", + "dev": true + }, + "node_modules/read-package-json": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/read-package-json/-/read-package-json-2.1.2.tgz", + "integrity": "sha512-D1KmuLQr6ZSJS0tW8hf3WGpRlwszJOXZ3E8Yd/DNRaM5d+1wVRZdHlpGBLAuovjr28LbWvjpWkBHMxpRGGjzNA==", + "dev": true, + "dependencies": { + "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" + } + }, + "node_modules/read-package-json-fast": { + "version": "2.0.3", + "resolved": "https://registry.npmmirror.com/read-package-json-fast/-/read-package-json-fast-2.0.3.tgz", + "integrity": "sha512-W/BKtbL+dUjTuRL2vziuYhp76s5HZ9qQhd/dKfWIZveD0O40453QNyZhC0e63lqZrAQ4jiOapVoeJ7JrszenQQ==", + "dev": true, + "dependencies": { + "json-parse-even-better-errors": "^2.3.0", + "npm-normalize-package-bin": "^1.0.1" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/read-package-tree": { + "version": "5.3.1", + "resolved": "https://registry.npmmirror.com/read-package-tree/-/read-package-tree-5.3.1.tgz", + "integrity": "sha512-mLUDsD5JVtlZxjSlPPx1RETkNjjvQYuweKwNVt1Sn8kP5Jh44pvYuUHCp6xSVDZWbNxVxG5lyZJ921aJH61sTw==", + "deprecated": "The functionality that this package provided is now in @npmcli/arborist", + "dev": true, + "dependencies": { + "read-package-json": "^2.0.0", + "readdir-scoped-modules": "^1.0.0", + "util-promisify": "^2.1.0" + } + }, + "node_modules/read-pkg": { + "version": "5.2.0", + "resolved": "https://registry.npmmirror.com/read-pkg/-/read-pkg-5.2.0.tgz", + "integrity": "sha512-Ug69mNOpfvKDAc2Q8DRpMjjzdtrnv9HcSMX+4VsZxD1aZ6ZzrIE7rlzXBtWTyhULSMKg076AW6WR5iZpD0JiOg==", + "dev": true, + "dependencies": { + "@types/normalize-package-data": "^2.4.0", + "normalize-package-data": "^2.5.0", + "parse-json": "^5.0.0", + "type-fest": "^0.6.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/read-pkg-up": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/read-pkg-up/-/read-pkg-up-3.0.0.tgz", + "integrity": "sha512-YFzFrVvpC6frF1sz8psoHDBGF7fLPc+llq/8NB43oagqWkx8ar5zYtsTORtOjw9W2RHLpWP+zTWwBvf1bCmcSw==", + "dev": true, + "dependencies": { + "find-up": "^2.0.0", + "read-pkg": "^3.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/read-pkg-up/node_modules/find-up": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/find-up/-/find-up-2.1.0.tgz", + "integrity": "sha512-NWzkk0jSJtTt08+FBFMvXoeZnOJD+jTtsRmBYbAIzJdX6l7dLgR7CTubCM5/eDdPUBvLCeVasP1brfVR/9/EZQ==", + "dev": true, + "dependencies": { + "locate-path": "^2.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/read-pkg-up/node_modules/load-json-file": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/load-json-file/-/load-json-file-4.0.0.tgz", + "integrity": "sha512-Kx8hMakjX03tiGTLAIdJ+lL0htKnXjEZN6hk/tozf/WOuYGdZBJrZ+rCJRbVCugsjB3jMLn9746NsQIf5VjBMw==", + "dev": true, + "dependencies": { + "graceful-fs": "^4.1.2", + "parse-json": "^4.0.0", + "pify": "^3.0.0", + "strip-bom": "^3.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/read-pkg-up/node_modules/locate-path": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/locate-path/-/locate-path-2.0.0.tgz", + "integrity": "sha512-NCI2kiDkyR7VeEKm27Kda/iQHyKJe1Bu0FlTbYp3CqJu+9IFe9bLyAjMxf5ZDDbEg+iMPzB5zYyUTSm8wVTKmA==", + "dev": true, + "dependencies": { + "p-locate": "^2.0.0", + "path-exists": "^3.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/read-pkg-up/node_modules/p-limit": { + "version": "1.3.0", + "resolved": "https://registry.npmmirror.com/p-limit/-/p-limit-1.3.0.tgz", + "integrity": "sha512-vvcXsLAJ9Dr5rQOPk7toZQZJApBl2K4J6dANSsEuh6QI41JYcsS/qhTGa9ErIUUgK3WNQoJYvylxvjqmiqEA9Q==", + "dev": true, + "dependencies": { + "p-try": "^1.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/read-pkg-up/node_modules/p-locate": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/p-locate/-/p-locate-2.0.0.tgz", + "integrity": "sha512-nQja7m7gSKuewoVRen45CtVfODR3crN3goVQ0DDZ9N3yHxgpkuBhZqsaiotSQRrADUrne346peY7kT3TSACykg==", + "dev": true, + "dependencies": { + "p-limit": "^1.1.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/read-pkg-up/node_modules/p-try": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/p-try/-/p-try-1.0.0.tgz", + "integrity": "sha512-U1etNYuMJoIz3ZXSrrySFjsXQTWOx2/jdi86L+2pRvph/qMKL6sbcCYdH23fqsbm8TH2Gn0OybpT4eSFlCVHww==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/read-pkg-up/node_modules/pify": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/pify/-/pify-3.0.0.tgz", + "integrity": "sha512-C3FsVNH1udSEX48gGX1xfvwTWfsYWj5U+8/uK15BGzIGrKoUpghX8hWZwa/OFnakBiiVNmBvemTJR5mcy7iPcg==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/read-pkg-up/node_modules/read-pkg": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/read-pkg/-/read-pkg-3.0.0.tgz", + "integrity": "sha512-BLq/cCO9two+lBgiTYNqD6GdtK8s4NpaWrl6/rCO9w0TUS8oJl7cmToOZfRYllKTISY6nt1U7jQ53brmKqY6BA==", + "dev": true, + "dependencies": { + "load-json-file": "^4.0.0", + "normalize-package-data": "^2.3.2", + "path-type": "^3.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/read-pkg/node_modules/parse-json": { + "version": "5.2.0", + "resolved": "https://registry.npmmirror.com/parse-json/-/parse-json-5.2.0.tgz", + "integrity": "sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==", + "dev": true, + "dependencies": { + "@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" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/readable-stream": { + "version": "2.3.7", + "resolved": "https://registry.npmmirror.com/readable-stream/-/readable-stream-2.3.7.tgz", + "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", + "dependencies": { + "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" + } + }, + "node_modules/readdir-scoped-modules": { + "version": "1.1.0", + "resolved": "https://registry.npmmirror.com/readdir-scoped-modules/-/readdir-scoped-modules-1.1.0.tgz", + "integrity": "sha512-asaikDeqAQg7JifRsZn1NJZXo9E+VwlyCfbkZhwyISinqk5zNS6266HS5kah6P0SaQKGF6SkNnZVHUzHFYxYDw==", + "deprecated": "This functionality has been moved to @npmcli/fs", + "dev": true, + "dependencies": { + "debuglog": "^1.0.1", + "dezalgo": "^1.0.0", + "graceful-fs": "^4.1.2", + "once": "^1.3.0" + } + }, + "node_modules/readdirp": { + "version": "3.6.0", + "resolved": "https://registry.npmmirror.com/readdirp/-/readdirp-3.6.0.tgz", + "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", + "optional": true, + "dependencies": { + "picomatch": "^2.2.1" + }, + "engines": { + "node": ">=8.10.0" + } + }, + "node_modules/reconnecting-websocket": { + "version": "4.4.0", + "resolved": "https://registry.npmmirror.com/reconnecting-websocket/-/reconnecting-websocket-4.4.0.tgz", + "integrity": "sha512-D2E33ceRPga0NvTDhJmphEgJ7FUYF0v4lr1ki0csq06OdlxKfugGzN0dSkxM/NfqCxYELK4KcaTOUOjTV6Dcng==" + }, + "node_modules/redent": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/redent/-/redent-3.0.0.tgz", + "integrity": "sha512-6tDA8g98We0zd0GvVeMT9arEOnTw9qM03L9cJXaCjrip1OO764RDBLBfrB4cwzNGDj5OA5ioymC9GkizgWJDUg==", + "dev": true, + "dependencies": { + "indent-string": "^4.0.0", + "strip-indent": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/redent/node_modules/strip-indent": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/strip-indent/-/strip-indent-3.0.0.tgz", + "integrity": "sha512-laJTa3Jb+VQpaC6DseHhF7dXVqHTfJPCRDaEbid/drOhgitgYku/letMUqOXFoWV0zIIUbjpdH2t+tYj4bQMRQ==", + "dev": true, + "dependencies": { + "min-indent": "^1.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/regenerate": { + "version": "1.4.2", + "resolved": "https://registry.npmmirror.com/regenerate/-/regenerate-1.4.2.tgz", + "integrity": "sha512-zrceR/XhGYU/d/opr2EKO7aRHUeiBI8qjtfHqADTwZd6Szfy16la6kqD0MIUs5z5hx6AaKa+PixpPrR289+I0A==", + "dev": true + }, + "node_modules/regenerate-unicode-properties": { + "version": "10.0.1", + "resolved": "https://registry.npmmirror.com/regenerate-unicode-properties/-/regenerate-unicode-properties-10.0.1.tgz", + "integrity": "sha512-vn5DU6yg6h8hP/2OkQo3K7uVILvY4iu0oI4t3HFa81UPkhGJwkRwM10JEc3upjdhHjs/k8GJY1sRBhk5sr69Bw==", + "dev": true, + "dependencies": { + "regenerate": "^1.4.2" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/regenerator-runtime": { + "version": "0.13.9", + "resolved": "https://registry.npmmirror.com/regenerator-runtime/-/regenerator-runtime-0.13.9.tgz", + "integrity": "sha512-p3VT+cOEgxFsRRA9X4lkI1E+k2/CtnKtU4gcxyaCUreilL/vqI6CdZ3wxVUx3UOUg+gnUOQQcRI7BmSI656MYA==" + }, + "node_modules/regenerator-transform": { + "version": "0.15.0", + "resolved": "https://registry.npmmirror.com/regenerator-transform/-/regenerator-transform-0.15.0.tgz", + "integrity": "sha512-LsrGtPmbYg19bcPHwdtmXwbW+TqNvtY4riE3P83foeHRroMbH6/2ddFBfab3t7kbzc7v7p4wbkIecHImqt0QNg==", + "dev": true, + "dependencies": { + "@babel/runtime": "^7.8.4" + } + }, + "node_modules/regex-cache": { + "version": "0.4.4", + "resolved": "https://registry.npmmirror.com/regex-cache/-/regex-cache-0.4.4.tgz", + "integrity": "sha512-nVIZwtCjkC9YgvWkpM55B5rBhBYRZhAaJbgcFYXXsHnbZ9UZI9nnVWYZpBlCqv9ho2eZryPnWrZGsOdPwVWXWQ==", + "dev": true, + "dependencies": { + "is-equal-shallow": "^0.1.3" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/regex-not": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/regex-not/-/regex-not-1.0.2.tgz", + "integrity": "sha512-J6SDjUgDxQj5NusnOtdFxDwN/+HWykR8GELwctJ7mdqhcyy1xEc4SRFHUXvxTp661YaVKAjfRLZ9cCqS6tn32A==", + "dependencies": { + "extend-shallow": "^3.0.2", + "safe-regex": "^1.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/regexp.prototype.flags": { + "version": "1.4.3", + "resolved": "https://registry.npmmirror.com/regexp.prototype.flags/-/regexp.prototype.flags-1.4.3.tgz", + "integrity": "sha512-fjggEOO3slI6Wvgjwflkc4NFRCTZAu5CnNfBd5qOMYhWdn67nJBBu34/TkD++eeFmd8C9r9jfXJ27+nSiRkSUA==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.3", + "functions-have-names": "^1.2.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/regexpp": { + "version": "1.1.0", + "resolved": "https://registry.npmmirror.com/regexpp/-/regexpp-1.1.0.tgz", + "integrity": "sha512-LOPw8FpgdQF9etWMaAfG/WRthIdXJGYp4mJ2Jgn/2lpkbod9jPn0t9UqN7AxBOKNfzRbYyVfgc7Vk4t/MpnXgw==", + "dev": true, + "optional": true, + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/regexpu-core": { + "version": "5.0.1", + "resolved": "https://registry.npmmirror.com/regexpu-core/-/regexpu-core-5.0.1.tgz", + "integrity": "sha512-CriEZlrKK9VJw/xQGJpQM5rY88BtuL8DM+AEwvcThHilbxiTAy8vq4iJnd2tqq8wLmjbGZzP7ZcKFjbGkmEFrw==", + "dev": true, + "dependencies": { + "regenerate": "^1.4.2", + "regenerate-unicode-properties": "^10.0.1", + "regjsgen": "^0.6.0", + "regjsparser": "^0.8.2", + "unicode-match-property-ecmascript": "^2.0.0", + "unicode-match-property-value-ecmascript": "^2.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/regjsgen": { + "version": "0.6.0", + "resolved": "https://registry.npmmirror.com/regjsgen/-/regjsgen-0.6.0.tgz", + "integrity": "sha512-ozE883Uigtqj3bx7OhL1KNbCzGyW2NQZPl6Hs09WTvCuZD5sTI4JY58bkbQWa/Y9hxIsvJ3M8Nbf7j54IqeZbA==", + "dev": true + }, + "node_modules/regjsparser": { + "version": "0.8.4", + "resolved": "https://registry.npmmirror.com/regjsparser/-/regjsparser-0.8.4.tgz", + "integrity": "sha512-J3LABycON/VNEu3abOviqGHuB/LOtOQj8SKmfP9anY5GfAVw/SPjwzSjxGjbZXIxbGfqTHtJw58C2Li/WkStmA==", + "dev": true, + "dependencies": { + "jsesc": "~0.5.0" + }, + "bin": { + "regjsparser": "bin/parser" + } + }, + "node_modules/regjsparser/node_modules/jsesc": { + "version": "0.5.0", + "resolved": "https://registry.npmmirror.com/jsesc/-/jsesc-0.5.0.tgz", + "integrity": "sha512-uZz5UnB7u4T9LvwmFqXii7pZSouaRPorGs5who1Ip7VO0wxanFvBL7GkM6dTHlgX+jhBApRetaWpnDabOeTcnA==", + "dev": true, + "bin": { + "jsesc": "bin/jsesc" + } + }, + "node_modules/relateurl": { + "version": "0.2.7", + "resolved": "https://registry.npmmirror.com/relateurl/-/relateurl-0.2.7.tgz", + "integrity": "sha512-G08Dxvm4iDN3MLM0EsP62EDV9IuhXPR6blNz6Utcp7zyV3tr4HVNINt6MpaRWbxoOHT3Q7YN2P+jaHX8vUbgog==", + "dev": true, + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/remove-trailing-separator": { + "version": "1.1.0", + "resolved": "https://registry.npmmirror.com/remove-trailing-separator/-/remove-trailing-separator-1.1.0.tgz", + "integrity": "sha512-/hS+Y0u3aOfIETiaiirUFwDBDzmXPvO+jAfKTitUngIPzdKc6Z0LoFjM/CK5PL4C+eKwHohlHAb6H0VFfmmUsw==", + "devOptional": true + }, + "node_modules/renderkid": { + "version": "2.0.7", + "resolved": "https://registry.npmmirror.com/renderkid/-/renderkid-2.0.7.tgz", + "integrity": "sha512-oCcFyxaMrKsKcTY59qnCAtmDVSLfPbrv6A3tVbPdFMMrv5jaK10V6m40cKsoPNhAqN6rmHW9sswW4o3ruSrwUQ==", + "dev": true, + "dependencies": { + "css-select": "^4.1.3", + "dom-converter": "^0.2.0", + "htmlparser2": "^6.1.0", + "lodash": "^4.17.21", + "strip-ansi": "^3.0.1" + } + }, + "node_modules/renderkid/node_modules/ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmmirror.com/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha512-TIGnTpdo+E3+pCyAluZvtED5p5wCqLdezCyhPZzKPcxvFplEt4i+W7OONCKgeZFT3+y5NZZfOOS/Bdcanm1MYA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/renderkid/node_modules/css-select": { + "version": "4.3.0", + "resolved": "https://registry.npmmirror.com/css-select/-/css-select-4.3.0.tgz", + "integrity": "sha512-wPpOYtnsVontu2mODhA19JrqWxNsfdatRKd64kmpRbQgh1KtItko5sTnEpPdpSaJszTOhEMlF/RPz28qj4HqhQ==", + "dev": true, + "dependencies": { + "boolbase": "^1.0.0", + "css-what": "^6.0.1", + "domhandler": "^4.3.1", + "domutils": "^2.8.0", + "nth-check": "^2.0.1" + } + }, + "node_modules/renderkid/node_modules/css-what": { + "version": "6.1.0", + "resolved": "https://registry.npmmirror.com/css-what/-/css-what-6.1.0.tgz", + "integrity": "sha512-HTUrgRJ7r4dsZKU6GjmpfRK1O76h97Z8MfS1G0FozR+oF2kG6Vfe8JE6zwrkbxigziPHinCJ+gCPjA9EaBDtRw==", + "dev": true, + "engines": { + "node": ">= 6" + } + }, + "node_modules/renderkid/node_modules/dom-serializer": { + "version": "1.4.1", + "resolved": "https://registry.npmmirror.com/dom-serializer/-/dom-serializer-1.4.1.tgz", + "integrity": "sha512-VHwB3KfrcOOkelEG2ZOfxqLZdfkil8PtJi4P8N2MMXucZq2yLp75ClViUlOVwyoHEDjYU433Aq+5zWP61+RGag==", + "dev": true, + "dependencies": { + "domelementtype": "^2.0.1", + "domhandler": "^4.2.0", + "entities": "^2.0.0" + } + }, + "node_modules/renderkid/node_modules/domelementtype": { + "version": "2.3.0", + "resolved": "https://registry.npmmirror.com/domelementtype/-/domelementtype-2.3.0.tgz", + "integrity": "sha512-OLETBj6w0OsagBwdXnPdN0cnMfF9opN69co+7ZrbfPGrdpPVNBUj02spi6B1N7wChLQiPn4CSH/zJvXw56gmHw==", + "dev": true + }, + "node_modules/renderkid/node_modules/domutils": { + "version": "2.8.0", + "resolved": "https://registry.npmmirror.com/domutils/-/domutils-2.8.0.tgz", + "integrity": "sha512-w96Cjofp72M5IIhpjgobBimYEfoPjx1Vx0BSX9P30WBdZW2WIKU0T1Bd0kz2eNZ9ikjKgHbEyKx8BB6H1L3h3A==", + "dev": true, + "dependencies": { + "dom-serializer": "^1.0.1", + "domelementtype": "^2.2.0", + "domhandler": "^4.2.0" + } + }, + "node_modules/renderkid/node_modules/lodash": { + "version": "4.17.21", + "resolved": "https://registry.npmmirror.com/lodash/-/lodash-4.17.21.tgz", + "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==", + "dev": true + }, + "node_modules/renderkid/node_modules/nth-check": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/nth-check/-/nth-check-2.0.1.tgz", + "integrity": "sha512-it1vE95zF6dTT9lBsYbxvqh0Soy4SPowchj0UBGj/V6cTPnXXtQOPUbhZ6CmGzAD/rW22LQK6E96pcdJXk4A4w==", + "dev": true, + "dependencies": { + "boolbase": "^1.0.0" + } + }, + "node_modules/renderkid/node_modules/strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmmirror.com/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha512-VhumSSbBqDTP8p2ZLKj40UjBCV4+v8bUSEpUb4KjRgWk9pbqGF4REFj6KEagidb2f/M6AzC0EmFyDNGaw9OCzg==", + "dev": true, + "dependencies": { + "ansi-regex": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/repeat-element": { + "version": "1.1.4", + "resolved": "https://registry.npmmirror.com/repeat-element/-/repeat-element-1.1.4.tgz", + "integrity": "sha512-LFiNfRcSu7KK3evMyYOuCzv3L10TW7yC1G2/+StMjK8Y6Vqd2MG7r/Qjw4ghtuCOjFvlnms/iMmLqpvW/ES/WQ==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/repeat-string": { + "version": "1.6.1", + "resolved": "https://registry.npmmirror.com/repeat-string/-/repeat-string-1.6.1.tgz", + "integrity": "sha512-PV0dzCYDNfRi1jCDbJzpW7jNNDRuCOG/jI5ctQcGKt/clZD+YcPS3yIlWuTJMmESC8aevCFmWJy5wjAFgNqN6w==", + "engines": { + "node": ">=0.10" + } + }, + "node_modules/request": { + "version": "2.88.2", + "resolved": "https://registry.npmmirror.com/request/-/request-2.88.2.tgz", + "integrity": "sha512-MsvtOrfG9ZcrOwAW+Qi+F6HbD0CWXEh9ou77uOb7FM2WPhwT7smM833PzanhJLsgXjN89Ir6V2PczXNnMpwKhw==", + "deprecated": "request has been deprecated, see https://github.com/request/request/issues/3142", + "dev": true, + "dependencies": { + "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" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/request-promise-core": { + "version": "1.1.4", + "resolved": "https://registry.npmmirror.com/request-promise-core/-/request-promise-core-1.1.4.tgz", + "integrity": "sha512-TTbAfBBRdWD7aNNOoVOBH4pN/KigV6LyapYNNlAPA8JwbovRti1E88m3sYAwsLi5ryhPKsE9APwnjFTgdUjTpw==", + "dev": true, + "dependencies": { + "lodash": "^4.17.19" + }, + "engines": { + "node": ">=0.10.0" + }, + "peerDependencies": { + "request": "^2.34" + } + }, + "node_modules/request-promise-native": { + "version": "1.0.9", + "resolved": "https://registry.npmmirror.com/request-promise-native/-/request-promise-native-1.0.9.tgz", + "integrity": "sha512-wcW+sIUiWnKgNY0dqCpOZkUbF/I+YPi+f09JZIDa39Ec+q82CpSYniDp+ISgTTbKmnpJWASeJBPZmoxH84wt3g==", + "deprecated": "request-promise-native has been deprecated because it extends the now deprecated request package, see https://github.com/request/request/issues/3142", + "dev": true, + "dependencies": { + "request-promise-core": "1.1.4", + "stealthy-require": "^1.1.1", + "tough-cookie": "^2.3.3" + }, + "engines": { + "node": ">=0.12.0" + }, + "peerDependencies": { + "request": "^2.34" + } + }, + "node_modules/request/node_modules/qs": { + "version": "6.5.3", + "resolved": "https://registry.npmmirror.com/qs/-/qs-6.5.3.tgz", + "integrity": "sha512-qxXIEh4pCGfHICj1mAJQ2/2XVZkjCDTcEgfoSQxc/fYivUZxTkk7L3bDBJSoNrEzXI17oUO5Dp07ktqE5KzczA==", + "dev": true, + "engines": { + "node": ">=0.6" + } + }, + "node_modules/require-directory": { + "version": "2.1.1", + "resolved": "https://registry.npmmirror.com/require-directory/-/require-directory-2.1.1.tgz", + "integrity": "sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/require-main-filename": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/require-main-filename/-/require-main-filename-2.0.0.tgz", + "integrity": "sha512-NKN5kMDylKuldxYLSUfrbo5Tuzh4hd+2E8NPPX02mZtn1VuREQToYe/ZdlJy+J3uCpfaiGF05e7B8W0iXbQHmg==", + "dev": true + }, + "node_modules/require-uncached": { + "version": "1.0.3", + "resolved": "https://registry.npmmirror.com/require-uncached/-/require-uncached-1.0.3.tgz", + "integrity": "sha512-Xct+41K3twrbBHdxAgMoOS+cNcoqIjfM2/VxBF4LL2hVph7YsF8VSKyQ3BDFZwEVbok9yeDl2le/qo0S77WG2w==", + "dev": true, + "optional": true, + "dependencies": { + "caller-path": "^0.1.0", + "resolve-from": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/requires-port": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/requires-port/-/requires-port-1.0.0.tgz", + "integrity": "sha512-KigOCHcocU3XODJxsu8i/j8T9tzT4adHiecwORRQ0ZZFcp7ahwXuRU1m+yuO90C5ZUyGeGfocHDI14M3L3yDAQ==", + "dev": true + }, + "node_modules/reselect": { + "version": "3.0.1", + "resolved": "https://registry.npmmirror.com/reselect/-/reselect-3.0.1.tgz", + "integrity": "sha512-b/6tFZCmRhtBMa4xGqiiRp9jh9Aqi2A687Lo265cN0/QohJQEBPiQ52f4QB6i0eF3yp3hmLL21LSGBcML2dlxA==", + "dev": true + }, + "node_modules/resolve": { + "version": "1.22.0", + "resolved": "https://registry.npmmirror.com/resolve/-/resolve-1.22.0.tgz", + "integrity": "sha512-Hhtrw0nLeSrFQ7phPp4OOcVjLPIeMnRlr5mcnVuMe7M/7eBn98A3hmFRLoFo3DLZkivSYwhRUJTyPyWAk56WLw==", + "dev": true, + "dependencies": { + "is-core-module": "^2.8.1", + "path-parse": "^1.0.7", + "supports-preserve-symlinks-flag": "^1.0.0" + }, + "bin": { + "resolve": "bin/resolve" + } + }, + "node_modules/resolve-cwd": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/resolve-cwd/-/resolve-cwd-2.0.0.tgz", + "integrity": "sha512-ccu8zQTrzVr954472aUVPLEcB3YpKSYR3cg/3lo1okzobPBM+1INXBbBZlDbnI/hbEocnf8j0QVo43hQKrbchg==", + "dev": true, + "dependencies": { + "resolve-from": "^3.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/resolve-cwd/node_modules/resolve-from": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/resolve-from/-/resolve-from-3.0.0.tgz", + "integrity": "sha512-GnlH6vxLymXJNMBo7XP1fJIzBFbdYt49CuTwmB/6N53t+kMPRMFKz783LlQ4tv28XoQfMWinAJX6WCGf2IlaIw==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/resolve-from": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/resolve-from/-/resolve-from-1.0.1.tgz", + "integrity": "sha512-kT10v4dhrlLNcnO084hEjvXCI1wUG9qZLoz2RogxqDQQYy7IxjI/iMUkOtQTNEh6rzHxvdQWHsJyel1pKOVCxg==", + "dev": true, + "optional": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/resolve-url": { + "version": "0.2.1", + "resolved": "https://registry.npmmirror.com/resolve-url/-/resolve-url-0.2.1.tgz", + "integrity": "sha512-ZuF55hVUQaaczgOIwqWzkEcEidmlD/xl44x1UZnhOXcYuFN2S6+rcxpG+C1N3So0wvNI3DmJICUFfu2SxhBmvg==", + "deprecated": "https://github.com/lydell/resolve-url#deprecated" + }, + "node_modules/restore-cursor": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/restore-cursor/-/restore-cursor-2.0.0.tgz", + "integrity": "sha512-6IzJLuGi4+R14vwagDHX+JrXmPVtPpn4mffDJ1UdR7/Edm87fl6yi8mMBIVvFtJaNTUvjughmW4hwLhRG7gC1Q==", + "dev": true, + "dependencies": { + "onetime": "^2.0.0", + "signal-exit": "^3.0.2" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/ret": { + "version": "0.1.15", + "resolved": "https://registry.npmmirror.com/ret/-/ret-0.1.15.tgz", + "integrity": "sha512-TTlYpa+OL+vMMNG24xSlQGEJ3B/RzEfUlLct7b5G/ytav+wPrplCpVMFuwzXbkecJrb6IYo1iFb0S9v37754mg==", + "engines": { + "node": ">=0.12" + } + }, + "node_modules/retry": { + "version": "0.12.0", + "resolved": "https://registry.npmmirror.com/retry/-/retry-0.12.0.tgz", + "integrity": "sha512-9LkiTwjUh6rT555DtE9rTX+BKByPfrMzEAtnlEtdEwr3Nkffwiihqe2bWADg+OQRjt9gl6ICdmB/ZFDCGAtSow==", + "dev": true, + "engines": { + "node": ">= 4" + } + }, + "node_modules/reusify": { + "version": "1.0.4", + "resolved": "https://registry.npmmirror.com/reusify/-/reusify-1.0.4.tgz", + "integrity": "sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==", + "dev": true, + "engines": { + "iojs": ">=1.0.0", + "node": ">=0.10.0" + } + }, + "node_modules/rgb-regex": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/rgb-regex/-/rgb-regex-1.0.1.tgz", + "integrity": "sha512-gDK5mkALDFER2YLqH6imYvK6g02gpNGM4ILDZ472EwWfXZnC2ZEpoB2ECXTyOVUKuk/bPJZMzwQPBYICzP+D3w==", + "dev": true + }, + "node_modules/rgba-regex": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/rgba-regex/-/rgba-regex-1.0.0.tgz", + "integrity": "sha512-zgn5OjNQXLUTdq8m17KdaicF6w89TZs8ZU8y0AYENIU6wG8GG6LLm0yLSiPY8DmaYmHdgRW8rnApjoT0fQRfMg==", + "dev": true + }, + "node_modules/rimraf": { + "version": "2.7.1", + "resolved": "https://registry.npmmirror.com/rimraf/-/rimraf-2.7.1.tgz", + "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", + "dependencies": { + "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" + } + }, + "node_modules/ripemd160": { + "version": "2.0.2", + "resolved": "https://registry.npmmirror.com/ripemd160/-/ripemd160-2.0.2.tgz", + "integrity": "sha512-ii4iagi25WusVoiC4B4lq7pbXfAp3D9v5CwfkY33vffw2+pkDjY1D8GaN7spsxvCSx8dkPqOZCEZyfxcmJG2IA==", + "dependencies": { + "hash-base": "^3.0.0", + "inherits": "^2.0.1" + } + }, + "node_modules/run-async": { + "version": "2.4.1", + "resolved": "https://registry.npmmirror.com/run-async/-/run-async-2.4.1.tgz", + "integrity": "sha512-tvVnVv01b8c1RrA6Ep7JkStj85Guv/YrMcwqYQnwjsAS2cTmmPGBBjAjpCW7RrSodNSoE2/qg9O4bceNvUuDgQ==", + "dev": true, + "engines": { + "node": ">=0.12.0" + } + }, + "node_modules/run-node": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/run-node/-/run-node-1.0.0.tgz", + "integrity": "sha512-kc120TBlQ3mih1LSzdAJXo4xn/GWS2ec0l3S+syHDXP9uRr0JAT8Qd3mdMuyjqCzeZktgP3try92cEgf9Nks8A==", + "dev": true, + "bin": { + "run-node": "run-node" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/run-parallel": { + "version": "1.2.0", + "resolved": "https://registry.npmmirror.com/run-parallel/-/run-parallel-1.2.0.tgz", + "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==", + "dev": true, + "dependencies": { + "queue-microtask": "^1.2.2" + } + }, + "node_modules/run-queue": { + "version": "1.0.3", + "resolved": "https://registry.npmmirror.com/run-queue/-/run-queue-1.0.3.tgz", + "integrity": "sha512-ntymy489o0/QQplUDnpYAYUsO50K9SBrIVaKCWDOJzYJts0f9WH9RFJkyagebkw5+y1oi00R7ynNW/d12GBumg==", + "dependencies": { + "aproba": "^1.1.1" + } + }, + "node_modules/rx-lite": { + "version": "4.0.8", + "resolved": "https://registry.npmmirror.com/rx-lite/-/rx-lite-4.0.8.tgz", + "integrity": "sha512-Cun9QucwK6MIrp3mry/Y7hqD1oFqTYLQ4pGxaHTjIdaFDWRGGLikqp6u8LcWJnzpoALg9hap+JGk8sFIUuEGNA==", + "dev": true, + "optional": true + }, + "node_modules/rx-lite-aggregates": { + "version": "4.0.8", + "resolved": "https://registry.npmmirror.com/rx-lite-aggregates/-/rx-lite-aggregates-4.0.8.tgz", + "integrity": "sha512-3xPNZGW93oCjiO7PtKxRK6iOVYBWBvtf9QHDfU23Oc+dLIQmAV//UnyXV/yihv81VS/UqoQPk4NegS8EFi55Hg==", + "dev": true, + "optional": true, + "dependencies": { + "rx-lite": "*" + } + }, + "node_modules/rxjs": { + "version": "6.6.7", + "resolved": "https://registry.npmmirror.com/rxjs/-/rxjs-6.6.7.tgz", + "integrity": "sha512-hTdwr+7yYNIT5n4AMYp85KA6yw2Va0FLa3Rguvbpa4W3I5xynaBZo41cM3XM+4Q6fRMj3sBYIR1VAmZMXYJvRQ==", + "dev": true, + "dependencies": { + "tslib": "^1.9.0" + }, + "engines": { + "npm": ">=2.0.0" + } + }, + "node_modules/rxjs/node_modules/tslib": { + "version": "1.14.1", + "resolved": "https://registry.npmmirror.com/tslib/-/tslib-1.14.1.tgz", + "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==", + "dev": true + }, + "node_modules/safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmmirror.com/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" + }, + "node_modules/safe-regex": { + "version": "1.1.0", + "resolved": "https://registry.npmmirror.com/safe-regex/-/safe-regex-1.1.0.tgz", + "integrity": "sha512-aJXcif4xnaNUzvUuC5gcb46oTS7zvg4jpMTnuqtrEPlR3vFr4pxtdTwaF1Qs3Enjn9HK+ZlwQui+a7z0SywIzg==", + "dependencies": { + "ret": "~0.1.10" + } + }, + "node_modules/safer-buffer": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/safer-buffer/-/safer-buffer-2.1.2.tgz", + "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" + }, + "node_modules/sass-graph": { + "version": "2.2.5", + "resolved": "https://registry.npmmirror.com/sass-graph/-/sass-graph-2.2.5.tgz", + "integrity": "sha512-VFWDAHOe6mRuT4mZRd4eKE+d8Uedrk6Xnh7Sh9b4NGufQLQjOrvf/MQoOdx+0s92L89FeyUUNfU597j/3uNpag==", + "dev": true, + "dependencies": { + "glob": "^7.0.0", + "lodash": "^4.0.0", + "scss-tokenizer": "^0.2.3", + "yargs": "^13.3.2" + }, + "bin": { + "sassgraph": "bin/sassgraph" + } + }, + "node_modules/sass-graph/node_modules/emoji-regex": { + "version": "7.0.3", + "resolved": "https://registry.npmmirror.com/emoji-regex/-/emoji-regex-7.0.3.tgz", + "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", + "dev": true + }, + "node_modules/sass-graph/node_modules/string-width": { + "version": "3.1.0", + "resolved": "https://registry.npmmirror.com/string-width/-/string-width-3.1.0.tgz", + "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", + "dev": true, + "dependencies": { + "emoji-regex": "^7.0.1", + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^5.1.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/sass-graph/node_modules/yargs": { + "version": "13.3.2", + "resolved": "https://registry.npmmirror.com/yargs/-/yargs-13.3.2.tgz", + "integrity": "sha512-AX3Zw5iPruN5ie6xGRIDgqkT+ZhnRlZMLMHAs8tg7nRruy2Nb+i5o9bwghAogtM08q1dpr2LVoS8KSTMYpWXUw==", + "dev": true, + "dependencies": { + "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" + } + }, + "node_modules/sass-graph/node_modules/yargs-parser": { + "version": "13.1.2", + "resolved": "https://registry.npmmirror.com/yargs-parser/-/yargs-parser-13.1.2.tgz", + "integrity": "sha512-3lbsNRf/j+A4QuSZfDRA7HRSfWrzO0YjqTJd5kjAq37Zep1CEgaYmrH9Q3GwPiB9cHyd1Y1UwggGhJGoxipbzg==", + "dev": true, + "dependencies": { + "camelcase": "^5.0.0", + "decamelize": "^1.2.0" + } + }, + "node_modules/sass-loader": { + "version": "10.2.1", + "resolved": "https://registry.npmmirror.com/sass-loader/-/sass-loader-10.2.1.tgz", + "integrity": "sha512-RRvWl+3K2LSMezIsd008ErK4rk6CulIMSwrcc2aZvjymUgKo/vjXGp1rSWmfTUX7bblEOz8tst4wBwWtCGBqKA==", + "dev": true, + "dependencies": { + "klona": "^2.0.4", + "loader-utils": "^2.0.0", + "neo-async": "^2.6.2", + "schema-utils": "^3.0.0", + "semver": "^7.3.2" + }, + "engines": { + "node": ">= 10.13.0" + }, + "peerDependencies": { + "fibers": ">= 3.1.0", + "node-sass": "^4.0.0 || ^5.0.0 || ^6.0.0", + "sass": "^1.3.0", + "webpack": "^4.36.0 || ^5.0.0" + }, + "peerDependenciesMeta": { + "fibers": { + "optional": true + }, + "node-sass": { + "optional": true + }, + "sass": { + "optional": true + } + } + }, + "node_modules/sass-loader/node_modules/json5": { + "version": "2.2.1", + "resolved": "https://registry.npmmirror.com/json5/-/json5-2.2.1.tgz", + "integrity": "sha512-1hqLFMSrGHRHxav9q9gNjJ5EXznIxGVO09xQRrwplcS8qs28pZ8s8hupZAmqDwZUmVZ2Qb2jnyPOWcDH8m8dlA==", + "dev": true, + "bin": { + "json5": "lib/cli.js" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/sass-loader/node_modules/loader-utils": { + "version": "2.0.4", + "resolved": "https://registry.npmmirror.com/loader-utils/-/loader-utils-2.0.4.tgz", + "integrity": "sha512-xXqpXoINfFhgua9xiqD8fPFHgkoq1mmmpE92WlDbm9rNRd/EbRb+Gqf908T2DMfuHjjJlksiK2RbHVOdD/MqSw==", + "dev": true, + "dependencies": { + "big.js": "^5.2.2", + "emojis-list": "^3.0.0", + "json5": "^2.1.2" + }, + "engines": { + "node": ">=8.9.0" + } + }, + "node_modules/sass-loader/node_modules/lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/sass-loader/node_modules/schema-utils": { + "version": "3.1.1", + "resolved": "https://registry.npmmirror.com/schema-utils/-/schema-utils-3.1.1.tgz", + "integrity": "sha512-Y5PQxS4ITlC+EahLuXaY86TXfR7Dc5lw294alXOq86JAHCihAIZfqv8nNCWvaEJvaC51uN9hbLGeV0cFBdH+Fw==", + "dev": true, + "dependencies": { + "@types/json-schema": "^7.0.8", + "ajv": "^6.12.5", + "ajv-keywords": "^3.5.2" + }, + "engines": { + "node": ">= 10.13.0" + } + }, + "node_modules/sass-loader/node_modules/semver": { + "version": "7.3.8", + "resolved": "https://registry.npmmirror.com/semver/-/semver-7.3.8.tgz", + "integrity": "sha512-NB1ctGL5rlHrPJtFDVIVzTyQylMLu9N9VICA6HSFJo8MCGVTMW6gfpicwKmmK/dAjTOrqu5l63JJOpDSrAis3A==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/sass-loader/node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/sax": { + "version": "1.2.4", + "resolved": "https://registry.npmmirror.com/sax/-/sax-1.2.4.tgz", + "integrity": "sha512-NqVDv9TpANUjFm0N8uM5GxL36UgKi9/atZw+x7YFnQ8ckwFGKrl4xX4yWtrey3UJm5nP1kUbnYgLopqWNSRhWw==" + }, + "node_modules/schema-utils": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/schema-utils/-/schema-utils-1.0.0.tgz", + "integrity": "sha512-i27Mic4KovM/lnGsy8whRCHhc7VicJajAjTrYg11K9zfZXnYIt4k5F+kZkwjnrhKzLic/HLU4j11mjsz2G/75g==", + "dependencies": { + "ajv": "^6.1.0", + "ajv-errors": "^1.0.0", + "ajv-keywords": "^3.1.0" + }, + "engines": { + "node": ">= 4" + } + }, + "node_modules/scss-tokenizer": { + "version": "0.2.3", + "resolved": "https://registry.npmmirror.com/scss-tokenizer/-/scss-tokenizer-0.2.3.tgz", + "integrity": "sha512-dYE8LhncfBUar6POCxMTm0Ln+erjeczqEvCJib5/7XNkdw1FkUGgwMPY360FY0FgPWQxHWCx29Jl3oejyGLM9Q==", + "dev": true, + "dependencies": { + "js-base64": "^2.1.8", + "source-map": "^0.4.2" + } + }, + "node_modules/scss-tokenizer/node_modules/source-map": { + "version": "0.4.4", + "resolved": "https://registry.npmmirror.com/source-map/-/source-map-0.4.4.tgz", + "integrity": "sha512-Y8nIfcb1s/7DcobUz1yOO1GSp7gyL+D9zLHDehT7iRESqGSxjJ448Sg7rvfgsRJCnKLdSl11uGf0s9X80cH0/A==", + "dev": true, + "dependencies": { + "amdefine": ">=0.0.4" + }, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/select": { + "version": "1.1.2", + "resolved": "https://registry.npmmirror.com/select/-/select-1.1.2.tgz", + "integrity": "sha512-OwpTSOfy6xSs1+pwcNrv0RBMOzI39Lp3qQKUTPVVPRjCdNa5JH/oPRiqsesIskK8TVgmRiHwO4KXlV2Li9dANA==" + }, + "node_modules/select-hose": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/select-hose/-/select-hose-2.0.0.tgz", + "integrity": "sha512-mEugaLK+YfkijB4fx0e6kImuJdCIt2LxCRcbEYPqRGCs4F2ogyfZU5IAZRdjCP8JPq2AtdNoC/Dux63d9Kiryg==", + "dev": true + }, + "node_modules/selfsigned": { + "version": "1.10.14", + "resolved": "https://registry.npmmirror.com/selfsigned/-/selfsigned-1.10.14.tgz", + "integrity": "sha512-lkjaiAye+wBZDCBsu5BGi0XiLRxeUlsGod5ZP924CRSEoGuZAw/f7y9RKu28rwTfiHVhdavhB0qH0INV6P1lEA==", + "dev": true, + "dependencies": { + "node-forge": "^0.10.0" + } + }, + "node_modules/semver": { + "version": "5.7.1", + "resolved": "https://registry.npmmirror.com/semver/-/semver-5.7.1.tgz", + "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", + "bin": { + "semver": "bin/semver" + } + }, + "node_modules/semver-compare": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/semver-compare/-/semver-compare-1.0.0.tgz", + "integrity": "sha512-YM3/ITh2MJ5MtzaM429anh+x2jiLVjqILF4m4oyQB18W7Ggea7BfqdH/wGMK7dDiMghv/6WG7znWMwUDzJiXow==", + "dev": true + }, + "node_modules/send": { + "version": "0.18.0", + "resolved": "https://registry.npmmirror.com/send/-/send-0.18.0.tgz", + "integrity": "sha512-qqWzuOjSFOuqPjFe4NOsMLafToQQwBSOEpS+FwEt3A2V3vKubTquT3vmLTQpFgMXp8AlFWFuP1qKaJZOtPpVXg==", + "dev": true, + "dependencies": { + "debug": "2.6.9", + "depd": "2.0.0", + "destroy": "1.2.0", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "etag": "~1.8.1", + "fresh": "0.5.2", + "http-errors": "2.0.0", + "mime": "1.6.0", + "ms": "2.1.3", + "on-finished": "2.4.1", + "range-parser": "~1.2.1", + "statuses": "2.0.1" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/send/node_modules/mime": { + "version": "1.6.0", + "resolved": "https://registry.npmmirror.com/mime/-/mime-1.6.0.tgz", + "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==", + "dev": true, + "bin": { + "mime": "cli.js" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/send/node_modules/ms": { + "version": "2.1.3", + "resolved": "https://registry.npmmirror.com/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", + "dev": true + }, + "node_modules/serialize-javascript": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/serialize-javascript/-/serialize-javascript-4.0.0.tgz", + "integrity": "sha512-GaNA54380uFefWghODBWEGisLZFj00nS5ACs6yHa9nLqlLpVLO8ChDGeKRjZnV4Nh4n0Qi7nhYZD/9fCPzEqkw==", + "dependencies": { + "randombytes": "^2.1.0" + } + }, + "node_modules/serve-index": { + "version": "1.9.1", + "resolved": "https://registry.npmmirror.com/serve-index/-/serve-index-1.9.1.tgz", + "integrity": "sha512-pXHfKNP4qujrtteMrSBb0rc8HJ9Ms/GrXwcUtUtD5s4ewDJI8bT3Cz2zTVRMKtri49pLx2e0Ya8ziP5Ya2pZZw==", + "dev": true, + "dependencies": { + "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" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/serve-index/node_modules/depd": { + "version": "1.1.2", + "resolved": "https://registry.npmmirror.com/depd/-/depd-1.1.2.tgz", + "integrity": "sha512-7emPTl6Dpo6JRXOXjLRxck+FlLRX5847cLKEn00PLAgc3g2hTZZgr+e4c2v6QpSmLeFP3n5yUo7ft6avBK/5jQ==", + "dev": true, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/serve-index/node_modules/http-errors": { + "version": "1.6.3", + "resolved": "https://registry.npmmirror.com/http-errors/-/http-errors-1.6.3.tgz", + "integrity": "sha512-lks+lVC8dgGyh97jxvxeYTWQFvh4uw4yC12gVl63Cg30sjPX4wuGcdkICVXDAESr6OJGjqGA8Iz5mkeN6zlD7A==", + "dev": true, + "dependencies": { + "depd": "~1.1.2", + "inherits": "2.0.3", + "setprototypeof": "1.1.0", + "statuses": ">= 1.4.0 < 2" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/serve-index/node_modules/inherits": { + "version": "2.0.3", + "resolved": "https://registry.npmmirror.com/inherits/-/inherits-2.0.3.tgz", + "integrity": "sha512-x00IRNXNy63jwGkJmzPigoySHbaqpNuzKbBOmzK+g2OdZpQ9w+sxCN+VSB3ja7IAge2OP2qpfxTjeNcyjmW1uw==", + "dev": true + }, + "node_modules/serve-index/node_modules/setprototypeof": { + "version": "1.1.0", + "resolved": "https://registry.npmmirror.com/setprototypeof/-/setprototypeof-1.1.0.tgz", + "integrity": "sha512-BvE/TwpZX4FXExxOxZyRGQQv651MSwmWKZGqvmPcRIjDqWub67kTKuIMx43cZZrS/cBBzwBcNDWoFxt2XEFIpQ==", + "dev": true + }, + "node_modules/serve-index/node_modules/statuses": { + "version": "1.5.0", + "resolved": "https://registry.npmmirror.com/statuses/-/statuses-1.5.0.tgz", + "integrity": "sha512-OpZ3zP+jT1PI7I8nemJX4AKmAX070ZkYPVWV/AaKTJl+tXCTGyVdC1a4SL8RUQYEwk/f34ZX8UTykN68FwrqAA==", + "dev": true, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/serve-static": { + "version": "1.15.0", + "resolved": "https://registry.npmmirror.com/serve-static/-/serve-static-1.15.0.tgz", + "integrity": "sha512-XGuRDNjXUijsUL0vl6nSD7cwURuzEgglbOaFuZM9g3kwDXOWVTck0jLzjPzGD+TazWbboZYu52/9/XPdUgne9g==", + "dev": true, + "dependencies": { + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "parseurl": "~1.3.3", + "send": "0.18.0" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/set-blocking": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/set-blocking/-/set-blocking-2.0.0.tgz", + "integrity": "sha512-KiKBS8AnWGEyLzofFfmvKwpdPzqiy16LvQfK3yv/fVH7Bj13/wl3JSR1J+rfgRE9q7xUJK4qvgS8raSOeLUehw==", + "dev": true + }, + "node_modules/set-value": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/set-value/-/set-value-2.0.1.tgz", + "integrity": "sha512-JxHc1weCN68wRY0fhCoXpyK55m/XPHafOmK4UWD7m2CI14GMcFypt4w/0+NV5f/ZMby2F6S2wwA7fgynh9gWSw==", + "dependencies": { + "extend-shallow": "^2.0.1", + "is-extendable": "^0.1.1", + "is-plain-object": "^2.0.3", + "split-string": "^3.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/set-value/node_modules/extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha512-zCnTtlxNoAiDc3gqY2aYAWFx7XWWiasuF2K8Me5WbN8otHKTUKBwjPtNpRs/rbUZm7KxWAaNj7P1a/p52GbVug==", + "dependencies": { + "is-extendable": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/setimmediate": { + "version": "1.0.5", + "resolved": "https://registry.npmmirror.com/setimmediate/-/setimmediate-1.0.5.tgz", + "integrity": "sha512-MATJdZp8sLqDl/68LfQmbP8zKPLQNV6BIZoIgrscFDQ+RsvK/BxeDQOgyxKKoh0y/8h3BqVFnCqQ/gd+reiIXA==" + }, + "node_modules/setprototypeof": { + "version": "1.2.0", + "resolved": "https://registry.npmmirror.com/setprototypeof/-/setprototypeof-1.2.0.tgz", + "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==", + "dev": true + }, + "node_modules/sha.js": { + "version": "2.4.11", + "resolved": "https://registry.npmmirror.com/sha.js/-/sha.js-2.4.11.tgz", + "integrity": "sha512-QMEp5B7cftE7APOjk5Y6xgrbWu+WkLVQwk8JNjZ8nKRciZaByEW6MubieAiToS7+dwvrjGhH8jRXz3MVd0AYqQ==", + "dependencies": { + "inherits": "^2.0.1", + "safe-buffer": "^5.0.1" + }, + "bin": { + "sha.js": "bin.js" + } + }, + "node_modules/shallow-clone": { + "version": "3.0.1", + "resolved": "https://registry.npmmirror.com/shallow-clone/-/shallow-clone-3.0.1.tgz", + "integrity": "sha512-/6KqX+GVUdqPuPPd2LxDDxzX6CAbjJehAAOKlNpqqUpAqPM6HeL8f+o3a+JsyGjn2lv0WY8UsTgUJjU9Ok55NA==", + "dev": true, + "dependencies": { + "kind-of": "^6.0.2" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/shebang-command": { + "version": "1.2.0", + "resolved": "https://registry.npmmirror.com/shebang-command/-/shebang-command-1.2.0.tgz", + "integrity": "sha512-EV3L1+UQWGor21OmnvojK36mhg+TyIKDh3iFBKBohr5xeXIhNBcx8oWdgkTEEQ+BEFFYdLRuqMfd5L84N1V5Vg==", + "dev": true, + "dependencies": { + "shebang-regex": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/shebang-regex": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/shebang-regex/-/shebang-regex-1.0.0.tgz", + "integrity": "sha512-wpoSFAxys6b2a2wHZ1XpDSgD7N9iVjg29Ph9uV/uaP9Ex/KXlkTZTeddxDPSYQpgvzKLGJke2UU0AzoGCjNIvQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/shell-quote": { + "version": "1.7.3", + "resolved": "https://registry.npmmirror.com/shell-quote/-/shell-quote-1.7.3.tgz", + "integrity": "sha512-Vpfqwm4EnqGdlsBFNmHhxhElJYrdfcxPThu+ryKS5J8L/fhAwLazFZtq+S+TWZ9ANj2piSQLGj6NQg+lKPmxrw==", + "dev": true + }, + "node_modules/side-channel": { + "version": "1.0.4", + "resolved": "https://registry.npmmirror.com/side-channel/-/side-channel-1.0.4.tgz", + "integrity": "sha512-q5XPytqFEIKHkGdiMIrY10mvLRvnQh42/+GoBlFW3b2LXLE2xxJpZFdm94we0BaoV3RwJyGqg5wS7epxTv0Zvw==", + "dependencies": { + "call-bind": "^1.0.0", + "get-intrinsic": "^1.0.2", + "object-inspect": "^1.9.0" + } + }, + "node_modules/signal-exit": { + "version": "3.0.7", + "resolved": "https://registry.npmmirror.com/signal-exit/-/signal-exit-3.0.7.tgz", + "integrity": "sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==", + "dev": true + }, + "node_modules/simple-git": { + "version": "1.132.0", + "resolved": "https://registry.npmmirror.com/simple-git/-/simple-git-1.132.0.tgz", + "integrity": "sha512-xauHm1YqCTom1sC9eOjfq3/9RKiUA9iPnxBbrY2DdL8l4ADMu0jjM5l5lphQP5YWNqAL2aXC/OeuQ76vHtW5fg==", + "dev": true, + "dependencies": { + "debug": "^4.0.1" + } + }, + "node_modules/simple-git/node_modules/debug": { + "version": "4.3.4", + "resolved": "https://registry.npmmirror.com/debug/-/debug-4.3.4.tgz", + "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", + "dev": true, + "dependencies": { + "ms": "2.1.2" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/simple-git/node_modules/ms": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", + "dev": true + }, + "node_modules/simple-swizzle": { + "version": "0.2.2", + "resolved": "https://registry.npmmirror.com/simple-swizzle/-/simple-swizzle-0.2.2.tgz", + "integrity": "sha512-JA//kQgZtbuY83m+xT+tXJkmJncGMTFT+C+g2h2R9uxkYIrE2yy9sgmcLhCnw57/WSD+Eh3J97FPEDFnbXnDUg==", + "dev": true, + "dependencies": { + "is-arrayish": "^0.3.1" + } + }, + "node_modules/simple-swizzle/node_modules/is-arrayish": { + "version": "0.3.2", + "resolved": "https://registry.npmmirror.com/is-arrayish/-/is-arrayish-0.3.2.tgz", + "integrity": "sha512-eVRqCvVlZbuw3GrM63ovNSNAeA1K16kaR/LRY/92w0zxQ5/1YzwblUX652i4Xs9RwAGjW9d9y6X88t8OaAJfWQ==", + "dev": true + }, + "node_modules/slash": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/slash/-/slash-2.0.0.tgz", + "integrity": "sha512-ZYKh3Wh2z1PpEXWr0MpSBZ0V6mZHAQfYevttO11c51CaWjGTaadiKZ+wVt1PbMlDV5qhMFslpZCemhwOK7C89A==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/slice-ansi": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/slice-ansi/-/slice-ansi-1.0.0.tgz", + "integrity": "sha512-POqxBK6Lb3q6s047D/XsDVNPnF9Dl8JSaqe9h9lURl0OdNqy/ujDrOiIHtsqXMGbWWTIomRzAMaTyawAU//Reg==", + "dev": true, + "optional": true, + "dependencies": { + "is-fullwidth-code-point": "^2.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/slide": { + "version": "1.1.6", + "resolved": "https://registry.npmmirror.com/slide/-/slide-1.1.6.tgz", + "integrity": "sha512-NwrtjCg+lZoqhFU8fOwl4ay2ei8PaqCBOUV3/ektPY9trO1yQ1oXEfmHAhKArUVUr/hOHvy5f6AdP17dCM0zMw==", + "dev": true, + "engines": { + "node": "*" + } + }, + "node_modules/smart-buffer": { + "version": "4.2.0", + "resolved": "https://registry.npmmirror.com/smart-buffer/-/smart-buffer-4.2.0.tgz", + "integrity": "sha512-94hK0Hh8rPqQl2xXc3HsaBoOXKV20MToPkcXvwbISWLEs+64sBq5kFgn2kJDHb1Pry9yrP0dxrCI9RRci7RXKg==", + "dev": true, + "engines": { + "node": ">= 6.0.0", + "npm": ">= 3.0.0" + } + }, + "node_modules/snapdragon": { + "version": "0.8.2", + "resolved": "https://registry.npmmirror.com/snapdragon/-/snapdragon-0.8.2.tgz", + "integrity": "sha512-FtyOnWN/wCHTVXOMwvSv26d+ko5vWlIDD6zoUJ7LW8vh+ZBC8QdljveRP+crNrtBwioEUWy/4dMtbBjA4ioNlg==", + "dependencies": { + "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" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/snapdragon-node": { + "version": "2.1.1", + "resolved": "https://registry.npmmirror.com/snapdragon-node/-/snapdragon-node-2.1.1.tgz", + "integrity": "sha512-O27l4xaMYt/RSQ5TR3vpWCAB5Kb/czIcqUFOM/C4fYcLnbZUc1PkjTAMjof2pBWaSTwOUd6qUHcFGVGj7aIwnw==", + "dependencies": { + "define-property": "^1.0.0", + "isobject": "^3.0.0", + "snapdragon-util": "^3.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/snapdragon-node/node_modules/define-property": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/define-property/-/define-property-1.0.0.tgz", + "integrity": "sha512-cZTYKFWspt9jZsMscWo8sc/5lbPC9Q0N5nBLgb+Yd915iL3udB1uFgS3B8YCx66UVHq018DAVFoee7x+gxggeA==", + "dependencies": { + "is-descriptor": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/snapdragon-node/node_modules/is-accessor-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", + "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", + "dependencies": { + "kind-of": "^6.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/snapdragon-node/node_modules/is-data-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", + "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", + "dependencies": { + "kind-of": "^6.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/snapdragon-node/node_modules/is-descriptor": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/is-descriptor/-/is-descriptor-1.0.2.tgz", + "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", + "dependencies": { + "is-accessor-descriptor": "^1.0.0", + "is-data-descriptor": "^1.0.0", + "kind-of": "^6.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/snapdragon-util": { + "version": "3.0.1", + "resolved": "https://registry.npmmirror.com/snapdragon-util/-/snapdragon-util-3.0.1.tgz", + "integrity": "sha512-mbKkMdQKsjX4BAL4bRYTj21edOf8cN7XHdYUJEe+Zn99hVEYcMvKPct1IqNe7+AZPirn8BCDOQBHQZknqmKlZQ==", + "dependencies": { + "kind-of": "^3.2.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/snapdragon-util/node_modules/kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmmirror.com/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha512-NOW9QQXMoZGg/oqnVNoNTTIFEIid1627WCffUBJEdMxYApq7mNE7CpzucIPc+ZQg25Phej7IJSmX3hO+oblOtQ==", + "dependencies": { + "is-buffer": "^1.1.5" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/snapdragon/node_modules/define-property": { + "version": "0.2.5", + "resolved": "https://registry.npmmirror.com/define-property/-/define-property-0.2.5.tgz", + "integrity": "sha512-Rr7ADjQZenceVOAKop6ALkkRAmH1A4Gx9hV/7ZujPUN2rkATqFO0JZLZInbAjpZYoJ1gUx8MRMQVkYemcbMSTA==", + "dependencies": { + "is-descriptor": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/snapdragon/node_modules/extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha512-zCnTtlxNoAiDc3gqY2aYAWFx7XWWiasuF2K8Me5WbN8otHKTUKBwjPtNpRs/rbUZm7KxWAaNj7P1a/p52GbVug==", + "dependencies": { + "is-extendable": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/sockjs": { + "version": "0.3.24", + "resolved": "https://registry.npmmirror.com/sockjs/-/sockjs-0.3.24.tgz", + "integrity": "sha512-GJgLTZ7vYb/JtPSSZ10hsOYIvEYsjbNU+zPdIHcUaWVNUEPivzxku31865sSSud0Da0W4lEeOPlmw93zLQchuQ==", + "dev": true, + "dependencies": { + "faye-websocket": "^0.11.3", + "uuid": "^8.3.2", + "websocket-driver": "^0.7.4" + } + }, + "node_modules/sockjs-client": { + "version": "1.6.0", + "resolved": "https://registry.npmmirror.com/sockjs-client/-/sockjs-client-1.6.0.tgz", + "integrity": "sha512-qVHJlyfdHFht3eBFZdKEXKTlb7I4IV41xnVNo8yUKA1UHcPJwgW2SvTq9LhnjjCywSkSK7c/e4nghU0GOoMCRQ==", + "dev": true, + "dependencies": { + "debug": "^3.2.7", + "eventsource": "^1.1.0", + "faye-websocket": "^0.11.4", + "inherits": "^2.0.4", + "url-parse": "^1.5.10" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/sockjs-client/node_modules/debug": { + "version": "3.2.7", + "resolved": "https://registry.npmmirror.com/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dev": true, + "dependencies": { + "ms": "^2.1.1" + } + }, + "node_modules/sockjs-client/node_modules/ms": { + "version": "2.1.3", + "resolved": "https://registry.npmmirror.com/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", + "dev": true + }, + "node_modules/sockjs/node_modules/uuid": { + "version": "8.3.2", + "resolved": "https://registry.npmmirror.com/uuid/-/uuid-8.3.2.tgz", + "integrity": "sha512-+NYs2QeMWy+GWFOEm9xnn6HCDp0l7QBD7ml8zLUmJ+93Q5NF0NocErnwkTkXVFNiX3/fpC6afS8Dhb/gz7R7eg==", + "dev": true, + "bin": { + "uuid": "dist/bin/uuid" + } + }, + "node_modules/socks": { + "version": "2.6.2", + "resolved": "https://registry.npmmirror.com/socks/-/socks-2.6.2.tgz", + "integrity": "sha512-zDZhHhZRY9PxRruRMR7kMhnf3I8hDs4S3f9RecfnGxvcBHQcKcIH/oUcEWffsfl1XxdYlA7nnlGbbTvPz9D8gA==", + "dev": true, + "dependencies": { + "ip": "^1.1.5", + "smart-buffer": "^4.2.0" + }, + "engines": { + "node": ">= 10.13.0", + "npm": ">= 3.0.0" + } + }, + "node_modules/socks-proxy-agent": { + "version": "6.2.0", + "resolved": "https://registry.npmmirror.com/socks-proxy-agent/-/socks-proxy-agent-6.2.0.tgz", + "integrity": "sha512-wWqJhjb32Q6GsrUqzuFkukxb/zzide5quXYcMVpIjxalDBBYy2nqKCFQ/9+Ie4dvOYSQdOk3hUlZSdzZOd3zMQ==", + "dev": true, + "dependencies": { + "agent-base": "^6.0.2", + "debug": "^4.3.3", + "socks": "^2.6.2" + }, + "engines": { + "node": ">= 10" + } + }, + "node_modules/socks-proxy-agent/node_modules/debug": { + "version": "4.3.4", + "resolved": "https://registry.npmmirror.com/debug/-/debug-4.3.4.tgz", + "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", + "dev": true, + "dependencies": { + "ms": "2.1.2" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/socks-proxy-agent/node_modules/ms": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", + "dev": true + }, + "node_modules/sort-keys": { + "version": "1.1.2", + "resolved": "https://registry.npmmirror.com/sort-keys/-/sort-keys-1.1.2.tgz", + "integrity": "sha512-vzn8aSqKgytVik0iwdBEi+zevbTYZogewTUM6dtpmGwEcdzbub/TX4bCzRhebDCRC3QzXgJsLRKB2V/Oof7HXg==", + "dev": true, + "dependencies": { + "is-plain-obj": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/sortablejs": { + "version": "1.10.2", + "resolved": "https://registry.npmmirror.com/sortablejs/-/sortablejs-1.10.2.tgz", + "integrity": "sha512-YkPGufevysvfwn5rfdlGyrGjt7/CRHwvRPogD/lC+TnvcN29jDpCifKP+rBqf+LRldfXSTh+0CGLcSg0VIxq3A==" + }, + "node_modules/source-list-map": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/source-list-map/-/source-list-map-2.0.1.tgz", + "integrity": "sha512-qnQ7gVMxGNxsiL4lEuJwe/To8UnK7fAnmbGEEH8RpLouuKbeEm0lhbQVFIrNSuB+G7tVrAlVsZgETT5nljf+Iw==" + }, + "node_modules/source-map": { + "version": "0.5.7", + "resolved": "https://registry.npmmirror.com/source-map/-/source-map-0.5.7.tgz", + "integrity": "sha512-LbrmJOMUSdEVxIKvdcJzQC+nQhe8FUZQTXQy6+I75skNgn3OoQ0DZA8YnFa7gp8tqtL3KPf1kmo0R5DoApeSGQ==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/source-map-resolve": { + "version": "0.5.3", + "resolved": "https://registry.npmmirror.com/source-map-resolve/-/source-map-resolve-0.5.3.tgz", + "integrity": "sha512-Htz+RnsXWk5+P2slx5Jh3Q66vhQj1Cllm0zvnaY98+NFx+Dv2CF/f5O/t8x+KaNdrdIAsruNzoh/KpialbqAnw==", + "deprecated": "See https://github.com/lydell/source-map-resolve#deprecated", + "dependencies": { + "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" + } + }, + "node_modules/source-map-support": { + "version": "0.5.21", + "resolved": "https://registry.npmmirror.com/source-map-support/-/source-map-support-0.5.21.tgz", + "integrity": "sha512-uBHU3L3czsIyYXKX88fdrGovxdSCoTGDRZ6SYXtSRxLZUzHg5P/66Ht6uoUlHu9EZod+inXhKo3qQgwXUT/y1w==", + "dependencies": { + "buffer-from": "^1.0.0", + "source-map": "^0.6.0" + } + }, + "node_modules/source-map-support/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmmirror.com/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/source-map-url": { + "version": "0.4.1", + "resolved": "https://registry.npmmirror.com/source-map-url/-/source-map-url-0.4.1.tgz", + "integrity": "sha512-cPiFOTLUKvJFIg4SKVScy4ilPPW6rFgMgfuZJPNoDuMs3nC1HbMUycBoJw77xFIp6z1UJQJOfx6C9GMH80DiTw==", + "deprecated": "See https://github.com/lydell/source-map-url#deprecated" + }, + "node_modules/spdx-correct": { + "version": "3.1.1", + "resolved": "https://registry.npmmirror.com/spdx-correct/-/spdx-correct-3.1.1.tgz", + "integrity": "sha512-cOYcUWwhCuHCXi49RhFRCyJEK3iPj1Ziz9DpViV3tbZOwXD49QzIN3MpOLJNxh2qwq2lJJZaKMVw9qNi4jTC0w==", + "dev": true, + "dependencies": { + "spdx-expression-parse": "^3.0.0", + "spdx-license-ids": "^3.0.0" + } + }, + "node_modules/spdx-exceptions": { + "version": "2.3.0", + "resolved": "https://registry.npmmirror.com/spdx-exceptions/-/spdx-exceptions-2.3.0.tgz", + "integrity": "sha512-/tTrYOC7PPI1nUAgx34hUpqXuyJG+DTHJTnIULG4rDygi4xu/tfgmq1e1cIRwRzwZgo4NLySi+ricLkZkw4i5A==", + "dev": true + }, + "node_modules/spdx-expression-parse": { + "version": "3.0.1", + "resolved": "https://registry.npmmirror.com/spdx-expression-parse/-/spdx-expression-parse-3.0.1.tgz", + "integrity": "sha512-cbqHunsQWnJNE6KhVSMsMeH5H/L9EpymbzqTQ3uLwNCLZ1Q481oWaofqH7nO6V07xlXwY6PhQdQ2IedWx/ZK4Q==", + "dev": true, + "dependencies": { + "spdx-exceptions": "^2.1.0", + "spdx-license-ids": "^3.0.0" + } + }, + "node_modules/spdx-license-ids": { + "version": "3.0.11", + "resolved": "https://registry.npmmirror.com/spdx-license-ids/-/spdx-license-ids-3.0.11.tgz", + "integrity": "sha512-Ctl2BrFiM0X3MANYgj3CkygxhRmr9mi6xhejbdO960nF6EDJApTYpn0BQnDKlnNBULKiCN1n3w9EBkHK8ZWg+g==", + "dev": true + }, + "node_modules/spdy": { + "version": "4.0.2", + "resolved": "https://registry.npmmirror.com/spdy/-/spdy-4.0.2.tgz", + "integrity": "sha512-r46gZQZQV+Kl9oItvl1JZZqJKGr+oEkB08A6BzkiR7593/7IbtuncXHd2YoYeTsG4157ZssMu9KYvUHLcjcDoA==", + "dev": true, + "dependencies": { + "debug": "^4.1.0", + "handle-thing": "^2.0.0", + "http-deceiver": "^1.2.7", + "select-hose": "^2.0.0", + "spdy-transport": "^3.0.0" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/spdy-transport": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/spdy-transport/-/spdy-transport-3.0.0.tgz", + "integrity": "sha512-hsLVFE5SjA6TCisWeJXFKniGGOpBgMLmerfO2aCyCU5s7nJ/rpAepqmFifv/GCbSbueEeAJJnmSQ2rKC/g8Fcw==", + "dev": true, + "dependencies": { + "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" + } + }, + "node_modules/spdy-transport/node_modules/debug": { + "version": "4.3.4", + "resolved": "https://registry.npmmirror.com/debug/-/debug-4.3.4.tgz", + "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", + "dev": true, + "dependencies": { + "ms": "2.1.2" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/spdy-transport/node_modules/ms": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", + "dev": true + }, + "node_modules/spdy-transport/node_modules/readable-stream": { + "version": "3.6.0", + "resolved": "https://registry.npmmirror.com/readable-stream/-/readable-stream-3.6.0.tgz", + "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", + "dev": true, + "dependencies": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/spdy/node_modules/debug": { + "version": "4.3.4", + "resolved": "https://registry.npmmirror.com/debug/-/debug-4.3.4.tgz", + "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", + "dev": true, + "dependencies": { + "ms": "2.1.2" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/spdy/node_modules/ms": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", + "dev": true + }, + "node_modules/speed-measure-webpack-plugin": { + "version": "1.5.0", + "resolved": "https://registry.npmmirror.com/speed-measure-webpack-plugin/-/speed-measure-webpack-plugin-1.5.0.tgz", + "integrity": "sha512-Re0wX5CtM6gW7bZA64ONOfEPEhwbiSF/vz6e2GvadjuaPrQcHTQdRGsD8+BE7iUOysXH8tIenkPCQBEcspXsNg==", + "dev": true, + "dependencies": { + "chalk": "^4.1.0" + }, + "engines": { + "node": ">=6.0.0" + }, + "peerDependencies": { + "webpack": "^1 || ^2 || ^3 || ^4 || ^5" + } + }, + "node_modules/speed-measure-webpack-plugin/node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmmirror.com/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/speed-measure-webpack-plugin/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmmirror.com/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/speed-measure-webpack-plugin/node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/speed-measure-webpack-plugin/node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmmirror.com/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "node_modules/speed-measure-webpack-plugin/node_modules/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/speed-measure-webpack-plugin/node_modules/supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmmirror.com/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/split": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/split/-/split-1.0.1.tgz", + "integrity": "sha512-mTyOoPbrivtXnwnIxZRFYRrPNtEFKlpB2fvjSnCQUiAA6qAZzqwna5envK4uk6OIeP17CsdF3rSBGYVBsU0Tkg==", + "dev": true, + "dependencies": { + "through": "2" + }, + "engines": { + "node": "*" + } + }, + "node_modules/split-on-first": { + "version": "1.1.0", + "resolved": "https://registry.npmmirror.com/split-on-first/-/split-on-first-1.1.0.tgz", + "integrity": "sha512-43ZssAJaMusuKWL8sKUBQXHWOpq8d6CfN/u1p4gUzfJkM05C8rxTmYrkIPTXapZpORA6LkkzcUulJ8FqA7Uudw==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/split-string": { + "version": "3.1.0", + "resolved": "https://registry.npmmirror.com/split-string/-/split-string-3.1.0.tgz", + "integrity": "sha512-NzNVhJDYpwceVVii8/Hu6DKfD2G+NrQHlS/V/qgv763EYudVwEcMQNxd2lh+0VrUByXN/oJkl5grOhYWvQUYiw==", + "dependencies": { + "extend-shallow": "^3.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/split2": { + "version": "3.2.2", + "resolved": "https://registry.npmmirror.com/split2/-/split2-3.2.2.tgz", + "integrity": "sha512-9NThjpgZnifTkJpzTZ7Eue85S49QwpNhZTq6GRJwObb6jnLFNGB7Qm73V5HewTROPyxD0C29xqmaI68bQtV+hg==", + "dev": true, + "dependencies": { + "readable-stream": "^3.0.0" + } + }, + "node_modules/split2/node_modules/readable-stream": { + "version": "3.6.0", + "resolved": "https://registry.npmmirror.com/readable-stream/-/readable-stream-3.6.0.tgz", + "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", + "dev": true, + "dependencies": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/sprintf-js": { + "version": "1.0.3", + "resolved": "https://registry.npmmirror.com/sprintf-js/-/sprintf-js-1.0.3.tgz", + "integrity": "sha512-D9cPgkvLlV3t3IzL0D0YLvGA9Ahk4PcvVwUbN0dSGr1aP0Nrt4AEnTUbuGvquEC0mA64Gqt1fzirlRs5ibXx8g==" + }, + "node_modules/sshpk": { + "version": "1.17.0", + "resolved": "https://registry.npmmirror.com/sshpk/-/sshpk-1.17.0.tgz", + "integrity": "sha512-/9HIEs1ZXGhSPE8X6Ccm7Nam1z8KcoCqPdI7ecm1N33EzAetWahvQWVqLZtaZQ+IDKX4IyA2o0gBzqIMkAagHQ==", + "dev": true, + "dependencies": { + "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" + }, + "bin": { + "sshpk-conv": "bin/sshpk-conv", + "sshpk-sign": "bin/sshpk-sign", + "sshpk-verify": "bin/sshpk-verify" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/ssri": { + "version": "6.0.2", + "resolved": "https://registry.npmmirror.com/ssri/-/ssri-6.0.2.tgz", + "integrity": "sha512-cepbSq/neFK7xB6A50KHN0xHDotYzq58wWCa5LeWqnPrHG8GzfEjO/4O8kpmcGW+oaxkvhEJCWgbgNk4/ZV93Q==", + "dependencies": { + "figgy-pudding": "^3.5.1" + } + }, + "node_modules/stable": { + "version": "0.1.8", + "resolved": "https://registry.npmmirror.com/stable/-/stable-0.1.8.tgz", + "integrity": "sha512-ji9qxRnOVfcuLDySj9qzhGSEFVobyt1kIOSkj1qZzYLzq7Tos/oUUWvotUPQLlrsidqsK6tBH89Bc9kL5zHA6w==", + "deprecated": "Modern JS already guarantees Array#sort() is a stable sort, so this library is deprecated. See the compatibility table on MDN: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/sort#browser_compatibility" + }, + "node_modules/stackframe": { + "version": "1.2.1", + "resolved": "https://registry.npmmirror.com/stackframe/-/stackframe-1.2.1.tgz", + "integrity": "sha512-h88QkzREN/hy8eRdyNhhsO7RSJ5oyTqxxmmn0dzBIMUclZsjpfmrsg81vp8mjjAs2vAZ72nyWxRUwSwmh0e4xg==", + "dev": true + }, + "node_modules/staged-git-files": { + "version": "1.1.2", + "resolved": "https://registry.npmmirror.com/staged-git-files/-/staged-git-files-1.1.2.tgz", + "integrity": "sha512-0Eyrk6uXW6tg9PYkhi/V/J4zHp33aNyi2hOCmhFLqLTIhbgqWn5jlSzI+IU0VqrZq6+DbHcabQl/WP6P3BG0QA==", + "dev": true, + "bin": { + "sgf": "bin/cli.js" + } + }, + "node_modules/static-extend": { + "version": "0.1.2", + "resolved": "https://registry.npmmirror.com/static-extend/-/static-extend-0.1.2.tgz", + "integrity": "sha512-72E9+uLc27Mt718pMHt9VMNiAL4LMsmDbBva8mxWUCkT07fSzEGMYUCk0XWY6lp0j6RBAG4cJ3mWuZv2OE3s0g==", + "dependencies": { + "define-property": "^0.2.5", + "object-copy": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/static-extend/node_modules/define-property": { + "version": "0.2.5", + "resolved": "https://registry.npmmirror.com/define-property/-/define-property-0.2.5.tgz", + "integrity": "sha512-Rr7ADjQZenceVOAKop6ALkkRAmH1A4Gx9hV/7ZujPUN2rkATqFO0JZLZInbAjpZYoJ1gUx8MRMQVkYemcbMSTA==", + "dependencies": { + "is-descriptor": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/statuses": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/statuses/-/statuses-2.0.1.tgz", + "integrity": "sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ==", + "dev": true, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/stdout-stream": { + "version": "1.4.1", + "resolved": "https://registry.npmmirror.com/stdout-stream/-/stdout-stream-1.4.1.tgz", + "integrity": "sha512-j4emi03KXqJWcIeF8eIXkjMFN1Cmb8gUlDYGeBALLPo5qdyTfA9bOtl8m33lRoC+vFMkP3gl0WsDr6+gzxbbTA==", + "dev": true, + "dependencies": { + "readable-stream": "^2.0.1" + } + }, + "node_modules/stealthy-require": { + "version": "1.1.1", + "resolved": "https://registry.npmmirror.com/stealthy-require/-/stealthy-require-1.1.1.tgz", + "integrity": "sha512-ZnWpYnYugiOVEY5GkcuJK1io5V8QmNYChG62gSit9pQVGErXtrKuPC55ITaVSukmMta5qpMU7vqLt2Lnni4f/g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/stream-browserify": { + "version": "2.0.2", + "resolved": "https://registry.npmmirror.com/stream-browserify/-/stream-browserify-2.0.2.tgz", + "integrity": "sha512-nX6hmklHs/gr2FuxYDltq8fJA1GDlxKQCz8O/IM4atRqBH8OORmBNgfvW5gG10GT/qQ9u0CzIvr2X5Pkt6ntqg==", + "dependencies": { + "inherits": "~2.0.1", + "readable-stream": "^2.0.2" + } + }, + "node_modules/stream-each": { + "version": "1.2.3", + "resolved": "https://registry.npmmirror.com/stream-each/-/stream-each-1.2.3.tgz", + "integrity": "sha512-vlMC2f8I2u/bZGqkdfLQW/13Zihpej/7PmSiMQsbYddxuTsJp8vRe2x2FvVExZg7FaOds43ROAuFJwPR4MTZLw==", + "dependencies": { + "end-of-stream": "^1.1.0", + "stream-shift": "^1.0.0" + } + }, + "node_modules/stream-http": { + "version": "2.8.3", + "resolved": "https://registry.npmmirror.com/stream-http/-/stream-http-2.8.3.tgz", + "integrity": "sha512-+TSkfINHDo4J+ZobQLWiMouQYB+UVYFttRA94FpEzzJ7ZdqcL4uUUQ7WkdkI4DSozGmgBUE/a47L+38PenXhUw==", + "dependencies": { + "builtin-status-codes": "^3.0.0", + "inherits": "^2.0.1", + "readable-stream": "^2.3.6", + "to-arraybuffer": "^1.0.0", + "xtend": "^4.0.0" + } + }, + "node_modules/stream-shift": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/stream-shift/-/stream-shift-1.0.1.tgz", + "integrity": "sha512-AiisoFqQ0vbGcZgQPY1cdP2I76glaVA/RauYR4G4thNFgkTqr90yXTo4LYX60Jl+sIlPNHHdGSwo01AvbKUSVQ==" + }, + "node_modules/strict-uri-encode": { + "version": "1.1.0", + "resolved": "https://registry.npmmirror.com/strict-uri-encode/-/strict-uri-encode-1.1.0.tgz", + "integrity": "sha512-R3f198pcvnB+5IpnBlRkphuE9n46WyVl8I39W/ZUTZLz4nqSP/oLYUrcnJrw462Ds8he4YKMov2efsTIw1BDGQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/string_decoder": { + "version": "1.1.1", + "resolved": "https://registry.npmmirror.com/string_decoder/-/string_decoder-1.1.1.tgz", + "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", + "dependencies": { + "safe-buffer": "~5.1.0" + } + }, + "node_modules/string-argv": { + "version": "0.0.2", + "resolved": "https://registry.npmmirror.com/string-argv/-/string-argv-0.0.2.tgz", + "integrity": "sha512-p6/Mqq0utTQWUeGMi/m0uBtlLZEwXSY3+mXzeRRqw7fz5ezUb28Wr0R99NlfbWaMmL/jCyT9be4jpn7Yz8IO8w==", + "dev": true, + "engines": { + "node": ">=0.6.19" + } + }, + "node_modules/string-width": { + "version": "2.1.1", + "resolved": "https://registry.npmmirror.com/string-width/-/string-width-2.1.1.tgz", + "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", + "dev": true, + "dependencies": { + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^4.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/string-width/node_modules/ansi-regex": { + "version": "3.0.1", + "resolved": "https://registry.npmmirror.com/ansi-regex/-/ansi-regex-3.0.1.tgz", + "integrity": "sha512-+O9Jct8wf++lXxxFc4hc8LsjaSq0HFzzL7cVsw8pRDIPdjKD2mT4ytDZlLuSBZ4cLKZFXIrMGO7DbQCtMJJMKw==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/string-width/node_modules/strip-ansi": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/strip-ansi/-/strip-ansi-4.0.0.tgz", + "integrity": "sha512-4XaJ2zQdCzROZDivEVIDPkcQn8LMFSa8kj8Gxb/Lnwzv9A8VctNZ+lfivC/sV3ivW8ElJTERXZoPBRrZKkNKow==", + "dev": true, + "dependencies": { + "ansi-regex": "^3.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/string.prototype.padend": { + "version": "3.1.3", + "resolved": "https://registry.npmmirror.com/string.prototype.padend/-/string.prototype.padend-3.1.3.tgz", + "integrity": "sha512-jNIIeokznm8SD/TZISQsZKYu7RJyheFNt84DUPrh482GC8RVp2MKqm2O5oBRdGxbDQoXrhhWtPIWQOiy20svUg==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.3", + "es-abstract": "^1.19.1" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/string.prototype.padstart": { + "version": "3.1.3", + "resolved": "https://registry.npmmirror.com/string.prototype.padstart/-/string.prototype.padstart-3.1.3.tgz", + "integrity": "sha512-NZydyOMtYxpTjGqp0VN5PYUF/tsU15yDMZnUdj16qRUIUiMJkHHSDElYyQFrMu+/WloTpA7MQSiADhBicDfaoA==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.3", + "es-abstract": "^1.19.1" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/string.prototype.trimend": { + "version": "1.0.4", + "resolved": "https://registry.npmmirror.com/string.prototype.trimend/-/string.prototype.trimend-1.0.4.tgz", + "integrity": "sha512-y9xCjw1P23Awk8EvTpcyL2NIr1j7wJ39f+k6lvRnSMz+mz9CGz9NYPelDk42kOz6+ql8xjfK8oYzy3jAP5QU5A==", + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.3" + } + }, + "node_modules/string.prototype.trimstart": { + "version": "1.0.4", + "resolved": "https://registry.npmmirror.com/string.prototype.trimstart/-/string.prototype.trimstart-1.0.4.tgz", + "integrity": "sha512-jh6e984OBfvxS50tdY2nRZnoC5/mLFKOREQfw8t5yytkoUsJRNxvI/E39qu1sD0OtWI3OC0XgKSmcWwziwYuZw==", + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.3" + } + }, + "node_modules/stringify-object": { + "version": "3.3.0", + "resolved": "https://registry.npmmirror.com/stringify-object/-/stringify-object-3.3.0.tgz", + "integrity": "sha512-rHqiFh1elqCQ9WPLIC8I0Q/g/wj5J1eMkyoiD6eoQApWHP0FtlK7rqnhmabL5VUY9JQCcqwwvlOaSuutekgyrw==", + "dev": true, + "dependencies": { + "get-own-enumerable-property-symbols": "^3.0.0", + "is-obj": "^1.0.1", + "is-regexp": "^1.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/stringify-object/node_modules/is-obj": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/is-obj/-/is-obj-1.0.1.tgz", + "integrity": "sha512-l4RyHgRqGN4Y3+9JHVrNqO+tN0rV5My76uW5/nuO4K1b6vw5G8d/cmFjP9tRfEsdhZNt0IFdZuK/c2Vr4Nb+Qg==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmmirror.com/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "dev": true, + "dependencies": { + "ansi-regex": "^4.1.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/strip-bom": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/strip-bom/-/strip-bom-3.0.0.tgz", + "integrity": "sha512-vavAMRXOgBVNF6nyEEmL3DBK19iRpDcoIwW+swQ+CbGiu7lju6t+JklA1MHweoWtadgt4ISVUsXLyDq34ddcwA==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/strip-eof": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/strip-eof/-/strip-eof-1.0.0.tgz", + "integrity": "sha512-7FCwGGmx8mD5xQd3RPUvnSpUXHM3BWuzjtpD4TXsfcZ9EL4azvVVUscFYwD9nx8Kh+uCBC00XBtAykoMHwTh8Q==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/strip-final-newline": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/strip-final-newline/-/strip-final-newline-2.0.0.tgz", + "integrity": "sha512-BrpvfNAE3dcvq7ll3xVumzjKjZQ5tI1sEUIKr3Uoks0XUl45St3FlatVqef9prk4jRDzhW6WZg+3bk93y6pLjA==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/strip-indent": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/strip-indent/-/strip-indent-2.0.0.tgz", + "integrity": "sha512-RsSNPLpq6YUL7QYy44RnPVTn/lcVZtb48Uof3X5JLbF4zD/Gs7ZFDv2HWol+leoQN2mT86LAzSshGfkTlSOpsA==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/strip-json-comments": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/strip-json-comments/-/strip-json-comments-2.0.1.tgz", + "integrity": "sha512-4gB8na07fecVVkOI6Rs4e7T6NOTki5EmL7TUduTs6bu3EdnSycntVJ4re8kgZA+wx9IueI2Y11bfbgwtzuE0KQ==", + "dev": true, + "optional": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/strong-log-transformer": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/strong-log-transformer/-/strong-log-transformer-2.1.0.tgz", + "integrity": "sha512-B3Hgul+z0L9a236FAUC9iZsL+nVHgoCJnqCbN588DjYxvGXaXaaFbfmQ/JhvKjZwsOukuR72XbHv71Qkug0HxA==", + "dev": true, + "dependencies": { + "duplexer": "^0.1.1", + "minimist": "^1.2.0", + "through": "^2.3.4" + }, + "bin": { + "sl-log-transformer": "bin/sl-log-transformer.js" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/style-mod": { + "version": "4.0.3", + "resolved": "https://registry.npmmirror.com/style-mod/-/style-mod-4.0.3.tgz", + "integrity": "sha512-78Jv8kYJdjbvRwwijtCevYADfsI0lGzYJe4mMFdceO8l75DFFDoqBhR1jVDicDRRaX4//g1u9wKeo+ztc2h1Rw==" + }, + "node_modules/stylehacks": { + "version": "4.0.3", + "resolved": "https://registry.npmmirror.com/stylehacks/-/stylehacks-4.0.3.tgz", + "integrity": "sha512-7GlLk9JwlElY4Y6a/rmbH2MhVlTyVmiJd1PfTCqFaIBEGMYNsrO/v3SeGTdhBThLg4Z+NbOk/qFMwCa+J+3p/g==", + "dev": true, + "dependencies": { + "browserslist": "^4.0.0", + "postcss": "^7.0.0", + "postcss-selector-parser": "^3.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/stylehacks/node_modules/postcss-selector-parser": { + "version": "3.1.2", + "resolved": "https://registry.npmmirror.com/postcss-selector-parser/-/postcss-selector-parser-3.1.2.tgz", + "integrity": "sha512-h7fJ/5uWuRVyOtkO45pnt1Ih40CEleeyCHzipqAZO2e5H20g25Y48uYnFUiShvY4rZWNJ/Bib/KVPmanaCtOhA==", + "dev": true, + "dependencies": { + "dot-prop": "^5.2.0", + "indexes-of": "^1.0.1", + "uniq": "^1.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/subarg": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/subarg/-/subarg-1.0.0.tgz", + "integrity": "sha512-RIrIdRY0X1xojthNcVtgT9sjpOGagEUKpZdgBUi054OEPFo282yg+zE+t1Rj3+RqKq2xStL7uUHhY+AjbC4BXg==", + "dev": true, + "dependencies": { + "minimist": "^1.1.0" + } + }, + "node_modules/supports-color": { + "version": "5.5.0", + "resolved": "https://registry.npmmirror.com/supports-color/-/supports-color-5.5.0.tgz", + "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", + "dependencies": { + "has-flag": "^3.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/supports-preserve-symlinks-flag": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz", + "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==", + "dev": true, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/svg-baker": { + "version": "1.7.0", + "resolved": "https://registry.npmmirror.com/svg-baker/-/svg-baker-1.7.0.tgz", + "integrity": "sha512-nibslMbkXOIkqKVrfcncwha45f97fGuAOn1G99YwnwTj8kF9YiM6XexPcUso97NxOm6GsP0SIvYVIosBis1xLg==", + "dev": true, + "dependencies": { + "bluebird": "^3.5.0", + "clone": "^2.1.1", + "he": "^1.1.1", + "image-size": "^0.5.1", + "loader-utils": "^1.1.0", + "merge-options": "1.0.1", + "micromatch": "3.1.0", + "postcss": "^5.2.17", + "postcss-prefix-selector": "^1.6.0", + "posthtml-rename-id": "^1.0", + "posthtml-svg-mode": "^1.0.3", + "query-string": "^4.3.2", + "traverse": "^0.6.6" + } + }, + "node_modules/svg-baker-runtime": { + "version": "1.4.7", + "resolved": "https://registry.npmmirror.com/svg-baker-runtime/-/svg-baker-runtime-1.4.7.tgz", + "integrity": "sha512-Zorfwwj5+lWjk/oxwSMsRdS2sPQQdTmmsvaSpzU+i9ZWi3zugHLt6VckWfnswphQP0LmOel3nggpF5nETbt6xw==", + "dev": true, + "dependencies": { + "deepmerge": "1.3.2", + "mitt": "1.1.2", + "svg-baker": "^1.7.0" + } + }, + "node_modules/svg-baker-runtime/node_modules/deepmerge": { + "version": "1.3.2", + "resolved": "https://registry.npmmirror.com/deepmerge/-/deepmerge-1.3.2.tgz", + "integrity": "sha512-qjMjTrk+RKv/sp4RPDpV5CnKhxjFI9p+GkLBOls5A8EEElldYWCWA9zceAkmfd0xIo2aU1nxiaLFoiya2sb6Cg==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/svg-baker/node_modules/ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmmirror.com/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha512-TIGnTpdo+E3+pCyAluZvtED5p5wCqLdezCyhPZzKPcxvFplEt4i+W7OONCKgeZFT3+y5NZZfOOS/Bdcanm1MYA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/svg-baker/node_modules/ansi-styles": { + "version": "2.2.1", + "resolved": "https://registry.npmmirror.com/ansi-styles/-/ansi-styles-2.2.1.tgz", + "integrity": "sha512-kmCevFghRiWM7HB5zTPULl4r9bVFSWjz62MhqizDGUrq2NWuNMQyuv4tHHoKJHs69M/MF64lEcHdYIocrdWQYA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/svg-baker/node_modules/chalk": { + "version": "1.1.3", + "resolved": "https://registry.npmmirror.com/chalk/-/chalk-1.1.3.tgz", + "integrity": "sha512-U3lRVLMSlsCfjqYPbLyVv11M9CPW4I728d6TCKMAOJueEeB9/8o+eSsMnxPJD+Q+K909sdESg7C+tIkoH6on1A==", + "dev": true, + "dependencies": { + "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" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/svg-baker/node_modules/chalk/node_modules/supports-color": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/supports-color/-/supports-color-2.0.0.tgz", + "integrity": "sha512-KKNVtd6pCYgPIKU4cp2733HWYCpplQhddZLBUryaAHou723x+FRzQ5Df824Fj+IyyuiQTRoub4SnIFfIcrp70g==", + "dev": true, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/svg-baker/node_modules/clone": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/clone/-/clone-2.1.2.tgz", + "integrity": "sha512-3Pe/CF1Nn94hyhIYpjtiLhdCoEoz0DqQ+988E9gmeEdQZlojxnOb74wctFyuwWQHzqyf9X7C7MG8juUpqBJT8w==", + "dev": true, + "engines": { + "node": ">=0.8" + } + }, + "node_modules/svg-baker/node_modules/define-property": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/define-property/-/define-property-1.0.0.tgz", + "integrity": "sha512-cZTYKFWspt9jZsMscWo8sc/5lbPC9Q0N5nBLgb+Yd915iL3udB1uFgS3B8YCx66UVHq018DAVFoee7x+gxggeA==", + "dev": true, + "dependencies": { + "is-descriptor": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/svg-baker/node_modules/extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha512-zCnTtlxNoAiDc3gqY2aYAWFx7XWWiasuF2K8Me5WbN8otHKTUKBwjPtNpRs/rbUZm7KxWAaNj7P1a/p52GbVug==", + "dev": true, + "dependencies": { + "is-extendable": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/svg-baker/node_modules/has-flag": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/has-flag/-/has-flag-1.0.0.tgz", + "integrity": "sha512-DyYHfIYwAJmjAjSSPKANxI8bFY9YtFrgkAfinBojQ8YJTOuOuav64tMUJv584SES4xl74PmuaevIyaLESHdTAA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/svg-baker/node_modules/is-accessor-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", + "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", + "dev": true, + "dependencies": { + "kind-of": "^6.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/svg-baker/node_modules/is-accessor-descriptor/node_modules/kind-of": { + "version": "6.0.3", + "resolved": "https://registry.npmmirror.com/kind-of/-/kind-of-6.0.3.tgz", + "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/svg-baker/node_modules/is-data-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", + "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", + "dev": true, + "dependencies": { + "kind-of": "^6.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/svg-baker/node_modules/is-data-descriptor/node_modules/kind-of": { + "version": "6.0.3", + "resolved": "https://registry.npmmirror.com/kind-of/-/kind-of-6.0.3.tgz", + "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/svg-baker/node_modules/is-descriptor": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/is-descriptor/-/is-descriptor-1.0.2.tgz", + "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", + "dev": true, + "dependencies": { + "is-accessor-descriptor": "^1.0.0", + "is-data-descriptor": "^1.0.0", + "kind-of": "^6.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/svg-baker/node_modules/is-descriptor/node_modules/kind-of": { + "version": "6.0.3", + "resolved": "https://registry.npmmirror.com/kind-of/-/kind-of-6.0.3.tgz", + "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/svg-baker/node_modules/kind-of": { + "version": "5.1.0", + "resolved": "https://registry.npmmirror.com/kind-of/-/kind-of-5.1.0.tgz", + "integrity": "sha512-NGEErnH6F2vUuXDh+OlbcKW7/wOcfdRHaZ7VWtqCztfHri/++YKmP51OdWeGPuqCOba6kk2OTe5d02VmTB80Pw==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/svg-baker/node_modules/micromatch": { + "version": "3.1.0", + "resolved": "https://registry.npmmirror.com/micromatch/-/micromatch-3.1.0.tgz", + "integrity": "sha512-3StSelAE+hnRvMs8IdVW7Uhk8CVed5tp+kLLGlBP6WiRAXS21GPGu/Nat4WNPXj2Eoc24B02SaeoyozPMfj0/g==", + "dev": true, + "dependencies": { + "arr-diff": "^4.0.0", + "array-unique": "^0.3.2", + "braces": "^2.2.2", + "define-property": "^1.0.0", + "extend-shallow": "^2.0.1", + "extglob": "^2.0.2", + "fragment-cache": "^0.2.1", + "kind-of": "^5.0.2", + "nanomatch": "^1.2.1", + "object.pick": "^1.3.0", + "regex-not": "^1.0.0", + "snapdragon": "^0.8.1", + "to-regex": "^3.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/svg-baker/node_modules/postcss": { + "version": "5.2.18", + "resolved": "https://registry.npmmirror.com/postcss/-/postcss-5.2.18.tgz", + "integrity": "sha512-zrUjRRe1bpXKsX1qAJNJjqZViErVuyEkMTRrwu4ud4sbTtIBRmtaYDrHmcGgmrbsW3MHfmtIf+vJumgQn+PrXg==", + "dev": true, + "dependencies": { + "chalk": "^1.1.3", + "js-base64": "^2.1.9", + "source-map": "^0.5.6", + "supports-color": "^3.2.3" + }, + "engines": { + "node": ">=0.12" + } + }, + "node_modules/svg-baker/node_modules/strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmmirror.com/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha512-VhumSSbBqDTP8p2ZLKj40UjBCV4+v8bUSEpUb4KjRgWk9pbqGF4REFj6KEagidb2f/M6AzC0EmFyDNGaw9OCzg==", + "dev": true, + "dependencies": { + "ansi-regex": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/svg-baker/node_modules/supports-color": { + "version": "3.2.3", + "resolved": "https://registry.npmmirror.com/supports-color/-/supports-color-3.2.3.tgz", + "integrity": "sha512-Jds2VIYDrlp5ui7t8abHN2bjAu4LV/q4N2KivFPpGH0lrka0BMq/33AmECUXlKPcHigkNaqfXRENFju+rlcy+A==", + "dev": true, + "dependencies": { + "has-flag": "^1.0.0" + }, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/svg-sprite-loader": { + "version": "5.0.0", + "resolved": "https://registry.npmmirror.com/svg-sprite-loader/-/svg-sprite-loader-5.0.0.tgz", + "integrity": "sha512-/hedkRC2IS0E+kFIb+OUCfqQlbVx72/WEEeRGw2uPsNgOOgJEONXzjfSm+CJ3pB9gOHytlBmPMS8ijMCp5s2Eg==", + "dev": true, + "dependencies": { + "bluebird": "^3.5.0", + "deepmerge": "1.3.2", + "domready": "1.0.8", + "escape-string-regexp": "1.0.5", + "html-webpack-plugin": "^3.2.0", + "loader-utils": "^1.1.0", + "svg-baker": "^1.5.0", + "svg-baker-runtime": "^1.4.7", + "url-slug": "2.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/svg-sprite-loader/node_modules/deepmerge": { + "version": "1.3.2", + "resolved": "https://registry.npmmirror.com/deepmerge/-/deepmerge-1.3.2.tgz", + "integrity": "sha512-qjMjTrk+RKv/sp4RPDpV5CnKhxjFI9p+GkLBOls5A8EEElldYWCWA9zceAkmfd0xIo2aU1nxiaLFoiya2sb6Cg==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/svg-tags": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/svg-tags/-/svg-tags-1.0.0.tgz", + "integrity": "sha512-ovssysQTa+luh7A5Weu3Rta6FJlFBBbInjOh722LIt6klpU2/HtdUbszju/G4devcvk8PGt7FCLv5wftu3THUA==", + "dev": true + }, + "node_modules/svgo": { + "version": "1.3.0", + "resolved": "https://registry.npmmirror.com/svgo/-/svgo-1.3.0.tgz", + "integrity": "sha512-MLfUA6O+qauLDbym+mMZgtXCGRfIxyQoeH6IKVcFslyODEe/ElJNwr0FohQ3xG4C6HK6bk3KYPPXwHVJk3V5NQ==", + "deprecated": "This SVGO version is no longer supported. Upgrade to v2.x.x.", + "dependencies": { + "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.33", + "csso": "^3.5.1", + "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" + }, + "bin": { + "svgo": "bin/svgo" + }, + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/symbol-observable": { + "version": "1.2.0", + "resolved": "https://registry.npmmirror.com/symbol-observable/-/symbol-observable-1.2.0.tgz", + "integrity": "sha512-e900nM8RRtGhlV36KGEU9k65K3mPb1WV70OdjfxlG2EAuM1noi/E/BaW/uMhL7bPEssK8QV57vN3esixjUvcXQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/synchronous-promise": { + "version": "2.0.15", + "resolved": "https://registry.npmmirror.com/synchronous-promise/-/synchronous-promise-2.0.15.tgz", + "integrity": "sha512-k8uzYIkIVwmT+TcglpdN50pS2y1BDcUnBPK9iJeGu0Pl1lOI8pD6wtzgw91Pjpe+RxtTncw32tLxs/R0yNL2Mg==", + "dev": true + }, + "node_modules/table": { + "version": "4.0.2", + "resolved": "https://registry.npmmirror.com/table/-/table-4.0.2.tgz", + "integrity": "sha512-UUkEAPdSGxtRpiV9ozJ5cMTtYiqz7Ni1OGqLXRCynrvzdtR1p+cfOWe2RJLwvUG8hNanaSRjecIqwOjqeatDsA==", + "dev": true, + "optional": true, + "dependencies": { + "ajv": "^5.2.3", + "ajv-keywords": "^2.1.0", + "chalk": "^2.1.0", + "lodash": "^4.17.4", + "slice-ansi": "1.0.0", + "string-width": "^2.1.1" + } + }, + "node_modules/table/node_modules/ajv": { + "version": "5.5.2", + "resolved": "https://registry.npmmirror.com/ajv/-/ajv-5.5.2.tgz", + "integrity": "sha512-Ajr4IcMXq/2QmMkEmSvxqfLN5zGmJ92gHXAeOXq1OekoH2rfDNsgdDoL2f7QaRCy7G/E6TpxBVdRuNraMztGHw==", + "dev": true, + "optional": true, + "dependencies": { + "co": "^4.6.0", + "fast-deep-equal": "^1.0.0", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.3.0" + } + }, + "node_modules/table/node_modules/ajv-keywords": { + "version": "2.1.1", + "resolved": "https://registry.npmmirror.com/ajv-keywords/-/ajv-keywords-2.1.1.tgz", + "integrity": "sha512-ZFztHzVRdGLAzJmpUT9LNFLe1YiVOEylcaNpEutM26PVTCtOD919IMfD01CgbRouB42Dd9atjx1HseC15DgOZA==", + "dev": true, + "optional": true, + "peerDependencies": { + "ajv": "^5.0.0" + } + }, + "node_modules/table/node_modules/fast-deep-equal": { + "version": "1.1.0", + "resolved": "https://registry.npmmirror.com/fast-deep-equal/-/fast-deep-equal-1.1.0.tgz", + "integrity": "sha512-fueX787WZKCV0Is4/T2cyAdM4+x1S3MXXOAhavE1ys/W42SHAPacLTQhucja22QBYrfGw50M2sRiXPtTGv9Ymw==", + "dev": true, + "optional": true + }, + "node_modules/table/node_modules/json-schema-traverse": { + "version": "0.3.1", + "resolved": "https://registry.npmmirror.com/json-schema-traverse/-/json-schema-traverse-0.3.1.tgz", + "integrity": "sha512-4JD/Ivzg7PoW8NzdrBSr3UFwC9mHgvI7Z6z3QGBsSHgKaRTUDmyZAAKJo2UbG1kUVfS9WS8bi36N49U1xw43DA==", + "dev": true, + "optional": true + }, + "node_modules/tapable": { + "version": "1.1.3", + "resolved": "https://registry.npmmirror.com/tapable/-/tapable-1.1.3.tgz", + "integrity": "sha512-4WK/bYZmj8xLr+HUCODHGF1ZFzsYffasLUgEiMBY4fgtltdO6B4WJtlSbPaDTLpYTcGVwM2qLnFTICEcNxs3kA==", + "engines": { + "node": ">=6" + } + }, + "node_modules/tar": { + "version": "4.4.19", + "resolved": "https://registry.npmmirror.com/tar/-/tar-4.4.19.tgz", + "integrity": "sha512-a20gEsvHnWe0ygBY8JbxoM4w3SJdhc7ZAuxkLqh+nvNQN2IOt0B5lLgM490X5Hl8FF0dl0tOf2ewFYAlIFgzVA==", + "dev": true, + "dependencies": { + "chownr": "^1.1.4", + "fs-minipass": "^1.2.7", + "minipass": "^2.9.0", + "minizlib": "^1.3.3", + "mkdirp": "^0.5.5", + "safe-buffer": "^5.2.1", + "yallist": "^3.1.1" + }, + "engines": { + "node": ">=4.5" + } + }, + "node_modules/tar-stream": { + "version": "2.2.0", + "resolved": "https://registry.npmmirror.com/tar-stream/-/tar-stream-2.2.0.tgz", + "integrity": "sha512-ujeqbceABgwMZxEJnk2HDY2DlnUZ+9oEcb1KzTVfYHio0UE6dG71n60d8D2I4qNvleWrrXpmjpt7vZeF1LnMZQ==", + "dev": true, + "dependencies": { + "bl": "^4.0.3", + "end-of-stream": "^1.4.1", + "fs-constants": "^1.0.0", + "inherits": "^2.0.3", + "readable-stream": "^3.1.1" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/tar-stream/node_modules/readable-stream": { + "version": "3.6.0", + "resolved": "https://registry.npmmirror.com/readable-stream/-/readable-stream-3.6.0.tgz", + "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", + "dev": true, + "dependencies": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/tar/node_modules/safe-buffer": { + "version": "5.2.1", + "resolved": "https://registry.npmmirror.com/safe-buffer/-/safe-buffer-5.2.1.tgz", + "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", + "dev": true + }, + "node_modules/temp-dir": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/temp-dir/-/temp-dir-1.0.0.tgz", + "integrity": "sha512-xZFXEGbG7SNC3itwBzI3RYjq/cEhBkx2hJuKGIUOcEULmkQExXiHat2z/qkISYsuR+IKumhEfKKbV5qXmhICFQ==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/temp-write": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/temp-write/-/temp-write-4.0.0.tgz", + "integrity": "sha512-HIeWmj77uOOHb0QX7siN3OtwV3CTntquin6TNVg6SHOqCP3hYKmox90eeFOGaY1MqJ9WYDDjkyZrW6qS5AWpbw==", + "dev": true, + "dependencies": { + "graceful-fs": "^4.1.15", + "is-stream": "^2.0.0", + "make-dir": "^3.0.0", + "temp-dir": "^1.0.0", + "uuid": "^3.3.2" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/temp-write/node_modules/is-stream": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/is-stream/-/is-stream-2.0.1.tgz", + "integrity": "sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/temp-write/node_modules/make-dir": { + "version": "3.1.0", + "resolved": "https://registry.npmmirror.com/make-dir/-/make-dir-3.1.0.tgz", + "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==", + "dev": true, + "dependencies": { + "semver": "^6.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/temp-write/node_modules/semver": { + "version": "6.3.0", + "resolved": "https://registry.npmmirror.com/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/terser": { + "version": "4.8.0", + "resolved": "https://registry.npmmirror.com/terser/-/terser-4.8.0.tgz", + "integrity": "sha512-EAPipTNeWsb/3wLPeup1tVPaXfIaU68xMnVdPafIL1TV05OhASArYyIfFvnvJCNrR2NIOvDVNNTFRa+Re2MWyw==", + "dependencies": { + "commander": "^2.20.0", + "source-map": "~0.6.1", + "source-map-support": "~0.5.12" + }, + "bin": { + "terser": "bin/terser" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/terser-webpack-plugin": { + "version": "1.4.5", + "resolved": "https://registry.npmmirror.com/terser-webpack-plugin/-/terser-webpack-plugin-1.4.5.tgz", + "integrity": "sha512-04Rfe496lN8EYruwi6oPQkG0vo8C+HT49X687FZnpPF0qMAIHONI6HEXYPKDOE8e5HjXTyKfqRd/agHtH0kOtw==", + "dependencies": { + "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" + }, + "engines": { + "node": ">= 6.9.0" + }, + "peerDependencies": { + "webpack": "^4.0.0" + } + }, + "node_modules/terser-webpack-plugin/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmmirror.com/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/terser/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmmirror.com/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/text-extensions": { + "version": "1.9.0", + "resolved": "https://registry.npmmirror.com/text-extensions/-/text-extensions-1.9.0.tgz", + "integrity": "sha512-wiBrwC1EhBelW12Zy26JeOUkQ5mRu+5o8rpsJk5+2t+Y5vE7e842qtZDQ2g1NpX/29HdyFeJ4nSIhI47ENSxlQ==", + "dev": true, + "engines": { + "node": ">=0.10" + } + }, + "node_modules/text-table": { + "version": "0.2.0", + "resolved": "https://registry.npmmirror.com/text-table/-/text-table-0.2.0.tgz", + "integrity": "sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==", + "dev": true + }, + "node_modules/thenify": { + "version": "3.3.1", + "resolved": "https://registry.npmmirror.com/thenify/-/thenify-3.3.1.tgz", + "integrity": "sha512-RVZSIV5IG10Hk3enotrhvz0T9em6cyHBLkH/YAZuKqd8hRkKhSfCGIcP2KUY0EPxndzANBmNllzWPwak+bheSw==", + "dev": true, + "dependencies": { + "any-promise": "^1.0.0" + } + }, + "node_modules/thenify-all": { + "version": "1.6.0", + "resolved": "https://registry.npmmirror.com/thenify-all/-/thenify-all-1.6.0.tgz", + "integrity": "sha512-RNxQH/qI8/t3thXJDwcstUO4zeqo64+Uy/+sNVRBx4Xn2OX+OZ9oP+iJnNFqplFra2ZUVeKCSa2oVWi3T4uVmA==", + "dev": true, + "dependencies": { + "thenify": ">= 3.1.0 < 4" + }, + "engines": { + "node": ">=0.8" + } + }, + "node_modules/thread-loader": { + "version": "2.1.3", + "resolved": "https://registry.npmmirror.com/thread-loader/-/thread-loader-2.1.3.tgz", + "integrity": "sha512-wNrVKH2Lcf8ZrWxDF/khdlLlsTMczdcwPA9VEK4c2exlEPynYWxi9op3nPTo5lAnDIkE0rQEB3VBP+4Zncc9Hg==", + "dev": true, + "dependencies": { + "loader-runner": "^2.3.1", + "loader-utils": "^1.1.0", + "neo-async": "^2.6.0" + }, + "engines": { + "node": ">= 6.9.0 <7.0.0 || >= 8.9.0" + }, + "peerDependencies": { + "webpack": "^2.0.0 || ^3.0.0 || ^4.0.0" + } + }, + "node_modules/through": { + "version": "2.3.8", + "resolved": "https://registry.npmmirror.com/through/-/through-2.3.8.tgz", + "integrity": "sha512-w89qg7PI8wAdvX60bMDP+bFoD5Dvhm9oLheFp5O4a2QF0cSBGsBX4qZmadPMvVqlLJBBci+WqGGOAPvcDeNSVg==", + "dev": true + }, + "node_modules/through2": { + "version": "2.0.5", + "resolved": "https://registry.npmmirror.com/through2/-/through2-2.0.5.tgz", + "integrity": "sha512-/mrRod8xqpA+IHSLyGCQ2s8SPHiCDEeQJSep1jqLYeEUClOFG2Qsh+4FU6G9VeqpZnGW/Su8LQGc4YKni5rYSQ==", + "dependencies": { + "readable-stream": "~2.3.6", + "xtend": "~4.0.1" + } + }, + "node_modules/thunky": { + "version": "1.1.0", + "resolved": "https://registry.npmmirror.com/thunky/-/thunky-1.1.0.tgz", + "integrity": "sha512-eHY7nBftgThBqOyHGVN+l8gF0BucP09fMo0oO/Lb0w1OF80dJv+lDVpXG60WMQvkcxAkNybKsrEIE3ZtKGmPrA==", + "dev": true + }, + "node_modules/timers-browserify": { + "version": "2.0.12", + "resolved": "https://registry.npmmirror.com/timers-browserify/-/timers-browserify-2.0.12.tgz", + "integrity": "sha512-9phl76Cqm6FhSX9Xe1ZUAMLtm1BLkKj2Qd5ApyWkXzsMRaA7dgr81kf4wJmQf/hAvg8EEyJxDo3du/0KlhPiKQ==", + "dependencies": { + "setimmediate": "^1.0.4" + }, + "engines": { + "node": ">=0.6.0" + } + }, + "node_modules/timsort": { + "version": "0.3.0", + "resolved": "https://registry.npmmirror.com/timsort/-/timsort-0.3.0.tgz", + "integrity": "sha512-qsdtZH+vMoCARQtyod4imc2nIJwg9Cc7lPRrw9CzF8ZKR0khdr8+2nX80PBhET3tcyTtJDxAffGh2rXH4tyU8A==", + "dev": true + }, + "node_modules/tiny-emitter": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/tiny-emitter/-/tiny-emitter-2.1.0.tgz", + "integrity": "sha512-NB6Dk1A9xgQPMoGqC5CVXn123gWyte215ONT5Pp5a0yt4nlEoO1ZWeCwpncaekPHXO60i47ihFnZPiRPjRMq4Q==" + }, + "node_modules/tinycolor2": { + "version": "1.4.2", + "resolved": "https://registry.npmmirror.com/tinycolor2/-/tinycolor2-1.4.2.tgz", + "integrity": "sha512-vJhccZPs965sV/L2sU4oRQVAos0pQXwsvTLkWYdqJ+a8Q5kPFzJTuOFwy7UniPli44NKQGAglksjvOcpo95aZA==", + "engines": { + "node": "*" + } + }, + "node_modules/tmp": { + "version": "0.0.33", + "resolved": "https://registry.npmmirror.com/tmp/-/tmp-0.0.33.tgz", + "integrity": "sha512-jRCJlojKnZ3addtTOjdIqoRuPEKBvNXcGYqzO6zWZX8KfKEpnGY5jfggJQ3EjKuu8D4bJRr0y+cYJFmYbImXGw==", + "dev": true, + "dependencies": { + "os-tmpdir": "~1.0.2" + }, + "engines": { + "node": ">=0.6.0" + } + }, + "node_modules/to-arraybuffer": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/to-arraybuffer/-/to-arraybuffer-1.0.1.tgz", + "integrity": "sha512-okFlQcoGTi4LQBG/PgSYblw9VOyptsz2KJZqc6qtgGdes8VktzUQkj4BI2blit072iS8VODNcMA+tvnS9dnuMA==" + }, + "node_modules/to-fast-properties": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/to-fast-properties/-/to-fast-properties-2.0.0.tgz", + "integrity": "sha512-/OaKK0xYrs3DmxRYqL/yDc+FxFUVYhDlXMhRmv3z915w2HF1tnN1omB354j8VUGO/hbRzyD6Y3sA7v7GS/ceog==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/to-object-path": { + "version": "0.3.0", + "resolved": "https://registry.npmmirror.com/to-object-path/-/to-object-path-0.3.0.tgz", + "integrity": "sha512-9mWHdnGRuh3onocaHzukyvCZhzvr6tiflAy/JRFXcJX0TjgfWA9pk9t8CMbzmBE4Jfw58pXbkngtBtqYxzNEyg==", + "dependencies": { + "kind-of": "^3.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/to-object-path/node_modules/kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmmirror.com/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha512-NOW9QQXMoZGg/oqnVNoNTTIFEIid1627WCffUBJEdMxYApq7mNE7CpzucIPc+ZQg25Phej7IJSmX3hO+oblOtQ==", + "dependencies": { + "is-buffer": "^1.1.5" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/to-regex": { + "version": "3.0.2", + "resolved": "https://registry.npmmirror.com/to-regex/-/to-regex-3.0.2.tgz", + "integrity": "sha512-FWtleNAtZ/Ki2qtqej2CXTOayOH9bHDQF+Q48VpWyDXjbYxA4Yz8iDB31zXOBUlOHHKidDbqGVrTUvQMPmBGBw==", + "dependencies": { + "define-property": "^2.0.2", + "extend-shallow": "^3.0.2", + "regex-not": "^1.0.2", + "safe-regex": "^1.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/to-regex-range": { + "version": "2.1.1", + "resolved": "https://registry.npmmirror.com/to-regex-range/-/to-regex-range-2.1.1.tgz", + "integrity": "sha512-ZZWNfCjUokXXDGXFpZehJIkZqq91BcULFq/Pi7M5i4JnxXdhMKAK682z8bCW3o8Hj1wuuzoKcW3DfVzaP6VuNg==", + "dependencies": { + "is-number": "^3.0.0", + "repeat-string": "^1.6.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/toidentifier": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/toidentifier/-/toidentifier-1.0.1.tgz", + "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==", + "dev": true, + "engines": { + "node": ">=0.6" + } + }, + "node_modules/toposort": { + "version": "1.0.7", + "resolved": "https://registry.npmmirror.com/toposort/-/toposort-1.0.7.tgz", + "integrity": "sha512-FclLrw8b9bMWf4QlCJuHBEVhSRsqDj6u3nIjAzPeJvgl//1hBlffdlk0MALceL14+koWEdU4ofRAXofbODxQzg==", + "dev": true + }, + "node_modules/tough-cookie": { + "version": "2.5.0", + "resolved": "https://registry.npmmirror.com/tough-cookie/-/tough-cookie-2.5.0.tgz", + "integrity": "sha512-nlLsUzgm1kfLXSXfRZMc1KLAugd4hqJHDTvc2hDIwS3mZAfMEuMbc03SujMF+GEcpaX/qboeycw6iO8JwVv2+g==", + "dev": true, + "dependencies": { + "psl": "^1.1.28", + "punycode": "^2.1.1" + }, + "engines": { + "node": ">=0.8" + } + }, + "node_modules/tr46": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/tr46/-/tr46-2.1.0.tgz", + "integrity": "sha512-15Ih7phfcdP5YxqiB+iDtLoaTz4Nd35+IiAv0kQ5FNKHzXgdWqPoTIqEDDJmXceQt4JZk6lVPT8lnDlPpGDppw==", + "dev": true, + "dependencies": { + "punycode": "^2.1.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/traverse": { + "version": "0.6.6", + "resolved": "https://registry.npmmirror.com/traverse/-/traverse-0.6.6.tgz", + "integrity": "sha512-kdf4JKs8lbARxWdp7RKdNzoJBhGUcIalSYibuGyHJbmk40pOysQ0+QPvlkCOICOivDWU2IJo2rkrxyTK2AH4fw==", + "dev": true + }, + "node_modules/trim-newlines": { + "version": "3.0.1", + "resolved": "https://registry.npmmirror.com/trim-newlines/-/trim-newlines-3.0.1.tgz", + "integrity": "sha512-c1PTsA3tYrIsLGkJkzHF+w9F2EyxfXGo4UyJc4pFL++FMjnq0HJS69T3M7d//gKrFKwy429bouPescbjecU+Zw==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/true-case-path": { + "version": "1.0.3", + "resolved": "https://registry.npmmirror.com/true-case-path/-/true-case-path-1.0.3.tgz", + "integrity": "sha512-m6s2OdQe5wgpFMC+pAJ+q9djG82O2jcHPOI6RNg1yy9rCYR+WD6Nbpl32fDpfC56nirdRy+opFa/Vk7HYhqaew==", + "dev": true, + "dependencies": { + "glob": "^7.1.2" + } + }, + "node_modules/tryer": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/tryer/-/tryer-1.0.1.tgz", + "integrity": "sha512-c3zayb8/kWWpycWYg87P71E1S1ZL6b6IJxfb5fvsUgsf0S2MVGaDhDXXjDMpdCpfWXqptc+4mXwmiy1ypXqRAA==", + "dev": true + }, + "node_modules/tsconfig-paths": { + "version": "3.14.1", + "resolved": "https://registry.npmmirror.com/tsconfig-paths/-/tsconfig-paths-3.14.1.tgz", + "integrity": "sha512-fxDhWnFSLt3VuTwtvJt5fpwxBHg5AdKWMsgcPOOIilyjymcYVZoCQF8fvFRezCNfblEXmi+PcM1eYHeOAgXCOQ==", + "dev": true, + "dependencies": { + "@types/json5": "^0.0.29", + "json5": "^1.0.1", + "minimist": "^1.2.6", + "strip-bom": "^3.0.0" + } + }, + "node_modules/tslib": { + "version": "2.4.0", + "resolved": "https://registry.npmmirror.com/tslib/-/tslib-2.4.0.tgz", + "integrity": "sha512-d6xOpEDfsi2CZVlPQzGeux8XMwLT9hssAsaPYExaQMuYskwb+x1x7J371tWlbBdWHroy99KnVB6qIkUbs5X3UQ==" + }, + "node_modules/tty-browserify": { + "version": "0.0.0", + "resolved": "https://registry.npmmirror.com/tty-browserify/-/tty-browserify-0.0.0.tgz", + "integrity": "sha512-JVa5ijo+j/sOoHGjw0sxw734b1LhBkQ3bvUGNdxnVXDCX81Yx7TFgnZygxrIIWn23hbfTaMYLwRmAxFyDuFmIw==" + }, + "node_modules/tunnel-agent": { + "version": "0.6.0", + "resolved": "https://registry.npmmirror.com/tunnel-agent/-/tunnel-agent-0.6.0.tgz", + "integrity": "sha512-McnNiV1l8RYeY8tBgEpuodCC1mLUdbSN+CYBL7kJsJNInOP8UjDDEwdk6Mw60vdLLrr5NHKZhMAOSrR2NZuQ+w==", + "dev": true, + "dependencies": { + "safe-buffer": "^5.0.1" + }, + "engines": { + "node": "*" + } + }, + "node_modules/tweetnacl": { + "version": "0.14.5", + "resolved": "https://registry.npmmirror.com/tweetnacl/-/tweetnacl-0.14.5.tgz", + "integrity": "sha512-KXXFFdAbFXY4geFIwoyNK+f5Z1b7swfXABfL7HXCmoIWMKU3dmS26672A4EeQtDzLKy7SXmfBu51JolvEKwtGA==", + "dev": true + }, + "node_modules/type-check": { + "version": "0.3.2", + "resolved": "https://registry.npmmirror.com/type-check/-/type-check-0.3.2.tgz", + "integrity": "sha512-ZCmOJdvOWDBYJlzAoFkC+Q0+bUyEOS1ltgp1MGU03fqHG+dbi9tBFU2Rd9QKiDZFAYrhPh2JUf7rZRIuHRKtOg==", + "dev": true, + "dependencies": { + "prelude-ls": "~1.1.2" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/type-fest": { + "version": "0.6.0", + "resolved": "https://registry.npmmirror.com/type-fest/-/type-fest-0.6.0.tgz", + "integrity": "sha512-q+MB8nYR1KDLrgr4G5yemftpMC7/QLqVndBmEEdqzmNj5dcFOO4Oo8qlwZE3ULT3+Zim1F8Kq4cBnikNhlCMlg==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/type-is": { + "version": "1.6.18", + "resolved": "https://registry.npmmirror.com/type-is/-/type-is-1.6.18.tgz", + "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", + "dev": true, + "dependencies": { + "media-typer": "0.3.0", + "mime-types": "~2.1.24" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/typedarray": { + "version": "0.0.6", + "resolved": "https://registry.npmmirror.com/typedarray/-/typedarray-0.0.6.tgz", + "integrity": "sha512-/aCDEGatGvZ2BIk+HmLf4ifCJFwvKFNb9/JeZPMulfgFracn9QFcAf5GO8B/mweUjSoblS5In0cWhqpfs/5PQA==" + }, + "node_modules/typedarray-to-buffer": { + "version": "3.1.5", + "resolved": "https://registry.npmmirror.com/typedarray-to-buffer/-/typedarray-to-buffer-3.1.5.tgz", + "integrity": "sha512-zdu8XMNEDepKKR+XYOXAVPtWui0ly0NtohUscw+UmaHiAWT8hrV1rr//H6V+0DvJ3OQ19S979M0laLfX8rm82Q==", + "dev": true, + "dependencies": { + "is-typedarray": "^1.0.0" + } + }, + "node_modules/uglify-js": { + "version": "3.4.10", + "resolved": "https://registry.npmmirror.com/uglify-js/-/uglify-js-3.4.10.tgz", + "integrity": "sha512-Y2VsbPVs0FIshJztycsO2SfPk7/KAF/T72qzv9u5EpQ4kB2hQoHlhNQTsNyy6ul7lQtqJN/AoWeS23OzEiEFxw==", + "dev": true, + "dependencies": { + "commander": "~2.19.0", + "source-map": "~0.6.1" + }, + "bin": { + "uglifyjs": "bin/uglifyjs" + }, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/uglify-js/node_modules/commander": { + "version": "2.19.0", + "resolved": "https://registry.npmmirror.com/commander/-/commander-2.19.0.tgz", + "integrity": "sha512-6tvAOO+D6OENvRAh524Dh9jcfKTYDQAqvqezbCW82xj5X0pSrcpxtvRKHLG0yBY6SD7PSDrJaj+0AiOcKVd1Xg==", + "dev": true + }, + "node_modules/uglify-js/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmmirror.com/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/uid-number": { + "version": "0.0.6", + "resolved": "https://registry.npmmirror.com/uid-number/-/uid-number-0.0.6.tgz", + "integrity": "sha512-c461FXIljswCuscZn67xq9PpszkPT6RjheWFQTgCyabJrTUozElanb0YEqv2UGgk247YpcJkFBuSGNvBlpXM9w==", + "dev": true, + "engines": { + "node": "*" + } + }, + "node_modules/umask": { + "version": "1.1.0", + "resolved": "https://registry.npmmirror.com/umask/-/umask-1.1.0.tgz", + "integrity": "sha512-lE/rxOhmiScJu9L6RTNVgB/zZbF+vGC0/p6D3xnkAePI2o0sMyFG966iR5Ki50OI/0mNi2yaRnxfLsPmEZF/JA==", + "dev": true + }, + "node_modules/unbox-primitive": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/unbox-primitive/-/unbox-primitive-1.0.2.tgz", + "integrity": "sha512-61pPlCD9h51VoreyJ0BReideM3MDKMKnh6+V9L08331ipq6Q8OFXZYiqP6n/tbHx4s5I9uRhcye6BrbkizkBDw==", + "dependencies": { + "call-bind": "^1.0.2", + "has-bigints": "^1.0.2", + "has-symbols": "^1.0.3", + "which-boxed-primitive": "^1.0.2" + } + }, + "node_modules/unicode-canonical-property-names-ecmascript": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/unicode-canonical-property-names-ecmascript/-/unicode-canonical-property-names-ecmascript-2.0.0.tgz", + "integrity": "sha512-yY5PpDlfVIU5+y/BSCxAJRBIS1Zc2dDG3Ujq+sR0U+JjUevW2JhocOF+soROYDSaAezOzOKuyyixhD6mBknSmQ==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/unicode-match-property-ecmascript": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/unicode-match-property-ecmascript/-/unicode-match-property-ecmascript-2.0.0.tgz", + "integrity": "sha512-5kaZCrbp5mmbz5ulBkDkbY0SsPOjKqVS35VpL9ulMPfSl0J0Xsm+9Evphv9CoIZFwre7aJoa94AY6seMKGVN5Q==", + "dev": true, + "dependencies": { + "unicode-canonical-property-names-ecmascript": "^2.0.0", + "unicode-property-aliases-ecmascript": "^2.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/unicode-match-property-value-ecmascript": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/unicode-match-property-value-ecmascript/-/unicode-match-property-value-ecmascript-2.0.0.tgz", + "integrity": "sha512-7Yhkc0Ye+t4PNYzOGKedDhXbYIBe1XEQYQxOPyhcXNMJ0WCABqqj6ckydd6pWRZTHV4GuCPKdBAUiMc60tsKVw==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/unicode-property-aliases-ecmascript": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/unicode-property-aliases-ecmascript/-/unicode-property-aliases-ecmascript-2.0.0.tgz", + "integrity": "sha512-5Zfuy9q/DFr4tfO7ZPeVXb1aPoeQSdeFMLpYuFebehDAhbuevLs5yxSZmIFN1tP5F9Wl4IpJrYojg85/zgyZHQ==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/unidecode": { + "version": "0.1.8", + "resolved": "https://registry.npmmirror.com/unidecode/-/unidecode-0.1.8.tgz", + "integrity": "sha512-SdoZNxCWpN2tXTCrGkPF/0rL2HEq+i2gwRG1ReBvx8/0yTzC3enHfugOf8A9JBShVwwrRIkLX0YcDUGbzjbVCA==", + "dev": true, + "engines": { + "node": ">= 0.4.12" + } + }, + "node_modules/union-value": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/union-value/-/union-value-1.0.1.tgz", + "integrity": "sha512-tJfXmxMeWYnczCVs7XAEvIV7ieppALdyepWMkHkwciRpZraG/xwT+s2JN8+pr1+8jCRf80FFzvr+MpQeeoF4Xg==", + "dependencies": { + "arr-union": "^3.1.0", + "get-value": "^2.0.6", + "is-extendable": "^0.1.1", + "set-value": "^2.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/uniq": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/uniq/-/uniq-1.0.1.tgz", + "integrity": "sha512-Gw+zz50YNKPDKXs+9d+aKAjVwpjNwqzvNpLigIruT4HA9lMZNdMqs9x07kKHB/L9WRzqp4+DlTU5s4wG2esdoA==", + "dev": true + }, + "node_modules/uniqs": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/uniqs/-/uniqs-2.0.0.tgz", + "integrity": "sha512-mZdDpf3vBV5Efh29kMw5tXoup/buMgxLzOt/XKFKcVmi+15ManNQWr6HfZ2aiZTYlYixbdNJ0KFmIZIv52tHSQ==", + "dev": true + }, + "node_modules/unique-filename": { + "version": "1.1.1", + "resolved": "https://registry.npmmirror.com/unique-filename/-/unique-filename-1.1.1.tgz", + "integrity": "sha512-Vmp0jIp2ln35UTXuryvjzkjGdRyf9b2lTXuSYUiPmzRcl3FDtYqAwOnTJkAngD9SWhnoJzDbTKwaOrZ+STtxNQ==", + "dependencies": { + "unique-slug": "^2.0.0" + } + }, + "node_modules/unique-slug": { + "version": "2.0.2", + "resolved": "https://registry.npmmirror.com/unique-slug/-/unique-slug-2.0.2.tgz", + "integrity": "sha512-zoWr9ObaxALD3DOPfjPSqxt4fnZiWblxHIgeWqW8x7UqDzEtHEQLzji2cuJYQFCU6KmoJikOYAZlrTHHebjx2w==", + "dependencies": { + "imurmurhash": "^0.1.4" + } + }, + "node_modules/universal-user-agent": { + "version": "6.0.0", + "resolved": "https://registry.npmmirror.com/universal-user-agent/-/universal-user-agent-6.0.0.tgz", + "integrity": "sha512-isyNax3wXoKaulPDZWHQqbmIx1k2tb9fb3GGDBRxCscfYV2Ch7WxPArBsFEG8s/safwXTT7H4QGhaIkTp9447w==", + "dev": true + }, + "node_modules/universalify": { + "version": "0.1.2", + "resolved": "https://registry.npmmirror.com/universalify/-/universalify-0.1.2.tgz", + "integrity": "sha512-rBJeI5CXAlmy1pV+617WB9J63U6XcazHHF2f2dbJix4XzpUF0RS3Zbj0FGIOCAva5P/d/GBOYaACQ1w+0azUkg==", + "dev": true, + "engines": { + "node": ">= 4.0.0" + } + }, + "node_modules/unpipe": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/unpipe/-/unpipe-1.0.0.tgz", + "integrity": "sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ==", + "dev": true, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/unquote": { + "version": "1.1.1", + "resolved": "https://registry.npmmirror.com/unquote/-/unquote-1.1.1.tgz", + "integrity": "sha512-vRCqFv6UhXpWxZPyGDh/F3ZpNv8/qo7w6iufLpQg9aKnQ71qM4B5KiI7Mia9COcjEhrO9LueHpMYjYzsWH3OIg==" + }, + "node_modules/unset-value": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/unset-value/-/unset-value-1.0.0.tgz", + "integrity": "sha512-PcA2tsuGSF9cnySLHTLSh2qrQiJ70mn+r+Glzxv2TWZblxsxCC52BDlZoPCsz7STd9pN7EZetkWZBAvk4cgZdQ==", + "dependencies": { + "has-value": "^0.3.1", + "isobject": "^3.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/unset-value/node_modules/has-value": { + "version": "0.3.1", + "resolved": "https://registry.npmmirror.com/has-value/-/has-value-0.3.1.tgz", + "integrity": "sha512-gpG936j8/MzaeID5Yif+577c17TxaDmhuyVgSwtnL/q8UUTySg8Mecb+8Cf1otgLoD7DDH75axp86ER7LFsf3Q==", + "dependencies": { + "get-value": "^2.0.3", + "has-values": "^0.1.4", + "isobject": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/unset-value/node_modules/has-value/node_modules/isobject": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/isobject/-/isobject-2.1.0.tgz", + "integrity": "sha512-+OUdGJlgjOBZDfxnDjYYG6zp487z0JGNQq3cYQYg5f5hKR+syHMsaztzGeml/4kGG55CSpKSpWTY+jYGgsHLgA==", + "dependencies": { + "isarray": "1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/unset-value/node_modules/has-values": { + "version": "0.1.4", + "resolved": "https://registry.npmmirror.com/has-values/-/has-values-0.1.4.tgz", + "integrity": "sha512-J8S0cEdWuQbqD9//tlZxiMuMNmxB8PlEwvYwuxsTmR1G5RXUePEX/SJn7aD0GMLieuZYSwNH0cQuJGwnYunXRQ==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/upath": { + "version": "1.2.0", + "resolved": "https://registry.npmmirror.com/upath/-/upath-1.2.0.tgz", + "integrity": "sha512-aZwGpamFO61g3OlfT7OQCHqhGnW43ieH9WZeP7QxN/G/jS4jfqUkZxoryvJgVPEcrl5NL/ggHsSmLMHuH64Lhg==", + "devOptional": true, + "engines": { + "node": ">=4", + "yarn": "*" + } + }, + "node_modules/upper-case": { + "version": "1.1.3", + "resolved": "https://registry.npmmirror.com/upper-case/-/upper-case-1.1.3.tgz", + "integrity": "sha512-WRbjgmYzgXkCV7zNVpy5YgrHgbBv126rMALQQMrmzOVC4GM2waQ9x7xtm8VU+1yF2kWyPzI9zbZ48n4vSxwfSA==", + "dev": true + }, + "node_modules/uri-js": { + "version": "4.4.1", + "resolved": "https://registry.npmmirror.com/uri-js/-/uri-js-4.4.1.tgz", + "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==", + "dependencies": { + "punycode": "^2.1.0" + } + }, + "node_modules/urix": { + "version": "0.1.0", + "resolved": "https://registry.npmmirror.com/urix/-/urix-0.1.0.tgz", + "integrity": "sha512-Am1ousAhSLBeB9cG/7k7r2R0zj50uDRlZHPGbazid5s9rlF1F/QKYObEKSIunSjIOkJZqwRRLpvewjEkM7pSqg==", + "deprecated": "Please see https://github.com/lydell/urix#deprecated" + }, + "node_modules/url": { + "version": "0.11.0", + "resolved": "https://registry.npmmirror.com/url/-/url-0.11.0.tgz", + "integrity": "sha512-kbailJa29QrtXnxgq+DdCEGlbTeYM2eJUxsz6vjZavrCYPMIFHMKQmSKYAIuUK2i7hgPm28a8piX5NTUtM/LKQ==", + "dependencies": { + "punycode": "1.3.2", + "querystring": "0.2.0" + } + }, + "node_modules/url-loader": { + "version": "1.1.2", + "resolved": "https://registry.npmmirror.com/url-loader/-/url-loader-1.1.2.tgz", + "integrity": "sha512-dXHkKmw8FhPqu8asTc1puBfe3TehOCo2+RmOOev5suNCIYBcT626kxiWg1NBVkwc4rO8BGa7gP70W7VXuqHrjg==", + "dev": true, + "dependencies": { + "loader-utils": "^1.1.0", + "mime": "^2.0.3", + "schema-utils": "^1.0.0" + }, + "engines": { + "node": ">= 6.9.0" + }, + "peerDependencies": { + "webpack": "^3.0.0 || ^4.0.0" + } + }, + "node_modules/url-parse": { + "version": "1.5.10", + "resolved": "https://registry.npmmirror.com/url-parse/-/url-parse-1.5.10.tgz", + "integrity": "sha512-WypcfiRhfeUP9vvF0j6rw0J3hrWrw6iZv3+22h6iRMJ/8z1Tj6XfLP4DsUix5MhMPnXpiHDoKyoZ/bdCkwBCiQ==", + "dev": true, + "dependencies": { + "querystringify": "^2.1.1", + "requires-port": "^1.0.0" + } + }, + "node_modules/url-slug": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/url-slug/-/url-slug-2.0.0.tgz", + "integrity": "sha512-aiNmSsVgrjCiJ2+KWPferjT46YFKoE8i0YX04BlMVDue022Xwhg/zYlnZ6V9/mP3p8Wj7LEp0myiTkC/p6sxew==", + "dev": true, + "dependencies": { + "unidecode": "0.1.8" + } + }, + "node_modules/url/node_modules/punycode": { + "version": "1.3.2", + "resolved": "https://registry.npmmirror.com/punycode/-/punycode-1.3.2.tgz", + "integrity": "sha512-RofWgt/7fL5wP1Y7fxE7/EmTLzQVnB0ycyibJ0OOHIlJqTNzglYFxVwETOcIoJqJmpDXJ9xImDv+Fq34F/d4Dw==" + }, + "node_modules/use": { + "version": "3.1.1", + "resolved": "https://registry.npmmirror.com/use/-/use-3.1.1.tgz", + "integrity": "sha512-cwESVXlO3url9YWlFW/TA9cshCEhtu7IKJ/p5soJ/gGpj7vbvFrAY/eIioQ6Dw23KjZhYgiIo8HOs1nQ2vr/oQ==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/util": { + "version": "0.11.1", + "resolved": "https://registry.npmmirror.com/util/-/util-0.11.1.tgz", + "integrity": "sha512-HShAsny+zS2TZfaXxD9tYj4HQGlBezXZMZuM/S5PKLLoZkShZiGk9o5CzukI1LVHZvjdvZ2Sj1aW/Ndn2NB/HQ==", + "dependencies": { + "inherits": "2.0.3" + } + }, + "node_modules/util-deprecate": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/util-deprecate/-/util-deprecate-1.0.2.tgz", + "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==" + }, + "node_modules/util-promisify": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/util-promisify/-/util-promisify-2.1.0.tgz", + "integrity": "sha512-K+5eQPYs14b3+E+hmE2J6gCZ4JmMl9DbYS6BeP2CHq6WMuNxErxf5B/n0fz85L8zUuoO6rIzNNmIQDu/j+1OcA==", + "dev": true, + "dependencies": { + "object.getownpropertydescriptors": "^2.0.3" + } + }, + "node_modules/util.promisify": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/util.promisify/-/util.promisify-1.0.1.tgz", + "integrity": "sha512-g9JpC/3He3bm38zsLupWryXHoEcS22YHthuPQSJdMy6KNrzIRzWqcsHzD/WUnqe45whVou4VIsPew37DoXWNrA==", + "dependencies": { + "define-properties": "^1.1.3", + "es-abstract": "^1.17.2", + "has-symbols": "^1.0.1", + "object.getownpropertydescriptors": "^2.1.0" + } + }, + "node_modules/util/node_modules/inherits": { + "version": "2.0.3", + "resolved": "https://registry.npmmirror.com/inherits/-/inherits-2.0.3.tgz", + "integrity": "sha512-x00IRNXNy63jwGkJmzPigoySHbaqpNuzKbBOmzK+g2OdZpQ9w+sxCN+VSB3ja7IAge2OP2qpfxTjeNcyjmW1uw==" + }, + "node_modules/utila": { + "version": "0.4.0", + "resolved": "https://registry.npmmirror.com/utila/-/utila-0.4.0.tgz", + "integrity": "sha512-Z0DbgELS9/L/75wZbro8xAnT50pBVFQZ+hUEueGDU5FN51YSCYM+jdxsfCiHjwNP/4LCDD0i/graKpeBnOXKRA==", + "dev": true + }, + "node_modules/utils-merge": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/utils-merge/-/utils-merge-1.0.1.tgz", + "integrity": "sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA==", + "dev": true, + "engines": { + "node": ">= 0.4.0" + } + }, + "node_modules/uuid": { + "version": "3.4.0", + "resolved": "https://registry.npmmirror.com/uuid/-/uuid-3.4.0.tgz", + "integrity": "sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A==", + "deprecated": "Please upgrade to version 7 or higher. Older versions may use Math.random() in certain circumstances, which is known to be problematic. See https://v8.dev/blog/math-random for details.", + "dev": true, + "bin": { + "uuid": "bin/uuid" + } + }, + "node_modules/v-click-outside-x": { + "version": "4.1.3", + "resolved": "https://registry.npmmirror.com/v-click-outside-x/-/v-click-outside-x-4.1.3.tgz", + "integrity": "sha512-qK4wwuDHK406fGBSJ4DbioPb6LQpRkWqk8i1TZhnVwObU+W4Ra6H7Cn+VLy/dTNTH/sgZJzk9YMYnB/s5RJ0Hg==", + "engines": { + "node": ">=8.11.4", + "npm": ">=6.10.1" + } + }, + "node_modules/v8-compile-cache": { + "version": "2.3.0", + "resolved": "https://registry.npmmirror.com/v8-compile-cache/-/v8-compile-cache-2.3.0.tgz", + "integrity": "sha512-l8lCEmLcLYZh4nbunNZvQCJc5pv7+RCwa8q/LdUx8u7lsWvPDKmpodJAJNwkAhJC//dFY48KuIEmjtd4RViDrA==", + "dev": true + }, + "node_modules/validate-npm-package-license": { + "version": "3.0.4", + "resolved": "https://registry.npmmirror.com/validate-npm-package-license/-/validate-npm-package-license-3.0.4.tgz", + "integrity": "sha512-DpKm2Ui/xN7/HQKCtpZxoRWBhZ9Z0kqtygG8XCgNQ8ZlDnxuQmWhj566j8fN4Cu3/JmbhsDo7fcAJq4s9h27Ew==", + "dev": true, + "dependencies": { + "spdx-correct": "^3.0.0", + "spdx-expression-parse": "^3.0.0" + } + }, + "node_modules/validate-npm-package-name": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/validate-npm-package-name/-/validate-npm-package-name-3.0.0.tgz", + "integrity": "sha512-M6w37eVCMMouJ9V/sdPGnC5H4uDr73/+xdq0FBLO3TFFX1+7wiUY6Es328NN+y43tmY+doUdN9g9J21vqB7iLw==", + "dev": true, + "dependencies": { + "builtins": "^1.0.3" + } + }, + "node_modules/vary": { + "version": "1.1.2", + "resolved": "https://registry.npmmirror.com/vary/-/vary-1.1.2.tgz", + "integrity": "sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg==", + "dev": true, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/vendors": { + "version": "1.0.4", + "resolved": "https://registry.npmmirror.com/vendors/-/vendors-1.0.4.tgz", + "integrity": "sha512-/juG65kTL4Cy2su4P8HjtkTxk6VmJDiOPBufWniqQ6wknac6jNiXS9vU+hO3wgusiyqWlzTbVHi0dyJqRONg3w==", + "dev": true + }, + "node_modules/verror": { + "version": "1.10.0", + "resolved": "https://registry.npmmirror.com/verror/-/verror-1.10.0.tgz", + "integrity": "sha512-ZZKSmDAEFOijERBLkmYfJ+vmk3w+7hOLYDNkRCuRuMJGEmqYNCNLyBBFwWKVMhfwaEF3WOd0Zlw86U/WC/+nYw==", + "dev": true, + "engines": [ + "node >=0.6.0" + ], + "dependencies": { + "assert-plus": "^1.0.0", + "core-util-is": "1.0.2", + "extsprintf": "^1.2.0" + } + }, + "node_modules/verror/node_modules/core-util-is": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/core-util-is/-/core-util-is-1.0.2.tgz", + "integrity": "sha512-3lqz5YjWTYnW6dlDa5TLaTCcShfar1e40rmcJVwCBJC6mWlFuj0eCHIElmG1g5kyuJ/GD+8Wn4FFCcz4gJPfaQ==", + "dev": true + }, + "node_modules/vm-browserify": { + "version": "1.1.2", + "resolved": "https://registry.npmmirror.com/vm-browserify/-/vm-browserify-1.1.2.tgz", + "integrity": "sha512-2ham8XPWTONajOR0ohOKOHXkm3+gaBmGut3SRuu75xLd/RRaY6vqgh8NBYYk7+RW3u5AtzPQZG8F10LHkl0lAQ==" + }, + "node_modules/vscode-jsonrpc": { + "version": "5.0.1", + "resolved": "https://registry.npmmirror.com/vscode-jsonrpc/-/vscode-jsonrpc-5.0.1.tgz", + "integrity": "sha512-JvONPptw3GAQGXlVV2utDcHx0BiY34FupW/kI6mZ5x06ER5DdPG/tXWMVHjTNULF5uKPOUUD0SaXg5QaubJL0A==", + "engines": { + "node": ">=8.0.0 || >=10.0.0" + } + }, + "node_modules/vscode-languageclient": { + "version": "6.1.4", + "resolved": "https://registry.npmmirror.com/vscode-languageclient/-/vscode-languageclient-6.1.4.tgz", + "integrity": "sha512-EUOU+bJu6axmt0RFNo3nrglQLPXMfanbYViJee3Fbn2VuQoX0ZOI4uTYhSRvYLP2vfwTP/juV62P/mksCdTZMA==", + "dependencies": { + "semver": "^6.3.0", + "vscode-languageserver-protocol": "3.15.3" + }, + "engines": { + "vscode": "^1.41.0" + } + }, + "node_modules/vscode-languageclient/node_modules/semver": { + "version": "6.3.0", + "resolved": "https://registry.npmmirror.com/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/vscode-languageserver-protocol": { + "version": "3.15.3", + "resolved": "https://registry.npmmirror.com/vscode-languageserver-protocol/-/vscode-languageserver-protocol-3.15.3.tgz", + "integrity": "sha512-zrMuwHOAQRhjDSnflWdJG+O2ztMWss8GqUUB8dXLR/FPenwkiBNkMIJJYfSN6sgskvsF0rHAoBowNQfbyZnnvw==", + "dependencies": { + "vscode-jsonrpc": "^5.0.1", + "vscode-languageserver-types": "3.15.1" + } + }, + "node_modules/vscode-languageserver-types": { + "version": "3.15.1", + "resolved": "https://registry.npmmirror.com/vscode-languageserver-types/-/vscode-languageserver-types-3.15.1.tgz", + "integrity": "sha512-+a9MPUQrNGRrGU630OGbYVQ+11iOIovjCkqxajPa9w57Sd5ruK8WQNsslzpa0x/QJqC8kRc2DUxWjIFwoNm4ZQ==" + }, + "node_modules/vscode-uri": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/vscode-uri/-/vscode-uri-2.1.2.tgz", + "integrity": "sha512-8TEXQxlldWAuIODdukIb+TR5s+9Ds40eSJrw+1iDDA9IFORPjMELarNQE3myz5XIkWWpdprmJjm1/SxMlWOC8A==" + }, + "node_modules/vscode-ws-jsonrpc": { + "version": "0.2.0", + "resolved": "https://registry.npmmirror.com/vscode-ws-jsonrpc/-/vscode-ws-jsonrpc-0.2.0.tgz", + "integrity": "sha512-NE9HNRgPjCaPyTJvIudcpyIWPImxwRDtuTX16yks7SAiZgSXigxAiZOvSvVBGmD1G/OMfrFo6BblOtjVR9DdVA==", + "dependencies": { + "vscode-jsonrpc": "^5.0.0" + } + }, + "node_modules/vue": { + "version": "2.6.12", + "resolved": "https://registry.npmmirror.com/vue/-/vue-2.6.12.tgz", + "integrity": "sha512-uhmLFETqPPNyuLLbsKz6ioJ4q7AZHzD8ZVFNATNyICSZouqP2Sz0rotWQC8UNBF6VGSCs5abnKJoStA6JbCbfg==" + }, + "node_modules/vue-cli-plugin-mockjs": { + "version": "0.1.3", + "resolved": "https://registry.npmmirror.com/vue-cli-plugin-mockjs/-/vue-cli-plugin-mockjs-0.1.3.tgz", + "integrity": "sha512-BK7EaGhrLYYMOAPuhLxZjpBGJOLtpCFfVT2FDhDqBZIF+DimmAG/06P+gFwMDSQb8j8ROirLLBgLDjqIXJj56A==", + "dev": true, + "dependencies": { + "express": "^4.16.4", + "mockjs": "^1.0.1-beta3" + } + }, + "node_modules/vue-codemirror": { + "version": "4.0.6", + "resolved": "https://registry.npmmirror.com/vue-codemirror/-/vue-codemirror-4.0.6.tgz", + "integrity": "sha512-ilU7Uf0mqBNSSV3KT7FNEeRIxH4s1fmpG4TfHlzvXn0QiQAbkXS9lLfwuZpaBVEnpP5CSE62iGJjoliTuA8poQ==", + "dependencies": { + "codemirror": "^5.41.0", + "diff-match-patch": "^1.0.0" + }, + "engines": { + "node": ">= 4.0.0", + "npm": ">= 3.0.0" + } + }, + "node_modules/vue-eslint-parser": { + "version": "2.0.3", + "resolved": "https://registry.npmmirror.com/vue-eslint-parser/-/vue-eslint-parser-2.0.3.tgz", + "integrity": "sha512-ZezcU71Owm84xVF6gfurBQUGg8WQ+WZGxgDEQu1IHFBZNx7BFZg3L1yHxrCBNNwbwFtE1GuvfJKMtb6Xuwc/Bw==", + "dev": true, + "optional": true, + "dependencies": { + "debug": "^3.1.0", + "eslint-scope": "^3.7.1", + "eslint-visitor-keys": "^1.0.0", + "espree": "^3.5.2", + "esquery": "^1.0.0", + "lodash": "^4.17.4" + }, + "engines": { + "node": ">=4" + }, + "peerDependencies": { + "eslint": ">=3.9.0" + } + }, + "node_modules/vue-eslint-parser/node_modules/debug": { + "version": "3.2.7", + "resolved": "https://registry.npmmirror.com/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dev": true, + "optional": true, + "dependencies": { + "ms": "^2.1.1" + } + }, + "node_modules/vue-eslint-parser/node_modules/eslint-scope": { + "version": "3.7.3", + "resolved": "https://registry.npmmirror.com/eslint-scope/-/eslint-scope-3.7.3.tgz", + "integrity": "sha512-W+B0SvF4gamyCTmUc+uITPY0989iXVfKvhwtmJocTaYoc/3khEHmEmvfY/Gn9HA9VV75jrQECsHizkNw1b68FA==", + "dev": true, + "optional": true, + "dependencies": { + "esrecurse": "^4.1.0", + "estraverse": "^4.1.1" + }, + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/vue-eslint-parser/node_modules/ms": { + "version": "2.1.3", + "resolved": "https://registry.npmmirror.com/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", + "dev": true, + "optional": true + }, + "node_modules/vue-hot-reload-api": { + "version": "2.3.4", + "resolved": "https://registry.npmmirror.com/vue-hot-reload-api/-/vue-hot-reload-api-2.3.4.tgz", + "integrity": "sha512-BXq3jwIagosjgNVae6tkHzzIk6a8MHFtzAdwhnV5VlvPTFxDCvIttgSiHWjdGoTJvXtmRu5HacExfdarRcFhog==", + "dev": true + }, + "node_modules/vue-i18n": { + "version": "8.22.1", + "resolved": "https://registry.npmmirror.com/vue-i18n/-/vue-i18n-8.22.1.tgz", + "integrity": "sha512-JNgiEJ5a8YPfk5y2lKyfOAGLmkpAVfhaUi+T4wGpSppRYZ3XSyawSDDketY5KV2CsAiBLAGEIO6jO+0l2hQubg==" + }, + "node_modules/vue-loader": { + "version": "15.9.8", + "resolved": "https://registry.npmmirror.com/vue-loader/-/vue-loader-15.9.8.tgz", + "integrity": "sha512-GwSkxPrihfLR69/dSV3+5CdMQ0D+jXg8Ma1S4nQXKJAznYFX14vHdc/NetQc34Dw+rBbIJyP7JOuVb9Fhprvog==", + "dev": true, + "dependencies": { + "@vue/component-compiler-utils": "^3.1.0", + "hash-sum": "^1.0.2", + "loader-utils": "^1.1.0", + "vue-hot-reload-api": "^2.3.0", + "vue-style-loader": "^4.1.0" + }, + "peerDependencies": { + "css-loader": "*", + "webpack": "^3.0.0 || ^4.1.0 || ^5.0.0-0" + }, + "peerDependenciesMeta": { + "cache-loader": { + "optional": true + }, + "vue-template-compiler": { + "optional": true + } + } + }, + "node_modules/vue-router": { + "version": "3.4.8", + "resolved": "https://registry.npmmirror.com/vue-router/-/vue-router-3.4.8.tgz", + "integrity": "sha512-3BsR84AqarcmweXjItxw3jwQsiYNssYg090yi4rlzTnCJxmHtkyCvhNz9Z7qRSOkmiV485KkUCReTp5AjNY4wg==" + }, + "node_modules/vue-style-loader": { + "version": "4.1.3", + "resolved": "https://registry.npmmirror.com/vue-style-loader/-/vue-style-loader-4.1.3.tgz", + "integrity": "sha512-sFuh0xfbtpRlKfm39ss/ikqs9AbKCoXZBpHeVZ8Tx650o0k0q/YCM7FRvigtxpACezfq6af+a7JeqVTWvncqDg==", + "dev": true, + "dependencies": { + "hash-sum": "^1.0.2", + "loader-utils": "^1.0.2" + } + }, + "node_modules/vue-template-compiler": { + "version": "2.6.12", + "resolved": "https://registry.npmmirror.com/vue-template-compiler/-/vue-template-compiler-2.6.12.tgz", + "integrity": "sha512-OzzZ52zS41YUbkCBfdXShQTe69j1gQDZ9HIX8miuC9C3rBCk9wIRjLiZZLrmX9V+Ftq/YEyv1JaVr5Y/hNtByg==", + "dev": true, + "dependencies": { + "de-indent": "^1.0.2", + "he": "^1.1.0" + } + }, + "node_modules/vue-template-es2015-compiler": { + "version": "1.9.1", + "resolved": "https://registry.npmmirror.com/vue-template-es2015-compiler/-/vue-template-es2015-compiler-1.9.1.tgz", + "integrity": "sha512-4gDntzrifFnCEvyoO8PqyJDmguXgVPxKiIxrBKjIowvL9l+N66196+72XVYR8BBf1Uv1Fgt3bGevJ+sEmxfZzw==", + "dev": true + }, + "node_modules/vuedraggable": { + "version": "2.24.3", + "resolved": "https://registry.npmmirror.com/vuedraggable/-/vuedraggable-2.24.3.tgz", + "integrity": "sha512-6/HDXi92GzB+Hcs9fC6PAAozK1RLt1ewPTLjK0anTYguXLAeySDmcnqE8IC0xa7shvSzRjQXq3/+dsZ7ETGF3g==", + "dependencies": { + "sortablejs": "1.10.2" + } + }, + "node_modules/vuescroll": { + "version": "4.16.1", + "resolved": "https://registry.npmmirror.com/vuescroll/-/vuescroll-4.16.1.tgz", + "integrity": "sha512-7fRsG2Yw5Z07LUz/IIu9barpmYiN9q+ZTC+CrVamvCbmsxyhz8mU1OuYFbfORysaUskioNMxTGDo+HOzeDfSyQ==", + "peerDependencies": { + "vue": "^2.0.0" + } + }, + "node_modules/w3c-keyname": { + "version": "2.2.8", + "resolved": "https://registry.npmmirror.com/w3c-keyname/-/w3c-keyname-2.2.8.tgz", + "integrity": "sha512-dpojBhNsCNN7T82Tm7k26A6G9ML3NkhDsnw9n/eoxSRlVBB4CEtIQ/KTCLI2Fwf3ataSXRhYFkQi3SlnFwPvPQ==" + }, + "node_modules/watch-size": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/watch-size/-/watch-size-2.0.0.tgz", + "integrity": "sha512-M92R89dNoTPWyCD+HuUEDdhaDnh9jxPGOwlDc0u51jAgmjUvzqaEMynXSr3BaWs+QdHYk4KzibPy1TFtjLmOZQ==" + }, + "node_modules/watchpack": { + "version": "1.7.5", + "resolved": "https://registry.npmmirror.com/watchpack/-/watchpack-1.7.5.tgz", + "integrity": "sha512-9P3MWk6SrKjHsGkLT2KHXdQ/9SNkyoJbabxnKOoJepsvJjJG8uYTR3yTPxPQvNDI3w4Nz1xnE0TLHK4RIVe/MQ==", + "dependencies": { + "graceful-fs": "^4.1.2", + "neo-async": "^2.5.0" + }, + "optionalDependencies": { + "chokidar": "^3.4.1", + "watchpack-chokidar2": "^2.0.1" + } + }, + "node_modules/watchpack-chokidar2": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/watchpack-chokidar2/-/watchpack-chokidar2-2.0.1.tgz", + "integrity": "sha512-nCFfBIPKr5Sh61s4LPpy1Wtfi0HE8isJ3d2Yb5/Ppw2P2B/3eVSEBjKfN0fmHJSK14+31KwMKmcrzs2GM4P0Ww==", + "optional": true, + "dependencies": { + "chokidar": "^2.1.8" + } + }, + "node_modules/watchpack-chokidar2/node_modules/anymatch": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/anymatch/-/anymatch-2.0.0.tgz", + "integrity": "sha512-5teOsQWABXHHBFP9y3skS5P3d/WfWXpv3FUpy+LorMrNYaT9pI4oLMQX7jzQ2KklNpGpWHzdCXTDT2Y3XGlZBw==", + "optional": true, + "dependencies": { + "micromatch": "^3.1.4", + "normalize-path": "^2.1.1" + } + }, + "node_modules/watchpack-chokidar2/node_modules/anymatch/node_modules/normalize-path": { + "version": "2.1.1", + "resolved": "https://registry.npmmirror.com/normalize-path/-/normalize-path-2.1.1.tgz", + "integrity": "sha512-3pKJwH184Xo/lnH6oyP1q2pMd7HcypqqmRs91/6/i2CGtWwIKGCkOOMTm/zXbgTEWHw1uNpNi/igc3ePOYHb6w==", + "optional": true, + "dependencies": { + "remove-trailing-separator": "^1.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/watchpack-chokidar2/node_modules/binary-extensions": { + "version": "1.13.1", + "resolved": "https://registry.npmmirror.com/binary-extensions/-/binary-extensions-1.13.1.tgz", + "integrity": "sha512-Un7MIEDdUC5gNpcGDV97op1Ywk748MpHcFTHoYs6qnj1Z3j7I53VG3nwZhKzoBZmbdRNnb6WRdFlwl7tSDuZGw==", + "optional": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/watchpack-chokidar2/node_modules/chokidar": { + "version": "2.1.8", + "resolved": "https://registry.npmmirror.com/chokidar/-/chokidar-2.1.8.tgz", + "integrity": "sha512-ZmZUazfOzf0Nve7duiCKD23PFSCs4JPoYyccjUFF3aQkQadqBhfzhjkwBH2mNOG9cTBwhamM37EIsIkZw3nRgg==", + "deprecated": "Chokidar 2 does not receive security updates since 2019. Upgrade to chokidar 3 with 15x fewer dependencies", + "optional": true, + "dependencies": { + "anymatch": "^2.0.0", + "async-each": "^1.0.1", + "braces": "^2.3.2", + "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" + }, + "optionalDependencies": { + "fsevents": "^1.2.7" + } + }, + "node_modules/watchpack-chokidar2/node_modules/fsevents": { + "version": "1.2.13", + "resolved": "https://registry.npmmirror.com/fsevents/-/fsevents-1.2.13.tgz", + "integrity": "sha512-oWb1Z6mkHIskLzEJ/XWX0srkpkTQ7vaopMQkyaEIoq0fmtFVxOthb8cCxeT+p3ynTdkk/RZwbgG4brR5BeWECw==", + "deprecated": "fsevents 1 will break on node v14+ and could be using insecure binaries. Upgrade to fsevents 2.", + "hasInstallScript": true, + "optional": true, + "os": [ + "darwin" + ], + "dependencies": { + "bindings": "^1.5.0", + "nan": "^2.12.1" + }, + "engines": { + "node": ">= 4.0" + } + }, + "node_modules/watchpack-chokidar2/node_modules/glob-parent": { + "version": "3.1.0", + "resolved": "https://registry.npmmirror.com/glob-parent/-/glob-parent-3.1.0.tgz", + "integrity": "sha512-E8Ak/2+dZY6fnzlR7+ueWvhsH1SjHr4jjss4YS/h4py44jY9MhK/VFdaZJAWDz6BbL21KeteKxFSFpq8OS5gVA==", + "optional": true, + "dependencies": { + "is-glob": "^3.1.0", + "path-dirname": "^1.0.0" + } + }, + "node_modules/watchpack-chokidar2/node_modules/glob-parent/node_modules/is-glob": { + "version": "3.1.0", + "resolved": "https://registry.npmmirror.com/is-glob/-/is-glob-3.1.0.tgz", + "integrity": "sha512-UFpDDrPgM6qpnFNI+rh/p3bUaq9hKLZN8bMUWzxmcnZVS3omf4IPK+BrewlnWjO1WmUsMYuSjKh4UJuV4+Lqmw==", + "optional": true, + "dependencies": { + "is-extglob": "^2.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/watchpack-chokidar2/node_modules/is-binary-path": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/is-binary-path/-/is-binary-path-1.0.1.tgz", + "integrity": "sha512-9fRVlXc0uCxEDj1nQzaWONSpbTfx0FmJfzHF7pwlI8DkWGoHBBea4Pg5Ky0ojwwxQmnSifgbKkI06Qv0Ljgj+Q==", + "optional": true, + "dependencies": { + "binary-extensions": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/watchpack-chokidar2/node_modules/readdirp": { + "version": "2.2.1", + "resolved": "https://registry.npmmirror.com/readdirp/-/readdirp-2.2.1.tgz", + "integrity": "sha512-1JU/8q+VgFZyxwrJ+SVIOsh+KywWGpds3NTqikiKpDMZWScmAYyKIgqkO+ARvNWJfXeXR1zxz7aHF4u4CyH6vQ==", + "optional": true, + "dependencies": { + "graceful-fs": "^4.1.11", + "micromatch": "^3.1.10", + "readable-stream": "^2.0.2" + }, + "engines": { + "node": ">=0.10" + } + }, + "node_modules/wbuf": { + "version": "1.7.3", + "resolved": "https://registry.npmmirror.com/wbuf/-/wbuf-1.7.3.tgz", + "integrity": "sha512-O84QOnr0icsbFGLS0O3bI5FswxzRr8/gHwWkDlQFskhSPryQXvrTMxjxGP4+iWYoauLoBvfDpkrOauZ+0iZpDA==", + "dev": true, + "dependencies": { + "minimalistic-assert": "^1.0.0" + } + }, + "node_modules/wcwidth": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/wcwidth/-/wcwidth-1.0.1.tgz", + "integrity": "sha512-XHPEwS0q6TaxcvG85+8EYkbiCux2XtWG2mkc47Ng2A77BQu9+DqIOJldST4HgPkuea7dvKSj5VgX3P1d4rW8Tg==", + "dev": true, + "dependencies": { + "defaults": "^1.0.3" + } + }, + "node_modules/webidl-conversions": { + "version": "6.1.0", + "resolved": "https://registry.npmmirror.com/webidl-conversions/-/webidl-conversions-6.1.0.tgz", + "integrity": "sha512-qBIvFLGiBpLjfwmYAaHPXsn+ho5xZnGvyGvsarywGNc8VyQJUMHJ8OBKGGrPER0okBeMDaan4mNBlgBROxuI8w==", + "dev": true, + "engines": { + "node": ">=10.4" + } + }, + "node_modules/webpack": { + "version": "4.46.0", + "resolved": "https://registry.npmmirror.com/webpack/-/webpack-4.46.0.tgz", + "integrity": "sha512-6jJuJjg8znb/xRItk7bkT0+Q7AHCYjjFnvKIWQPkNIOyRqoCGvkOs0ipeQzrqz4l5FtN5ZI/ukEHroeX/o1/5Q==", + "dependencies": { + "@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.5.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.7.4", + "webpack-sources": "^1.4.1" + }, + "bin": { + "webpack": "bin/webpack.js" + }, + "engines": { + "node": ">=6.11.5" + }, + "peerDependenciesMeta": { + "webpack-cli": { + "optional": true + }, + "webpack-command": { + "optional": true + } + } + }, + "node_modules/webpack-bundle-analyzer": { + "version": "3.9.0", + "resolved": "https://registry.npmmirror.com/webpack-bundle-analyzer/-/webpack-bundle-analyzer-3.9.0.tgz", + "integrity": "sha512-Ob8amZfCm3rMB1ScjQVlbYYUEJyEjdEtQ92jqiFUYt5VkEeO2v5UMbv49P/gnmCZm3A6yaFQzCBvpZqN4MUsdA==", + "dev": true, + "dependencies": { + "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.19", + "mkdirp": "^0.5.1", + "opener": "^1.5.1", + "ws": "^6.0.0" + }, + "bin": { + "webpack-bundle-analyzer": "lib/bin/analyzer.js" + }, + "engines": { + "node": ">= 6.14.4" + } + }, + "node_modules/webpack-bundle-analyzer/node_modules/acorn": { + "version": "7.4.1", + "resolved": "https://registry.npmmirror.com/acorn/-/acorn-7.4.1.tgz", + "integrity": "sha512-nQyp0o1/mNdbTO1PO6kHkwSrmgZ0MT/jCCpNiwbUjGoRN4dlBhqJtoQuCnEOKzgTVwg0ZWiCoQy6SxMebQVh8A==", + "dev": true, + "bin": { + "acorn": "bin/acorn" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/webpack-bundle-analyzer/node_modules/acorn-walk": { + "version": "7.2.0", + "resolved": "https://registry.npmmirror.com/acorn-walk/-/acorn-walk-7.2.0.tgz", + "integrity": "sha512-OPdCF6GsMIP+Az+aWfAAOEt2/+iVDKE7oy6lJ098aoe59oAmK76qV6Gw60SbZ8jHuG2wH058GF4pLFbYamYrVA==", + "dev": true, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/webpack-chain": { + "version": "4.12.1", + "resolved": "https://registry.npmmirror.com/webpack-chain/-/webpack-chain-4.12.1.tgz", + "integrity": "sha512-BCfKo2YkDe2ByqkEWe1Rw+zko4LsyS75LVr29C6xIrxAg9JHJ4pl8kaIZ396SUSNp6b4815dRZPSTAS8LlURRQ==", + "dev": true, + "dependencies": { + "deepmerge": "^1.5.2", + "javascript-stringify": "^1.6.0" + } + }, + "node_modules/webpack-chain/node_modules/deepmerge": { + "version": "1.5.2", + "resolved": "https://registry.npmmirror.com/deepmerge/-/deepmerge-1.5.2.tgz", + "integrity": "sha512-95k0GDqvBjZavkuvzx/YqVLv/6YYa17fz6ILMSf7neqQITCPbnfEnQvEgMPNjH4kgobe7+WIL0yJEHku+H3qtQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/webpack-dev-middleware": { + "version": "3.7.3", + "resolved": "https://registry.npmmirror.com/webpack-dev-middleware/-/webpack-dev-middleware-3.7.3.tgz", + "integrity": "sha512-djelc/zGiz9nZj/U7PTBi2ViorGJXEWo/3ltkPbDyxCXhhEXkW0ce99falaok4TPj+AsxLiXJR0EBOb0zh9fKQ==", + "dev": true, + "dependencies": { + "memory-fs": "^0.4.1", + "mime": "^2.4.4", + "mkdirp": "^0.5.1", + "range-parser": "^1.2.1", + "webpack-log": "^2.0.0" + }, + "engines": { + "node": ">= 6" + }, + "peerDependencies": { + "webpack": "^4.0.0 || ^5.0.0" + } + }, + "node_modules/webpack-dev-server": { + "version": "3.11.3", + "resolved": "https://registry.npmmirror.com/webpack-dev-server/-/webpack-dev-server-3.11.3.tgz", + "integrity": "sha512-3x31rjbEQWKMNzacUZRE6wXvUFuGpH7vr0lIEbYpMAG9BOxi0928QU1BBswOAP3kg3H1O4hiS+sq4YyAn6ANnA==", + "dev": true, + "dependencies": { + "ansi-html-community": "0.0.8", + "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.8", + "semver": "^6.3.0", + "serve-index": "^1.9.1", + "sockjs": "^0.3.21", + "sockjs-client": "^1.5.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" + }, + "bin": { + "webpack-dev-server": "bin/webpack-dev-server.js" + }, + "engines": { + "node": ">= 6.11.5" + }, + "peerDependencies": { + "webpack": "^4.0.0 || ^5.0.0" + }, + "peerDependenciesMeta": { + "webpack-cli": { + "optional": true + } + } + }, + "node_modules/webpack-dev-server/node_modules/ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmmirror.com/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha512-TIGnTpdo+E3+pCyAluZvtED5p5wCqLdezCyhPZzKPcxvFplEt4i+W7OONCKgeZFT3+y5NZZfOOS/Bdcanm1MYA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/webpack-dev-server/node_modules/anymatch": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/anymatch/-/anymatch-2.0.0.tgz", + "integrity": "sha512-5teOsQWABXHHBFP9y3skS5P3d/WfWXpv3FUpy+LorMrNYaT9pI4oLMQX7jzQ2KklNpGpWHzdCXTDT2Y3XGlZBw==", + "dev": true, + "dependencies": { + "micromatch": "^3.1.4", + "normalize-path": "^2.1.1" + } + }, + "node_modules/webpack-dev-server/node_modules/anymatch/node_modules/normalize-path": { + "version": "2.1.1", + "resolved": "https://registry.npmmirror.com/normalize-path/-/normalize-path-2.1.1.tgz", + "integrity": "sha512-3pKJwH184Xo/lnH6oyP1q2pMd7HcypqqmRs91/6/i2CGtWwIKGCkOOMTm/zXbgTEWHw1uNpNi/igc3ePOYHb6w==", + "dev": true, + "dependencies": { + "remove-trailing-separator": "^1.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/webpack-dev-server/node_modules/binary-extensions": { + "version": "1.13.1", + "resolved": "https://registry.npmmirror.com/binary-extensions/-/binary-extensions-1.13.1.tgz", + "integrity": "sha512-Un7MIEDdUC5gNpcGDV97op1Ywk748MpHcFTHoYs6qnj1Z3j7I53VG3nwZhKzoBZmbdRNnb6WRdFlwl7tSDuZGw==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/webpack-dev-server/node_modules/chokidar": { + "version": "2.1.8", + "resolved": "https://registry.npmmirror.com/chokidar/-/chokidar-2.1.8.tgz", + "integrity": "sha512-ZmZUazfOzf0Nve7duiCKD23PFSCs4JPoYyccjUFF3aQkQadqBhfzhjkwBH2mNOG9cTBwhamM37EIsIkZw3nRgg==", + "deprecated": "Chokidar 2 does not receive security updates since 2019. Upgrade to chokidar 3 with 15x fewer dependencies", + "dev": true, + "dependencies": { + "anymatch": "^2.0.0", + "async-each": "^1.0.1", + "braces": "^2.3.2", + "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" + }, + "optionalDependencies": { + "fsevents": "^1.2.7" + } + }, + "node_modules/webpack-dev-server/node_modules/debug": { + "version": "4.3.4", + "resolved": "https://registry.npmmirror.com/debug/-/debug-4.3.4.tgz", + "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", + "dev": true, + "dependencies": { + "ms": "2.1.2" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/webpack-dev-server/node_modules/emoji-regex": { + "version": "7.0.3", + "resolved": "https://registry.npmmirror.com/emoji-regex/-/emoji-regex-7.0.3.tgz", + "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", + "dev": true + }, + "node_modules/webpack-dev-server/node_modules/fsevents": { + "version": "1.2.13", + "resolved": "https://registry.npmmirror.com/fsevents/-/fsevents-1.2.13.tgz", + "integrity": "sha512-oWb1Z6mkHIskLzEJ/XWX0srkpkTQ7vaopMQkyaEIoq0fmtFVxOthb8cCxeT+p3ynTdkk/RZwbgG4brR5BeWECw==", + "deprecated": "fsevents 1 will break on node v14+ and could be using insecure binaries. Upgrade to fsevents 2.", + "dev": true, + "hasInstallScript": true, + "optional": true, + "os": [ + "darwin" + ], + "dependencies": { + "bindings": "^1.5.0", + "nan": "^2.12.1" + }, + "engines": { + "node": ">= 4.0" + } + }, + "node_modules/webpack-dev-server/node_modules/glob-parent": { + "version": "3.1.0", + "resolved": "https://registry.npmmirror.com/glob-parent/-/glob-parent-3.1.0.tgz", + "integrity": "sha512-E8Ak/2+dZY6fnzlR7+ueWvhsH1SjHr4jjss4YS/h4py44jY9MhK/VFdaZJAWDz6BbL21KeteKxFSFpq8OS5gVA==", + "dev": true, + "dependencies": { + "is-glob": "^3.1.0", + "path-dirname": "^1.0.0" + } + }, + "node_modules/webpack-dev-server/node_modules/glob-parent/node_modules/is-glob": { + "version": "3.1.0", + "resolved": "https://registry.npmmirror.com/is-glob/-/is-glob-3.1.0.tgz", + "integrity": "sha512-UFpDDrPgM6qpnFNI+rh/p3bUaq9hKLZN8bMUWzxmcnZVS3omf4IPK+BrewlnWjO1WmUsMYuSjKh4UJuV4+Lqmw==", + "dev": true, + "dependencies": { + "is-extglob": "^2.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/webpack-dev-server/node_modules/is-absolute-url": { + "version": "3.0.3", + "resolved": "https://registry.npmmirror.com/is-absolute-url/-/is-absolute-url-3.0.3.tgz", + "integrity": "sha512-opmNIX7uFnS96NtPmhWQgQx6/NYFgsUXYMllcfzwWKUMwfo8kku1TvE6hkNcH+Q1ts5cMVrsY7j0bxXQDciu9Q==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/webpack-dev-server/node_modules/is-binary-path": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/is-binary-path/-/is-binary-path-1.0.1.tgz", + "integrity": "sha512-9fRVlXc0uCxEDj1nQzaWONSpbTfx0FmJfzHF7pwlI8DkWGoHBBea4Pg5Ky0ojwwxQmnSifgbKkI06Qv0Ljgj+Q==", + "dev": true, + "dependencies": { + "binary-extensions": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/webpack-dev-server/node_modules/ms": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", + "dev": true + }, + "node_modules/webpack-dev-server/node_modules/readdirp": { + "version": "2.2.1", + "resolved": "https://registry.npmmirror.com/readdirp/-/readdirp-2.2.1.tgz", + "integrity": "sha512-1JU/8q+VgFZyxwrJ+SVIOsh+KywWGpds3NTqikiKpDMZWScmAYyKIgqkO+ARvNWJfXeXR1zxz7aHF4u4CyH6vQ==", + "dev": true, + "dependencies": { + "graceful-fs": "^4.1.11", + "micromatch": "^3.1.10", + "readable-stream": "^2.0.2" + }, + "engines": { + "node": ">=0.10" + } + }, + "node_modules/webpack-dev-server/node_modules/semver": { + "version": "6.3.0", + "resolved": "https://registry.npmmirror.com/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/webpack-dev-server/node_modules/string-width": { + "version": "3.1.0", + "resolved": "https://registry.npmmirror.com/string-width/-/string-width-3.1.0.tgz", + "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", + "dev": true, + "dependencies": { + "emoji-regex": "^7.0.1", + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^5.1.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/webpack-dev-server/node_modules/string-width/node_modules/ansi-regex": { + "version": "4.1.1", + "resolved": "https://registry.npmmirror.com/ansi-regex/-/ansi-regex-4.1.1.tgz", + "integrity": "sha512-ILlv4k/3f6vfQ4OoP2AGvirOktlQ98ZEL1k9FaQjxa3L1abBgbuTDAdPOpvbGncC0BTVQrl+OM8xZGK6tWXt7g==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/webpack-dev-server/node_modules/string-width/node_modules/strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmmirror.com/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "dev": true, + "dependencies": { + "ansi-regex": "^4.1.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/webpack-dev-server/node_modules/strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmmirror.com/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha512-VhumSSbBqDTP8p2ZLKj40UjBCV4+v8bUSEpUb4KjRgWk9pbqGF4REFj6KEagidb2f/M6AzC0EmFyDNGaw9OCzg==", + "dev": true, + "dependencies": { + "ansi-regex": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/webpack-dev-server/node_modules/supports-color": { + "version": "6.1.0", + "resolved": "https://registry.npmmirror.com/supports-color/-/supports-color-6.1.0.tgz", + "integrity": "sha512-qe1jfm1Mg7Nq/NSh6XE24gPXROEVsWHxC1LIx//XNlD9iw7YZQGjZNjYN7xGaEG6iKdA8EtNFW6R0gjnVXp+wQ==", + "dev": true, + "dependencies": { + "has-flag": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/webpack-dev-server/node_modules/yargs": { + "version": "13.3.2", + "resolved": "https://registry.npmmirror.com/yargs/-/yargs-13.3.2.tgz", + "integrity": "sha512-AX3Zw5iPruN5ie6xGRIDgqkT+ZhnRlZMLMHAs8tg7nRruy2Nb+i5o9bwghAogtM08q1dpr2LVoS8KSTMYpWXUw==", + "dev": true, + "dependencies": { + "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" + } + }, + "node_modules/webpack-dev-server/node_modules/yargs-parser": { + "version": "13.1.2", + "resolved": "https://registry.npmmirror.com/yargs-parser/-/yargs-parser-13.1.2.tgz", + "integrity": "sha512-3lbsNRf/j+A4QuSZfDRA7HRSfWrzO0YjqTJd5kjAq37Zep1CEgaYmrH9Q3GwPiB9cHyd1Y1UwggGhJGoxipbzg==", + "dev": true, + "dependencies": { + "camelcase": "^5.0.0", + "decamelize": "^1.2.0" + } + }, + "node_modules/webpack-log": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/webpack-log/-/webpack-log-2.0.0.tgz", + "integrity": "sha512-cX8G2vR/85UYG59FgkoMamwHUIkSSlV3bBMRsbxVXVUk2j6NleCKjQ/WE9eYg9WY4w25O9w8wKP4rzNZFmUcUg==", + "dev": true, + "dependencies": { + "ansi-colors": "^3.0.0", + "uuid": "^3.3.2" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/webpack-merge": { + "version": "4.2.2", + "resolved": "https://registry.npmmirror.com/webpack-merge/-/webpack-merge-4.2.2.tgz", + "integrity": "sha512-TUE1UGoTX2Cd42j3krGYqObZbOD+xF7u28WB7tfUordytSjbWTIjK/8V0amkBfTYN4/pB/GIDlJZZ657BGG19g==", + "dev": true, + "dependencies": { + "lodash": "^4.17.15" + } + }, + "node_modules/webpack-sources": { + "version": "1.4.3", + "resolved": "https://registry.npmmirror.com/webpack-sources/-/webpack-sources-1.4.3.tgz", + "integrity": "sha512-lgTS3Xhv1lCOKo7SA5TjKXMjpSM4sBjNV5+q2bqesbSPs5FjGmU6jjtBSkX9b4qW87vDIsCIlUPOEhbZrMdjeQ==", + "dependencies": { + "source-list-map": "^2.0.0", + "source-map": "~0.6.1" + } + }, + "node_modules/webpack-sources/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmmirror.com/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/webpack-virtual-modules": { + "version": "0.3.2", + "resolved": "https://registry.npmmirror.com/webpack-virtual-modules/-/webpack-virtual-modules-0.3.2.tgz", + "integrity": "sha512-RXQXioY6MhzM4CNQwmBwKXYgBs6ulaiQ8bkNQEl2J6Z+V+s7lgl/wGvaI/I0dLnYKB8cKsxQc17QOAVIphPLDw==", + "dev": true, + "dependencies": { + "debug": "^3.0.0" + } + }, + "node_modules/webpack-virtual-modules/node_modules/debug": { + "version": "3.2.7", + "resolved": "https://registry.npmmirror.com/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dev": true, + "dependencies": { + "ms": "^2.1.1" + } + }, + "node_modules/webpack-virtual-modules/node_modules/ms": { + "version": "2.1.3", + "resolved": "https://registry.npmmirror.com/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", + "dev": true + }, + "node_modules/websocket-driver": { + "version": "0.7.4", + "resolved": "https://registry.npmmirror.com/websocket-driver/-/websocket-driver-0.7.4.tgz", + "integrity": "sha512-b17KeDIQVjvb0ssuSDF2cYXSg2iztliJ4B9WdsuB6J952qCPKmnVq4DyW5motImXHDC1cBT/1UezrJVsKw5zjg==", + "dev": true, + "dependencies": { + "http-parser-js": ">=0.5.1", + "safe-buffer": ">=5.1.0", + "websocket-extensions": ">=0.1.1" + }, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/websocket-extensions": { + "version": "0.1.4", + "resolved": "https://registry.npmmirror.com/websocket-extensions/-/websocket-extensions-0.1.4.tgz", + "integrity": "sha512-OqedPIGOfsDlo31UNwYbCFMSaO9m9G/0faIHj5/dZFDMFqPTcx6UwqyOy3COEaEOg/9VsGIpdqn62W5KhoKSpg==", + "dev": true, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/whatwg-url": { + "version": "8.7.0", + "resolved": "https://registry.npmmirror.com/whatwg-url/-/whatwg-url-8.7.0.tgz", + "integrity": "sha512-gAojqb/m9Q8a5IV96E3fHJM70AzCkgt4uXYX2O7EmuyOnLrViCQlsEBmF9UQIu3/aeAIp2U17rtbpZWNntQqdg==", + "dev": true, + "dependencies": { + "lodash": "^4.7.0", + "tr46": "^2.1.0", + "webidl-conversions": "^6.1.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/which": { + "version": "1.3.1", + "resolved": "https://registry.npmmirror.com/which/-/which-1.3.1.tgz", + "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==", + "dev": true, + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "which": "bin/which" + } + }, + "node_modules/which-boxed-primitive": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/which-boxed-primitive/-/which-boxed-primitive-1.0.2.tgz", + "integrity": "sha512-bwZdv0AKLpplFY2KZRX6TvyuN7ojjr7lwkg6ml0roIy9YeuSr7JS372qlNW18UQYzgYK9ziGcerWqZOmEn9VNg==", + "dependencies": { + "is-bigint": "^1.0.1", + "is-boolean-object": "^1.1.0", + "is-number-object": "^1.0.4", + "is-string": "^1.0.5", + "is-symbol": "^1.0.3" + } + }, + "node_modules/which-module": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/which-module/-/which-module-2.0.0.tgz", + "integrity": "sha512-B+enWhmw6cjfVC7kS8Pj9pCrKSc5txArRyaYGe088shv/FGWH+0Rjx/xPgtsWfsUtS27FkP697E4DDhgrgoc0Q==", + "dev": true + }, + "node_modules/wide-align": { + "version": "1.1.5", + "resolved": "https://registry.npmmirror.com/wide-align/-/wide-align-1.1.5.tgz", + "integrity": "sha512-eDMORYaPNZ4sQIuuYPDHdQvf4gyCF9rEEV/yPxGfwPkRodwEgiMUUXTx/dex+Me0wxx53S+NgUHaP7y3MGlDmg==", + "dev": true, + "dependencies": { + "string-width": "^1.0.2 || 2 || 3 || 4" + } + }, + "node_modules/word-wrap": { + "version": "1.2.3", + "resolved": "https://registry.npmmirror.com/word-wrap/-/word-wrap-1.2.3.tgz", + "integrity": "sha512-Hz/mrNwitNRh/HUAtM/VT/5VH+ygD6DV7mYKZAtHOrbs8U7lvPS6xf7EJKMF0uW1KJCl0H701g3ZGus+muE5vQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/wordwrap": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/wordwrap/-/wordwrap-1.0.0.tgz", + "integrity": "sha512-gvVzJFlPycKc5dZN4yPkP8w7Dc37BtP1yczEneOb4uq34pXZcvrtRTmWV8W+Ume+XCxKgbjM+nevkyFPMybd4Q==", + "dev": true + }, + "node_modules/worker-farm": { + "version": "1.7.0", + "resolved": "https://registry.npmmirror.com/worker-farm/-/worker-farm-1.7.0.tgz", + "integrity": "sha512-rvw3QTZc8lAxyVrqcSGVm5yP/IJ2UcB3U0graE3LCFoZ0Yn2x4EoVSqJKdB/T5M+FLcRPjz4TDacRf3OCfNUzw==", + "dependencies": { + "errno": "~0.1.7" + } + }, + "node_modules/worker-loader": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/worker-loader/-/worker-loader-2.0.0.tgz", + "integrity": "sha512-tnvNp4K3KQOpfRnD20m8xltE3eWh89Ye+5oj7wXEEHKac1P4oZ6p9oTj8/8ExqoSBnk9nu5Pr4nKfQ1hn2APJw==", + "dependencies": { + "loader-utils": "^1.0.0", + "schema-utils": "^0.4.0" + }, + "engines": { + "node": ">= 6.9.0 || >= 8.9.0" + }, + "peerDependencies": { + "webpack": "^3.0.0 || ^4.0.0-alpha.0 || ^4.0.0" + } + }, + "node_modules/worker-loader/node_modules/schema-utils": { + "version": "0.4.7", + "resolved": "https://registry.npmmirror.com/schema-utils/-/schema-utils-0.4.7.tgz", + "integrity": "sha512-v/iwU6wvwGK8HbU9yi3/nhGzP0yGSuhQMzL6ySiec1FSrZZDkhm4noOSWzrNFo/jEc+SJY6jRTwuwbSXJPDUnQ==", + "dependencies": { + "ajv": "^6.1.0", + "ajv-keywords": "^3.1.0" + }, + "engines": { + "node": ">= 4" + } + }, + "node_modules/wrap-ansi": { + "version": "7.0.0", + "resolved": "https://registry.npmmirror.com/wrap-ansi/-/wrap-ansi-7.0.0.tgz", + "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/wrap-ansi/node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmmirror.com/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/wrap-ansi/node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmmirror.com/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/wrap-ansi/node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmmirror.com/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/wrap-ansi/node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmmirror.com/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "node_modules/wrap-ansi/node_modules/is-fullwidth-code-point": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", + "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/wrap-ansi/node_modules/string-width": { + "version": "4.2.3", + "resolved": "https://registry.npmmirror.com/string-width/-/string-width-4.2.3.tgz", + "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", + "dev": true, + "dependencies": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/wrap-ansi/node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmmirror.com/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/wrappy": { + "version": "1.0.2", + "resolved": "https://registry.npmmirror.com/wrappy/-/wrappy-1.0.2.tgz", + "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==" + }, + "node_modules/write": { + "version": "0.2.1", + "resolved": "https://registry.npmmirror.com/write/-/write-0.2.1.tgz", + "integrity": "sha512-CJ17OoULEKXpA5pef3qLj5AxTJ6mSt7g84he2WIskKwqFO4T97d5V7Tadl0DYDk7qyUOQD5WlUlOMChaYrhxeA==", + "dev": true, + "optional": true, + "dependencies": { + "mkdirp": "^0.5.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/write-file-atomic": { + "version": "2.4.3", + "resolved": "https://registry.npmmirror.com/write-file-atomic/-/write-file-atomic-2.4.3.tgz", + "integrity": "sha512-GaETH5wwsX+GcnzhPgKcKjJ6M2Cq3/iZp1WyY/X1CSqrW+jVNM9Y7D8EC2sM4ZG/V8wZlSniJnCKWPmBYAucRQ==", + "dev": true, + "dependencies": { + "graceful-fs": "^4.1.11", + "imurmurhash": "^0.1.4", + "signal-exit": "^3.0.2" + } + }, + "node_modules/write-json-file": { + "version": "4.3.0", + "resolved": "https://registry.npmmirror.com/write-json-file/-/write-json-file-4.3.0.tgz", + "integrity": "sha512-PxiShnxf0IlnQuMYOPPhPkhExoCQuTUNPOa/2JWCYTmBquU9njyyDuwRKN26IZBlp4yn1nt+Agh2HOOBl+55HQ==", + "dev": true, + "dependencies": { + "detect-indent": "^6.0.0", + "graceful-fs": "^4.1.15", + "is-plain-obj": "^2.0.0", + "make-dir": "^3.0.0", + "sort-keys": "^4.0.0", + "write-file-atomic": "^3.0.0" + }, + "engines": { + "node": ">=8.3" + } + }, + "node_modules/write-json-file/node_modules/detect-indent": { + "version": "6.1.0", + "resolved": "https://registry.npmmirror.com/detect-indent/-/detect-indent-6.1.0.tgz", + "integrity": "sha512-reYkTUJAZb9gUuZ2RvVCNhVHdg62RHnJ7WJl8ftMi4diZ6NWlciOzQN88pUhSELEwflJht4oQDv0F0BMlwaYtA==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/write-json-file/node_modules/is-plain-obj": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/is-plain-obj/-/is-plain-obj-2.1.0.tgz", + "integrity": "sha512-YWnfyRwxL/+SsrWYfOpUtz5b3YD+nyfkHvjbcanzk8zgyO4ASD67uVMRt8k5bM4lLMDnXfriRhOpemw+NfT1eA==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/write-json-file/node_modules/make-dir": { + "version": "3.1.0", + "resolved": "https://registry.npmmirror.com/make-dir/-/make-dir-3.1.0.tgz", + "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==", + "dev": true, + "dependencies": { + "semver": "^6.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/write-json-file/node_modules/semver": { + "version": "6.3.0", + "resolved": "https://registry.npmmirror.com/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/write-json-file/node_modules/sort-keys": { + "version": "4.2.0", + "resolved": "https://registry.npmmirror.com/sort-keys/-/sort-keys-4.2.0.tgz", + "integrity": "sha512-aUYIEU/UviqPgc8mHR6IW1EGxkAXpeRETYcrzg8cLAvUPZcpAlleSXHV2mY7G12GphSH6Gzv+4MMVSSkbdteHg==", + "dev": true, + "dependencies": { + "is-plain-obj": "^2.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/write-json-file/node_modules/write-file-atomic": { + "version": "3.0.3", + "resolved": "https://registry.npmmirror.com/write-file-atomic/-/write-file-atomic-3.0.3.tgz", + "integrity": "sha512-AvHcyZ5JnSfq3ioSyjrBkH9yW4m7Ayk8/9My/DD9onKeu/94fwrMocemO2QAJFAlnnDN+ZDS+ZjAR5ua1/PV/Q==", + "dev": true, + "dependencies": { + "imurmurhash": "^0.1.4", + "is-typedarray": "^1.0.0", + "signal-exit": "^3.0.2", + "typedarray-to-buffer": "^3.1.5" + } + }, + "node_modules/write-pkg": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/write-pkg/-/write-pkg-4.0.0.tgz", + "integrity": "sha512-v2UQ+50TNf2rNHJ8NyWttfm/EJUBWMJcx6ZTYZr6Qp52uuegWw/lBkCtCbnYZEmPRNL61m+u67dAmGxo+HTULA==", + "dev": true, + "dependencies": { + "sort-keys": "^2.0.0", + "type-fest": "^0.4.1", + "write-json-file": "^3.2.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/write-pkg/node_modules/sort-keys": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/sort-keys/-/sort-keys-2.0.0.tgz", + "integrity": "sha512-/dPCrG1s3ePpWm6yBbxZq5Be1dXGLyLn9Z791chDC3NFrpkVbWGzkBwPN1knaciexFXgRJ7hzdnwZ4stHSDmjg==", + "dev": true, + "dependencies": { + "is-plain-obj": "^1.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/write-pkg/node_modules/type-fest": { + "version": "0.4.1", + "resolved": "https://registry.npmmirror.com/type-fest/-/type-fest-0.4.1.tgz", + "integrity": "sha512-IwzA/LSfD2vC1/YDYMv/zHP4rDF1usCwllsDpbolT3D4fUepIO7f9K70jjmUewU/LmGUKJcwcVtDCpnKk4BPMw==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/write-pkg/node_modules/write-json-file": { + "version": "3.2.0", + "resolved": "https://registry.npmmirror.com/write-json-file/-/write-json-file-3.2.0.tgz", + "integrity": "sha512-3xZqT7Byc2uORAatYiP3DHUUAVEkNOswEWNs9H5KXiicRTvzYzYqKjYc4G7p+8pltvAw641lVByKVtMpf+4sYQ==", + "dev": true, + "dependencies": { + "detect-indent": "^5.0.0", + "graceful-fs": "^4.1.15", + "make-dir": "^2.1.0", + "pify": "^4.0.1", + "sort-keys": "^2.0.0", + "write-file-atomic": "^2.4.2" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/ws": { + "version": "6.2.2", + "resolved": "https://registry.npmmirror.com/ws/-/ws-6.2.2.tgz", + "integrity": "sha512-zmhltoSR8u1cnDsD43TX59mzoMZsLKqUweyYBAIvTngR3shc0W6aOZylZmq/7hqyVxPdi+5Ud2QInblgyE72fw==", + "dev": true, + "dependencies": { + "async-limiter": "~1.0.0" + } + }, + "node_modules/xss": { + "version": "1.0.11", + "resolved": "https://registry.npmmirror.com/xss/-/xss-1.0.11.tgz", + "integrity": "sha512-EimjrjThZeK2MO7WKR9mN5ZC1CSqivSl55wvUK5EtU6acf0rzEE1pN+9ZDrFXJ82BRp3JL38pPE6S4o/rpp1zQ==", + "dependencies": { + "commander": "^2.20.3", + "cssfilter": "0.0.10" + }, + "bin": { + "xss": "bin/xss" + }, + "engines": { + "node": ">= 0.10.0" + } + }, + "node_modules/xtend": { + "version": "4.0.2", + "resolved": "https://registry.npmmirror.com/xtend/-/xtend-4.0.2.tgz", + "integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==", + "engines": { + "node": ">=0.4" + } + }, + "node_modules/y18n": { + "version": "4.0.3", + "resolved": "https://registry.npmmirror.com/y18n/-/y18n-4.0.3.tgz", + "integrity": "sha512-JKhqTOwSrqNA1NY5lSztJ1GrBiUodLMmIZuLiDaMRJ+itFd+ABVE8XBjOvIWL+rSqNDC74LCSFmlb/U4UZ4hJQ==" + }, + "node_modules/yallist": { + "version": "3.1.1", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-3.1.1.tgz", + "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==" + }, + "node_modules/yaml": { + "version": "1.10.2", + "resolved": "https://registry.npmmirror.com/yaml/-/yaml-1.10.2.tgz", + "integrity": "sha512-r3vXyErRCYJ7wg28yvBY5VSoAF8ZvlcW9/BwUzEtUsjvX/DKs24dIkuwjtuprwJJHsbyUbLApepYTR1BN4uHrg==", + "dev": true, + "engines": { + "node": ">= 6" + } + }, + "node_modules/yargs": { + "version": "16.2.0", + "resolved": "https://registry.npmmirror.com/yargs/-/yargs-16.2.0.tgz", + "integrity": "sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw==", + "dev": true, + "dependencies": { + "cliui": "^7.0.2", + "escalade": "^3.1.1", + "get-caller-file": "^2.0.5", + "require-directory": "^2.1.1", + "string-width": "^4.2.0", + "y18n": "^5.0.5", + "yargs-parser": "^20.2.2" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/yargs-parser": { + "version": "20.2.9", + "resolved": "https://registry.npmmirror.com/yargs-parser/-/yargs-parser-20.2.9.tgz", + "integrity": "sha512-y11nGElTIV+CT3Zv9t7VKl+Q3hTQoT9a1Qzezhhl6Rp21gJ/IVTW7Z3y9EWXhuUBC2Shnf+DX0antecpAwSP8w==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/yargs/node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmmirror.com/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/yargs/node_modules/cliui": { + "version": "7.0.4", + "resolved": "https://registry.npmmirror.com/cliui/-/cliui-7.0.4.tgz", + "integrity": "sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==", + "dev": true, + "dependencies": { + "string-width": "^4.2.0", + "strip-ansi": "^6.0.0", + "wrap-ansi": "^7.0.0" + } + }, + "node_modules/yargs/node_modules/is-fullwidth-code-point": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", + "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/yargs/node_modules/string-width": { + "version": "4.2.3", + "resolved": "https://registry.npmmirror.com/string-width/-/string-width-4.2.3.tgz", + "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", + "dev": true, + "dependencies": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/yargs/node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmmirror.com/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/yargs/node_modules/y18n": { + "version": "5.0.8", + "resolved": "https://registry.npmmirror.com/y18n/-/y18n-5.0.8.tgz", + "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/yorkie": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/yorkie/-/yorkie-2.0.0.tgz", + "integrity": "sha512-jcKpkthap6x63MB4TxwCyuIGkV0oYP/YRyuQU5UO0Yz/E/ZAu+653/uov+phdmO54n6BcvFRyyt0RRrWdN2mpw==", + "dev": true, + "hasInstallScript": true, + "dependencies": { + "execa": "^0.8.0", + "is-ci": "^1.0.10", + "normalize-path": "^1.0.0", + "strip-indent": "^2.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/yorkie/node_modules/cross-spawn": { + "version": "5.1.0", + "resolved": "https://registry.npmmirror.com/cross-spawn/-/cross-spawn-5.1.0.tgz", + "integrity": "sha512-pTgQJ5KC0d2hcY8eyL1IzlBPYjTkyH72XRZPnLyKus2mBfNjQs3klqbJU2VILqZryAZUt9JOb3h/mWMy23/f5A==", + "dev": true, + "dependencies": { + "lru-cache": "^4.0.1", + "shebang-command": "^1.2.0", + "which": "^1.2.9" + } + }, + "node_modules/yorkie/node_modules/execa": { + "version": "0.8.0", + "resolved": "https://registry.npmmirror.com/execa/-/execa-0.8.0.tgz", + "integrity": "sha512-zDWS+Rb1E8BlqqhALSt9kUhss8Qq4nN3iof3gsOdyINksElaPyNBtKUMTR62qhvgVWR0CqCX7sdnKe4MnUbFEA==", + "dev": true, + "dependencies": { + "cross-spawn": "^5.0.1", + "get-stream": "^3.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" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/yorkie/node_modules/get-stream": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/get-stream/-/get-stream-3.0.0.tgz", + "integrity": "sha512-GlhdIUuVakc8SJ6kK0zAFbiGzRFzNnY4jUuEbV9UROo4Y+0Ny4fjvcZFVTeDA4odpFyOQzaw6hXukJSq/f28sQ==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/yorkie/node_modules/lru-cache": { + "version": "4.1.5", + "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-4.1.5.tgz", + "integrity": "sha512-sWZlbEP2OsHNkXrMl5GYk/jKk70MBng6UU4YI/qGDYbgf6YbP4EvmqISbXCoJiRKs+1bSpFHVgQxvJ17F2li5g==", + "dev": true, + "dependencies": { + "pseudomap": "^1.0.2", + "yallist": "^2.1.2" + } + }, + "node_modules/yorkie/node_modules/normalize-path": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/normalize-path/-/normalize-path-1.0.0.tgz", + "integrity": "sha512-7WyT0w8jhpDStXRq5836AMmihQwq2nrUVQrgjvUo/p/NZf9uy/MeJ246lBJVmWuYXMlJuG9BNZHF0hWjfTbQUA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/yorkie/node_modules/yallist": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-2.1.2.tgz", + "integrity": "sha512-ncTzHV7NvsQZkYe1DW7cbDLm0YpzHmZF5r/iyP3ZnQtMiJ+pjzisCiMNI+Sj+xQF5pXhSHxSB3uDbsBTzY/c2A==", + "dev": true + }, + "node_modules/yup": { + "version": "0.27.0", + "resolved": "https://registry.npmmirror.com/yup/-/yup-0.27.0.tgz", + "integrity": "sha512-v1yFnE4+u9za42gG/b/081E7uNW9mUj3qtkmelLbW5YPROZzSH/KUUyJu9Wt8vxFJcT9otL/eZopS0YK1L5yPQ==", + "dev": true, + "dependencies": { + "@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" + } + }, + "node_modules/yup/node_modules/toposort": { + "version": "2.0.2", + "resolved": "https://registry.npmmirror.com/toposort/-/toposort-2.0.2.tgz", + "integrity": "sha512-0a5EOkAUp8D4moMi2W8ZF8jcga7BgZd91O/yabJCFY8az+XSzeGyTKs0Aoo897iV1Nj6guFq8orWDS96z91oGg==", + "dev": true + }, + "node_modules/zip-stream": { + "version": "2.1.3", + "resolved": "https://registry.npmmirror.com/zip-stream/-/zip-stream-2.1.3.tgz", + "integrity": "sha512-EkXc2JGcKhO5N5aZ7TmuNo45budRaFGHOmz24wtJR7znbNqDPmdZtUauKX6et8KAVseAMBOyWJqEpXcHTBsh7Q==", + "dev": true, + "dependencies": { + "archiver-utils": "^2.1.0", + "compress-commons": "^2.1.1", + "readable-stream": "^3.4.0" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/zip-stream/node_modules/readable-stream": { + "version": "3.6.0", + "resolved": "https://registry.npmmirror.com/readable-stream/-/readable-stream-3.6.0.tgz", + "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", + "dev": true, + "dependencies": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/zrender": { + "version": "4.3.2", + "resolved": "https://registry.npmmirror.com/zrender/-/zrender-4.3.2.tgz", + "integrity": "sha512-bIusJLS8c4DkIcdiK+s13HiQ/zjQQVgpNohtd8d94Y2DnJqgM1yjh/jpDb8DoL6hd7r8Awagw8e3qK/oLaWr3g==" + } + }, "dependencies": { "@ampproject/remapping": { "version": "2.1.2", @@ -906,7 +28744,6 @@ "version": "7.17.9", "resolved": "https://registry.npmmirror.com/@babel/runtime/-/runtime-7.17.9.tgz", "integrity": "sha512-lSiBBvodq29uShpWGNbgFdKYNiFDo5/HIYsaCEY9ff4sb10x9jizo2+pRrSyF4jKZCXqgzuqBOQKbUm90gQwJg==", - "dev": true, "requires": { "regenerator-runtime": "^0.13.4" } @@ -977,6 +28814,43 @@ "to-fast-properties": "^2.0.0" } }, + "@codemirror/lang-json": { + "version": "6.0.1", + "resolved": "https://registry.npmmirror.com/@codemirror/lang-json/-/lang-json-6.0.1.tgz", + "integrity": "sha512-+T1flHdgpqDDlJZ2Lkil/rLiRy684WMLc74xUnjJH48GQdfJo/pudlTRreZmKwzP8/tGdKf83wlbAdOCzlJOGQ==", + "requires": { + "@codemirror/language": "^6.0.0", + "@lezer/json": "^1.0.0" + } + }, + "@codemirror/language": { + "version": "6.8.0", + "resolved": "https://registry.npmmirror.com/@codemirror/language/-/language-6.8.0.tgz", + "integrity": "sha512-r1paAyWOZkfY0RaYEZj3Kul+MiQTEbDvYqf8gPGaRvNneHXCmfSaAVFjwRUPlgxS8yflMxw2CTu6uCMp8R8A2g==", + "requires": { + "@codemirror/state": "^6.0.0", + "@codemirror/view": "^6.0.0", + "@lezer/common": "^1.0.0", + "@lezer/highlight": "^1.0.0", + "@lezer/lr": "^1.0.0", + "style-mod": "^4.0.0" + } + }, + "@codemirror/state": { + "version": "6.2.1", + "resolved": "https://registry.npmmirror.com/@codemirror/state/-/state-6.2.1.tgz", + "integrity": "sha512-RupHSZ8+OjNT38zU9fKH2sv+Dnlr8Eb8sl4NOnnqz95mCFTZUaiRP8Xv5MeeaG0px2b8Bnfe7YGwCV3nsBhbuw==" + }, + "@codemirror/view": { + "version": "6.14.0", + "resolved": "https://registry.npmmirror.com/@codemirror/view/-/view-6.14.0.tgz", + "integrity": "sha512-I263FPs4In42MNmrdwN2DfmYPFMVMXgT7o/mxdGp4jv5LPs8i0FOxzmxF5yeeQdYSTztb2ZhmPIu0ahveInVTg==", + "requires": { + "@codemirror/state": "^6.1.4", + "style-mod": "^4.0.0", + "w3c-keyname": "^2.2.4" + } + }, "@gar/promisify": { "version": "1.1.3", "resolved": "https://registry.npmmirror.com/@gar/promisify/-/promisify-1.1.3.tgz", @@ -4123,6 +31997,36 @@ } } }, + "@lezer/common": { + "version": "1.0.3", + "resolved": "https://registry.npmmirror.com/@lezer/common/-/common-1.0.3.tgz", + "integrity": "sha512-JH4wAXCgUOcCGNekQPLhVeUtIqjH0yPBs7vvUdSjyQama9618IOKFJwkv2kcqdhF0my8hQEgCTEJU0GIgnahvA==" + }, + "@lezer/highlight": { + "version": "1.1.6", + "resolved": "https://registry.npmmirror.com/@lezer/highlight/-/highlight-1.1.6.tgz", + "integrity": "sha512-cmSJYa2us+r3SePpRCjN5ymCqCPv+zyXmDl0ciWtVaNiORT/MxM7ZgOMQZADD0o51qOaOg24qc/zBViOIwAjJg==", + "requires": { + "@lezer/common": "^1.0.0" + } + }, + "@lezer/json": { + "version": "1.0.1", + "resolved": "https://registry.npmmirror.com/@lezer/json/-/json-1.0.1.tgz", + "integrity": "sha512-nkVC27qiEZEjySbi6gQRuMwa2sDu2PtfjSgz0A4QF81QyRGm3kb2YRzLcOPcTEtmcwvrX/cej7mlhbwViA4WJw==", + "requires": { + "@lezer/highlight": "^1.0.0", + "@lezer/lr": "^1.0.0" + } + }, + "@lezer/lr": { + "version": "1.3.9", + "resolved": "https://registry.npmmirror.com/@lezer/lr/-/lr-1.3.9.tgz", + "integrity": "sha512-XPz6dzuTHlnsbA5M2DZgjflNQ+9Hi5Swhic0RULdp3oOs3rh6bqGZolosVqN/fQIT8uNiepzINJDnS39oweTHQ==", + "requires": { + "@lezer/common": "^1.0.0" + } + }, "@mrmlnc/readdir-enhanced": { "version": "2.2.1", "resolved": "https://registry.npmmirror.com/@mrmlnc/readdir-enhanced/-/readdir-enhanced-2.2.1.tgz", @@ -4534,7 +32438,8 @@ "version": "1.0.4", "resolved": "https://registry.npmmirror.com/@octokit/plugin-request-log/-/plugin-request-log-1.0.4.tgz", "integrity": "sha512-mLUsMkgP7K/cnFEw07kWqXGF5LKrOkD+lhCrKvPHXWDywAwuDUeDwWBpc69XK3pNX0uKiVt8g5z96PJ6z9xCFA==", - "dev": true + "dev": true, + "requires": {} }, "@octokit/plugin-rest-endpoint-methods": { "version": "5.13.0", @@ -4600,6 +32505,21 @@ "@octokit/openapi-types": "^11.2.0" } }, + "@riophae/vue-treeselect": { + "version": "0.4.0", + "resolved": "https://registry.npmmirror.com/@riophae/vue-treeselect/-/vue-treeselect-0.4.0.tgz", + "integrity": "sha512-J4atYmBqXQmiPFK/0B5sXKjtnGc21mBJEiyKIDZwk0Q9XuynVFX6IJ4EpaLmUgL5Tve7HAS7wkiGGSti6Uaxcg==", + "requires": { + "@babel/runtime": "^7.3.1", + "babel-helper-vue-jsx-merge-props": "^2.0.3", + "easings-css": "^1.0.0", + "fuzzysearch": "^1.0.3", + "is-promise": "^2.1.0", + "lodash": "^4.0.0", + "material-colors": "^1.2.6", + "watch-size": "^2.0.0" + } + }, "@samverschueren/stream-to-observable": { "version": "0.3.1", "resolved": "https://registry.npmmirror.com/@samverschueren/stream-to-observable/-/stream-to-observable-0.3.1.tgz", @@ -5254,7 +33174,8 @@ "version": "1.1.2", "resolved": "https://registry.npmmirror.com/@vue/preload-webpack-plugin/-/preload-webpack-plugin-1.1.2.tgz", "integrity": "sha512-LIZMuJk38pk9U9Ur4YzHjlIyMuxPlACdBIHH9/nGYVTsaGKOSnSuELiE8vS9wa+dJpIYspYUOqk+L1Q4pgHQHQ==", - "dev": true + "dev": true, + "requires": {} }, "@vue/web-component-wrapper": { "version": "1.3.0", @@ -5435,16 +33356,6 @@ "integrity": "sha512-GpSwvyXOcOOlV70vbnzjj4fW5xW/FdUF6nQEt1ENy7m4ZCczi1+/buVUPAqmGfqznsORNFzUMjctTIp8a9tuCQ==", "dev": true }, - "JSONStream": { - "version": "1.3.5", - "resolved": "https://registry.npmmirror.com/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" - } - }, "abbrev": { "version": "1.1.1", "resolved": "https://registry.npmmirror.com/abbrev/-/abbrev-1.1.1.tgz", @@ -5587,12 +33498,14 @@ "ajv-errors": { "version": "1.0.1", "resolved": "https://registry.npmmirror.com/ajv-errors/-/ajv-errors-1.0.1.tgz", - "integrity": "sha512-DCRfO/4nQ+89p/RK43i8Ezd41EqdGIU4ld7nGF8OQ14oc/we5rEntLCUa7+jrn3nn83BosfwZA0wb4pon2o8iQ==" + "integrity": "sha512-DCRfO/4nQ+89p/RK43i8Ezd41EqdGIU4ld7nGF8OQ14oc/we5rEntLCUa7+jrn3nn83BosfwZA0wb4pon2o8iQ==", + "requires": {} }, "ajv-keywords": { "version": "3.5.2", "resolved": "https://registry.npmmirror.com/ajv-keywords/-/ajv-keywords-3.5.2.tgz", - "integrity": "sha512-5p6WTN0DdTGVQk6VjcEju19IgaHudalcfabD7yhDGeA6bcQnmL+CpveLJq/3hvfwd1aof6L386Ougkx6RfyMIQ==" + "integrity": "sha512-5p6WTN0DdTGVQk6VjcEju19IgaHudalcfabD7yhDGeA6bcQnmL+CpveLJq/3hvfwd1aof6L386Ougkx6RfyMIQ==", + "requires": {} }, "alphanum-sort": { "version": "1.0.2", @@ -5756,12 +33669,6 @@ "integrity": "sha512-THtfYS6KtME/yIAhKjZ2ul7XI96lQGHRputJQHO80LAWQnuGP4iCIN8vdMRboGbIEYBwU33q8Tch1os2+X0kMg==", "dev": true }, - "array-find-index": { - "version": "1.0.2", - "resolved": "https://registry.npmmirror.com/array-find-index/-/array-find-index-1.0.2.tgz", - "integrity": "sha512-M1HQyIXcBGtVywBt8WVdim+lrNaK7VHp99Qt5pSNziXznKHViIBbXWtfRTpEFpF/c4FdfxNAsCCwPp5phBYJtw==", - "dev": true - }, "array-flatten": { "version": "1.1.1", "resolved": "https://registry.npmmirror.com/array-flatten/-/array-flatten-1.1.1.tgz", @@ -5911,7 +33818,8 @@ "async-each": { "version": "1.0.3", "resolved": "https://registry.npmmirror.com/async-each/-/async-each-1.0.3.tgz", - "integrity": "sha512-z/WhQ5FPySLdvREByI2vZiTWwCnF0moMJ1hK9YQwDTHKh6I7/uSckMetoRGb5UBZPC1z0jlw+n/XCgjeH7y1AQ==" + "integrity": "sha512-z/WhQ5FPySLdvREByI2vZiTWwCnF0moMJ1hK9YQwDTHKh6I7/uSckMetoRGb5UBZPC1z0jlw+n/XCgjeH7y1AQ==", + "devOptional": true }, "async-foreach": { "version": "0.1.3", @@ -6063,6 +33971,11 @@ "resolve": "^1.12.0" } }, + "babel-helper-vue-jsx-merge-props": { + "version": "2.0.3", + "resolved": "https://registry.npmmirror.com/babel-helper-vue-jsx-merge-props/-/babel-helper-vue-jsx-merge-props-2.0.3.tgz", + "integrity": "sha512-gsLiKK7Qrb7zYJNgiXKpXblxbV5ffSwR0f5whkPAaBAR4fhi6bwRZxX9wBlIc5M/v8CCkXUbXZL4N/nSE97cqg==" + }, "babel-loader": { "version": "8.2.5", "resolved": "https://registry.npmmirror.com/babel-loader/-/babel-loader-8.2.5.tgz", @@ -6234,6 +34147,11 @@ "@babel/helper-define-polyfill-provider": "^0.3.1" } }, + "babel-plugin-transform-remove-console": { + "version": "6.9.4", + "resolved": "https://registry.npmmirror.com/babel-plugin-transform-remove-console/-/babel-plugin-transform-remove-console-6.9.4.tgz", + "integrity": "sha512-88blrUrMX3SPiGkT1GnvVY8E/7A+k6oj3MNvUtTIxJflFzXTw1bHkuJ/y039ouhFMp2prRn5cQGzokViYi1dsg==" + }, "babel-polyfill": { "version": "6.26.0", "resolved": "https://registry.npmmirror.com/babel-polyfill/-/babel-polyfill-6.26.0.tgz", @@ -6419,15 +34337,6 @@ } } }, - "block-stream": { - "version": "0.0.9", - "resolved": "https://registry.npmmirror.com/block-stream/-/block-stream-0.0.9.tgz", - "integrity": "sha512-OorbnJVPII4DuUKbjARAe8u8EfqOmkEEaSFIyoQ7OjTHn6kafxWl0wLgoZ2rXaYd7MyLcDaU4TmhfxtwgcccMQ==", - "dev": true, - "requires": { - "inherits": "~2.0.0" - } - }, "bluebird": { "version": "3.7.2", "resolved": "https://registry.npmmirror.com/bluebird/-/bluebird-3.7.2.tgz", @@ -6496,6 +34405,11 @@ "resolved": "https://registry.npmmirror.com/boolbase/-/boolbase-1.0.0.tgz", "integrity": "sha512-JZOSA7Mo9sNGB8+UjSgzdLtokWAky1zbztM3WRLCbZ70/3cTANmQmOdR7y2g+J0e2WXywy1yS468tY+IruqEww==" }, + "bootstrap": { + "version": "3.4.1", + "resolved": "https://registry.npmmirror.com/bootstrap/-/bootstrap-3.4.1.tgz", + "integrity": "sha512-yN5oZVmRCwe5aKwzRj6736nSmKDX7pLYwsXiCj/EYmo16hODaBiT4En5btW/jhBF/seV+XMx3aYwukYC3A49DA==" + }, "brace-expansion": { "version": "1.1.11", "resolved": "https://registry.npmmirror.com/brace-expansion/-/brace-expansion-1.1.11.tgz", @@ -7284,6 +35198,16 @@ "dev": true, "optional": true }, + "clipboard": { + "version": "2.0.8", + "resolved": "https://registry.npmmirror.com/clipboard/-/clipboard-2.0.8.tgz", + "integrity": "sha512-Y6WO0unAIQp5bLmk1zdThRhgJt/x3ks6f30s3oE3H1mgIEU33XyQjEf8gsf6DxC7NPX8Y1SsNWjUjL/ywLnnbQ==", + "requires": { + "good-listener": "^1.2.2", + "select": "^1.1.2", + "tiny-emitter": "^2.0.0" + } + }, "clipboardy": { "version": "2.3.0", "resolved": "https://registry.npmmirror.com/clipboardy/-/clipboardy-2.3.0.tgz", @@ -7396,6 +35320,11 @@ "integrity": "sha512-RpAVKQA5T63xEj6/giIbUEtZwJ4UFIc3ZtvEkiaUERylqe8xb5IvqcgOurZLahv93CLKfxcw5YI+DZcUBRyLXA==", "dev": true }, + "codemirror": { + "version": "5.65.13", + "resolved": "https://registry.npmmirror.com/codemirror/-/codemirror-5.65.13.tgz", + "integrity": "sha512-SVWEzKXmbHmTQQWaz03Shrh4nybG0wXx2MEu3FO4ezbPW8IbnZEd5iGHGEffSUaitKYa3i+pHpBsSvw8sPHtzg==" + }, "collection-visit": { "version": "1.0.0", "resolved": "https://registry.npmmirror.com/collection-visit/-/collection-visit-1.0.0.tgz", @@ -7844,8 +35773,8 @@ "integrity": "sha512-nK7sAtfi+QXbxHCYfhpZsfRtaitZLIA6889kFIouLvz6repszQDgxBu7wf2WbU+Dco7sAnNCJYERCwt54WPC2Q==", "dev": true, "requires": { - "JSONStream": "^1.0.4", "is-text-path": "^1.0.1", + "JSONStream": "^1.0.4", "lodash": "^4.17.15", "meow": "^8.0.0", "split2": "^3.0.0", @@ -8893,6 +36822,11 @@ "integrity": "sha512-/Tb/JcjK111nNScGob5MNtsntNM1aCNUDipB/TkwZFhyDrrE47SOx/18wF2bbjgc3ZzCSKW1T5nt5EbFoAz/Vg==", "dev": true }, + "cssfilter": { + "version": "0.0.10", + "resolved": "https://registry.npmmirror.com/cssfilter/-/cssfilter-0.0.10.tgz", + "integrity": "sha512-FAaLDaplstoRsDR8XGYH51znUN0UY7nMc6Z9/fvE8EXGwvJE9hu7W2vHwx1+bd6gCYnln9nLbzxFTrcO9YQDZw==" + }, "cssnano": { "version": "4.1.11", "resolved": "https://registry.npmmirror.com/cssnano/-/cssnano-4.1.11.tgz", @@ -9005,20 +36939,16 @@ "integrity": "sha512-qv8s+G47V6Hq+g2kRE5th+ASzzrL7b6l+tap1DHKK25ZQJv3yIFhH96XaQ7NGL+zRW3t/RDbweJf/dJDe5Z5KA==", "dev": true }, - "currently-unhandled": { - "version": "0.4.1", - "resolved": "https://registry.npmmirror.com/currently-unhandled/-/currently-unhandled-0.4.1.tgz", - "integrity": "sha512-/fITjgjGU50vjQ4FH6eUoYu+iUoUKIXws2hL15JJpIR+BbTxaXQsMuuyjtNh2WqsSBS5nsaZHFsFecyw5CCAng==", - "dev": true, - "requires": { - "array-find-index": "^1.0.1" - } - }, "cyclist": { "version": "1.0.1", "resolved": "https://registry.npmmirror.com/cyclist/-/cyclist-1.0.1.tgz", "integrity": "sha512-NJGVKPS81XejHcLhaLJS7plab0fK3slPh11mESeeDq2W4ZI5kUKK/LRRdVDvjJseojbPB7ZwjnyOybg3Igea/A==" }, + "d3": { + "version": "3.5.17", + "resolved": "https://registry.npmmirror.com/d3/-/d3-3.5.17.tgz", + "integrity": "sha512-yFk/2idb8OHPKkbAL8QaOaqENNoMhIaSHZerk3oQsECwkObkCpJyjYwCe+OHiq6UEdhe1m8ZGARRRO3ljFjlKg==" + }, "d3-dispatch": { "version": "2.0.0", "resolved": "https://registry.npmmirror.com/d3-dispatch/-/d3-dispatch-2.0.0.tgz", @@ -9080,6 +37010,11 @@ "integrity": "sha512-jyCETtSl3VMZMWeRo7iY1FL19ges1t55hMo5yaam4Jrsm5EPL89UQkoQRyiI+Yf4k8r2ZpdngkV8hr1lIdjb3Q==", "dev": true }, + "dayjs": { + "version": "1.10.7", + "resolved": "https://registry.npmmirror.com/dayjs/-/dayjs-1.10.7.tgz", + "integrity": "sha512-P6twpd70BcPK34K26uJ1KT3wlhpuOAPoMwJzpsIWUxHZ7wpmbdZL/hQqBDfz7hGurYSa5PhzdhDHtt319hL3ig==" + }, "de-indent": { "version": "1.0.2", "resolved": "https://registry.npmmirror.com/de-indent/-/de-indent-1.0.2.tgz", @@ -9374,6 +37309,11 @@ "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==", "dev": true }, + "delegate": { + "version": "3.2.0", + "resolved": "https://registry.npmmirror.com/delegate/-/delegate-3.2.0.tgz", + "integrity": "sha512-IofjkYBZaZivn0V8nnsMJGBr4jVLxHDheKSW88PyxS5QC4Vo9ZbZVvhzlSxY87fVq3STR6r+4cGepyHkcWOQSw==" + }, "delegates": { "version": "1.0.0", "resolved": "https://registry.npmmirror.com/delegates/-/delegates-1.0.0.tgz", @@ -9419,6 +37359,11 @@ "integrity": "sha512-T0NIuQpnTvFDATNuHN5roPwSBG83rFsuO+MXXH9/3N1eFbn4wcPjttvjMLEPWJ0RGUYgQE7cGgS3tNxbqCGM7g==", "dev": true }, + "dexie": { + "version": "2.0.4", + "resolved": "https://registry.npmmirror.com/dexie/-/dexie-2.0.4.tgz", + "integrity": "sha512-aQ/s1U2wHxwBKRrt2Z/mwFNHMQWhESerFsMYzE+5P5OsIe5o1kgpFMWkzKTtkvkyyEni6mWr/T4HUJuY9xIHLA==" + }, "dezalgo": { "version": "1.0.4", "resolved": "https://registry.npmmirror.com/dezalgo/-/dezalgo-1.0.4.tgz", @@ -9429,6 +37374,11 @@ "wrappy": "1" } }, + "diff-match-patch": { + "version": "1.0.5", + "resolved": "https://registry.npmmirror.com/diff-match-patch/-/diff-match-patch-1.0.5.tgz", + "integrity": "sha512-IayShXAgj/QMXgB0IWmKx+rOPuGMhqm5w6jvFxmVenXKIzRqTAAsbBPT3kWQeGANj3jGgvcvv4yK6SxqYmikgw==" + }, "diffie-hellman": { "version": "5.0.3", "resolved": "https://registry.npmmirror.com/diffie-hellman/-/diffie-hellman-5.0.3.tgz", @@ -9582,6 +37532,11 @@ "integrity": "sha512-YXQl1DSa4/PQyRfgrv6aoNjhasp/p4qs9FjJ4q4cQk+8m4r6k4ZSiEyytKG8f8W9gi8WsQtIObNmKd+tMzNTmA==", "dev": true }, + "dt-sql-parser": { + "version": "1.2.1", + "resolved": "https://registry.npmmirror.com/dt-sql-parser/-/dt-sql-parser-1.2.1.tgz", + "integrity": "sha512-bp6Rtm8N3m2Z3tavtKGo8J5ffaPQdPodYV2xGi9f6IsWbHzOkpNBxHlORcWtnSLtA/JvZcvImQxgbkdaGiv5cw==" + }, "duplexer": { "version": "0.1.2", "resolved": "https://registry.npmmirror.com/duplexer/-/duplexer-0.1.2.tgz", @@ -9599,6 +37554,11 @@ "stream-shift": "^1.0.0" } }, + "easings-css": { + "version": "1.0.0", + "resolved": "https://registry.npmmirror.com/easings-css/-/easings-css-1.0.0.tgz", + "integrity": "sha512-7Uq7NdazNfVtr0RNmPAys8it0zKCuaqxJStYKEl72D3j4gbvXhhaM7iWNbqhA4C94ygCye6VuyhzBRQC4szeBg==" + }, "easy-stack": { "version": "1.0.1", "resolved": "https://registry.npmmirror.com/easy-stack/-/easy-stack-1.0.1.tgz", @@ -9911,7 +37871,8 @@ "version": "5.3.2", "resolved": "https://registry.npmmirror.com/acorn-jsx/-/acorn-jsx-5.3.2.tgz", "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", - "dev": true + "dev": true, + "requires": {} }, "ansi-escapes": { "version": "4.3.2", @@ -10341,7 +38302,8 @@ "version": "12.0.0", "resolved": "https://registry.npmmirror.com/eslint-config-standard/-/eslint-config-standard-12.0.0.tgz", "integrity": "sha512-COUz8FnXhqFitYj4DTqHzidjIL/t4mumGZto5c7DrBpvWoie+Sn3P4sLEzUGeYhRElWuFEf8K1S1EfvD1vixCQ==", - "dev": true + "dev": true, + "requires": {} }, "eslint-import-resolver-node": { "version": "0.3.6", @@ -10524,7 +38486,8 @@ "version": "4.1.0", "resolved": "https://registry.npmmirror.com/eslint-plugin-standard/-/eslint-plugin-standard-4.1.0.tgz", "integrity": "sha512-ZL7+QRixjTR6/528YNGyDotyffm5OQst/sGxKDwGb9Uqs4In5Egi4+jbobhqJoyoCM6/7v/1A5fhQ7ScMtDjaQ==", - "dev": true + "dev": true, + "requires": {} }, "eslint-plugin-vue": { "version": "6.2.2", @@ -10547,7 +38510,8 @@ "version": "5.3.2", "resolved": "https://registry.npmmirror.com/acorn-jsx/-/acorn-jsx-5.3.2.tgz", "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", - "dev": true + "dev": true, + "requires": {} }, "debug": { "version": "4.3.4", @@ -11431,18 +39395,6 @@ "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", "optional": true }, - "fstream": { - "version": "1.0.12", - "resolved": "https://registry.npmmirror.com/fstream/-/fstream-1.0.12.tgz", - "integrity": "sha512-WvJ193OHa0GHPEL+AycEJgxvBEwyfRkN1vhjca23OaPVMCaLCXTd5qAu82AjTcgP1UJmytkOKb63Ypde7raDIg==", - "dev": true, - "requires": { - "graceful-fs": "^4.1.2", - "inherits": "~2.0.0", - "mkdirp": ">=0.5 0", - "rimraf": "2" - } - }, "function-bind": { "version": "1.1.1", "resolved": "https://registry.npmmirror.com/function-bind/-/function-bind-1.1.1.tgz", @@ -11460,6 +39412,11 @@ "integrity": "sha512-xckBUXyTIqT97tq2x2AMb+g163b5JFysYk0x4qxNFwbfQkmNZoiRHb6sPzI9/QV33WeuvVYBUIiD4NzNIyqaRQ==", "dev": true }, + "fuzzysearch": { + "version": "1.0.3", + "resolved": "https://registry.npmmirror.com/fuzzysearch/-/fuzzysearch-1.0.3.tgz", + "integrity": "sha512-s+kNWQuI3mo9OALw0HJ6YGmMbLqEufCh2nX/zzV5CrICQ/y4AwPxM+6TIiF9ItFCHXFCyM/BfCCmN57NTIJuPg==" + }, "g-status": { "version": "2.0.2", "resolved": "https://registry.npmmirror.com/g-status/-/g-status-2.0.2.tgz", @@ -11808,6 +39765,7 @@ "version": "5.1.2", "resolved": "https://registry.npmmirror.com/glob-parent/-/glob-parent-5.1.2.tgz", "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", + "devOptional": true, "requires": { "is-glob": "^4.0.1" } @@ -11815,8 +39773,7 @@ "glob-to-regexp": { "version": "0.3.0", "resolved": "https://registry.npmmirror.com/glob-to-regexp/-/glob-to-regexp-0.3.0.tgz", - "integrity": "sha512-Iozmtbqv0noj0uDDqoL0zNq0VBEfK2YFoMAZoxJe4cwphvLR+JskfF30QhXHOR4m3KrE6NLRYw+U9MRXvifyig==", - "dev": true + "integrity": "sha512-Iozmtbqv0noj0uDDqoL0zNq0VBEfK2YFoMAZoxJe4cwphvLR+JskfF30QhXHOR4m3KrE6NLRYw+U9MRXvifyig==" }, "glob2base": { "version": "0.0.12", @@ -11893,6 +39850,14 @@ } } }, + "good-listener": { + "version": "1.2.2", + "resolved": "https://registry.npmmirror.com/good-listener/-/good-listener-1.2.2.tgz", + "integrity": "sha512-goW1b+d9q/HIwbVYZzZ6SsTr4IgE+WA44A0GmPIQstuOrgsFcT7VEJ48nmr9GaRtNu0XTKacFLGnBPAM6Afouw==", + "requires": { + "delegate": "^3.1.2" + } + }, "graceful-fs": { "version": "4.2.10", "resolved": "https://registry.npmmirror.com/graceful-fs/-/graceful-fs-4.2.10.tgz", @@ -12110,6 +40075,11 @@ "integrity": "sha512-l9sfDFsuqtOqKDsQdqrMRk0U85RZc0RtOR9yPI7mRVOa4FsR/BVnZ0shmQRM96Ji99kYZP/7hn1cedc1+ApsTQ==", "dev": true }, + "highlight.js": { + "version": "9.18.3", + "resolved": "https://registry.npmmirror.com/highlight.js/-/highlight.js-9.18.3.tgz", + "integrity": "sha512-zBZAmhSupHIl5sITeMqIJnYCDfAEc3Gdkqj65wC1lpI468MMQeeQkhcIAvk+RylAkxrCcI9xy9piHiXeQ1BdzQ==" + }, "hmac-drbg": { "version": "1.0.1", "resolved": "https://registry.npmmirror.com/hmac-drbg/-/hmac-drbg-1.0.1.tgz", @@ -12624,12 +40594,6 @@ "resolved": "https://registry.npmmirror.com/imurmurhash/-/imurmurhash-0.1.4.tgz", "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==" }, - "in-publish": { - "version": "2.0.1", - "resolved": "https://registry.npmmirror.com/in-publish/-/in-publish-2.0.1.tgz", - "integrity": "sha512-oDM0kUSNFC31ShNxHKUyfZKy8ZeXZBWMjMdZHKLOk13uvT27VTL/QzRGfRUcevJhpkZAvlhPYuXkF7eNWrtyxQ==", - "dev": true - }, "indent-string": { "version": "4.0.0", "resolved": "https://registry.npmmirror.com/indent-string/-/indent-string-4.0.0.tgz", @@ -13033,13 +40997,8 @@ "is-extglob": { "version": "2.1.1", "resolved": "https://registry.npmmirror.com/is-extglob/-/is-extglob-2.1.1.tgz", - "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==" - }, - "is-finite": { - "version": "1.1.0", - "resolved": "https://registry.npmmirror.com/is-finite/-/is-finite-1.1.0.tgz", - "integrity": "sha512-cdyMtqX/BOqqNBBiKlIVkytNHm49MtMlYyn1zxzvJKWmFMlGzm+ry5BBfYyeY9YmNKbRSo/o7OX9w9ale0wg3w==", - "dev": true + "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", + "devOptional": true }, "is-fullwidth-code-point": { "version": "2.0.0", @@ -13051,6 +41010,7 @@ "version": "4.0.3", "resolved": "https://registry.npmmirror.com/is-glob/-/is-glob-4.0.3.tgz", "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", + "devOptional": true, "requires": { "is-extglob": "^2.1.1" } @@ -13160,8 +41120,7 @@ "is-promise": { "version": "2.2.2", "resolved": "https://registry.npmmirror.com/is-promise/-/is-promise-2.2.2.tgz", - "integrity": "sha512-+lP4/6lKUBfQjZ2pdxThZvLUAafmZb8OAxFb8XXtiQmS35INgr85hdOGoEs124ez1FCnZJt6jau/T+alh58QFQ==", - "dev": true + "integrity": "sha512-+lP4/6lKUBfQjZ2pdxThZvLUAafmZb8OAxFb8XXtiQmS35INgr85hdOGoEs124ez1FCnZJt6jau/T+alh58QFQ==" }, "is-regex": { "version": "1.1.4", @@ -13238,12 +41197,6 @@ "integrity": "sha512-cyA56iCMHAh5CdzjJIa4aohJyeO1YbwLi3Jc35MmRU6poroFjIGZzUzupGiRPOjgHg9TLu43xbpwXk523fMxKA==", "dev": true }, - "is-utf8": { - "version": "0.2.1", - "resolved": "https://registry.npmmirror.com/is-utf8/-/is-utf8-0.2.1.tgz", - "integrity": "sha512-rMYPYvCzsXywIsldgLaSoPlw5PfoB/ssr7hY4pLfcodrA5M/eArza1a9VmTiNIBNMjOGr1Ow9mTyU2o69U6U9Q==", - "dev": true - }, "is-weakref": { "version": "1.0.2", "resolved": "https://registry.npmmirror.com/is-weakref/-/is-weakref-1.0.2.tgz", @@ -13311,6 +41264,11 @@ "integrity": "sha512-fnjC0up+0SjEJtgmmG+teeel68kutkvzfctO/KxE3qJlbunkJYAshgH3boU++gSBHP8z5/r0ts0qRIrHf0RTQQ==", "dev": true }, + "jquery": { + "version": "3.6.0", + "resolved": "https://registry.npmmirror.com/jquery/-/jquery-3.6.0.tgz", + "integrity": "sha512-JVzAR/AjBvVt2BmYhxRCSYysDsPcssdmTFnzyLEts9qNwmjmu4JTAMYubEfwVOSwpQ1I1sKKFcxhZCI2buerfw==" + }, "js-base64": { "version": "2.6.4", "resolved": "https://registry.npmmirror.com/js-base64/-/js-base64-2.6.4.tgz", @@ -13432,6 +41390,16 @@ "integrity": "sha512-POQXvpdL69+CluYsillJ7SUhKvytYjW9vG/GKpnf+xP8UWgYEM/RaMzHHofbALDiKbbP1W8UEYmgGl39WkPZsg==", "dev": true }, + "JSONStream": { + "version": "1.3.5", + "resolved": "https://registry.npmmirror.com/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" + } + }, "jsprim": { "version": "1.4.2", "resolved": "https://registry.npmmirror.com/jsprim/-/jsprim-1.4.2.tgz", @@ -13464,6 +41432,12 @@ "graceful-fs": "^4.1.11" } }, + "klona": { + "version": "2.0.5", + "resolved": "https://registry.npmmirror.com/klona/-/klona-2.0.5.tgz", + "integrity": "sha512-pJiBpiXMbt7dkzXe8Ghj/u4FfXOOa98fPW+bihOJ4SjnoijweJrNThJfd3ifXpXhREjpoF2mZVH1GfS9LV3kHQ==", + "dev": true + }, "launch-editor": { "version": "2.3.0", "resolved": "https://registry.npmmirror.com/launch-editor/-/launch-editor-2.3.0.tgz", @@ -14324,16 +42298,6 @@ "js-tokens": "^3.0.0 || ^4.0.0" } }, - "loud-rejection": { - "version": "1.6.0", - "resolved": "https://registry.npmmirror.com/loud-rejection/-/loud-rejection-1.6.0.tgz", - "integrity": "sha512-RPNliZOFkqFumDhvYqOaNY4Uz9oJM2K9tC6JWsJJsNdhuONW4LQHRBpb0qf4pJApVffI5N39SwzWZJuEhfd7eQ==", - "dev": true, - "requires": { - "currently-unhandled": "^0.4.1", - "signal-exit": "^3.0.0" - } - }, "lower-case": { "version": "1.1.4", "resolved": "https://registry.npmmirror.com/lower-case/-/lower-case-1.1.4.tgz", @@ -14533,12 +42497,25 @@ "escape-string-regexp": "^1.0.4" } }, + "material-colors": { + "version": "1.2.6", + "resolved": "https://registry.npmmirror.com/material-colors/-/material-colors-1.2.6.tgz", + "integrity": "sha512-6qE4B9deFBIa9YSpOc9O0Sgc43zTeVYbgDT5veRKSlB2+ZuHNoVVxA1L/ckMUayV9Ay9y7Z/SZCLcGteW9i7bg==" + }, "math-random": { "version": "1.0.4", "resolved": "https://registry.npmmirror.com/math-random/-/math-random-1.0.4.tgz", "integrity": "sha512-rUxjysqif/BZQH2yhd5Aaq7vXMSx9NdEsQcyA07uEzIvxgI7zIr33gGsh+RU0/XjmQpCW7RsVof1vlkvQVCK5A==", "dev": true }, + "mavon-editor": { + "version": "2.10.4", + "resolved": "https://registry.npmmirror.com/mavon-editor/-/mavon-editor-2.10.4.tgz", + "integrity": "sha512-CFsBLkgt/KZBDg+SJYe2fyYv4zClY149PiwpH0rDAiiP4ae1XNs0GC8nBsoTeipsHcebDLN1QMkt3bUsnMDjQw==", + "requires": { + "xss": "^1.0.6" + } + }, "md5": { "version": "2.3.0", "resolved": "https://registry.npmmirror.com/md5/-/md5-2.3.0.tgz", @@ -15207,9 +43184,17 @@ }, "moment": { "version": "2.29.4", - "resolved": "https://registry.npmjs.org/moment/-/moment-2.29.4.tgz", + "resolved": "https://registry.npmmirror.com/moment/-/moment-2.29.4.tgz", "integrity": "sha512-5LC9SOxjSc2HF6vO2CyuTDNivEdoz2IvyJJGj6X8DJ0eFyfszE0QiEd+iXmBvUP3WHxSjFH/vIsA0EN00cgr8w==" }, + "moment-timezone": { + "version": "0.5.34", + "resolved": "https://registry.npmmirror.com/moment-timezone/-/moment-timezone-0.5.34.tgz", + "integrity": "sha512-3zAEHh2hKUs3EXLESx/wsgw6IQdusOT8Bxm3D9UrHPQR7zlMmzwybC8zHEM1tQ4LJwP7fcxrWr8tuBg05fFCbg==", + "requires": { + "moment": ">= 2.9.0" + } + }, "monaco-editor": { "version": "0.19.3", "resolved": "https://registry.npmmirror.com/monaco-editor/-/monaco-editor-0.19.3.tgz", @@ -15224,6 +43209,17 @@ "loader-utils": "^1.2.3" } }, + "monaco-languageclient": { + "version": "0.13.0", + "resolved": "https://registry.npmmirror.com/monaco-languageclient/-/monaco-languageclient-0.13.0.tgz", + "integrity": "sha512-aCwd33dTitwV5QwY56rpYHwzEGXei8TZ+yvZcvP3gEMd6Mizr8m3pOuoknDi2SUfLuNAHS6+ulvLgZlNQB5awg==", + "requires": { + "glob-to-regexp": "^0.3.0", + "vscode-jsonrpc": "^5.0.0", + "vscode-languageclient": "^6.0.0", + "vscode-uri": "^2.1.1" + } + }, "move-concurrently": { "version": "1.0.1", "resolved": "https://registry.npmmirror.com/move-concurrently/-/move-concurrently-1.0.1.tgz", @@ -15334,7 +43330,8 @@ "nan": { "version": "2.15.0", "resolved": "https://registry.npmmirror.com/nan/-/nan-2.15.0.tgz", - "integrity": "sha512-8ZtvEnA2c5aYCZYd1cvgdnU6cqwixRoYg70xPLWUws5ORTa/lnw+u4amixRS/Ac5U5mQVgp9pnlSUnbNWFaWZQ==" + "integrity": "sha512-8ZtvEnA2c5aYCZYd1cvgdnU6cqwixRoYg70xPLWUws5ORTa/lnw+u4amixRS/Ac5U5mQVgp9pnlSUnbNWFaWZQ==", + "devOptional": true }, "nanomatch": { "version": "1.2.13", @@ -15512,23 +43509,21 @@ "dev": true }, "node-sass": { - "version": "4.14.1", - "resolved": "https://registry.npmmirror.com/node-sass/-/node-sass-4.14.1.tgz", - "integrity": "sha512-sjCuOlvGyCJS40R8BscF5vhVlQjNN069NtQ1gSxyK1u9iqvn6tf7O1R4GNowVZfiZUCRt5MmMs1xd+4V/7Yr0g==", + "version": "6.0.1", + "resolved": "https://registry.npmmirror.com/node-sass/-/node-sass-6.0.1.tgz", + "integrity": "sha512-f+Rbqt92Ful9gX0cGtdYwjTrWAaGURgaK5rZCWOgCNyGWusFYHhbqCCBoFBeat+HKETOU02AyTxNhJV0YZf2jQ==", "dev": true, "requires": { "async-foreach": "^0.1.3", "chalk": "^1.1.1", - "cross-spawn": "^3.0.0", + "cross-spawn": "^7.0.3", "gaze": "^1.0.0", "get-stdin": "^4.0.1", "glob": "^7.0.3", - "in-publish": "^2.0.0", "lodash": "^4.17.15", - "meow": "^3.7.0", - "mkdirp": "^0.5.1", + "meow": "^9.0.0", "nan": "^2.13.2", - "node-gyp": "^3.8.0", + "node-gyp": "^7.1.0", "npmlog": "^4.0.0", "request": "^2.88.0", "sass-graph": "2.2.5", @@ -15548,22 +43543,6 @@ "integrity": "sha512-kmCevFghRiWM7HB5zTPULl4r9bVFSWjz62MhqizDGUrq2NWuNMQyuv4tHHoKJHs69M/MF64lEcHdYIocrdWQYA==", "dev": true }, - "camelcase": { - "version": "2.1.1", - "resolved": "https://registry.npmmirror.com/camelcase/-/camelcase-2.1.1.tgz", - "integrity": "sha512-DLIsRzJVBQu72meAKPkWQOLcujdXT32hwdfnkI1frSiSRMK1MofjKHf+MEx0SB6fjEFXL8fBDv1dKymBlOp4Qw==", - "dev": true - }, - "camelcase-keys": { - "version": "2.1.0", - "resolved": "https://registry.npmmirror.com/camelcase-keys/-/camelcase-keys-2.1.0.tgz", - "integrity": "sha512-bA/Z/DERHKqoEOrp+qeGKw1QlvEQkGZSc0XaY6VnTxZr+Kv1G5zFwttpjv8qxZ/sBPT4nthwZaAcsAZTJlSKXQ==", - "dev": true, - "requires": { - "camelcase": "^2.0.0", - "map-obj": "^1.0.0" - } - }, "chalk": { "version": "1.1.3", "resolved": "https://registry.npmmirror.com/chalk/-/chalk-1.1.3.tgz", @@ -15577,24 +43556,40 @@ "supports-color": "^2.0.0" } }, + "chownr": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/chownr/-/chownr-2.0.0.tgz", + "integrity": "sha512-bIomtDF5KGpdogkLd9VspvFzk9KfpyyGlS8YFVZl7TGPBHL5snIOnxeshwVgPteQ9b4Eydl+pVbIyE1DcvCWgQ==", + "dev": true + }, "cross-spawn": { - "version": "3.0.1", - "resolved": "https://registry.npmmirror.com/cross-spawn/-/cross-spawn-3.0.1.tgz", - "integrity": "sha512-eZ+m1WNhSZutOa/uRblAc9Ut5MQfukFrFMtPSm3bZCA888NmMd5AWXWdgRZ80zd+pTk1P2JrGjg9pUPTvl2PWQ==", + "version": "7.0.3", + "resolved": "https://registry.npmmirror.com/cross-spawn/-/cross-spawn-7.0.3.tgz", + "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", "dev": true, "requires": { - "lru-cache": "^4.0.1", - "which": "^1.2.9" + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" } }, "find-up": { - "version": "1.1.2", - "resolved": "https://registry.npmmirror.com/find-up/-/find-up-1.1.2.tgz", - "integrity": "sha512-jvElSjyuo4EMQGoTwo1uJU5pQMwTW5lS1x05zzfJuTIyLR3zwO27LYrxNg+dlvKpGOuGy/MzBdXh80g0ve5+HA==", + "version": "4.1.0", + "resolved": "https://registry.npmmirror.com/find-up/-/find-up-4.1.0.tgz", + "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", "dev": true, "requires": { - "path-exists": "^2.0.0", - "pinkie-promise": "^2.0.0" + "locate-path": "^5.0.0", + "path-exists": "^4.0.0" + } + }, + "fs-minipass": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/fs-minipass/-/fs-minipass-2.1.0.tgz", + "integrity": "sha512-V/JgOLFCS+R6Vcq0slCuaeWEdNC3ouDlJMNIsacH2VtALiu9mV4LPrHc5cDl8k5aw6J8jwgWWpiTo5RYhmIzvg==", + "dev": true, + "requires": { + "minipass": "^3.0.0" } }, "get-stdin": { @@ -15603,161 +43598,188 @@ "integrity": "sha512-F5aQMywwJ2n85s4hJPTT9RPxGmubonuB10MNYo17/xph174n2MIR33HRguhzVag10O/npM7SPk73LMZNP+FaWw==", "dev": true }, - "indent-string": { - "version": "2.1.0", - "resolved": "https://registry.npmmirror.com/indent-string/-/indent-string-2.1.0.tgz", - "integrity": "sha512-aqwDFWSgSgfRaEwao5lg5KEcVd/2a+D1rvoG7NdilmYz0NwRk6StWpWdz/Hpk34MKPpx7s8XxUqimfcQK6gGlg==", + "hosted-git-info": { + "version": "4.1.0", + "resolved": "https://registry.npmmirror.com/hosted-git-info/-/hosted-git-info-4.1.0.tgz", + "integrity": "sha512-kyCuEOWjJqZuDbRHzL8V93NzQhwIB71oFWSyzVo+KPZI+pnQPPxucdkrOZvkLRnrf5URsQM+IJ09Dw29cRALIA==", "dev": true, "requires": { - "repeating": "^2.0.0" + "lru-cache": "^6.0.0" } }, - "load-json-file": { - "version": "1.1.0", - "resolved": "https://registry.npmmirror.com/load-json-file/-/load-json-file-1.1.0.tgz", - "integrity": "sha512-cy7ZdNRXdablkXYNI049pthVeXFurRyb9+hA/dZzerZ0pGTx42z+y+ssxBaVV2l70t1muq5IdKhn4UtcoGUY9A==", + "locate-path": { + "version": "5.0.0", + "resolved": "https://registry.npmmirror.com/locate-path/-/locate-path-5.0.0.tgz", + "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", "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" + "p-locate": "^4.1.0" } }, "lru-cache": { - "version": "4.1.5", - "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-4.1.5.tgz", - "integrity": "sha512-sWZlbEP2OsHNkXrMl5GYk/jKk70MBng6UU4YI/qGDYbgf6YbP4EvmqISbXCoJiRKs+1bSpFHVgQxvJ17F2li5g==", + "version": "6.0.0", + "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", "dev": true, "requires": { - "pseudomap": "^1.0.2", - "yallist": "^2.1.2" + "yallist": "^4.0.0" } }, - "map-obj": { - "version": "1.0.1", - "resolved": "https://registry.npmmirror.com/map-obj/-/map-obj-1.0.1.tgz", - "integrity": "sha512-7N/q3lyZ+LVCp7PzuxrJr4KMbBE2hW7BT7YNia330OFxIf4d3r5zVpicP2650l7CPN6RM9zOJRl3NGpqSiw3Eg==", - "dev": true - }, "meow": { - "version": "3.7.0", - "resolved": "https://registry.npmmirror.com/meow/-/meow-3.7.0.tgz", - "integrity": "sha512-TNdwZs0skRlpPpCUK25StC4VH+tP5GgeY1HQOOGP+lQ2xtdkN2VtT/5tiX9k3IWpkBPV9b3LsAWXn4GGi/PrSA==", + "version": "9.0.0", + "resolved": "https://registry.npmmirror.com/meow/-/meow-9.0.0.tgz", + "integrity": "sha512-+obSblOQmRhcyBt62furQqRAQpNyWXo8BuQ5bN7dG8wmwQ+vwHKp/rCFD4CrTP8CsDQD1sjoZ94K417XEUk8IQ==", "dev": true, "requires": { - "camelcase-keys": "^2.0.0", - "decamelize": "^1.1.2", - "loud-rejection": "^1.0.0", - "map-obj": "^1.0.1", - "minimist": "^1.1.3", - "normalize-package-data": "^2.3.4", - "object-assign": "^4.0.1", - "read-pkg-up": "^1.0.1", - "redent": "^1.0.0", - "trim-newlines": "^1.0.0" + "@types/minimist": "^1.2.0", + "camelcase-keys": "^6.2.2", + "decamelize": "^1.2.0", + "decamelize-keys": "^1.1.0", + "hard-rejection": "^2.1.0", + "minimist-options": "4.1.0", + "normalize-package-data": "^3.0.0", + "read-pkg-up": "^7.0.1", + "redent": "^3.0.0", + "trim-newlines": "^3.0.0", + "type-fest": "^0.18.0", + "yargs-parser": "^20.2.3" + } + }, + "minipass": { + "version": "3.3.4", + "resolved": "https://registry.npmmirror.com/minipass/-/minipass-3.3.4.tgz", + "integrity": "sha512-I9WPbWHCGu8W+6k1ZiGpPu0GkoKBeorkfKNuAFBNS1HNFJvke82sxvI5bzcCNpWPorkOO5QQ+zomzzwRxejXiw==", + "dev": true, + "requires": { + "yallist": "^4.0.0" + } + }, + "minizlib": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/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.npmmirror.com/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true + }, "node-gyp": { - "version": "3.8.0", - "resolved": "https://registry.npmmirror.com/node-gyp/-/node-gyp-3.8.0.tgz", - "integrity": "sha512-3g8lYefrRRzvGeSowdJKAKyks8oUpLEd/DyPV4eMhVlhJ0aNaZqIrNUIPuEWWTAoPqyFkfGrM67MC69baqn6vA==", + "version": "7.1.2", + "resolved": "https://registry.npmmirror.com/node-gyp/-/node-gyp-7.1.2.tgz", + "integrity": "sha512-CbpcIo7C3eMu3dL1c3d0xw449fHIGALIJsRP4DDPHpyiW8vcriNY7ubh9TE4zEKfSxscY7PjeFnshE7h75ynjQ==", "dev": true, "requires": { - "fstream": "^1.0.0", - "glob": "^7.0.3", - "graceful-fs": "^4.1.2", - "mkdirp": "^0.5.0", - "nopt": "2 || 3", - "npmlog": "0 || 1 || 2 || 3 || 4", - "osenv": "0", - "request": "^2.87.0", - "rimraf": "2", - "semver": "~5.3.0", - "tar": "^2.0.0", - "which": "1" + "env-paths": "^2.2.0", + "glob": "^7.1.4", + "graceful-fs": "^4.2.3", + "nopt": "^5.0.0", + "npmlog": "^4.1.2", + "request": "^2.88.2", + "rimraf": "^3.0.2", + "semver": "^7.3.2", + "tar": "^6.0.2", + "which": "^2.0.2" } }, "nopt": { - "version": "3.0.6", - "resolved": "https://registry.npmmirror.com/nopt/-/nopt-3.0.6.tgz", - "integrity": "sha512-4GUt3kSEYmk4ITxzB/b9vaIDfUVWN/Ml1Fwl11IlnIG2iaJ9O6WXZ9SrYM9NLI8OCBieN2Y8SWC2oJV0RQ7qYg==", + "version": "5.0.0", + "resolved": "https://registry.npmmirror.com/nopt/-/nopt-5.0.0.tgz", + "integrity": "sha512-Tbj67rffqceeLpcRXrT7vKAN8CwfPeIBgM7E6iBkmKLV7bEMwpGgYLGv0jACUsECaa/vuxP0IjEont6umdMgtQ==", "dev": true, "requires": { "abbrev": "1" } }, - "parse-json": { - "version": "2.2.0", - "resolved": "https://registry.npmmirror.com/parse-json/-/parse-json-2.2.0.tgz", - "integrity": "sha512-QR/GGaKCkhwk1ePQNYDRKYZ3mwU9ypsKhB0XyFnLQdomyEqk3e8wpW3V5Jp88zbxK4n5ST1nqo+g9juTpownhQ==", + "normalize-package-data": { + "version": "3.0.3", + "resolved": "https://registry.npmmirror.com/normalize-package-data/-/normalize-package-data-3.0.3.tgz", + "integrity": "sha512-p2W1sgqij3zMMyRC067Dg16bfzVH+w7hyegmpIvZ4JNjqtGOVAIvLmjBx3yP7YTe9vKJgkoNOPjwQGogDoMXFA==", + "dev": true, + "requires": { + "hosted-git-info": "^4.0.1", + "is-core-module": "^2.5.0", + "semver": "^7.3.4", + "validate-npm-package-license": "^3.0.1" + } + }, + "p-locate": { + "version": "4.1.0", + "resolved": "https://registry.npmmirror.com/p-locate/-/p-locate-4.1.0.tgz", + "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", "dev": true, "requires": { - "error-ex": "^1.2.0" + "p-limit": "^2.2.0" } }, "path-exists": { - "version": "2.1.0", - "resolved": "https://registry.npmmirror.com/path-exists/-/path-exists-2.1.0.tgz", - "integrity": "sha512-yTltuKuhtNeFJKa1PiRzfLAU5182q1y4Eb4XCJ3PBqyzEDkAZRzBrKKBct682ls9reBVHf9udYLN5Nd+K1B9BQ==", - "dev": true, - "requires": { - "pinkie-promise": "^2.0.0" - } + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/path-exists/-/path-exists-4.0.0.tgz", + "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", + "dev": true }, - "path-type": { - "version": "1.1.0", - "resolved": "https://registry.npmmirror.com/path-type/-/path-type-1.1.0.tgz", - "integrity": "sha512-S4eENJz1pkiQn9Znv33Q+deTOKmbl+jj1Fl+qiP/vYezj+S8x+J3Uo0ISrx/QoEvIlOaDWJhPaRd1flJ9HXZqg==", + "path-key": { + "version": "3.1.1", + "resolved": "https://registry.npmmirror.com/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", + "dev": true + }, + "read-pkg-up": { + "version": "7.0.1", + "resolved": "https://registry.npmmirror.com/read-pkg-up/-/read-pkg-up-7.0.1.tgz", + "integrity": "sha512-zK0TB7Xd6JpCLmlLmufqykGE+/TlOePD6qKClNW7hHDKFh/J7/7gCWGR7joEQEW1bKq3a3yUZSObOoWLFQ4ohg==", "dev": true, "requires": { - "graceful-fs": "^4.1.2", - "pify": "^2.0.0", - "pinkie-promise": "^2.0.0" + "find-up": "^4.1.0", + "read-pkg": "^5.2.0", + "type-fest": "^0.8.1" + }, + "dependencies": { + "type-fest": { + "version": "0.8.1", + "resolved": "https://registry.npmmirror.com/type-fest/-/type-fest-0.8.1.tgz", + "integrity": "sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==", + "dev": true + } } }, - "pify": { - "version": "2.3.0", - "resolved": "https://registry.npmmirror.com/pify/-/pify-2.3.0.tgz", - "integrity": "sha512-udgsAY+fTnvv7kI7aaxbqwWNb0AHiB0qBO89PZKPkoTmGOgdbrHDKD+0B2X4uTfJ/FT1R09r9gTsjUjNJotuog==", - "dev": true - }, - "read-pkg": { - "version": "1.1.0", - "resolved": "https://registry.npmmirror.com/read-pkg/-/read-pkg-1.1.0.tgz", - "integrity": "sha512-7BGwRHqt4s/uVbuyoeejRn4YmFnYZiFl4AuaeXHlgZf3sONF0SOGlxs2Pw8g6hCKupo08RafIO5YXFNOKTfwsQ==", + "rimraf": { + "version": "3.0.2", + "resolved": "https://registry.npmmirror.com/rimraf/-/rimraf-3.0.2.tgz", + "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", "dev": true, "requires": { - "load-json-file": "^1.0.0", - "normalize-package-data": "^2.3.2", - "path-type": "^1.0.0" + "glob": "^7.1.3" } }, - "read-pkg-up": { - "version": "1.0.1", - "resolved": "https://registry.npmmirror.com/read-pkg-up/-/read-pkg-up-1.0.1.tgz", - "integrity": "sha512-WD9MTlNtI55IwYUS27iHh9tK3YoIVhxis8yKhLpTqWtml739uXc9NWTpxoHkfZf3+DkCCsXox94/VWZniuZm6A==", + "semver": { + "version": "7.3.8", + "resolved": "https://registry.npmmirror.com/semver/-/semver-7.3.8.tgz", + "integrity": "sha512-NB1ctGL5rlHrPJtFDVIVzTyQylMLu9N9VICA6HSFJo8MCGVTMW6gfpicwKmmK/dAjTOrqu5l63JJOpDSrAis3A==", "dev": true, "requires": { - "find-up": "^1.0.0", - "read-pkg": "^1.0.0" + "lru-cache": "^6.0.0" } }, - "redent": { - "version": "1.0.0", - "resolved": "https://registry.npmmirror.com/redent/-/redent-1.0.0.tgz", - "integrity": "sha512-qtW5hKzGQZqKoh6JNSD+4lfitfPKGz42e6QwiRmPM5mmKtR0N41AbJRYu0xJi7nhOJ4WDgRkKvAk6tw4WIwR4g==", + "shebang-command": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/shebang-command/-/shebang-command-2.0.0.tgz", + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", "dev": true, "requires": { - "indent-string": "^2.1.0", - "strip-indent": "^1.0.1" + "shebang-regex": "^3.0.0" } }, - "semver": { - "version": "5.3.0", - "resolved": "https://registry.npmmirror.com/semver/-/semver-5.3.0.tgz", - "integrity": "sha512-mfmm3/H9+67MCVix1h+IXTpDwL6710LyHuk7+cWC9T1mE0qz4iHhh6r4hU2wrIT9iTsAAC2XQRvfblL028cpLw==", + "shebang-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmmirror.com/shebang-regex/-/shebang-regex-3.0.0.tgz", + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", "dev": true }, "strip-ansi": { @@ -15769,24 +43791,6 @@ "ansi-regex": "^2.0.0" } }, - "strip-bom": { - "version": "2.0.0", - "resolved": "https://registry.npmmirror.com/strip-bom/-/strip-bom-2.0.0.tgz", - "integrity": "sha512-kwrX1y7czp1E69n2ajbG65mIo9dqvJ+8aBQXOGVxqwvNbsXdFM6Lq37dLAY3mknUwru8CfcCbfOLL/gMo+fi3g==", - "dev": true, - "requires": { - "is-utf8": "^0.2.0" - } - }, - "strip-indent": { - "version": "1.0.1", - "resolved": "https://registry.npmmirror.com/strip-indent/-/strip-indent-1.0.1.tgz", - "integrity": "sha512-I5iQq6aFMM62fBEAIB/hXzwJD6EEZ0xEGCX2t7oXqaKPIRgt4WruAQ285BISgdkP+HLGWyeGmNJcpIwFeRYRUA==", - "dev": true, - "requires": { - "get-stdin": "^4.0.1" - } - }, "supports-color": { "version": "2.0.0", "resolved": "https://registry.npmmirror.com/supports-color/-/supports-color-2.0.0.tgz", @@ -15794,26 +43798,38 @@ "dev": true }, "tar": { - "version": "2.2.2", - "resolved": "https://registry.npmmirror.com/tar/-/tar-2.2.2.tgz", - "integrity": "sha512-FCEhQ/4rE1zYv9rYXJw/msRqsnmlje5jHP6huWeBZ704jUTy02c5AZyWujpMR1ax6mVw9NyJMfuK2CMDWVIfgA==", + "version": "6.1.12", + "resolved": "https://registry.npmmirror.com/tar/-/tar-6.1.12.tgz", + "integrity": "sha512-jU4TdemS31uABHd+Lt5WEYJuzn+TJTCBLljvIAHZOz6M9Os5pJ4dD+vRFLxPa/n3T0iEFzpi+0x1UfuDZYbRMw==", "dev": true, "requires": { - "block-stream": "*", - "fstream": "^1.0.12", - "inherits": "2" + "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" } }, - "trim-newlines": { - "version": "1.0.0", - "resolved": "https://registry.npmmirror.com/trim-newlines/-/trim-newlines-1.0.0.tgz", - "integrity": "sha512-Nm4cF79FhSTzrLKGDMi3I4utBtFv8qKy4sq1enftf2gMdpqI8oVQTAfySkTz5r49giVzDj88SVZXP4CeYQwjaw==", + "type-fest": { + "version": "0.18.1", + "resolved": "https://registry.npmmirror.com/type-fest/-/type-fest-0.18.1.tgz", + "integrity": "sha512-OIAYXk8+ISY+qTOwkHtKqzAuxchoMiD9Udx+FSGQDuiRR+PJKJHc2NJAXlbhkGwTt/4/nKZxELY1w3ReWOL8mw==", "dev": true }, + "which": { + "version": "2.0.2", + "resolved": "https://registry.npmmirror.com/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "requires": { + "isexe": "^2.0.0" + } + }, "yallist": { - "version": "2.1.2", - "resolved": "https://registry.npmmirror.com/yallist/-/yallist-2.1.2.tgz", - "integrity": "sha512-ncTzHV7NvsQZkYe1DW7cbDLm0YpzHmZF5r/iyP3ZnQtMiJ+pjzisCiMNI+Sj+xQF5pXhSHxSB3uDbsBTzY/c2A==", + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", "dev": true } } @@ -15843,7 +43859,8 @@ "normalize-path": { "version": "3.0.0", "resolved": "https://registry.npmmirror.com/normalize-path/-/normalize-path-3.0.0.tgz", - "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==" + "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", + "devOptional": true }, "normalize-range": { "version": "0.1.2", @@ -16708,39 +44725,21 @@ "json-parse-better-errors": "^1.0.1" } }, - "parse-url": { - "version": "6.0.2", - "resolved": "https://registry.npmjs.org/parse-url/-/parse-url-6.0.2.tgz", - "integrity": "sha512-uCSjOvD3T+6B/sPWhR+QowAZcU/o4bjPrVBQBGFxcDF6J6FraCGIaDBsdoQawiaaAVdHvtqBe3w3vKlfBKySOQ==", + "parse-path": { + "version": "4.0.3", + "resolved": "https://registry.npmmirror.com/parse-path/-/parse-path-4.0.3.tgz", + "integrity": "sha512-9Cepbp2asKnWTJ9x2kpw6Fe8y9JDbqwahGCTvklzd/cEq5C5JC59x2Xb0Kx+x0QZ8bvNquGO8/BWP0cwBHzSAA==", "dev": true, "requires": { "is-ssh": "^1.3.0", - "normalize-url": "^6.1.0", - "parse-path": "^4.0.4", - "protocols": "^1.4.0" + "protocols": "^1.4.0", + "qs": "^6.9.4", + "query-string": "^6.13.8" }, "dependencies": { - "normalize-url": { - "version": "6.1.0", - "resolved": "https://registry.npmjs.org/normalize-url/-/normalize-url-6.1.0.tgz", - "integrity": "sha512-DlL+XwOy3NxAQ8xuC0okPgK46iuVNAK01YN7RueYBqqFeGsBjV9XmCAzAdgt+667bCl5kPh9EqKKDwnaPG1I7A==", - "dev": true - }, - "parse-path": { - "version": "4.0.4", - "resolved": "https://registry.npmjs.org/parse-path/-/parse-path-4.0.4.tgz", - "integrity": "sha512-Z2lWUis7jlmXC1jeOG9giRO2+FsuyNipeQ43HAjqAZjwSe3SEf+q/84FGPHoso3kyntbxa4c4i77t3m6fGf8cw==", - "dev": true, - "requires": { - "is-ssh": "^1.3.0", - "protocols": "^1.4.0", - "qs": "^6.9.4", - "query-string": "^6.13.8" - } - }, "query-string": { "version": "6.14.1", - "resolved": "https://registry.npmjs.org/query-string/-/query-string-6.14.1.tgz", + "resolved": "https://registry.npmmirror.com/query-string/-/query-string-6.14.1.tgz", "integrity": "sha512-XDxAeVmpfu1/6IjyT/gXHOl+S0vQ9owggJ30hhWKdHAsNPOcasn5o9BW0eejZqL2e4vMjhAxoW3jVHcD6mbcYw==", "dev": true, "requires": { @@ -16752,12 +44751,32 @@ }, "strict-uri-encode": { "version": "2.0.0", - "resolved": "https://registry.npmjs.org/strict-uri-encode/-/strict-uri-encode-2.0.0.tgz", + "resolved": "https://registry.npmmirror.com/strict-uri-encode/-/strict-uri-encode-2.0.0.tgz", "integrity": "sha512-QwiXZgpRcKkhTj2Scnn++4PKtWsH0kpzZ62L2R6c/LUVYv7hVnZqcg2+sMuT6R7Jusu1vviK/MFsu6kNJfWlEQ==", "dev": true } } }, + "parse-url": { + "version": "6.0.0", + "resolved": "https://registry.npmmirror.com/parse-url/-/parse-url-6.0.0.tgz", + "integrity": "sha512-cYyojeX7yIIwuJzledIHeLUBVJ6COVLeT4eF+2P6aKVzwvgKQPndCBv3+yQ7pcWjqToYwaligxzSYNNmGoMAvw==", + "dev": true, + "requires": { + "is-ssh": "^1.3.0", + "normalize-url": "^6.1.0", + "parse-path": "^4.0.0", + "protocols": "^1.4.0" + }, + "dependencies": { + "normalize-url": { + "version": "6.1.0", + "resolved": "https://registry.npmmirror.com/normalize-url/-/normalize-url-6.1.0.tgz", + "integrity": "sha512-DlL+XwOy3NxAQ8xuC0okPgK46iuVNAK01YN7RueYBqqFeGsBjV9XmCAzAdgt+667bCl5kPh9EqKKDwnaPG1I7A==", + "dev": true + } + } + }, "parse5": { "version": "5.1.1", "resolved": "https://registry.npmmirror.com/parse5/-/parse5-5.1.1.tgz", @@ -16837,7 +44856,8 @@ "path-dirname": { "version": "1.0.2", "resolved": "https://registry.npmmirror.com/path-dirname/-/path-dirname-1.0.2.tgz", - "integrity": "sha512-ALzNPpyNq9AqXMBjeymIjFDAkAFH06mHJH/cSBHAgU0s4vfpBn6b2nf8tiRLvagKD8RbTpq2FKTBg7cl9l3c7Q==" + "integrity": "sha512-ALzNPpyNq9AqXMBjeymIjFDAkAFH06mHJH/cSBHAgU0s4vfpBn6b2nf8tiRLvagKD8RbTpq2FKTBg7cl9l3c7Q==", + "devOptional": true }, "path-exists": { "version": "3.0.0", @@ -16917,7 +44937,8 @@ "picomatch": { "version": "2.3.1", "resolved": "https://registry.npmmirror.com/picomatch/-/picomatch-2.3.1.tgz", - "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==" + "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", + "devOptional": true }, "pify": { "version": "4.0.1", @@ -17620,7 +45641,8 @@ "version": "1.15.0", "resolved": "https://registry.npmmirror.com/postcss-prefix-selector/-/postcss-prefix-selector-1.15.0.tgz", "integrity": "sha512-9taaTPs6I4906QC03zBBt0LfTWAhrqEWlKSj0jRlxrg1yV+O91h0wcquu6krcA5L6aEv3QnCeG8B1vZ5WT4ecQ==", - "dev": true + "dev": true, + "requires": {} }, "postcss-reduce-initial": { "version": "4.0.3", @@ -18268,6 +46290,11 @@ "picomatch": "^2.2.1" } }, + "reconnecting-websocket": { + "version": "4.4.0", + "resolved": "https://registry.npmmirror.com/reconnecting-websocket/-/reconnecting-websocket-4.4.0.tgz", + "integrity": "sha512-D2E33ceRPga0NvTDhJmphEgJ7FUYF0v4lr1ki0csq06OdlxKfugGzN0dSkxM/NfqCxYELK4KcaTOUOjTV6Dcng==" + }, "redent": { "version": "3.0.0", "resolved": "https://registry.npmmirror.com/redent/-/redent-3.0.0.tgz", @@ -18307,8 +46334,7 @@ "regenerator-runtime": { "version": "0.13.9", "resolved": "https://registry.npmmirror.com/regenerator-runtime/-/regenerator-runtime-0.13.9.tgz", - "integrity": "sha512-p3VT+cOEgxFsRRA9X4lkI1E+k2/CtnKtU4gcxyaCUreilL/vqI6CdZ3wxVUx3UOUg+gnUOQQcRI7BmSI656MYA==", - "dev": true + "integrity": "sha512-p3VT+cOEgxFsRRA9X4lkI1E+k2/CtnKtU4gcxyaCUreilL/vqI6CdZ3wxVUx3UOUg+gnUOQQcRI7BmSI656MYA==" }, "regenerator-transform": { "version": "0.15.0", @@ -18401,7 +46427,8 @@ "remove-trailing-separator": { "version": "1.1.0", "resolved": "https://registry.npmmirror.com/remove-trailing-separator/-/remove-trailing-separator-1.1.0.tgz", - "integrity": "sha512-/hS+Y0u3aOfIETiaiirUFwDBDzmXPvO+jAfKTitUngIPzdKc6Z0LoFjM/CK5PL4C+eKwHohlHAb6H0VFfmmUsw==" + "integrity": "sha512-/hS+Y0u3aOfIETiaiirUFwDBDzmXPvO+jAfKTitUngIPzdKc6Z0LoFjM/CK5PL4C+eKwHohlHAb6H0VFfmmUsw==", + "devOptional": true }, "renderkid": { "version": "2.0.7", @@ -18505,15 +46532,6 @@ "resolved": "https://registry.npmmirror.com/repeat-string/-/repeat-string-1.6.1.tgz", "integrity": "sha512-PV0dzCYDNfRi1jCDbJzpW7jNNDRuCOG/jI5ctQcGKt/clZD+YcPS3yIlWuTJMmESC8aevCFmWJy5wjAFgNqN6w==" }, - "repeating": { - "version": "2.0.1", - "resolved": "https://registry.npmmirror.com/repeating/-/repeating-2.0.1.tgz", - "integrity": "sha512-ZqtSMuVybkISo2OWvqvm7iHSWngvdaW3IpsT9/uP8v4gMi591LY6h35wdOfvQdWCKFWZWm2Y1Opp4kV7vQKT6A==", - "dev": true, - "requires": { - "is-finite": "^1.0.0" - } - }, "request": { "version": "2.88.2", "resolved": "https://registry.npmmirror.com/request/-/request-2.88.2.tgz", @@ -18842,22 +46860,68 @@ } }, "sass-loader": { - "version": "7.3.1", - "resolved": "https://registry.npmmirror.com/sass-loader/-/sass-loader-7.3.1.tgz", - "integrity": "sha512-tuU7+zm0pTCynKYHpdqaPpe+MMTQ76I9TPZ7i4/5dZsigE350shQWe5EZNl5dBidM49TPET75tNqRbcsUZWeNA==", + "version": "10.2.1", + "resolved": "https://registry.npmmirror.com/sass-loader/-/sass-loader-10.2.1.tgz", + "integrity": "sha512-RRvWl+3K2LSMezIsd008ErK4rk6CulIMSwrcc2aZvjymUgKo/vjXGp1rSWmfTUX7bblEOz8tst4wBwWtCGBqKA==", "dev": true, "requires": { - "clone-deep": "^4.0.1", - "loader-utils": "^1.0.1", - "neo-async": "^2.5.0", - "pify": "^4.0.1", - "semver": "^6.3.0" + "klona": "^2.0.4", + "loader-utils": "^2.0.0", + "neo-async": "^2.6.2", + "schema-utils": "^3.0.0", + "semver": "^7.3.2" }, "dependencies": { + "json5": { + "version": "2.2.1", + "resolved": "https://registry.npmmirror.com/json5/-/json5-2.2.1.tgz", + "integrity": "sha512-1hqLFMSrGHRHxav9q9gNjJ5EXznIxGVO09xQRrwplcS8qs28pZ8s8hupZAmqDwZUmVZ2Qb2jnyPOWcDH8m8dlA==", + "dev": true + }, + "loader-utils": { + "version": "2.0.4", + "resolved": "https://registry.npmmirror.com/loader-utils/-/loader-utils-2.0.4.tgz", + "integrity": "sha512-xXqpXoINfFhgua9xiqD8fPFHgkoq1mmmpE92WlDbm9rNRd/EbRb+Gqf908T2DMfuHjjJlksiK2RbHVOdD/MqSw==", + "dev": true, + "requires": { + "big.js": "^5.2.2", + "emojis-list": "^3.0.0", + "json5": "^2.1.2" + } + }, + "lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmmirror.com/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "requires": { + "yallist": "^4.0.0" + } + }, + "schema-utils": { + "version": "3.1.1", + "resolved": "https://registry.npmmirror.com/schema-utils/-/schema-utils-3.1.1.tgz", + "integrity": "sha512-Y5PQxS4ITlC+EahLuXaY86TXfR7Dc5lw294alXOq86JAHCihAIZfqv8nNCWvaEJvaC51uN9hbLGeV0cFBdH+Fw==", + "dev": true, + "requires": { + "@types/json-schema": "^7.0.8", + "ajv": "^6.12.5", + "ajv-keywords": "^3.5.2" + } + }, "semver": { - "version": "6.3.0", - "resolved": "https://registry.npmmirror.com/semver/-/semver-6.3.0.tgz", - "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "version": "7.3.8", + "resolved": "https://registry.npmmirror.com/semver/-/semver-7.3.8.tgz", + "integrity": "sha512-NB1ctGL5rlHrPJtFDVIVzTyQylMLu9N9VICA6HSFJo8MCGVTMW6gfpicwKmmK/dAjTOrqu5l63JJOpDSrAis3A==", + "dev": true, + "requires": { + "lru-cache": "^6.0.0" + } + }, + "yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmmirror.com/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", "dev": true } } @@ -18898,6 +46962,11 @@ } } }, + "select": { + "version": "1.1.2", + "resolved": "https://registry.npmmirror.com/select/-/select-1.1.2.tgz", + "integrity": "sha512-OwpTSOfy6xSs1+pwcNrv0RBMOzI39Lp3qQKUTPVVPRjCdNa5JH/oPRiqsesIskK8TVgmRiHwO4KXlV2Li9dANA==" + }, "select-hose": { "version": "2.0.0", "resolved": "https://registry.npmmirror.com/select-hose/-/select-hose-2.0.0.tgz", @@ -19389,6 +47458,11 @@ "is-plain-obj": "^1.0.0" } }, + "sortablejs": { + "version": "1.10.2", + "resolved": "https://registry.npmmirror.com/sortablejs/-/sortablejs-1.10.2.tgz", + "integrity": "sha512-YkPGufevysvfwn5rfdlGyrGjt7/CRHwvRPogD/lC+TnvcN29jDpCifKP+rBqf+LRldfXSTh+0CGLcSg0VIxq3A==" + }, "source-list-map": { "version": "2.0.1", "resolved": "https://registry.npmmirror.com/source-list-map/-/source-list-map-2.0.1.tgz", @@ -19769,6 +47843,14 @@ "integrity": "sha512-R3f198pcvnB+5IpnBlRkphuE9n46WyVl8I39W/ZUTZLz4nqSP/oLYUrcnJrw462Ds8he4YKMov2efsTIw1BDGQ==", "dev": true }, + "string_decoder": { + "version": "1.1.1", + "resolved": "https://registry.npmmirror.com/string_decoder/-/string_decoder-1.1.1.tgz", + "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", + "requires": { + "safe-buffer": "~5.1.0" + } + }, "string-argv": { "version": "0.0.2", "resolved": "https://registry.npmmirror.com/string-argv/-/string-argv-0.0.2.tgz", @@ -19842,14 +47924,6 @@ "define-properties": "^1.1.3" } }, - "string_decoder": { - "version": "1.1.1", - "resolved": "https://registry.npmmirror.com/string_decoder/-/string_decoder-1.1.1.tgz", - "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", - "requires": { - "safe-buffer": "~5.1.0" - } - }, "stringify-object": { "version": "3.3.0", "resolved": "https://registry.npmmirror.com/stringify-object/-/stringify-object-3.3.0.tgz", @@ -19920,6 +47994,11 @@ "through": "^2.3.4" } }, + "style-mod": { + "version": "4.0.3", + "resolved": "https://registry.npmmirror.com/style-mod/-/style-mod-4.0.3.tgz", + "integrity": "sha512-78Jv8kYJdjbvRwwijtCevYADfsI0lGzYJe4mMFdceO8l75DFFDoqBhR1jVDicDRRaX4//g1u9wKeo+ztc2h1Rw==" + }, "stylehacks": { "version": "4.0.3", "resolved": "https://registry.npmmirror.com/stylehacks/-/stylehacks-4.0.3.tgz", @@ -20278,7 +48357,8 @@ "resolved": "https://registry.npmmirror.com/ajv-keywords/-/ajv-keywords-2.1.1.tgz", "integrity": "sha512-ZFztHzVRdGLAzJmpUT9LNFLe1YiVOEylcaNpEutM26PVTCtOD919IMfD01CgbRouB42Dd9atjx1HseC15DgOZA==", "dev": true, - "optional": true + "optional": true, + "requires": {} }, "fast-deep-equal": { "version": "1.1.0", @@ -20508,6 +48588,11 @@ "integrity": "sha512-qsdtZH+vMoCARQtyod4imc2nIJwg9Cc7lPRrw9CzF8ZKR0khdr8+2nX80PBhET3tcyTtJDxAffGh2rXH4tyU8A==", "dev": true }, + "tiny-emitter": { + "version": "2.1.0", + "resolved": "https://registry.npmmirror.com/tiny-emitter/-/tiny-emitter-2.1.0.tgz", + "integrity": "sha512-NB6Dk1A9xgQPMoGqC5CVXn123gWyte215ONT5Pp5a0yt4nlEoO1ZWeCwpncaekPHXO60i47ihFnZPiRPjRMq4Q==" + }, "tinycolor2": { "version": "1.4.2", "resolved": "https://registry.npmmirror.com/tinycolor2/-/tinycolor2-1.4.2.tgz", @@ -20887,7 +48972,8 @@ "upath": { "version": "1.2.0", "resolved": "https://registry.npmmirror.com/upath/-/upath-1.2.0.tgz", - "integrity": "sha512-aZwGpamFO61g3OlfT7OQCHqhGnW43ieH9WZeP7QxN/G/jS4jfqUkZxoryvJgVPEcrl5NL/ggHsSmLMHuH64Lhg==" + "integrity": "sha512-aZwGpamFO61g3OlfT7OQCHqhGnW43ieH9WZeP7QxN/G/jS4jfqUkZxoryvJgVPEcrl5NL/ggHsSmLMHuH64Lhg==", + "devOptional": true }, "upper-case": { "version": "1.1.3", @@ -21083,6 +49169,54 @@ "resolved": "https://registry.npmmirror.com/vm-browserify/-/vm-browserify-1.1.2.tgz", "integrity": "sha512-2ham8XPWTONajOR0ohOKOHXkm3+gaBmGut3SRuu75xLd/RRaY6vqgh8NBYYk7+RW3u5AtzPQZG8F10LHkl0lAQ==" }, + "vscode-jsonrpc": { + "version": "5.0.1", + "resolved": "https://registry.npmmirror.com/vscode-jsonrpc/-/vscode-jsonrpc-5.0.1.tgz", + "integrity": "sha512-JvONPptw3GAQGXlVV2utDcHx0BiY34FupW/kI6mZ5x06ER5DdPG/tXWMVHjTNULF5uKPOUUD0SaXg5QaubJL0A==" + }, + "vscode-languageclient": { + "version": "6.1.4", + "resolved": "https://registry.npmmirror.com/vscode-languageclient/-/vscode-languageclient-6.1.4.tgz", + "integrity": "sha512-EUOU+bJu6axmt0RFNo3nrglQLPXMfanbYViJee3Fbn2VuQoX0ZOI4uTYhSRvYLP2vfwTP/juV62P/mksCdTZMA==", + "requires": { + "semver": "^6.3.0", + "vscode-languageserver-protocol": "3.15.3" + }, + "dependencies": { + "semver": { + "version": "6.3.0", + "resolved": "https://registry.npmmirror.com/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==" + } + } + }, + "vscode-languageserver-protocol": { + "version": "3.15.3", + "resolved": "https://registry.npmmirror.com/vscode-languageserver-protocol/-/vscode-languageserver-protocol-3.15.3.tgz", + "integrity": "sha512-zrMuwHOAQRhjDSnflWdJG+O2ztMWss8GqUUB8dXLR/FPenwkiBNkMIJJYfSN6sgskvsF0rHAoBowNQfbyZnnvw==", + "requires": { + "vscode-jsonrpc": "^5.0.1", + "vscode-languageserver-types": "3.15.1" + } + }, + "vscode-languageserver-types": { + "version": "3.15.1", + "resolved": "https://registry.npmmirror.com/vscode-languageserver-types/-/vscode-languageserver-types-3.15.1.tgz", + "integrity": "sha512-+a9MPUQrNGRrGU630OGbYVQ+11iOIovjCkqxajPa9w57Sd5ruK8WQNsslzpa0x/QJqC8kRc2DUxWjIFwoNm4ZQ==" + }, + "vscode-uri": { + "version": "2.1.2", + "resolved": "https://registry.npmmirror.com/vscode-uri/-/vscode-uri-2.1.2.tgz", + "integrity": "sha512-8TEXQxlldWAuIODdukIb+TR5s+9Ds40eSJrw+1iDDA9IFORPjMELarNQE3myz5XIkWWpdprmJjm1/SxMlWOC8A==" + }, + "vscode-ws-jsonrpc": { + "version": "0.2.0", + "resolved": "https://registry.npmmirror.com/vscode-ws-jsonrpc/-/vscode-ws-jsonrpc-0.2.0.tgz", + "integrity": "sha512-NE9HNRgPjCaPyTJvIudcpyIWPImxwRDtuTX16yks7SAiZgSXigxAiZOvSvVBGmD1G/OMfrFo6BblOtjVR9DdVA==", + "requires": { + "vscode-jsonrpc": "^5.0.0" + } + }, "vue": { "version": "2.6.12", "resolved": "https://registry.npmmirror.com/vue/-/vue-2.6.12.tgz", @@ -21098,6 +49232,15 @@ "mockjs": "^1.0.1-beta3" } }, + "vue-codemirror": { + "version": "4.0.6", + "resolved": "https://registry.npmmirror.com/vue-codemirror/-/vue-codemirror-4.0.6.tgz", + "integrity": "sha512-ilU7Uf0mqBNSSV3KT7FNEeRIxH4s1fmpG4TfHlzvXn0QiQAbkXS9lLfwuZpaBVEnpP5CSE62iGJjoliTuA8poQ==", + "requires": { + "codemirror": "^5.41.0", + "diff-match-patch": "^1.0.0" + } + }, "vue-eslint-parser": { "version": "2.0.3", "resolved": "https://registry.npmmirror.com/vue-eslint-parser/-/vue-eslint-parser-2.0.3.tgz", @@ -21198,10 +49341,29 @@ "integrity": "sha512-4gDntzrifFnCEvyoO8PqyJDmguXgVPxKiIxrBKjIowvL9l+N66196+72XVYR8BBf1Uv1Fgt3bGevJ+sEmxfZzw==", "dev": true }, + "vuedraggable": { + "version": "2.24.3", + "resolved": "https://registry.npmmirror.com/vuedraggable/-/vuedraggable-2.24.3.tgz", + "integrity": "sha512-6/HDXi92GzB+Hcs9fC6PAAozK1RLt1ewPTLjK0anTYguXLAeySDmcnqE8IC0xa7shvSzRjQXq3/+dsZ7ETGF3g==", + "requires": { + "sortablejs": "1.10.2" + } + }, "vuescroll": { "version": "4.16.1", "resolved": "https://registry.npmmirror.com/vuescroll/-/vuescroll-4.16.1.tgz", - "integrity": "sha512-7fRsG2Yw5Z07LUz/IIu9barpmYiN9q+ZTC+CrVamvCbmsxyhz8mU1OuYFbfORysaUskioNMxTGDo+HOzeDfSyQ==" + "integrity": "sha512-7fRsG2Yw5Z07LUz/IIu9barpmYiN9q+ZTC+CrVamvCbmsxyhz8mU1OuYFbfORysaUskioNMxTGDo+HOzeDfSyQ==", + "requires": {} + }, + "w3c-keyname": { + "version": "2.2.8", + "resolved": "https://registry.npmmirror.com/w3c-keyname/-/w3c-keyname-2.2.8.tgz", + "integrity": "sha512-dpojBhNsCNN7T82Tm7k26A6G9ML3NkhDsnw9n/eoxSRlVBB4CEtIQ/KTCLI2Fwf3ataSXRhYFkQi3SlnFwPvPQ==" + }, + "watch-size": { + "version": "2.0.0", + "resolved": "https://registry.npmmirror.com/watch-size/-/watch-size-2.0.0.tgz", + "integrity": "sha512-M92R89dNoTPWyCD+HuUEDdhaDnh9jxPGOwlDc0u51jAgmjUvzqaEMynXSr3BaWs+QdHYk4KzibPy1TFtjLmOZQ==" }, "watchpack": { "version": "1.7.5", @@ -22073,6 +50235,15 @@ "async-limiter": "~1.0.0" } }, + "xss": { + "version": "1.0.11", + "resolved": "https://registry.npmmirror.com/xss/-/xss-1.0.11.tgz", + "integrity": "sha512-EimjrjThZeK2MO7WKR9mN5ZC1CSqivSl55wvUK5EtU6acf0rzEE1pN+9ZDrFXJ82BRp3JL38pPE6S4o/rpp1zQ==", + "requires": { + "commander": "^2.20.3", + "cssfilter": "0.0.10" + } + }, "xtend": { "version": "4.0.2", "resolved": "https://registry.npmmirror.com/xtend/-/xtend-4.0.2.tgz", diff --git a/web/package.json b/web/package.json index 7d905cc8d2..30bd9164cc 100644 --- a/web/package.json +++ b/web/package.json @@ -1,10 +1,14 @@ { "name": "dataspherestudio", - "version": "1.1.4", + "version": "1.1.6", "private": true, "scripts": { - "serve": "npm run postinstall && cd packages/dss && npm run serve", - "build": "npm run postinstall && cd packages/dss && npm run build", + "serve": "patch-package && cd packages/dss && npm run serve", + "build": "patch-package && cd packages/dss && npm run build", + "serve-sandbox": "patch-package && npm run serve --configfile=config.sandbox.json", + "build-sandbox": "patch-package && npm run build --configfile=config.sandbox.json", + "serve-scriptis": "patch-package && npm run serve --module=scriptis --micro_module=scriptis", + "build-scriptis": "patch-package && npm run build --module=scriptis --micro_module=scriptis", "lint": "vue-cli-service lint --no-fix", "fix": "vue-cli-service lint --fix", "precommit": "lint-staged", @@ -61,8 +65,8 @@ "less-loader": "6.1.0", "lint-staged": "8.2.1", "monaco-editor-webpack-plugin": "1.8.2", - "node-sass": "4.14.1", - "sass-loader": "7.3.1", + "node-sass": "6.0.1", + "sass-loader": "10.2.1", "speed-measure-webpack-plugin": "1.5.0", "svg-sprite-loader": "5.0.0", "patch-package": "6.2.2", diff --git a/web/packages/apiServices/i18n/en.json b/web/packages/apiServices/i18n/en.json index 05923fe8cd..ea3eceb9d0 100644 --- a/web/packages/apiServices/i18n/en.json +++ b/web/packages/apiServices/i18n/en.json @@ -67,7 +67,7 @@ "visibleRole": "Visible Role:", "test": "Test:", "applicant": "Applicant:", - "status": "Status:", + "status": "Status", "find": "Find", "path": "API Path:", "submitter": "Submitter", @@ -256,7 +256,17 @@ "notice": { "publishSuccess": "Publish Success" }, - "tagPlaceholder": "tag,click enter add" + "tagPlaceholder": "tag,click enter add", + "istitle": "This is the title", + "isdesc": "This is the description", + "cancel": "Cancel", + "tuned": "Stay tuned!", + "manage": "Management", + "disableenable": "Disable/Enable", + "more1024": "Cannot exceed 1024 characters!", + "moreline": "Too many rows, please query in batches!", + "outlimit": "Exceed the limit, please modify!", + "uninittask": "Nothing has been returned yet,Please try again later or contact administrator!" } } } diff --git a/web/packages/apiServices/i18n/zh.json b/web/packages/apiServices/i18n/zh.json index cd6505c215..3ed8a18b80 100644 --- a/web/packages/apiServices/i18n/zh.json +++ b/web/packages/apiServices/i18n/zh.json @@ -67,7 +67,7 @@ "visibleRole": "可见角色:", "test": "测试:", "applicant": "申请人:", - "status": "状态:", + "status": "状态", "path": "API路径:", "submitter": "提交人", "find": "查询", @@ -256,7 +256,17 @@ "buttonText": "查询" } }, - "tagPlaceholder": "标签,按 enter 创建" + "tagPlaceholder": "标签,按 enter 创建", + "istitle": "我是标题", + "isdesc": "我是描述内容", + "cancel": "取消", + "tuned": "暂未开放!", + "manage": "管理", + "disableenable": "禁用/启用", + "more1024": "不能超过1024个字符!", + "moreline": "行数过多请分批查询!", + "outlimit": "超出限制请修改!", + "uninittask": "任务暂时未生成返回,请稍后再试或联系管理员!" } } } diff --git a/web/packages/apiServices/module/apiServices/apiCard.vue b/web/packages/apiServices/module/apiServices/apiCard.vue index 0af6174a36..d8d90176f3 100644 --- a/web/packages/apiServices/module/apiServices/apiCard.vue +++ b/web/packages/apiServices/module/apiServices/apiCard.vue @@ -39,6 +39,7 @@ diff --git a/web/packages/dss/package.json b/web/packages/dss/package.json index f046eeffca..3abc2d65ef 100644 --- a/web/packages/dss/package.json +++ b/web/packages/dss/package.json @@ -1,19 +1,19 @@ { "name": "@dataspherestudio/dss", - "version": "1.1.4", + "version": "1.1.12", "dependencies": { - "@dataspherestudio/scriptis": "^1.1.4", - "@dataspherestudio/shared": "^1.1.4" + "@dataspherestudio/scriptis": "^1.1.12", + "@dataspherestudio/shared": "^1.1.12" }, "devDependencies": { "@vue/cli-plugin-babel": "3.12.1", "@vue/cli-plugin-eslint": "3.12.1", "@vue/cli-service": "3.12.1", - "@vue/eslint-config-standard": "4.0.0" + "@vue/eslint-config-standard": "4.0.0", + "babel-plugin-transform-remove-console": "6.9.4" }, "scripts": { "serve": "vue-cli-service serve", - "build": "vue-cli-service build", - "build-sandbox": "vue-cli-service build --mode sandbox" + "build": "vue-cli-service build" } } diff --git a/web/packages/dss/src/common-router.js b/web/packages/dss/src/common-router.js index 57bebd860a..83334c63b1 100644 --- a/web/packages/dss/src/common-router.js +++ b/web/packages/dss/src/common-router.js @@ -15,6 +15,15 @@ export const subAppRoutes = { component: () => import('../view/commonIframe/index.vue'), }, + { + path: '/microApp/:appName', + name: 'microAppContainer', + meta: { + publicPage: true, + }, + component: () => + import('../view/microApp/index.vue'), + }, // 新增一个redirect路由mock路由刷新效果,放在这里共享layout可保持header不会unmount,减少ajax { path: '/redirect/:path(.*)', diff --git a/web/packages/dss/src/dynamic-apps.js b/web/packages/dss/src/dynamic-apps.js index 3ee4510278..8d48071638 100644 --- a/web/packages/dss/src/dynamic-apps.js +++ b/web/packages/dss/src/dynamic-apps.js @@ -45,7 +45,6 @@ if (apps.appsI18n) { merge(i18nConfig, item) }); } - export { subRoutes, i18nConfig, diff --git a/web/packages/dss/src/main.js b/web/packages/dss/src/main.js index 09b3054947..cb7c762f1f 100644 --- a/web/packages/dss/src/main.js +++ b/web/packages/dss/src/main.js @@ -37,7 +37,24 @@ import 'iview/dist/styles/iview.css' import '@dataspherestudio/shared/common/style/theme/default.less' // Icon -import '@dataspherestudio/shared/components/svgIcon/index.js' +import SvgIcon from '@dataspherestudio/shared/components/svgIcon/index.vue'// svg component + + +// register globally +Vue.component('SvgIcon', SvgIcon) + +if (apps.vuecomps) { + Object.keys(apps.vuecomps).forEach(it => { + if (apps.vuecomps[it].default && apps.vuecomps[it].default.install) { + Vue.use(apps.vuecomps[it].default) + } else { + Vue.component(it, apps.vuecomps[it].default) + } + }) +} + +import('@dataspherestudio/shared/components/svgIcon/index.js') + import '../module/index.js' // 扩展模块 @@ -77,7 +94,7 @@ Vue.prototype.$API_PATH = API_PATH; Vue.prototype.$APP_CONF = apps.conf || {}; new Vue({ - router, + router: router(), i18n, render: (h) => h(App) }).$mount('#app') diff --git a/web/packages/dss/src/router.js b/web/packages/dss/src/router.js index cc0b81f112..be2e426b41 100644 --- a/web/packages/dss/src/router.js +++ b/web/packages/dss/src/router.js @@ -14,12 +14,29 @@ * limitations under the License. * */ - -import VueRouter from "vue-router"; +import axios from 'axios' +import VueRouter from "vue-router" import routes from "./common-router" import { apps, subRoutes } from './dynamic-apps' -import storage from '@dataspherestudio/shared/common/helper/storage'; +import storage from '@dataspherestudio/shared/common/helper/storage' +import plugin from '@dataspherestudio/shared/common/util/plugin' +/** + * 检查是否需要更新前端资源 + * 当前页面app.{hash}.js与服务端index.html文件里对应script文件hash不同则需要提示更新 + */ +async function checkNeedShowUpdate() { + const { data } = await axios.get(`/index.html?t=${Date.now()}`) + let serverAppHash = data.split(' diff --git a/web/packages/dss/view/commonIframe/index.vue b/web/packages/dss/view/commonIframe/index.vue index 4082b8a5ef..417d722f48 100644 --- a/web/packages/dss/view/commonIframe/index.vue +++ b/web/packages/dss/view/commonIframe/index.vue @@ -1,6 +1,7 @@ + + diff --git a/web/packages/dss/vue.config.js b/web/packages/dss/vue.config.js index 51d26fbaa0..7c0db0ac2f 100644 --- a/web/packages/dss/vue.config.js +++ b/web/packages/dss/vue.config.js @@ -22,9 +22,16 @@ const MonacoWebpackPlugin = require('monaco-editor-webpack-plugin'); const SpeedMeasurePlugin = require("speed-measure-webpack-plugin"); const VirtualModulesPlugin = require('webpack-virtual-modules'); const webpack = require("webpack"); -const { apps, exts, conf, version } = require('../../config.json') const child_process = require('child_process'); +let configFile= `../../config.json` +if (process.env.npm_config_configfile) { + configFile = `../../${process.env.npm_config_configfile}` +} else if (process.env.npm_config_micro_module) { + configFile = `../../config.${process.env.npm_config_micro_module}.json` +} +const { apps = {}, exts = {}, conf = {}, version, components = {} } = require(configFile) + if (version) { process.env.VUE_APP_VERSION = version } @@ -87,6 +94,13 @@ Object.keys(exts).forEach((item, index) => { } }) +//vue components +let vuecomps = [] +Object.keys(components).forEach(item => { + vuecomps.push(`${item}: require('@/${components[item]}')`) +}) + +// config Object.keys(conf).forEach(item=> { if(['app_logo'].includes(item)) { confs.push(`${item}: require('@/${conf[item]}')`) @@ -115,6 +129,7 @@ const virtualModules = new VirtualModulesPlugin({ modules: ${JSON.stringify(modules)}, exts: {${extsMoule.join(',')}}, appsRoutes: {${appsRoutes.join(',')}}, + vuecomps: {${vuecomps.join(',')}}, appsI18n: [${appsI18n.join(',')}], requireComponent: [${requireComponent.join(',')}], requireComponentVue: [${requireComponentVue.join(',')}], @@ -218,6 +233,7 @@ module.exports = { resolve: { alias: { '@': path.resolve(__dirname, '../'), + 'vscode': require.resolve('monaco-languageclient/lib/vscode-compatibility') } }, plugins diff --git a/web/packages/shared/components/editor/compare.vue b/web/packages/editor/compare.vue similarity index 85% rename from web/packages/shared/components/editor/compare.vue rename to web/packages/editor/compare.vue index 082b400608..9496a68d98 100644 --- a/web/packages/shared/components/editor/compare.vue +++ b/web/packages/editor/compare.vue @@ -8,7 +8,7 @@ {{ isFullscreen ? $t('message.scripts.constants.logPanelList.releaseFullScreen') : $t('message.scripts.constants.logPanelList.fullScreen') }} -

+
@@ -20,7 +20,7 @@ const defaultToolbar = { mergeDefaults: true } export default { - name: 'MonacoEditor', + name: 'WeEditorCompare', props: { original: String, value: { @@ -41,15 +41,6 @@ export default { type: [Boolean, Object], default: () => defaultToolbar }, - height: { - type: Number, - default: 120 - }, - //fix--esc退出全屏&modal退出事件冲突 - disableEsc: { - type: Boolean, - default: false - }, readOnly: { type: Boolean, default: false @@ -62,8 +53,7 @@ export default { data(){ return { - isFullscreen: false, - editorHeight: this.height + 'px' + isFullscreen: false } }, @@ -106,13 +96,6 @@ export default { } }, - height(newVal) { - if(typeof newVal === 'number') { - this.editorHeight = newVal + 'px' - this.layout(); - } - }, - diffEditor() { this.changeModel() } @@ -126,14 +109,14 @@ export default { document.addEventListener('keyup', this.esc, false); this.initMonaco(monaco) this.changeTheme(localStorage.getItem('theme')); - eventbus.on('monaco.change', this.changeTheme); + eventbus.on('theme.change', this.changeTheme); this.monaco = monaco; }, beforeDestroy() { document.removeEventListener('keyup', this.esc, false); - eventbus.off('monaco.change', this.changeTheme); + eventbus.off('theme.change', this.changeTheme); this.editor && this.editor.dispose() }, @@ -182,9 +165,6 @@ export default { const value = editor.getValue(); this.$emit('change', value, event); }) - if (this.height) { - this.$refs.editor.style.height = this.height + 'px'; - } this.layout(); this.$emit('editorDidMount', this.editor) }, @@ -202,7 +182,11 @@ export default { }, fullAction() { - this.isFullscreen = !this.isFullscreen + if (this.isFullscreen) { + this.exitFullScreen() + } else { + this.fullScreen() + } }, layout() { @@ -219,17 +203,15 @@ export default { }, exitFullScreen() { - this.editorHeight = this.height + 'px'; this.isFullscreen = false; this.layout(); - this.$emit('esc-key-event', false) + this.$emit('full-screen-change', false) }, fullScreen() { - this.editorHeight = '100%'; this.isFullscreen = true; this.layout(); - this.$emit('esc-key-event', true) + this.$emit('full-screen-change', true) }, changeTheme(theme) { @@ -248,6 +230,7 @@ export default { .wrap { position: relative; border: 1px solid #eee; + height: 100%; } .toolbar { height: 30px; @@ -261,6 +244,8 @@ export default { } .el-editor { width: 100%; + height: calc(100% - 32px); + min-height: 380px; } .workbench-body-navbar-item { margin: 0 16px; @@ -274,7 +259,7 @@ export default { border: 0; right: 0; width: 100%; - height: 99%; + height: 100%; background: #fff; z-index: 9999; } diff --git a/web/packages/shared/components/editor/editor.vue b/web/packages/editor/editor.vue similarity index 88% rename from web/packages/shared/components/editor/editor.vue rename to web/packages/editor/editor.vue index 8ad7388db6..692d5086c9 100644 --- a/web/packages/shared/components/editor/editor.vue +++ b/web/packages/editor/editor.vue @@ -51,12 +51,15 @@ export default { application: String, }, data() { + const closeSuggest = storage.get('close_db_table_suggest', 'local') + const autobreak = storage.get('editor_auto_breakline', 'local') return { editor: null, editorModel: null, decorations: null, isParserClose: true, // 默认关闭语法验证 - dbtbsuggest: true, // 默认打开库表联想 + dbtbsuggest: !closeSuggest, // 默认打开库表联想 + autobreak: autobreak, // 默认关闭自动换行 closeParser: null, openParser: null, sqlParser: null, @@ -137,12 +140,12 @@ export default { mounted() { this.initMonaco(); this.changeTheme(localStorage.getItem('theme')); - eventbus.on('monaco.change', this.changeTheme); + eventbus.on('theme.change', this.changeTheme); }, beforeDestroy: function() { // 销毁 editor,进行gc this.editor && this.editor.dispose(); - eventbus.off('monaco.change', this.changeTheme); + eventbus.off('theme.change', this.changeTheme); }, methods: { // 初始化 @@ -167,7 +170,7 @@ export default { }), 100); this.editor.onContextMenu(debounce(() => { // 需要调换文字的右键菜单功能 - const selectList = [{label: 'Change All Occurrences', text: '改变所有出现'}, {label: 'Format Document', text: '格式化'}, {label: 'Command Palette', text: '命令面板'}, {label: 'Cut', text: '剪切'}, {label: 'Copy', text: '复制'}]; + const selectList = [{label: 'Change All Occurrences', text: '改变所有出现'}, {label: 'Format Document', text: '格式化'}, {label: 'Command Palette', text: '命令面板'}, {label: 'Cut', text: this.$t('message.common.Cut')}, {label: 'Copy', text: this.$t('message.common.copy')}]; if (localStorage.getItem('locale') === 'zh-CN') { selectList.forEach((item) => { let elmentList = document.querySelectorAll(`.actions-container .action-label[aria-label="${item.label}"]`); @@ -397,7 +400,7 @@ export default { // 控制右键菜单的显示 vm.openDbTbSuggest.set(true); vm.closeDbTbSuggest.set(false); - storage.set('close_db_table_suggest', true) + storage.set('close_db_table_suggest', true, 'local') }, }); @@ -413,10 +416,64 @@ export default { vm.dbtbsuggest = true; vm.openDbTbSuggest.set(false); vm.closeDbTbSuggest.set(true); - storage.set('close_db_table_suggest', false) + storage.set('close_db_table_suggest', false, 'local') }, }); + // 打开、关闭自动换行 + this.closeAutoBreak = this.editor.createContextKey('closeAutoBreak', this.autobreak); + this.openAutoBreak = this.editor.createContextKey('openAutoBreak', !this.autobreak); + this.editor.addAction({ + id: 'closeAutoBreak', + label: this.$t('message.common.monacoMenu.GBZDHH'), + keybindings: [], + keybindingContext: null, + // 用于控制右键菜单的显示 + precondition: 'closeAutoBreak', + contextMenuGroupId: 'control', + contextMenuOrder: 2.4, + run() { + vm.autobreak = false; + // 控制右键菜单的显示 + vm.openAutoBreak.set(true); + vm.closeAutoBreak.set(false); + storage.set('editor_auto_breakline', true, 'local'); + vm.editor.updateOptions({wordWrap: 'off'}); + }, + }); + + this.editor.addAction({ + id: 'openAutoBreak', + label: this.$t('message.common.monacoMenu.DKZDHH'), + keybindings: [], + keybindingContext: null, + precondition: 'openAutoBreak', + contextMenuGroupId: 'control', + contextMenuOrder: 2.5, + run() { + vm.autobreak = true; + vm.openAutoBreak.set(false); + vm.closeAutoBreak.set(true); + storage.set('editor_auto_breakline', false, 'local'); + vm.editor.updateOptions({wordWrap: 'on'}); + }, + }); + + if (this.$APP_CONF && this.$APP_CONF.lsp_service) { + this.editor.addAction({ + id: 'newdbsuggest', + label: this.$t('message.common.monacoMenu.newdbcomplition'), + keybindings: [], + keybindingContext: null, + contextMenuGroupId: 'control', + contextMenuOrder: 2.5, + run() { + localStorage.setItem('scriptis-edditor-type', 'lsp'); + location.reload(); + }, + }); + } + if (this.language === 'hql') { // 控制语法检查 this.closeParser = this.editor.createContextKey('closeParser', !this.isParserClose); diff --git a/web/packages/shared/components/editor/highRiskGrammar.js b/web/packages/editor/highRiskGrammar.js similarity index 100% rename from web/packages/shared/components/editor/highRiskGrammar.js rename to web/packages/editor/highRiskGrammar.js diff --git a/web/packages/editor/index.js b/web/packages/editor/index.js new file mode 100644 index 0000000000..479de4e956 --- /dev/null +++ b/web/packages/editor/index.js @@ -0,0 +1,32 @@ +/* + * Copyright 2019 WeBank + * + * 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 weEditor from './editor.vue'; +import weEditorCompare from './compare.vue'; + +const install = function(Vue) { + const uselsp = localStorage.getItem('scriptis-edditor-type') === 'lsp' + if (!uselsp) { + Vue.component('WeEditor', weEditor); + Vue.component('WeEditorCompare', weEditorCompare); + } +}; + +export default { + install, +}; + diff --git a/web/packages/shared/components/editor/index.scss b/web/packages/editor/index.scss similarity index 94% rename from web/packages/shared/components/editor/index.scss rename to web/packages/editor/index.scss index 0cf733d9de..ef269f8ce0 100644 --- a/web/packages/shared/components/editor/index.scss +++ b/web/packages/editor/index.scss @@ -20,7 +20,9 @@ .we-editor{ height: 100%; width: 100%; - + .monaco-menu .monaco-action-bar.vertical .action-menu-item { + height: 1.6em; + } .monaco-menu-container a:hover { color: #fff !important; } diff --git a/web/packages/shared/components/editor/keyword/hql.js b/web/packages/editor/keyword/hql.js similarity index 100% rename from web/packages/shared/components/editor/keyword/hql.js rename to web/packages/editor/keyword/hql.js diff --git a/web/packages/shared/components/editor/keyword/python.js b/web/packages/editor/keyword/python.js similarity index 100% rename from web/packages/shared/components/editor/keyword/python.js rename to web/packages/editor/keyword/python.js diff --git a/web/packages/shared/components/editor/keyword/sh.js b/web/packages/editor/keyword/sh.js similarity index 99% rename from web/packages/shared/components/editor/keyword/sh.js rename to web/packages/editor/keyword/sh.js index ba25adc70c..438486116a 100644 --- a/web/packages/shared/components/editor/keyword/sh.js +++ b/web/packages/editor/keyword/sh.js @@ -616,7 +616,7 @@ export default { if (keywordMatch) { const matchList = keywordMatch[0].split(' '); const match = matchList[matchList.length - 1]; - const list = getReturnList(match, shellProposals, 'insertText'); + const list = getReturnList({match, proposals: shellProposals, fieldString: 'insertText'}); return list; } return []; diff --git a/web/packages/shared/components/editor/languages/hql.js b/web/packages/editor/languages/hql.js similarity index 100% rename from web/packages/shared/components/editor/languages/hql.js rename to web/packages/editor/languages/hql.js diff --git a/web/packages/shared/components/editor/languages/log.js b/web/packages/editor/languages/log.js similarity index 100% rename from web/packages/shared/components/editor/languages/log.js rename to web/packages/editor/languages/log.js diff --git a/web/packages/shared/components/editor/languages/out.js b/web/packages/editor/languages/out.js similarity index 100% rename from web/packages/shared/components/editor/languages/out.js rename to web/packages/editor/languages/out.js diff --git a/web/packages/shared/components/editor/languages/sh.js b/web/packages/editor/languages/sh.js similarity index 100% rename from web/packages/shared/components/editor/languages/sh.js rename to web/packages/editor/languages/sh.js diff --git a/web/packages/shared/components/editor/monaco-loader.js b/web/packages/editor/monaco-loader.js similarity index 90% rename from web/packages/shared/components/editor/monaco-loader.js rename to web/packages/editor/monaco-loader.js index 4c335452f2..0106eb279f 100644 --- a/web/packages/shared/components/editor/monaco-loader.js +++ b/web/packages/editor/monaco-loader.js @@ -17,14 +17,12 @@ import hql from './languages/hql'; import log from './languages/log'; -import sas from './languages/sas'; import sh from './languages/sh'; import out from './languages/out'; import defaultView from './theme/defaultView'; import logview from './theme/logView'; import hqlKeyword from './keyword/hql'; import pythonKeyword from './keyword/python'; -import sasKeyword from './keyword/sas'; import shKeyword from './keyword/sh'; import * as monaco from 'monaco-editor'; @@ -33,11 +31,11 @@ const languagesList = monaco.languages.getLanguages(); const findLang = find(languagesList, (lang) => { return lang.id === 'hql'; }); -if (!findLang) { +const uselsp = localStorage.getItem('scriptis-edditor-type') === 'lsp' +if (!findLang && !uselsp) { // 注册languages hql.register(monaco); log.register(monaco); - sas.register(monaco); sh.register(monaco); out.register(monaco); // 注册theme @@ -47,7 +45,6 @@ if (!findLang) { // 注册关键字联想 hqlKeyword.register(monaco); pythonKeyword.register(monaco); - sasKeyword.register(monaco); shKeyword.register(monaco); } diff --git a/web/packages/editor/package.json b/web/packages/editor/package.json new file mode 100644 index 0000000000..7860eaded0 --- /dev/null +++ b/web/packages/editor/package.json @@ -0,0 +1,8 @@ +{ + "name": "@dataspherestudio/editor", + "version": "1.1.6", + "dependencies": { + "@dataspherestudio/shared": "^1.1.6", + "dt-sql-parser": "1.2.1" + } +} diff --git a/web/packages/shared/components/editor/sqlFormatter/core/Formatter.js b/web/packages/editor/sqlFormatter/core/Formatter.js similarity index 100% rename from web/packages/shared/components/editor/sqlFormatter/core/Formatter.js rename to web/packages/editor/sqlFormatter/core/Formatter.js diff --git a/web/packages/shared/components/editor/sqlFormatter/core/Indentation.js b/web/packages/editor/sqlFormatter/core/Indentation.js similarity index 100% rename from web/packages/shared/components/editor/sqlFormatter/core/Indentation.js rename to web/packages/editor/sqlFormatter/core/Indentation.js diff --git a/web/packages/shared/components/editor/sqlFormatter/core/InlineBlock.js b/web/packages/editor/sqlFormatter/core/InlineBlock.js similarity index 100% rename from web/packages/shared/components/editor/sqlFormatter/core/InlineBlock.js rename to web/packages/editor/sqlFormatter/core/InlineBlock.js diff --git a/web/packages/shared/components/editor/sqlFormatter/core/Params.js b/web/packages/editor/sqlFormatter/core/Params.js similarity index 100% rename from web/packages/shared/components/editor/sqlFormatter/core/Params.js rename to web/packages/editor/sqlFormatter/core/Params.js diff --git a/web/packages/shared/components/editor/sqlFormatter/core/Tokenizer.js b/web/packages/editor/sqlFormatter/core/Tokenizer.js similarity index 100% rename from web/packages/shared/components/editor/sqlFormatter/core/Tokenizer.js rename to web/packages/editor/sqlFormatter/core/Tokenizer.js diff --git a/web/packages/shared/components/editor/sqlFormatter/core/tokenTypes.js b/web/packages/editor/sqlFormatter/core/tokenTypes.js similarity index 100% rename from web/packages/shared/components/editor/sqlFormatter/core/tokenTypes.js rename to web/packages/editor/sqlFormatter/core/tokenTypes.js diff --git a/web/packages/shared/components/editor/sqlFormatter/languages/Db2Formatter.js b/web/packages/editor/sqlFormatter/languages/Db2Formatter.js similarity index 100% rename from web/packages/shared/components/editor/sqlFormatter/languages/Db2Formatter.js rename to web/packages/editor/sqlFormatter/languages/Db2Formatter.js diff --git a/web/packages/shared/components/editor/sqlFormatter/languages/N1qlFormatter.js b/web/packages/editor/sqlFormatter/languages/N1qlFormatter.js similarity index 100% rename from web/packages/shared/components/editor/sqlFormatter/languages/N1qlFormatter.js rename to web/packages/editor/sqlFormatter/languages/N1qlFormatter.js diff --git a/web/packages/shared/components/editor/sqlFormatter/languages/PlSqlFormatter.js b/web/packages/editor/sqlFormatter/languages/PlSqlFormatter.js similarity index 100% rename from web/packages/shared/components/editor/sqlFormatter/languages/PlSqlFormatter.js rename to web/packages/editor/sqlFormatter/languages/PlSqlFormatter.js diff --git a/web/packages/shared/components/editor/sqlFormatter/languages/StandardSqlFormatter.js b/web/packages/editor/sqlFormatter/languages/StandardSqlFormatter.js similarity index 100% rename from web/packages/shared/components/editor/sqlFormatter/languages/StandardSqlFormatter.js rename to web/packages/editor/sqlFormatter/languages/StandardSqlFormatter.js diff --git a/web/packages/shared/components/editor/sqlFormatter/sqlFormatter.js b/web/packages/editor/sqlFormatter/sqlFormatter.js similarity index 100% rename from web/packages/shared/components/editor/sqlFormatter/sqlFormatter.js rename to web/packages/editor/sqlFormatter/sqlFormatter.js diff --git a/web/packages/shared/components/editor/theme/defaultView.js b/web/packages/editor/theme/defaultView.js similarity index 100% rename from web/packages/shared/components/editor/theme/defaultView.js rename to web/packages/editor/theme/defaultView.js diff --git a/web/packages/shared/components/editor/theme/logView.js b/web/packages/editor/theme/logView.js similarity index 100% rename from web/packages/shared/components/editor/theme/logView.js rename to web/packages/editor/theme/logView.js diff --git a/web/packages/shared/components/editor/util.js b/web/packages/editor/util.js similarity index 98% rename from web/packages/shared/components/editor/util.js rename to web/packages/editor/util.js index ca2499d43a..22e23a4b26 100644 --- a/web/packages/shared/components/editor/util.js +++ b/web/packages/editor/util.js @@ -25,7 +25,7 @@ import storage from '@dataspherestudio/shared/common/helper/storage'; * @param {*} lang */ const getHiveList = async (monaco, lang) => { - const closeSuggest = storage.get('close_db_table_suggest') + const closeSuggest = storage.get('close_db_table_suggest', 'local') let dbInfoProposals = []; let tableInfoProposals = []; let udfProposals = []; diff --git a/web/packages/editorLsp/compare.vue b/web/packages/editorLsp/compare.vue new file mode 100644 index 0000000000..abf3d062b9 --- /dev/null +++ b/web/packages/editorLsp/compare.vue @@ -0,0 +1,271 @@ + + + diff --git a/web/packages/editorLsp/editor.vue b/web/packages/editorLsp/editor.vue new file mode 100644 index 0000000000..af0e62760e --- /dev/null +++ b/web/packages/editorLsp/editor.vue @@ -0,0 +1,578 @@ + + + diff --git a/web/packages/workflows/module/common/workflowContentItem/index.js b/web/packages/editorLsp/highRiskGrammar.js similarity index 66% rename from web/packages/workflows/module/common/workflowContentItem/index.js rename to web/packages/editorLsp/highRiskGrammar.js index 07390072e4..3dc3cb8051 100644 --- a/web/packages/workflows/module/common/workflowContentItem/index.js +++ b/web/packages/editorLsp/highRiskGrammar.js @@ -15,5 +15,28 @@ * */ -import workflowContentitem from './index.vue'; -export default workflowContentitem; +const hql = [ + /Truncate\s*Table/i, + /Drop\s*Table/i, + /Drop\s*Function/i, + /Drop\s*Database/i, + /ALTER\s*DATABASE/i, +]; + +const python = [ + /sys/i, + /os/i, + /sc\.stop/i, + /spark\.stop/i, +]; + +const scala = [ + /sc\.stop/i, + /spark\.stop/i, +]; + +export default { + hql, + python, + scala, +}; diff --git a/web/packages/shared/components/editor/index.js b/web/packages/editorLsp/index.js similarity index 67% rename from web/packages/shared/components/editor/index.js rename to web/packages/editorLsp/index.js index fa3f25cea8..14e40469a5 100644 --- a/web/packages/shared/components/editor/index.js +++ b/web/packages/editorLsp/index.js @@ -14,7 +14,19 @@ * limitations under the License. * */ - import weEditor from './editor.vue'; +import weEditorCompare from './compare.vue'; + + +const install = function(Vue) { + const uselsp = localStorage.getItem('scriptis-edditor-type') === 'lsp' + if (uselsp) { + Vue.component('WeEditor', weEditor); + Vue.component('WeEditorCompare', weEditorCompare); + } +}; + +export default { + install, +}; -export default weEditor; diff --git a/web/packages/editorLsp/index.scss b/web/packages/editorLsp/index.scss new file mode 100644 index 0000000000..ef269f8ce0 --- /dev/null +++ b/web/packages/editorLsp/index.scss @@ -0,0 +1,56 @@ +/*! + * Copyright 2019 WeBank + * + * 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 '~@dataspherestudio/shared/common/style/variables.scss'; + +.we-editor{ + height: 100%; + width: 100%; + .monaco-menu .monaco-action-bar.vertical .action-menu-item { + height: 1.6em; + } + .monaco-menu-container a:hover { + color: #fff !important; + } + .glyphMarginClass { + background: #ff9900; + } + .contentClass { + background: #fad8a483; + } + .inlineDecoration { + position: relative; + color: #ff9900 !important; + cursor: pointer; + font-style: oblique; + // 波浪线 + &::after { + content: ''; + position: absolute; + bottom: -2px; + left: 5%; + width: 90%; + height: 2px; + background: -webkit-linear-gradient(315deg, transparent, transparent 45%, #ed4014, transparent 55%, transparent 100%),-webkit-linear-gradient(45deg, transparent, transparent 45%, #ed4014, transparent 55%, transparent 100%); + background-size: 4px 4px; + background-repeat: repeat-x; + } + } + .highRiskGrammar { + color: red; + } +} diff --git a/web/packages/editorLsp/keyword/sh.js b/web/packages/editorLsp/keyword/sh.js new file mode 100644 index 0000000000..438486116a --- /dev/null +++ b/web/packages/editorLsp/keyword/sh.js @@ -0,0 +1,626 @@ +/* + * Copyright 2019 WeBank + * + * 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 { getReturnList } from '../util'; +const shellKeywordInfoProposals = [ + { + label: 'if', + documentation: 'Keywords', + insertText: 'if', + detail: 'Keywords', + }, + { + label: 'then', + documentation: 'Keywords', + insertText: 'then', + detail: 'Keywords', + }, + { + label: 'do', + documentation: 'Keywords', + insertText: 'do', + detail: 'Keywords', + }, + { + label: 'else', + documentation: 'Keywords', + insertText: 'else', + detail: 'Keywords', + }, + { + label: 'elif', + documentation: 'Keywords', + insertText: 'elif', + detail: 'Keywords', + }, + { + label: 'while', + documentation: 'Keywords', + insertText: 'while', + detail: 'Keywords', + }, + { + label: 'until', + documentation: 'Keywords', + insertText: 'until', + detail: 'Keywords', + }, + { + label: 'for', + documentation: 'Keywords', + insertText: 'for', + detail: 'Keywords', + }, + { + label: 'in', + documentation: 'Keywords', + insertText: 'in', + detail: 'Keywords', + }, + { + label: 'esac', + documentation: 'Keywords', + insertText: 'esac', + detail: 'Keywords', + }, + { + label: 'fi', + documentation: 'Keywords', + insertText: 'fi', + detail: 'Keywords', + }, + { + label: 'fin', + documentation: 'Keywords', + insertText: 'fin', + detail: 'Keywords', + }, + { + label: 'fil', + documentation: 'Keywords', + insertText: 'fil', + detail: 'Keywords', + }, + { + label: 'done', + documentation: 'Keywords', + insertText: 'done', + detail: 'Keywords', + }, + { + label: 'exit', + documentation: 'Keywords', + insertText: 'exit', + detail: 'Keywords', + }, + { + label: 'set', + documentation: 'Keywords', + insertText: 'set', + detail: 'Keywords', + }, + { + label: 'unset', + documentation: 'Keywords', + insertText: 'unset', + detail: 'Keywords', + }, + { + label: 'export', + documentation: 'Keywords', + insertText: 'export', + detail: 'Keywords', + }, + { + label: 'function', + documentation: 'Keywords', + insertText: 'function', + detail: 'Keywords', + }, + { + label: 'awk', + documentation: 'Builtin', + insertText: 'awk', + detail: 'Builtin', + }, + { + + label: 'ab', + documentation: 'Builtin', + insertText: 'ab', + detail: 'Builtin', + }, + { + + label: 'bash', + documentation: 'Builtin', + insertText: 'bash', + detail: 'Builtin', + }, + { + + label: 'beep', + documentation: 'Builtin', + insertText: 'beep', + detail: 'Builtin', + }, + { + + label: 'cat', + documentation: 'Builtin', + insertText: 'cat', + detail: 'Builtin', + }, + { + + label: 'cc', + documentation: 'Builtin', + insertText: 'cc', + detail: 'Builtin', + }, + { + + label: 'cd', + documentation: 'Builtin', + insertText: 'cd', + detail: 'Builtin', + }, + { + + label: 'chown', + documentation: 'Builtin', + insertText: 'chown', + detail: 'Builtin', + }, + { + + label: 'chmod', + documentation: 'Builtin', + insertText: 'chmod', + detail: 'Builtin', + }, + { + + label: 'chroot', + documentation: 'Builtin', + insertText: 'chroot', + detail: 'Builtin', + }, + { + + label: 'clear', + documentation: 'Builtin', + insertText: 'clear', + detail: 'Builtin', + }, + { + + label: 'cp', + documentation: 'Builtin', + insertText: 'cp', + detail: 'Builtin', + }, + { + + label: 'curl', + documentation: 'Builtin', + insertText: 'curl', + detail: 'Builtin', + }, + { + + label: 'cut', + documentation: 'Builtin', + insertText: 'cut', + detail: 'Builtin', + }, + { + + label: 'diff', + documentation: 'Builtin', + insertText: 'diff', + detail: 'Builtin', + }, + { + + label: 'echo', + documentation: 'Builtin', + insertText: 'echo', + detail: 'Builtin', + }, + { + + label: 'find', + documentation: 'Builtin', + insertText: 'find', + detail: 'Builtin', + }, + { + + label: 'gawk', + documentation: 'Builtin', + insertText: 'gawk', + detail: 'Builtin', + }, + { + + label: 'gcc', + documentation: 'Builtin', + insertText: 'gcc', + detail: 'Builtin', + }, + { + + label: 'get', + documentation: 'Builtin', + insertText: 'get', + detail: 'Builtin', + }, + { + + label: 'git', + documentation: 'Builtin', + insertText: 'git', + detail: 'Builtin', + }, + { + + label: 'grep', + documentation: 'Builtin', + insertText: 'grep', + detail: 'Builtin', + }, + { + + label: 'hg', + documentation: 'Builtin', + insertText: 'hg', + detail: 'Builtin', + }, + { + + label: 'kill', + documentation: 'Builtin', + insertText: 'kill', + detail: 'Builtin', + }, + { + + label: 'killall', + documentation: 'Builtin', + insertText: 'killall', + detail: 'Builtin', + }, + { + + label: 'ln', + documentation: 'Builtin', + insertText: 'ln', + detail: 'Builtin', + }, + { + + label: 'ls', + documentation: 'Builtin', + insertText: 'ls', + detail: 'Builtin', + }, + { + + label: 'make', + documentation: 'Builtin', + insertText: 'make', + detail: 'Builtin', + }, + { + + label: 'mkdir', + documentation: 'Builtin', + insertText: 'mkdir', + detail: 'Builtin', + }, + { + + label: 'openssl', + documentation: 'Builtin', + insertText: 'openssl', + detail: 'Builtin', + }, + { + + label: 'mv', + documentation: 'Builtin', + insertText: 'mv', + detail: 'Builtin', + }, + { + + label: 'nc', + documentation: 'Builtin', + insertText: 'nc', + detail: 'Builtin', + }, + { + + label: 'node', + documentation: 'Builtin', + insertText: 'node', + detail: 'Builtin', + }, + { + + label: 'npm', + documentation: 'Builtin', + insertText: 'npm', + detail: 'Builtin', + }, + { + + label: 'ping', + documentation: 'Builtin', + insertText: 'ping', + detail: 'Builtin', + }, + { + + label: 'ps', + documentation: 'Builtin', + insertText: 'ps', + detail: 'Builtin', + }, + { + + label: 'restart', + documentation: 'Builtin', + insertText: 'restart', + detail: 'Builtin', + }, + { + + label: 'rm', + documentation: 'Builtin', + insertText: 'rm', + detail: 'Builtin', + }, + { + + label: 'rmdir', + documentation: 'Builtin', + insertText: 'rmdir', + detail: 'Builtin', + }, + { + + label: 'sed', + documentation: 'Builtin', + insertText: 'sed', + detail: 'Builtin', + }, + { + + label: 'service', + documentation: 'Builtin', + insertText: 'service', + detail: 'Builtin', + }, + { + + label: 'sh', + documentation: 'Builtin', + insertText: 'sh', + detail: 'Builtin', + }, + { + + label: 'shopt', + documentation: 'Builtin', + insertText: 'shopt', + detail: 'Builtin', + }, + { + + label: 'shred', + documentation: 'Builtin', + insertText: 'shred', + detail: 'Builtin', + }, + { + + label: 'source', + documentation: 'Builtin', + insertText: 'source', + detail: 'Builtin', + }, + { + + label: 'sort', + documentation: 'Builtin', + insertText: 'sort', + detail: 'Builtin', + }, + { + + label: 'sleep', + documentation: 'Builtin', + insertText: 'sleep', + detail: 'Builtin', + }, + { + + label: 'ssh', + documentation: 'Builtin', + insertText: 'ssh', + detail: 'Builtin', + }, + { + + label: 'start', + documentation: 'Builtin', + insertText: 'start', + detail: 'Builtin', + }, + { + + label: 'stop', + documentation: 'Builtin', + insertText: 'stop', + detail: 'Builtin', + }, + { + + label: 'su', + documentation: 'Builtin', + insertText: 'su', + detail: 'Builtin', + }, + { + + label: 'sudo', + documentation: 'Builtin', + insertText: 'sudo', + detail: 'Builtin', + }, + { + + label: 'svn', + documentation: 'Builtin', + insertText: 'svn', + detail: 'Builtin', + }, + { + + label: 'tee', + documentation: 'Builtin', + insertText: 'tee', + detail: 'Builtin', + }, + { + + label: 'telnet', + documentation: 'Builtin', + insertText: 'telnet', + detail: 'Builtin', + }, + { + + label: 'top', + documentation: 'Builtin', + insertText: 'top', + detail: 'Builtin', + }, + { + + label: 'touch', + documentation: 'Builtin', + insertText: 'touch', + detail: 'Builtin', + }, + { + + label: 'vi', + documentation: 'Builtin', + insertText: 'vi', + detail: 'Builtin', + }, + { + + label: 'vim', + documentation: 'Builtin', + insertText: 'vim', + detail: 'Builtin', + }, + { + + label: 'wall', + documentation: 'Builtin', + insertText: 'wall', + detail: 'Builtin', + }, + { + + label: 'wc', + documentation: 'Builtin', + insertText: 'wc', + detail: 'Builtin', + }, + { + + label: 'wget', + documentation: 'Builtin', + insertText: 'wget', + detail: 'Builtin', + }, + { + + label: 'who', + documentation: 'Builtin', + insertText: 'who', + detail: 'Builtin', + }, + { + + label: 'write', + documentation: 'Builtin', + insertText: 'write', + detail: 'Builtin', + }, + { + + label: 'yes', + documentation: 'Builtin', + insertText: 'yes', + detail: 'Builtin', + }, + { + label: 'zsh', + documentation: 'Builtin', + insertText: 'zsh', + detail: 'Builtin', + }, +]; + +export default { + async register(monaco) { + const shellProposals = shellKeywordInfoProposals.map((item) => ({ + label: item.label.toLowerCase(), + kind: monaco.languages.CompletionItemKind.Keyword, + insertText: item.insertText.toLowerCase(), + detail: item.detail, + documentation: item.documentation, + })); + + monaco.languages.registerCompletionItemProvider('sh', { + triggerCharacters: 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789._'.split(''), + async provideCompletionItems(model, position) { + const textUntilPosition = model.getValueInRange({ + startLineNumber: position.lineNumber, + startColumn: 1, + endLineNumber: position.lineNumber, + endColumn: position.column, + }); + const keywordMatch = textUntilPosition.match(/([^"]*)?$/i); + if (keywordMatch) { + const matchList = keywordMatch[0].split(' '); + const match = matchList[matchList.length - 1]; + const list = getReturnList({match, proposals: shellProposals, fieldString: 'insertText'}); + return list; + } + return []; + }, + }); + }, +}; diff --git a/web/packages/editorLsp/languages/log.js b/web/packages/editorLsp/languages/log.js new file mode 100644 index 0000000000..9f12ac5567 --- /dev/null +++ b/web/packages/editorLsp/languages/log.js @@ -0,0 +1,37 @@ +/* + * Copyright 2019 WeBank + * + * 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. + * + */ + +export default { + register(monaco) { + monaco.languages.register({ id: 'log' }); + + monaco.languages.setMonarchTokensProvider('log', { + tokenizer: { + root: [ + [/(^[=a-zA-Z].*|\d\s.*)/, 'log-normal'], + [/\sERROR\s.*/, 'log-error'], + [/\sWARN\s.*/, 'log-warn'], + [/\sINFO\s.*/, 'log-info'], + [/^([0-9]{4}||[0-9]{2})-[0-9]{2}-[0-9]{2} [0-9]{2}:[0-9]{2}:[0-9]{2}(.[0-9]{3})?/, 'log-date'], + [/^[0-9]{2}\/[0-9]{2}\/[0-9]{2} [0-9]{2}:[0-9]{2}:[0-9]{2}(.[0-9]{3})?/, 'log-date'], + [/(^\*\*Waiting queue:.*)/, 'log-info'], + [/(^\*\*result tips:.*)/, 'log-info'], + ], + }, + }); + }, +}; diff --git a/web/packages/editorLsp/languages/out.js b/web/packages/editorLsp/languages/out.js new file mode 100644 index 0000000000..042e85e1d5 --- /dev/null +++ b/web/packages/editorLsp/languages/out.js @@ -0,0 +1,1307 @@ +/* + * Copyright 2019 WeBank + * + * 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. + * + */ + + +const richLanguageConfiguration = { + comments: { + lineComment: '--', + blockComment: ['/*', '*/'], + }, + brackets: [ + ['{', '}'], + ['[', ']'], + ['(', ')'], + ], + autoClosingPairs: [ + { open: '{', close: '}' }, + { open: '[', close: ']' }, + { open: '(', close: ')' }, + { open: '"', close: '"' }, + { open: '\'', close: '\'' }, + ], + surroundingPairs: [ + { open: '{', close: '}' }, + { open: '[', close: ']' }, + { open: '(', close: ')' }, + { open: '"', close: '"' }, + { open: '\'', close: '\'' }, + ], +}; + +const langDefinition = { + defaultToken: '', + tokenPostfix: '.out', + ignoreCase: true, + + brackets: [ + { open: '[', close: ']', token: 'delimiter.square' }, + { open: '(', close: ')', token: 'delimiter.parenthesis' }, + ], + + keywords: [ + 'ABORT_AFTER_WAIT', + 'ABSENT', + 'ABSOLUTE', + 'ABORT', + 'ACCENT_SENSITIVITY', + 'ACTION', + 'ACTIVATION', + 'ACTIVE', + 'ADD', + 'ADDRESS', + 'ADMIN', + 'AES', + 'AES_128', + 'AES_192', + 'AES_256', + 'AFFINITY', + 'AFTER', + 'AGGREGATE', + 'ALGORITHM', + 'ALL_CONSTRAINTS', + 'ALL_ERRORMSGS', + 'ALL_INDEXES', + 'ALL_LEVELS', + 'ALL_SPARSE_COLUMNS', + 'ALLOW_CONNECTIONS', + 'ALLOW_MULTIPLE_EVENT_LOSS', + 'ALLOW_PAGE_LOCKS', + 'ALLOW_ROW_LOCKS', + 'ALLOW_SINGLE_EVENT_LOSS', + 'ALLOW_SNAPSHOT_ISOLATION', + 'ALLOWED', + 'ALTER', + 'ANALYZE', + 'ANONYMOUS', + 'ANSI_DEFAULTS', + 'ANSI_NULL_DEFAULT', + 'ANSI_NULL_DFLT_OFF', + 'ANSI_NULL_DFLT_ON', + 'ANSI_NULLS', + 'ANSI_PADDING', + 'ANSI_WARNINGS', + 'APPEND', + 'APPLICATION', + 'APPLICATION_LOG', + 'ARCHIVE', + 'ARITHABORT', + 'ARITHIGNORE', + 'ARRAY', + 'AS', + 'ASC', + 'ASSEMBLY', + 'ASYMMETRIC', + 'ASYNCHRONOUS_COMMIT', + 'AT', + 'ATOMIC', + 'ATTACH', + 'ATTACH_REBUILD_LOG', + 'AUDIT', + 'AUDIT_GUID', + 'AUTHENTICATION', + 'AUTHORIZATION', + 'AUTO', + 'AUTOCOMMIT', + 'AUTO_CLEANUP', + 'AUTO_CLOSE', + 'AUTO_CREATE_STATISTICS', + 'AUTO_SHRINK', + 'AUTO_UPDATE_STATISTICS', + 'AUTO_UPDATE_STATISTICS_ASYNC', + 'AUTOMATED_BACKUP_PREFERENCE', + 'AUTOMATIC', + 'AVAILABILITY', + 'AVAILABILITY_MODE', + 'BACKUP', + 'BACKUP_PRIORITY', + 'BASE64', + 'BATCHSIZE', + 'BEFORE', + 'BEGIN', + 'BEGIN_DIALOG', + 'BIGINT', + 'BINARY', + 'BINDING', + 'BIT', + 'BLOCKERS', + 'BLOCKSIZE', + 'BOOLEAN', + 'BOTH', + 'BOUNDING_BOX', + 'BREAK', + 'BROKER', + 'BROKER_INSTANCE', + 'BROWSE', + 'BUCKET', + 'BUCKETS', + 'BUCKET_COUNT', + 'BUFFER', + 'BUFFERCOUNT', + 'BULK', + 'BULK_LOGGED', + 'BY', + 'CACHE', + 'CALL', + 'CALLED', + 'CALLER', + 'CAP_CPU_PERCENT', + 'CASCADE', + 'CASE', + 'CATALOG', + 'CATCH', + 'CELLS_PER_OBJECT', + 'CERTIFICATE', + 'CHANGE', + 'CHANGE_RETENTION', + 'CHANGE_TRACKING', + 'CHANGES', + 'CHAR', + 'CHARACTER', + 'CHECK', + 'CHECK_CONSTRAINTS', + 'CHECK_EXPIRATION', + 'CHECK_POLICY', + 'CHECKALLOC', + 'CHECKCATALOG', + 'CHECKCONSTRAINTS', + 'CHECKDB', + 'CHECKFILEGROUP', + 'CHECKIDENT', + 'CHECKPOINT', + 'CHECKTABLE', + 'CLASSIFIER_FUNCTION', + 'CLEANTABLE', + 'CLEANUP', + 'CLEAR', + 'CLOSE', + 'CLUSTER', + 'CLUSTERED', + 'CLUSTERSTATUS', + 'CODEPAGE', + 'COLLATE', + 'COLLECTION', + 'COLUMN', + 'COLUMN_SET', + 'COLUMNS', + 'COLUMNSTORE', + 'COLUMNSTORE_ARCHIVE', + 'COMMENT', + 'COMMIT', + 'COMMITTED', + 'COMPACT', + 'COMPACTIONS', + 'COMPATIBILITY_LEVEL', + 'COMPRESSION', + 'COMPUTE', + 'CONCAT', + 'CONCAT_NULL_YIELDS_NULL', + 'CONCATENATE', + 'CONF', + 'CONFIGURATION', + 'CONNECT', + 'CONSTRAINT', + 'CONTAINMENT', + 'CONTENT', + 'CONTEXT', + 'CONTINUE', + 'CONTINUE_AFTER_ERROR', + 'CONTRACT', + 'CONTRACT_NAME', + 'CONTROL', + 'CONVERSATION', + 'COOKIE', + 'COPY_ONLY', + 'COUNTER', + 'CPU', + 'CREATE', + 'CREATE_NEW', + 'CREATION_DISPOSITION', + 'CREDENTIAL', + 'CRYPTOGRAPHIC', + 'CUBE', + 'CURRENT', + 'CURRENT_DATE', + 'CURSOR', + 'CURSOR_CLOSE_ON_COMMIT', + 'CURSOR_DEFAULT', + 'CYCLE', + 'DATA', + 'DATA_COMPRESSION', + 'DATA_PURITY', + 'DATABASE', + 'DATABASES', + 'DATABASE_DEFAULT', + 'DATABASE_MIRRORING', + 'DATABASE_SNAPSHOT', + 'DATAFILETYPE', + 'DATE', + 'DATE_CORRELATION_OPTIMIZATION', + 'DATEFIRST', + 'DATEFORMAT', + 'DATETIME', + 'DATETIME2', + 'DATETIMEOFFSET', + 'DAY', + 'DAYOFWEEK', + 'DAYOFYEAR', + 'DAYS', + 'DB_CHAINING', + 'DBCC', + 'DBREINDEX', + 'DBPROPERTIES', + 'DDL_DATABASE_LEVEL_EVENTS', + 'DEADLOCK_PRIORITY', + 'DEALLOCATE', + 'DEC', + 'DECIMAL', + 'DECLARE', + 'DECRYPTION', + 'DEFAULT', + 'DEFAULT_DATABASE', + 'DEFAULT_FULLTEXT_LANGUAGE', + 'DEFAULT_LANGUAGE', + 'DEFAULT_SCHEMA', + 'DEFERRED', + 'DEFINED', + 'DEFINITION', + 'DELAY', + 'DELAYED_DURABILITY', + 'DELETE', + 'DELETED', + 'DELIMITED', + 'DENSITY_VECTOR', + 'DENY', + 'DEPENDENCY', + 'DEPENDENTS', + 'DES', + 'DESC', + 'DESCRIPTION', + 'DESCRIBE', + 'DESX', + 'DETAIL', + 'DHCP', + 'DIAGNOSTICS', + 'DIALOG', + 'DIFFERENTIAL', + 'DIRECTORIES', + 'DIRECTORY', + 'DIRECTORY_NAME', + 'DISABLE', + 'DISABLE_BROKER', + 'DISABLED', + 'DISK', + 'DISTINCT', + 'DISTRIBUTE', + 'DISTRIBUTED', + 'DOCUMENT', + 'DOUBLE', + 'DOW', + 'DROP', + 'DROP_EXISTING', + 'DROPCLEANBUFFERS', + 'DUMP', + 'DURABILITY', + 'DYNAMIC', + 'EDITION', + 'ELEMENTS', + 'ELEM_TYPE', + 'ELSE', + 'EMERGENCY', + 'EMPTY', + 'EMPTYFILE', + 'ENABLE', + 'ENABLE_BROKER', + 'ENABLED', + 'ENCRYPTION', + 'END', + 'ENDPOINT', + 'ENDPOINT_URL', + 'ERRLVL', + 'ERROR', + 'ERROR_BROKER_CONVERSATIONS', + 'ERRORFILE', + 'ESCAPE', + 'ESCAPED', + 'ESTIMATEONLY', + 'EVENT', + 'EVENT_RETENTION_MODE', + 'EXCHANGE', + 'EXCLUSIVE', + 'EXEC', + 'EXECUTABLE', + 'EXECUTE', + 'EXIT', + 'EXPAND', + 'EXPORT', + 'EXPIREDATE', + 'EXPIRY_DATE', + 'EXPLAIN', + 'EXPLICIT', + 'EXPRESSION', + 'EXTENDED', + 'EXTENDED_LOGICAL_CHECKS', + 'EXTENSION', + 'EXTERNAL', + 'EXTERNAL_ACCESS', + 'EXTRACT', + 'FAIL_OPERATION', + 'FAILOVER', + 'FAILOVER_MODE', + 'FAILURE_CONDITION_LEVEL', + 'FALSE', + 'FAN_IN', + 'FAST', + 'FAST_FORWARD', + 'FETCH', + 'FIELDS', + 'FIELDTERMINATOR', + 'FILE', + 'FILEFORMAT', + 'FILEGROUP', + 'FILEGROWTH', + 'FILELISTONLY', + 'FILENAME', + 'FILEPATH', + 'FILESTREAM', + 'FILESTREAM_ON', + 'FILETABLE_COLLATE_FILENAME', + 'FILETABLE_DIRECTORY', + 'FILETABLE_FULLPATH_UNIQUE_CONSTRAINT_NAME', + 'FILETABLE_NAMESPACE', + 'FILETABLE_PRIMARY_KEY_CONSTRAINT_NAME', + 'FILETABLE_STREAMID_UNIQUE_CONSTRAINT_NAME', + 'FILLFACTOR', + 'FILTERING', + 'FIRE_TRIGGERS', + 'FIRST', + 'FIRSTROW', + 'FLOAT', + 'FMTONLY', + 'FOLLOWING', + 'FOR', + 'FORCE', + 'FORCE_FAILOVER_ALLOW_DATA_LOSS', + 'FORCE_SERVICE_ALLOW_DATA_LOSS', + 'FORCED', + 'FORCEPLAN', + 'FORCESCAN', + 'FORCESEEK', + 'FOREIGN', + 'FORMATFILE', + 'FORMATTED', + 'FORMSOF', + 'FORWARD_ONLY', + 'FREE', + 'FREEPROCCACHE', + 'FREESESSIONCACHE', + 'FREESYSTEMCACHE', + 'FROM', + 'FULL', + 'FULLSCAN', + 'FULLTEXT', + 'FUNCTION', + 'FUNCTIONS', + 'GB', + 'GEOGRAPHY_AUTO_GRID', + 'GEOGRAPHY_GRID', + 'GEOMETRY_AUTO_GRID', + 'GEOMETRY_GRID', + 'GET', + 'GLOBAL', + 'GO', + 'GOTO', + 'GOVERNOR', + 'GRANT', + 'GRIDS', + 'GROUP', + 'GROUP_MAX_REQUESTS', + 'HADR', + 'HASH', + 'HASHED', + 'HAVING', + 'HEADERONLY', + 'HEALTH_CHECK_TIMEOUT', + 'HELP', + 'HIERARCHYID', + 'HIGH', + 'HINT', + 'HISTOGRAM', + 'HOLDLOCK', + 'HOLD_DDLTIME', + 'HONOR_BROKER_PRIORITY', + 'HOUR', + 'HOURS', + 'IDENTITY', + 'IDENTITY_INSERT', + 'IDENTITY_VALUE', + 'IDENTITYCOL', + 'IDXPROPERTIES', + 'IF', + 'IGNORE', + 'IGNORE_CONSTRAINTS', + 'IGNORE_DUP_KEY', + 'IGNORE_NONCLUSTERED_COLUMNSTORE_INDEX', + 'IGNORE_TRIGGERS', + 'IMAGE', + 'IMMEDIATE', + 'IMPERSONATE', + 'IMPLICIT_TRANSACTIONS', + 'IMPORT', + 'IMPORTANCE', + 'INCLUDE', + 'INCREMENT', + 'INCREMENTAL', + 'INDEX', + 'INDEXES', + 'INDEXDEFRAG', + 'INFINITE', + 'INFLECTIONAL', + 'INIT', + 'INITIATOR', + 'INPATH', + 'INPUT', + 'INPUTDRIVER', + 'INPUTFORMAT', + 'INPUTBUFFER', + 'INSENSITIVE', + 'INSERT', + 'INSERTED', + 'INSTEAD', + 'INT', + 'INTEGER', + 'INTERVAL', + 'INTO', + 'IO', + 'IP', + 'ISABOUT', + 'ISOLATION', + 'ITEMS', + 'JAR', + 'JOB', + 'KB', + 'KEEP', + 'KEEP_CDC', + 'KEEP_NULLS', + 'KEEP_REPLICATION', + 'KEEPDEFAULTS', + 'KEEPFIXED', + 'KEEPIDENTITY', + 'KEEPNULLS', + 'KERBEROS', + 'KEY', + 'KEY_SOURCE', + 'KEY_TYPE', + 'KEYS', + 'KEYSET', + 'KILL', + 'KILOBYTES_PER_BATCH', + 'LABELONLY', + 'LANGUAGE', + 'LAST', + 'LASTROW', + 'LATERAL', + 'LESS', + 'LEVEL', + 'LEVEL_1', + 'LEVEL_2', + 'LEVEL_3', + 'LEVEL_4', + 'LIFETIME', + 'LIMIT', + 'LINENO', + 'LINES', + 'LIST', + 'LISTENER', + 'LISTENER_IP', + 'LISTENER_PORT', + 'LOAD', + 'LOADHISTORY', + 'LOB_COMPACTION', + 'LOCAL', + 'LOCAL_SERVICE_NAME', + 'LOCATION', + 'LOCK', + 'LOCKS', + 'LOCK_ESCALATION', + 'LOCK_TIMEOUT', + 'LOGICAL', + 'LOGIN', + 'LOGSPACE', + 'LONG', + 'LOOP', + 'LOW', + 'MACRO', + 'MAP', + 'MAPJOIN', + 'MANUAL', + 'MARK', + 'MARK_IN_USE_FOR_REMOVAL', + 'MASTER', + 'MATERIALIZED', + 'MAX_CPU_PERCENT', + 'MAX_DISPATCH_LATENCY', + 'MAX_DOP', + 'MAX_DURATION', + 'MAX_EVENT_SIZE', + 'MAX_FILES', + 'MAX_IOPS_PER_VOLUME', + 'MAX_MEMORY', + 'MAX_MEMORY_PERCENT', + 'MAX_QUEUE_READERS', + 'MAX_ROLLOVER_FILES', + 'MAX_SIZE', + 'MAXDOP', + 'MAXERRORS', + 'MAXLENGTH', + 'MAXRECURSION', + 'MAXSIZE', + 'MAXTRANSFERSIZE', + 'MAXVALUE', + 'MB', + 'MEDIADESCRIPTION', + 'MEDIANAME', + 'MEDIAPASSWORD', + 'MEDIUM', + 'MEMBER', + 'MEMORY_OPTIMIZED', + 'MEMORY_OPTIMIZED_DATA', + 'MEMORY_OPTIMIZED_ELEVATE_TO_SNAPSHOT', + 'MEMORY_PARTITION_MODE', + 'MERGE', + 'MESSAGE', + 'MESSAGE_FORWARD_SIZE', + 'MESSAGE_FORWARDING', + 'METADATA', + 'MICROSECOND', + 'MILLISECOND', + 'MIN_CPU_PERCENT', + 'MIN_IOPS_PER_VOLUME', + 'MIN_MEMORY_PERCENT', + 'MINUS', + 'MINUTE', + 'MINUTES', + 'MINVALUE', + 'MIRROR', + 'MIRROR_ADDRESS', + 'MODIFY', + 'MONEY', + 'MONTH', + 'MONTHS', + 'MORE', + 'MOVE', + 'MSCK', + 'MULTI_USER', + 'MUST_CHANGE', + 'NAME', + 'NANOSECOND', + 'NATIONAL', + 'NATIVE_COMPILATION', + 'NCHAR', + 'NEGOTIATE', + 'NESTED_TRIGGERS', + 'NEW_ACCOUNT', + 'NEW_BROKER', + 'NEW_PASSWORD', + 'NEWNAME', + 'NEXT', + 'NO', + 'NO_BROWSETABLE', + 'NO_CHECKSUM', + 'NO_COMPRESSION', + 'NO_EVENT_LOSS', + 'NO_INFOMSGS', + 'NO_TRUNCATE', + 'NO_WAIT', + 'NOCHECK', + 'NOCOUNT', + 'NO_DROP', + 'NOEXEC', + 'NOEXPAND', + 'NOFORMAT', + 'NOINDEX', + 'NOINIT', + 'NOLOCK', + 'NON', + 'NON_TRANSACTED_ACCESS', + 'NONCLUSTERED', + 'NONE', + 'NORECOMPUTE', + 'NORECOVERY', + 'NORELY', + 'NORESEED', + 'NORESET', + 'NOREWIND', + 'NORMAL', + 'NOSCAN', + 'NOSKIP', + 'NOTIFICATION', + 'NOTRUNCATE', + 'NOUNLOAD', + 'NOVALIDATE', + 'NOWAIT', + 'NTEXT', + 'NTLM', + 'NULLS', + 'NUMANODE', + 'NUMERIC', + 'NUMERIC_ROUNDABORT', + 'NVARCHAR', + 'OBJECT', + 'OF', + 'OFF', + 'OFFLINE', + 'OFFSET', + 'OFFSETS', + 'OLD_ACCOUNT', + 'OLD_PASSWORD', + 'ON', + 'ON_FAILURE', + 'ONLINE', + 'ONLY', + 'OPEN', + 'OPEN_EXISTING', + 'OPENTRAN', + 'OPERATOR', + 'OPTIMISTIC', + 'OPTIMIZE', + 'OPTION', + 'ORDER', + 'OUT', + 'OUTPUT', + 'OUTPUTDRIVER', + 'OUTPUTBUFFER', + 'OUTPUTFORMAT', + 'OVER', + 'OVERRIDE', + 'OWNER', + 'OWNERSHIP', + 'OVERWRITE', + 'PAD_INDEX', + 'PAGE', + 'PAGE_VERIFY', + 'PAGECOUNT', + 'PAGLOCK', + 'PARAMETERIZATION', + 'PARSEONLY', + 'PARTIAL', + 'PARTIALSCAN', + 'PARTITION', + 'PARTITIONED', + 'PARTITIONS', + 'PARTNER', + 'PASSWORD', + 'PATH', + 'PER_CPU', + 'PER_NODE', + 'PERCENT', + 'PERMISSION_SET', + 'PERSISTED', + 'PHYSICAL_ONLY', + 'PLAN', + 'PLUS', + 'POISON_MESSAGE_HANDLING', + 'POOL', + 'POPULATION', + 'PORT', + 'PRECEDING', + 'PRECISION', + 'PRETTY', + 'PRESERVE', + 'PRIMARY', + 'PRIMARY_ROLE', + 'PRINT', + 'PRIOR', + 'PRIORITY', + 'PRIORITY_LEVEL', + 'PRINCIPALS', + 'PRIVATE', + 'PRIVILEGES', + 'PROC', + 'PROCCACHE', + 'PROCEDURE', + 'PROCEDURE_NAME', + 'PROCESS', + 'PROFILE', + 'PROPERTY', + 'PROPERTY_DESCRIPTION', + 'PROPERTY_INT_ID', + 'PROPERTY_SET_GUID', + 'PROTECTION', + 'PROVIDER', + 'PROVIDER_KEY_NAME', + 'PUBLIC', + 'PURGE', + 'PUT', + 'QUARTER', + 'QUERY', + 'QUERY_GOVERNOR_COST_LIMIT', + 'QUEUE', + 'QUEUE_DELAY', + 'QUOTED_IDENTIFIER', + 'RAISERROR', + 'RANGE', + 'RAW', + 'RC2', + 'RC4', + 'RC4_128', + 'READ', + 'READ_COMMITTED_SNAPSHOT', + 'READ_ONLY', + 'READ_ONLY_ROUTING_LIST', + 'READ_ONLY_ROUTING_URL', + 'READ_WRITE', + 'READ_WRITE_FILEGROUPS', + 'READCOMMITTED', + 'READCOMMITTEDLOCK', + 'READONLY', + 'READPAST', + 'READS', + 'READTEXT', + 'READUNCOMMITTED', + 'READWRITE', + 'REAL', + 'REBUILD', + 'RECEIVE', + 'RECOMPILE', + 'RECONFIGURE', + 'RECORDREADER', + 'RECORDWRITER', + 'RECOVERY', + 'RECURSIVE', + 'RECURSIVE_TRIGGERS', + 'REDUCE', + 'REFERENCES', + 'REGENERATE', + 'REGEXP', + 'RELATED_CONVERSATION', + 'RELATED_CONVERSATION_GROUP', + 'RELATIVE', + 'RELOAD', + 'RELY', + 'REMOTE', + 'REMOTE_PROC_TRANSACTIONS', + 'REMOTE_SERVICE_NAME', + 'REMOVE', + 'RENAME', + 'REORGANIZE', + 'REPAIR', + 'REPAIR_ALLOW_DATA_LOSS', + 'REPAIR_FAST', + 'REPAIR_REBUILD', + 'REPEATABLE', + 'REPEATABLEREAD', + 'REPLICA', + 'REPLICATION', + 'REQUEST_MAX_CPU_TIME_SEC', + 'REQUEST_MAX_MEMORY_GRANT_PERCENT', + 'REQUEST_MEMORY_GRANT_TIMEOUT_SEC', + 'REQUIRED', + 'RESAMPLE', + 'RESEED', + 'RESERVE_DISK_SPACE', + 'RESET', + 'RESOURCE', + 'RESTART', + 'RESTORE', + 'RESTRICT', + 'RESTRICTED_USER', + 'RESULT', + 'RESUME', + 'RETAINDAYS', + 'RETENTION', + 'RETURN', + 'RETURNS', + 'REVERT', + 'REVOKE', + 'REWIND', + 'REWINDONLY', + 'REWRITE', + 'RLIKE', + 'ROBUST', + 'ROLE', + 'ROLES', + 'ROLLBACK', + 'ROLLUP', + 'ROOT', + 'ROUTE', + 'ROW', + 'ROWCOUNT', + 'ROWGUIDCOL', + 'ROWLOCK', + 'ROWS', + 'ROWS_PER_BATCH', + 'ROWTERMINATOR', + 'ROWVERSION', + 'RSA_1024', + 'RSA_2048', + 'RSA_512', + 'RULE', + 'SAFE', + 'SAFETY', + 'SAMPLE', + 'SAVE', + 'SCHEDULER', + 'SCHEMA', + 'SCHEMAS', + 'SCHEMA_AND_DATA', + 'SCHEMA_ONLY', + 'SCHEMABINDING', + 'SCHEME', + 'SCROLL', + 'SCROLL_LOCKS', + 'SEARCH', + 'SECOND', + 'SECONDARY', + 'SECONDARY_ONLY', + 'SECONDARY_ROLE', + 'SECONDS', + 'SECRET', + 'SECURITY_LOG', + 'SECURITYAUDIT', + 'SELECT', + 'SELECTIVE', + 'SELF', + 'SEMI', + 'SEND', + 'SENT', + 'SEQUENCE', + 'SERDE', + 'SERDEPROPERTIES', + 'SERIALIZABLE', + 'SERVER', + 'SERVICE', + 'SERVICE_BROKER', + 'SERVICE_NAME', + 'SESSION', + 'SESSION_TIMEOUT', + 'SET', + 'SETS', + 'SETUSER', + 'SHARED', + 'SHOW', + 'SHOW_DATABASE', + 'SHOW_STATISTICS', + 'SHOWCONTIG', + 'SHOWPLAN', + 'SHOWPLAN_ALL', + 'SHOWPLAN_TEXT', + 'SHOWPLAN_XML', + 'SHRINKDATABASE', + 'SHRINKFILE', + 'SHUTDOWN', + 'SID', + 'SIGNATURE', + 'SIMPLE', + 'SINGLE_BLOB', + 'SINGLE_CLOB', + 'SINGLE_NCLOB', + 'SINGLE_USER', + 'SINGLETON', + 'SIZE', + 'SKEWED', + 'SKIP', + 'SMALLDATETIME', + 'SMALLINT', + 'SMALLMONEY', + 'SNAPSHOT', + 'SORT', + 'SORTED', + 'SORT_IN_TEMPDB', + 'SOURCE', + 'SPARSE', + 'SPATIAL', + 'SPATIAL_WINDOW_MAX_CELLS', + 'SPECIFICATION', + 'SPLIT', + 'SQL', + 'SQL_VARIANT', + 'SQLPERF', + 'SSL', + 'STANDBY', + 'START', + 'START_DATE', + 'STARTED', + 'STARTUP_STATE', + 'STAT_HEADER', + 'STATE', + 'STATEMENT', + 'STATIC', + 'STATISTICAL_SEMANTICS', + 'STATISTICS', + 'STATISTICS_INCREMENTAL', + 'STATISTICS_NORECOMPUTE', + 'STATS', + 'STATS_STREAM', + 'STATUS', + 'STATUSONLY', + 'STOP', + 'STOP_ON_ERROR', + 'STOPAT', + 'STOPATMARK', + 'STOPBEFOREMARK', + 'STOPLIST', + 'STOPPED', + 'STORED', + 'STREAMTABLE', + 'STRING', + 'STRUCT', + 'SUBJECT', + 'SUBSCRIPTION', + 'SUMMARY', + 'SUPPORTED', + 'SUSPEND', + 'SWITCH', + 'SYMMETRIC', + 'SYNCHRONOUS_COMMIT', + 'SYNONYM', + 'SYSNAME', + 'SYSTEM', + 'TABLE', + 'TABLES', + 'TABLERESULTS', + 'TABLESAMPLE', + 'TABLOCK', + 'TABLOCKX', + 'TAKE', + 'TAPE', + 'TARGET', + 'TARGET_RECOVERY_TIME', + 'TB', + 'TBLPROPERTIES', + 'TCP', + 'TEMPORARY', + 'TERMINATED', + 'TEXT', + 'TEXTIMAGE_ON', + 'TEXTSIZE', + 'THEN', + 'THESAURUS', + 'THROW', + 'TIES', + 'TIME', + 'TIMEOUT', + 'TIMER', + 'TIMESTAMP', + 'TIMESTAMPTZ', + 'TINYINT', + 'TO', + 'TOP', + 'TORN_PAGE_DETECTION', + 'TOUCH', + 'TRACEOFF', + 'TRACEON', + 'TRACESTATUS', + 'TRACK_CAUSALITY', + 'TRACK_COLUMNS_UPDATED', + 'TRAN', + 'TRANSACTION', + 'TRANSACTIONS', + 'TRANSFER', + 'TRANSFORM', + 'TRANSFORM_NOISE_WORDS', + 'TRIGGER', + 'TRIPLE_DES', + 'TRIPLE_DES_3KEY', + 'TRUE', + 'TRUNCATE', + 'TRUNCATEONLY', + 'TRUSTWORTHY', + 'TRY', + 'TSQL', + 'TWO_DIGIT_YEAR_CUTOFF', + 'TYPE', + 'TYPE_WARNING', + 'UNARCHIVE', + 'UNBOUNDED', + 'UNCHECKED', + 'UNCOMMITTED', + 'UNDEFINED', + 'UNDO', + 'UNSET', + 'UNSIGNED', + 'UNIONTYPE', + 'UNIQUE', + 'UNIQUEIDENTIFIER', + 'UNIQUEJOIN', + 'UNKNOWN', + 'UNLIMITED', + 'UNLOAD', + 'UNLOCK', + 'UNSAFE', + 'UPDATE', + 'UPDATETEXT', + 'UPDATEUSAGE', + 'UPDLOCK', + 'URI', + 'URL', + 'USE', + 'USED', + 'USER', + 'USEROPTIONS', + 'USING', + 'UTC', + 'UTC_TMESTAMP', + 'UTCTIMESTAMP', + 'VALID_XML', + 'VALIDATE', + 'VALIDATION', + 'VALUE', + 'VALUE_TYPE', + 'VALUES', + 'VARBINARY', + 'VARCHAR', + 'VARYING', + 'VECTORIZATION', + 'VERIFYONLY', + 'VERSION', + 'VIEW', + 'VIEW_METADATA', + 'VIEWS', + 'VISIBILITY', + 'WAIT_AT_LOW_PRIORITY', + 'WAITFOR', + 'WEEK', + 'WEEKS', + 'WEIGHT', + 'WELL_FORMED_XML', + 'WHEN', + 'WHERE', + 'WHILE', + 'WINDOW', + 'WINDOWS', + 'WITH', + 'WITHIN', + 'WITHOUT', + 'WITNESS', + 'WORK', + 'WORKLOAD', + 'WRITE', + 'WRITETEXT', + 'XACT_ABORT', + 'XLOCK', + 'XMAX', + 'XMIN', + 'XML', + 'XMLDATA', + 'XMLNAMESPACES', + 'XMLSCHEMA', + 'XQUERY', + 'XSINIL', + 'YEAR', + 'YEARS', + 'YMAX', + 'YMIN', + 'ZONE', + ], + operators: [ + // Logical + 'ALL', 'AND', 'ANY', 'BETWEEN', 'EXISTS', 'IN', 'LIKE', 'NOT', 'OR', 'SOME', + // Set + 'EXCEPT', 'INTERSECT', 'UNION', + // Join + 'APPLY', 'CROSS', 'FULL', 'INNER', 'JOIN', 'LEFT', 'OUTER', 'RIGHT', + // Predicates + 'CONTAINS', 'FREETEXT', 'IS', 'NULL', + // Pivoting + 'PIVOT', 'UNPIVOT', + // Merging + 'MATCHED', + ], + builtinFunctions: [ + // Aggregate + 'AVG', 'CHECKSUM_AGG', 'COUNT', 'COUNT_BIG', 'GROUPING', 'GROUPING_ID', 'MAX', 'MIN', 'SUM', 'STDEV', 'STDEVP', 'VAR', 'VARP', + // Analytic + 'CUME_DIST', 'FIRST_VALUE', 'LAG', 'LAST_VALUE', 'LEAD', 'PERCENTILE_CONT', 'PERCENTILE_DISC', 'PERCENT_RANK', + // Collation + 'COLLATE', 'COLLATIONPROPERTY', 'TERTIARY_WEIGHTS', + // Azure + 'FEDERATION_FILTERING_VALUE', + // Conversion + 'CAST', 'CONVERT', 'PARSE', 'TRY_CAST', 'TRY_CONVERT', 'TRY_PARSE', + // Cryptographic + 'ASYMKEY_ID', 'ASYMKEYPROPERTY', 'CERTPROPERTY', 'CERT_ID', 'CRYPT_GEN_RANDOM', + 'DECRYPTBYASYMKEY', 'DECRYPTBYCERT', 'DECRYPTBYKEY', 'DECRYPTBYKEYAUTOASYMKEY', 'DECRYPTBYKEYAUTOCERT', 'DECRYPTBYPASSPHRASE', + 'ENCRYPTBYASYMKEY', 'ENCRYPTBYCERT', 'ENCRYPTBYKEY', 'ENCRYPTBYPASSPHRASE', 'HASHBYTES', 'IS_OBJECTSIGNED', + 'KEY_GUID', 'KEY_ID', 'KEY_NAME', 'SIGNBYASYMKEY', 'SIGNBYCERT', 'SYMKEYPROPERTY', 'VERIFYSIGNEDBYCERT', 'VERIFYSIGNEDBYASYMKEY', + // Cursor + 'CURSOR_STATUS', + // Datatype + 'DATALENGTH', 'IDENT_CURRENT', 'IDENT_INCR', 'IDENT_SEED', 'IDENTITY', 'SQL_VARIANT_PROPERTY', + // Datetime + 'CURRENT_TIMESTAMP', 'DATEADD', 'DATEDIFF', 'DATEFROMPARTS', 'DATENAME', 'DATEPART', 'DATETIME2FROMPARTS', 'DATETIMEFROMPARTS', + 'DATETIMEOFFSETFROMPARTS', 'DAY', 'EOMONTH', 'GETDATE', 'GETUTCDATE', 'ISDATE', 'MONTH', 'SMALLDATETIMEFROMPARTS', 'SWITCHOFFSET', + 'SYSDATETIME', 'SYSDATETIMEOFFSET', 'SYSUTCDATETIME', 'TIMEFROMPARTS', 'TODATETIMEOFFSET', 'YEAR', + // Logical + 'CHOOSE', 'COALESCE', 'IIF', 'NULLIF', + // Mathematical + 'ABS', 'ACOS', 'ASIN', 'ATAN', 'ATN2', 'CEILING', 'COS', 'COT', 'DEGREES', 'EXP', 'FLOOR', 'LOG', 'LOG10', + 'PI', 'POWER', 'RADIANS', 'RAND', 'ROUND', 'SIGN', 'SIN', 'SQRT', 'SQUARE', 'TAN', + // Metadata + 'APP_NAME', 'APPLOCK_MODE', 'APPLOCK_TEST', 'ASSEMBLYPROPERTY', 'COL_LENGTH', 'COL_NAME', 'COLUMNPROPERTY', + 'DATABASE_PRINCIPAL_ID', 'DATABASEPROPERTYEX', 'DB_ID', 'DB_NAME', 'FILE_ID', 'FILE_IDEX', 'FILE_NAME', 'FILEGROUP_ID', + 'FILEGROUP_NAME', 'FILEGROUPPROPERTY', 'FILEPROPERTY', 'FULLTEXTCATALOGPROPERTY', 'FULLTEXTSERVICEPROPERTY', + 'INDEX_COL', 'INDEXKEY_PROPERTY', 'INDEXPROPERTY', 'OBJECT_DEFINITION', 'OBJECT_ID', + 'OBJECT_NAME', 'OBJECT_SCHEMA_NAME', 'OBJECTPROPERTY', 'OBJECTPROPERTYEX', 'ORIGINAL_DB_NAME', 'PARSENAME', + 'SCHEMA_ID', 'SCHEMA_NAME', 'SCOPE_IDENTITY', 'SERVERPROPERTY', 'STATS_DATE', 'TYPE_ID', 'TYPE_NAME', 'TYPEPROPERTY', + // Ranking + 'DENSE_RANK', 'NTILE', 'RANK', 'ROW_NUMBER', + // Replication + 'PUBLISHINGSERVERNAME', + // Rowset + 'OPENDATASOURCE', 'OPENQUERY', 'OPENROWSET', 'OPENXML', + // Security + 'CERTENCODED', 'CERTPRIVATEKEY', 'CURRENT_USER', 'HAS_DBACCESS', 'HAS_PERMS_BY_NAME', 'IS_MEMBER', 'IS_ROLEMEMBER', 'IS_SRVROLEMEMBER', + 'LOGINPROPERTY', 'ORIGINAL_LOGIN', 'PERMISSIONS', 'PWDENCRYPT', 'PWDCOMPARE', 'SESSION_USER', 'SESSIONPROPERTY', 'SUSER_ID', 'SUSER_NAME', + 'SUSER_SID', 'SUSER_SNAME', 'SYSTEM_USER', 'USER', 'USER_ID', 'USER_NAME', + // String + 'ASCII', 'CHAR', 'CHARINDEX', 'CONCAT', 'DIFFERENCE', 'FORMAT', 'LEFT', 'LEN', 'LOWER', 'LTRIM', 'NCHAR', 'PATINDEX', + 'QUOTENAME', 'REPLACE', 'REPLICATE', 'REVERSE', 'RIGHT', 'RTRIM', 'SOUNDEX', 'SPACE', 'STR', 'STUFF', 'SUBSTRING', 'UNICODE', 'UPPER', + // System + 'BINARY_CHECKSUM', 'CHECKSUM', 'CONNECTIONPROPERTY', 'CONTEXT_INFO', 'CURRENT_REQUEST_ID', 'ERROR_LINE', 'ERROR_NUMBER', 'ERROR_MESSAGE', + 'ERROR_PROCEDURE', 'ERROR_SEVERITY', 'ERROR_STATE', 'FORMATMESSAGE', 'GETANSINULL', 'GET_FILESTREAM_TRANSACTION_CONTEXT', 'HOST_ID', + 'HOST_NAME', 'ISNULL', 'ISNUMERIC', 'MIN_ACTIVE_ROWVERSION', 'NEWID', 'NEWSEQUENTIALID', 'ROWCOUNT_BIG', 'XACT_STATE', + // TextImage + 'TEXTPTR', 'TEXTVALID', + // Trigger + 'COLUMNS_UPDATED', 'EVENTDATA', 'TRIGGER_NESTLEVEL', 'UPDATE', + // ChangeTracking + 'CHANGETABLE', 'CHANGE_TRACKING_CONTEXT', 'CHANGE_TRACKING_CURRENT_VERSION', 'CHANGE_TRACKING_IS_COLUMN_IN_MASK', 'CHANGE_TRACKING_MIN_VALID_VERSION', + // FullTextSearch + 'CONTAINSTABLE', 'FREETEXTTABLE', + // SemanticTextSearch + 'SEMANTICKEYPHRASETABLE', 'SEMANTICSIMILARITYDETAILSTABLE', 'SEMANTICSIMILARITYTABLE', + // FileStream + 'FILETABLEROOTPATH', 'GETFILENAMESPACEPATH', 'GETPATHLOCATOR', 'PATHNAME', + // ServiceBroker + 'GET_TRANSMISSION_STATUS', + ], + builtinVariables: [ + // Configuration + '@@DATEFIRST', '@@DBTS', '@@LANGID', '@@LANGUAGE', '@@LOCK_TIMEOUT', '@@MAX_CONNECTIONS', '@@MAX_PRECISION', '@@NESTLEVEL', + '@@OPTIONS', '@@REMSERVER', '@@SERVERNAME', '@@SERVICENAME', '@@SPID', '@@TEXTSIZE', '@@VERSION', + // Cursor + '@@CURSOR_ROWS', '@@FETCH_STATUS', + // Datetime + '@@DATEFIRST', + // Metadata + '@@PROCID', + // System + '@@ERROR', '@@IDENTITY', '@@ROWCOUNT', '@@TRANCOUNT', + // Stats + '@@CONNECTIONS', '@@CPU_BUSY', '@@IDLE', '@@IO_BUSY', '@@PACKET_ERRORS', '@@PACK_RECEIVED', '@@PACK_SENT', + '@@TIMETICKS', '@@TOTAL_ERRORS', '@@TOTAL_READ', '@@TOTAL_WRITE', + ], + pseudoColumns: [ + '$ACTION', '$IDENTITY', '$ROWGUID', '$PARTITION', + ], + tokenizer: { + root: [ + { include: '@comments' }, + { include: '@whitespace' }, + { include: '@pseudoColumns' }, + { include: '@numbers' }, + { include: '@strings' }, + { include: '@complexIdentifiers' }, + { include: '@scopes' }, + [/[;,.]/, 'delimiter'], + [/[()]/, '@brackets'], + [/[\w@#$]+/, { + cases: { + '@keywords': 'keyword', + '@operators': 'operator', + '@builtinVariables': 'predefined', + '@builtinFunctions': 'predefined', + '@default': 'identifier', + }, + }], + [/[<>=!%&+\-*/|~^]/, 'operator'], + ], + whitespace: [ + [/\s+/, 'white'], + ], + comments: [ + [/--+.*/, 'comment'], + [/\/\*/, { token: 'comment.quote', next: '@comment' }], + ], + comment: [ + [/[^*/]+/, 'comment'], + // Not supporting nested comments, as nested comments seem to not be standard? + // i.e. http://stackoverflow.com/questions/728172/are-there-multiline-comment-delimiters-in-sql-that-are-vendor-agnostic + // [/\/\*/, { token: 'comment.quote', next: '@push' }], // nested comment not allowed :-( + [/\*\//, { token: 'comment.quote', next: '@pop' }], + [/./, 'comment'], + ], + pseudoColumns: [ + [/[$][A-Za-z_][\w@#$]*/, { + cases: { + '@pseudoColumns': 'predefined', + '@default': 'identifier', + }, + }], + ], + numbers: [ + [/0[xX][0-9a-fA-F]*/, 'number'], + [/[$][+-]*\d*(\.\d*)?/, 'number'], + [/((\d+(\.\d*)?)|(\.\d+))([eE][-+]?\d+)?/, 'number'], + ], + strings: [ + [/N'/, { token: 'string', next: '@string' }], + [/'/, { token: 'string', next: '@string' }], + ], + string: [ + [/[^']+/, 'string'], + [/''/, 'string'], + [/'/, { token: 'string', next: '@pop' }], + ], + complexIdentifiers: [ + [/\[/, { token: 'identifier.quote', next: '@bracketedIdentifier' }], + [/"/, { token: 'identifier.quote', next: '@quotedIdentifier' }], + ], + bracketedIdentifier: [ + [/[^\]]+/, 'identifier'], + [/]]/, 'identifier'], + [/]/, { token: 'identifier.quote', next: '@pop' }], + ], + quotedIdentifier: [ + [/[^"]+/, 'identifier'], + [/""/, 'identifier'], + [/"/, { token: 'identifier.quote', next: '@pop' }], + ], + scopes: [ + [/BEGIN\s+(DISTRIBUTED\s+)?TRAN(SACTION)?\b/i, 'keyword'], + [/BEGIN\s+TRY\b/i, { token: 'keyword.try' }], + [/END\s+TRY\b/i, { token: 'keyword.try' }], + [/BEGIN\s+CATCH\b/i, { token: 'keyword.catch' }], + [/END\s+CATCH\b/i, { token: 'keyword.catch' }], + [/(BEGIN|CASE)\b/i, { token: 'keyword.block' }], + [/END\b/i, { token: 'keyword.block' }], + [/WHEN\b/i, { token: 'keyword.choice' }], + [/THEN\b/i, { token: 'keyword.choice' }], + ], + }, +}; + +export default { + config: richLanguageConfiguration, + definition: langDefinition, + register(monaco) { + monaco.languages.register({ id: 'out' }); + monaco.languages.setLanguageConfiguration('out', richLanguageConfiguration); + monaco.languages.setMonarchTokensProvider('out', langDefinition); + + }, +}; diff --git a/web/packages/editorLsp/languages/py.js b/web/packages/editorLsp/languages/py.js new file mode 100644 index 0000000000..0c898c49b1 --- /dev/null +++ b/web/packages/editorLsp/languages/py.js @@ -0,0 +1,118 @@ +import { + MonacoLanguageClient, + MonacoServices, + createConnection, + ErrorAction, + CloseAction +} from "monaco-languageclient"; +import { listen } from "vscode-ws-jsonrpc"; +import ReconnectingWebSocket from "reconnecting-websocket"; + +let installed = false +let languageClient +let maxRetries = 10 +window.languageClient = window.languageClient || {} +/** + * 创建websocket + * @param {*} url + */ +function createWebSocket(url) { + const socketOptions = { + maxReconnectionDelay: 10000, + minReconnectionDelay: 1000, + reconnectionDelayGrowFactor: 1.3, + connectionTimeout: 10000, + maxRetries, + debug: false, + }; + return new ReconnectingWebSocket(url, [], socketOptions); +} + +/** + * 创建client + * @param {*} connection + */ +function createLanguageClient(connection, documentSelector = ["python"]) { + return new MonacoLanguageClient({ + name: "Python Language Server MonacoClient", + clientOptions: { + documentSelector, + errorHandler: { + error: () => ErrorAction.Continue, + closed: () => CloseAction.DoNotRestart + }, + }, + connectionProvider: { + get: (errorHandler, closeHandler) => { + return Promise.resolve( + createConnection(connection, errorHandler, closeHandler) + ); + }, + }, + }); +} + +/** + * register language + * @param {*} monaco + */ +export function register(monaco) { + monaco.languages.register({ + id: 'python', + extensions: ['.py', '.python'], + aliases: ['py', 'python'], + mimetypes: ["application/json"], + }); +} + +/** + * connect language server + * @param {*} editor + */ +export function connectService(editor, url, cb) { + if (url) { + if (!installed) { + installed = true; + MonacoServices.install(editor); + } + if (window.languageClient.__connected_py_langserver !== true) { + window.languageClient.__connected_py_langserver = true; + const webSocket = createWebSocket(url); + webSocket.addEventListener('error', () => { + if (webSocket._retryCount >= maxRetries) { + cb({ + errMsg: 'connect-failded' + }) + } + }); + window.languageClient.__webSocket_py_langserver = webSocket; + listen({ + webSocket, + onConnection: (connection) => { + if (languageClient) { languageClient.cleanUp() } + // if (!languageClient) { + languageClient = createLanguageClient(connection); + const disposable = languageClient.start(); + connection.onClose(() => disposable.dispose()); + // } + if (cb) { + cb({ + client: languageClient + }) + } + }, + }); + } else { + if (cb) { + cb({ + client: languageClient + }) + } + } + } +} + +export default { + register, + connectService +} diff --git a/web/packages/editorLsp/languages/sh.js b/web/packages/editorLsp/languages/sh.js new file mode 100644 index 0000000000..9ba4bb9a7c --- /dev/null +++ b/web/packages/editorLsp/languages/sh.js @@ -0,0 +1,207 @@ +/* + * Copyright 2019 WeBank + * + * 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. + * + */ + +'use strict'; +export const conf = { + comments: { + lineComment: '#', + }, + brackets: [['{', '}'], ['[', ']'], ['(', ')']], + autoClosingPairs: [ + { open: '{', close: '}' }, + { open: '[', close: ']' }, + { open: '(', close: ')' }, + { open: '"', close: '"' }, + { open: '\'', close: '\'' }, + { open: '`', close: '`' }, + ], + surroundingPairs: [ + { open: '{', close: '}' }, + { open: '[', close: ']' }, + { open: '(', close: ')' }, + { open: '"', close: '"' }, + { open: '\'', close: '\'' }, + { open: '`', close: '`' }, + ], +}; +export const language = { + defaultToken: '', + ignoreCase: true, + tokenPostfix: '.shell', + brackets: [ + { token: 'delimiter.bracket', open: '{', close: '}' }, + { token: 'delimiter.parenthesis', open: '(', close: ')' }, + { token: 'delimiter.square', open: '[', close: ']' }, + ], + keywords: [ + 'if', + 'then', + 'do', + 'else', + 'elif', + 'while', + 'until', + 'for', + 'in', + 'esac', + 'fi', + 'fin', + 'fil', + 'done', + 'exit', + 'set', + 'unset', + 'export', + 'function', + ], + builtins: [ + 'ab', + 'awk', + 'bash', + 'beep', + 'cat', + 'cc', + 'cd', + 'chown', + 'chmod', + 'chroot', + 'clear', + 'cp', + 'curl', + 'cut', + 'diff', + 'echo', + 'find', + 'gawk', + 'gcc', + 'get', + 'git', + 'grep', + 'hg', + 'kill', + 'killall', + 'ln', + 'ls', + 'make', + 'mkdir', + 'openssl', + 'mv', + 'nc', + 'node', + 'npm', + 'ping', + 'ps', + 'restart', + 'rm', + 'rmdir', + 'sed', + 'service', + 'sh', + 'shopt', + 'shred', + 'source', + 'sort', + 'sleep', + 'ssh', + 'start', + 'stop', + 'su', + 'sudo', + 'svn', + 'tee', + 'telnet', + 'top', + 'touch', + 'vi', + 'vim', + 'wall', + 'wc', + 'wget', + 'who', + 'write', + 'yes', + 'zsh', + ], + // we include these common regular expressions + symbols: /[=>=!%&+\-*/|~^]/, 'operator'], + ], + whitespace: [ + [/\s+/, 'white'], + ], + comments: [ + [/--+.*/, 'comment'], + [/\/\*/, { token: 'comment.quote', next: '@comment' }], + ], + comment: [ + [/[^*/]+/, 'comment'], + // Not supporting nested comments, as nested comments seem to not be standard? + // i.e. http://stackoverflow.com/questions/728172/are-there-multiline-comment-delimiters-in-sql-that-are-vendor-agnostic + // [/\/\*/, { token: 'comment.quote', next: '@push' }], // nested comment not allowed :-( + [/\*\//, { token: 'comment.quote', next: '@pop' }], + [/./, 'comment'], + ], + pseudoColumns: [ + [/[$][A-Za-z_][\w@#$]*/, { + cases: { + '@pseudoColumns': 'predefined', + '@default': 'identifier', + }, + }], + ], + numbers: [ + [/0[xX][0-9a-fA-F]*/, 'number'], + [/[$][+-]*\d*(\.\d*)?/, 'number'], + [/((\d+(\.\d*)?)|(\.\d+))([eE][-+]?\d+)?/, 'number'], + ], + strings: [ + [/N'/, { token: 'string', next: '@string' }], + [/'/, { token: 'string', next: '@string' }], + ], + string: [ + [/[^']+/, 'string'], + [/''/, 'string'], + [/'/, { token: 'string', next: '@pop' }], + ], + complexIdentifiers: [ + [/\[/, { token: 'identifier.quote', next: '@bracketedIdentifier' }], + [/"/, { token: 'identifier.quote', next: '@quotedIdentifier' }], + ], + bracketedIdentifier: [ + [/[^\]]+/, 'identifier'], + [/]]/, 'identifier'], + [/]/, { token: 'identifier.quote', next: '@pop' }], + ], + quotedIdentifier: [ + [/[^"]+/, 'identifier'], + [/""/, 'identifier'], + [/"/, { token: 'identifier.quote', next: '@pop' }], + ], + scopes: [ + [/BEGIN\s+(DISTRIBUTED\s+)?TRAN(SACTION)?\b/i, 'keyword'], + [/BEGIN\s+TRY\b/i, { token: 'keyword.try' }], + [/END\s+TRY\b/i, { token: 'keyword.try' }], + [/BEGIN\s+CATCH\b/i, { token: 'keyword.catch' }], + [/END\s+CATCH\b/i, { token: 'keyword.catch' }], + [/(BEGIN|CASE)\b/i, { token: 'keyword.block' }], + [/END\b/i, { token: 'keyword.block' }], + [/WHEN\b/i, { token: 'keyword.choice' }], + [/THEN\b/i, { token: 'keyword.choice' }], + ], + }, +}; + +let installed = false +let languageClient +const maxRetries = 10 +window.languageClient = window.languageClient || {} + +/** + * 创建websocket + * @param {*} url + */ +function createWebSocket(url) { + const socketOptions = { + maxReconnectionDelay: 10000, + minReconnectionDelay: 1000, + reconnectionDelayGrowFactor: 1.3, + connectionTimeout: 10000, + maxRetries, + debug: false, + }; + return new ReconnectingWebSocket(url, [], socketOptions); +} + +/** + * 创建client + * @param {*} connection + */ +function createLanguageClient(connection, documentSelector = ["sql"]) { + return new MonacoLanguageClient({ + name: "SQL Language Server MonacoClient", + clientOptions: { + documentSelector, + errorHandler: { + error: () => ErrorAction.Continue, + closed: () => CloseAction.DoNotRestart + }, + }, + connectionProvider: { + get: (errorHandler, closeHandler) => { + return Promise.resolve( + createConnection(connection, errorHandler, closeHandler) + ); + }, + }, + }); +} + +/** + * connect language server + * @param {*} editor + */ +export function connectService(editor, url, cb) { + if (url) { + if (!installed) { + installed = true; + MonacoServices.install(editor); + } + if (window.languageClient.__connected_sql_langserver !== true) { + window.languageClient.__connected_sql_langserver = true; + const webSocket = createWebSocket(url); + webSocket.addEventListener('error', () => { + if (webSocket._retryCount >= maxRetries) { + cb({ + errMsg: 'connect-failded' + }) + } + }); + window.languageClient.__webSocket_sql_langserver = webSocket; + listen({ + webSocket, + onConnection: (connection) => { + if (languageClient) { languageClient.cleanUp() } + // if (!languageClient) { + languageClient = createLanguageClient(connection); + const disposable = languageClient.start(); + connection.onClose(() => disposable.dispose()); + + // } + if (cb) { + cb({ + client: languageClient + }) + } + }, + }); + } else { + if (cb) { + cb({ + client: languageClient + }) + } + } + } +} + +/** + * register language + * @param {*} monaco + */ +export function register(monaco) { + monaco.languages.register({ + id: 'sql', + extensions: ['.sql', '.hql', '.psql', '.tsql', '.jdbc', '.qmlsql', '.fql', '.ngql'], + aliases: ['sql', 'hql', 'SQL', 'HQL'], + mimetypes: ["application/json"], + }); + + monaco.languages.setLanguageConfiguration('sql', richLanguageConfiguration); + monaco.languages.setMonarchTokensProvider('sql', langDefinition); + + // 处理格式化 + monaco.languages.registerDocumentFormattingEditProvider('sql', { + provideDocumentFormattingEdits: function (model) { + let range = model.getFullModelRange(); + let value = model.getValue(); + let newValue = sqlFormatter.format(value); + return [ + { + range: range, + text: newValue, + }, + ]; + }, + }); +} + +export default { + config: richLanguageConfiguration, + definition: langDefinition, + register, + connectService +} diff --git a/web/packages/editorLsp/monaco-lsp.js b/web/packages/editorLsp/monaco-lsp.js new file mode 100644 index 0000000000..f883a81ce8 --- /dev/null +++ b/web/packages/editorLsp/monaco-lsp.js @@ -0,0 +1,121 @@ +import * as monaco from 'monaco-editor'; + +// 引入主题,语言,关键字 +import defaultView from './theme/defaultView'; +import logview from './theme/logView'; +import sqlLanguage from './languages/sql'; +import pyLanguage from './languages/py'; +import log from './languages/log'; +import sh from './languages/sh'; +import out from './languages/out'; +import shKeyword from './keyword/sh'; + +window.setImmediate = setTimeout; + +const languagesList = monaco.languages.getLanguages(); +const findLang = find(languagesList, (lang) => { + return lang.id === 'log'; +}); + +const sqlLang = ['sql', 'hql'] +window.languageClient = window.languageClient || {} + +if (!findLang) { + // 注册theme + defaultView.register(monaco); + logview.register(monaco); + + // 注册languages + sqlLanguage.register(monaco); + pyLanguage.register(monaco); + log.register(monaco); + sh.register(monaco); + out.register(monaco); + // 注册关键字联想 + shKeyword.register(monaco); +} + +/** + * 初始化编辑器 + */ +export function initClient({ el, value, service }, config, filePath, cb) { + const options = { ...config } + let model + if (filePath) { + const uri = monaco.Uri.parse(filePath) + model = monaco.editor.getModel(uri) + if (!model) { + model = monaco.editor.createModel( + value, + sqlLang.indexOf(config.language) > -1 ? 'sql' : config.language, + uri + ) + } else { + model.setValue(value) + } + } + if (model) { + options.model = model + } + const editor = monaco.editor.create(el, options); + const locationObj = { + protocol: location.protocol == 'https:' ? 'wss:' : 'ws:', + host: location.host + } + /** + * + * @param {*} type + */ + function callBack (type) { + return ({client, errMsg})=> { + if (client) { + window.languageClient[type] = client + } + if (errMsg) { + if (cb) { + cb({errMsg}) + } + } + } + } + // sql + if (sqlLang.indexOf(config.language) > -1 && service.sql) { + const wsurl = service.sql.replace(/\$\{([^}]*)}/g, function (a, b) { + return locationObj[b] + }) + sqlLanguage.connectService(editor, wsurl, callBack('sql')) + } + // python + if (config.language === 'python' && service.py) { + const wsurl = service.py.replace(/\$\{([^}]*)}/g, function (a, b) { + return locationObj[b] + }) + pyLanguage.connectService(editor, wsurl, callBack('python')) + } + + return { monaco, editor } +} + +/** + * get language client + * @param {*} lang + */ +export function getLanguageClient(lang) { + if (lang && sqlLang.indexOf(lang) > -1 ) lang = 'sql' + return window.languageClient[lang] +} + +/** + * 库表联想 + * @param {*} languageClient + */ +export function changeAssociation(lang, open) { + const client = getLanguageClient(lang) + if (client) { + const params = { + command: "changeAssociation", + arguments: open ? ['open'] : ['close'], + }; + client.sendRequest("workspace/executeCommand", params); + } +} diff --git a/web/packages/editorLsp/package.json b/web/packages/editorLsp/package.json new file mode 100644 index 0000000000..82387e9d9e --- /dev/null +++ b/web/packages/editorLsp/package.json @@ -0,0 +1,10 @@ +{ + "name": "@dataspherestudio/editor-lsp", + "version": "1.1.6", + "dependencies": { + "@dataspherestudio/shared": "^1.1.6", + "monaco-languageclient": "0.13.0", + "reconnecting-websocket": "4.4.0", + "vscode-ws-jsonrpc": "0.2.0" + } +} diff --git a/web/packages/editorLsp/sqlFormatter/core/Formatter.js b/web/packages/editorLsp/sqlFormatter/core/Formatter.js new file mode 100644 index 0000000000..f3e4028347 --- /dev/null +++ b/web/packages/editorLsp/sqlFormatter/core/Formatter.js @@ -0,0 +1,321 @@ +/* + * Copyright 2019 WeBank + * + * 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 trimEnd from 'lodash/trimEnd'; +import tokenTypes from './tokenTypes'; +import Indentation from './Indentation'; +import InlineBlock from './InlineBlock'; +import Params from './Params'; + +/** + * + */ +export default class Formatter { + /** + * @param {Object} cfg + * @param {Object} cfg.indent + * @param {Object} cfg.params + * @param {Tokenizer} tokenizer + */ + constructor(cfg, tokenizer) { + this.cfg = cfg || {}; + this.indentation = new Indentation(this.cfg.indent); + this.inlineBlock = new InlineBlock(); + this.params = new Params(this.cfg.params); + this.tokenizer = tokenizer; + this.previousReservedWord = {}; + this.tokens = []; + this.index = 0; + } + + /** + * Formats whitespaces in a SQL string to make it easier to read. + * + * @param {String} query The SQL query string + * @return {String} formatted query + */ + format(query) { + this.tokens = this.tokenizer.tokenize(query); + const formattedQuery = this.getFormattedQueryFromTokens(); + return formattedQuery.trim(); + } + + /** + * @return {*} + */ + getFormattedQueryFromTokens() { + let formattedQuery = ''; + this.tokens.forEach((token, index) => { + this.index = index; + if (token.type === tokenTypes.WHITESPACE) { + // ignore (we do our own whitespace formatting) + } else if (token.type === tokenTypes.LINE_COMMENT) { + formattedQuery = this.formatLineComment(token, formattedQuery); + } else if (token.type === tokenTypes.BLOCK_COMMENT) { + formattedQuery = this.formatBlockComment(token, formattedQuery); + } else if (token.type === tokenTypes.RESERVED_TOPLEVEL) { + formattedQuery = this.formatToplevelReservedWord(token, formattedQuery); + this.previousReservedWord = token; + } else if (token.type === tokenTypes.RESERVED_NEWLINE) { + formattedQuery = this.formatNewlineReservedWord(token, formattedQuery); + this.previousReservedWord = token; + } else if (token.type === tokenTypes.RESERVED) { + formattedQuery = this.formatWithSpaces(token, formattedQuery); + this.previousReservedWord = token; + } else if (token.type === tokenTypes.OPEN_PAREN) { + formattedQuery = this.formatOpeningParentheses(token, formattedQuery); + } else if (token.type === tokenTypes.CLOSE_PAREN) { + formattedQuery = this.formatClosingParentheses(token, formattedQuery); + } else if (token.type === tokenTypes.PLACEHOLDER) { + formattedQuery = this.formatPlaceholder(token, formattedQuery); + } else if (token.value === ',') { + formattedQuery = this.formatComma(token, formattedQuery); + } else if (token.value === ':') { + formattedQuery = this.formatWithSpaceAfter(token, formattedQuery); + } else if (token.value === '.') { + formattedQuery = this.formatWithoutSpaces(token, formattedQuery); + } else if (token.value === ';') { + formattedQuery = this.formatQuerySeparator(token, formattedQuery); + } else if (token.value === '$') { + formattedQuery = formattedQuery + token.value; + } else { + formattedQuery = this.formatWithSpaces(token, formattedQuery); + } + }); + return formattedQuery; + } + + /** + * + * @param {*} token + * @param {*} query + * @return {*} + */ + formatLineComment(token, query) { + return this.addNewline(query + token.value); + } + + /** + * + * @param {*} token + * @param {*} query + * @return {*} + */ + formatBlockComment(token, query) { + return this.addNewline(this.addNewline(query) + this.indentComment(token.value)); + } + + /** + * + * @param {*} comment + * @return {*} + */ + indentComment(comment) { + return comment.replace(/\n/g, '\n' + this.indentation.getIndent()); + } + + /** + * + * @param {*} token + * @param {*} query + * @return {*} + */ + formatToplevelReservedWord(token, query) { + this.indentation.decreaseTopLevel(); + + query = this.addNewline(query); + + this.indentation.increaseToplevel(); + + query += this.equalizeWhitespace(token.value); + return this.addNewline(query); + } + + /** + * + * @param {*} token + * @param {*} query + * @return {*} + */ + formatNewlineReservedWord(token, query) { + return this.addNewline(query) + this.equalizeWhitespace(token.value) + ' '; + } + + /** + * Replace any sequence of whitespace characters with single space + * @param {*} string + * @return {*} + */ + equalizeWhitespace(string) { + return string.replace(/\s+/g, ' '); + } + + /** + * Opening parentheses increase the block indent level and start a new line + * @param {*} token + * @param {*} query + * @return {*} + */ + formatOpeningParentheses(token, query) { + // Take out the preceding space unless there was whitespace there in the original query + // or another opening parens or line comment + const preserveWhitespaceFor = [ + tokenTypes.WHITESPACE, + tokenTypes.OPEN_PAREN, + tokenTypes.LINE_COMMENT, + ]; + if (!preserveWhitespaceFor.includes(this.previousToken().type)) { + query = trimEnd(query); + } + query += token.value; + + this.inlineBlock.beginIfPossible(this.tokens, this.index); + + if (!this.inlineBlock.isActive()) { + this.indentation.increaseBlockLevel(); + query = this.addNewline(query); + } + return query; + } + + /** + * Closing parentheses decrease the block indent level + * @param {*} token + * @param {*} query + * @return {*} + */ + formatClosingParentheses(token, query) { + if (this.inlineBlock.isActive()) { + this.inlineBlock.end(); + return this.formatWithSpaceAfter(token, query); + } else { + this.indentation.decreaseBlockLevel(); + return this.formatWithSpaces(token, this.addNewline(query)); + } + } + + /** + * + * @param {*} token + * @param {*} query + * @return {*} + */ + formatPlaceholder(token, query) { + return query + this.params.get(token) + ' '; + } + + /** + * Commas start a new line (unless within inline parentheses or SQL "LIMIT" clause) + * @param {*} token + * @param {*} query + * @return {*} + */ + formatComma(token, query) { + query = this.trimTrailingWhitespace(query) + token.value + ' '; + + if (this.inlineBlock.isActive()) { + return query; + } else if (/^LIMIT$/i.test(this.previousReservedWord.value)) { + return query; + } else { + return this.addNewline(query); + } + } + + /** + * + * @param {*} token + * @param {*} query + * @return {*} + */ + formatWithSpaceAfter(token, query) { + return this.trimTrailingWhitespace(query) + token.value + ' '; + } + + /** + * + * @param {*} token + * @param {*} query + * @return {*} + */ + formatWithoutSpaces(token, query) { + return this.trimTrailingWhitespace(query) + token.value; + } + + /** + * + * @param {*} token + * @param {*} query + * @return {*} + */ + formatWithSpaces(token, query) { + return query + token.value + ' '; + } + + /** + * + * @param {*} token + * @param {*} query + * @return {*} + */ + formatQuerySeparator(token, query) { + return this.trimTrailingWhitespace(query) + token.value + '\n'; + } + + /** + * + * @param {*} query + * @return {*} + */ + addNewline(query) { + return trimEnd(query) + '\n' + this.indentation.getIndent(); + } + + /** + * + * @param {*} query + * @return {*} + */ + trimTrailingWhitespace(query) { + if (this.previousNonWhitespaceToken().type === tokenTypes.LINE_COMMENT) { + return trimEnd(query) + '\n'; + } else { + return trimEnd(query); + } + } + + /** + * + * @return {*} + */ + previousNonWhitespaceToken() { + let n = 1; + while (this.previousToken(n).type === tokenTypes.WHITESPACE) { + n++; + } + return this.previousToken(n); + } + + /** + * + * @param {*} offset + * @return {*} + */ + previousToken(offset = 1) { + return this.tokens[this.index - offset] || {}; + } +} \ No newline at end of file diff --git a/web/packages/editorLsp/sqlFormatter/core/Indentation.js b/web/packages/editorLsp/sqlFormatter/core/Indentation.js new file mode 100644 index 0000000000..84de44ef42 --- /dev/null +++ b/web/packages/editorLsp/sqlFormatter/core/Indentation.js @@ -0,0 +1,86 @@ +/* + * Copyright 2019 WeBank + * + * 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 repeat from 'lodash/repeat'; +import last from 'lodash/last'; + +const INDENT_TYPE_TOP_LEVEL = 'top-level'; +const INDENT_TYPE_BLOCK_LEVEL = 'block-level'; + +/** + * Manages indentation levels. + * + * There are two types of indentation levels: + * + * - BLOCK_LEVEL : increased by open-parenthesis + * - TOP_LEVEL : increased by RESERVED_TOPLEVEL words + */ +export default class Indentation { + /** + * @param {String} indent Indent value, default is " " (2 spaces) + */ + constructor(indent) { + this.indent = indent || ' '; + this.indentTypes = []; + } + + /** + * Returns current indentation string. + * @return {String} + */ + getIndent() { + return repeat(this.indent, this.indentTypes.length); + } + + /** + * Increases indentation by one top-level indent. + */ + increaseToplevel() { + this.indentTypes.push(INDENT_TYPE_TOP_LEVEL); + } + + /** + * Increases indentation by one block-level indent. + */ + increaseBlockLevel() { + this.indentTypes.push(INDENT_TYPE_BLOCK_LEVEL); + } + + /** + * Decreases indentation by one top-level indent. + * Does nothing when the previous indent is not top-level. + */ + decreaseTopLevel() { + if (last(this.indentTypes) === INDENT_TYPE_TOP_LEVEL) { + this.indentTypes.pop(); + } + } + + /** + * Decreases indentation by one block-level indent. + * If there are top-level indents within the block-level indent, + * throws away these as well. + */ + decreaseBlockLevel() { + while (this.indentTypes.length > 0) { + const type = this.indentTypes.pop(); + if (type !== INDENT_TYPE_TOP_LEVEL) { + break; + } + } + } +} diff --git a/web/packages/editorLsp/sqlFormatter/core/InlineBlock.js b/web/packages/editorLsp/sqlFormatter/core/InlineBlock.js new file mode 100644 index 0000000000..1165e232ae --- /dev/null +++ b/web/packages/editorLsp/sqlFormatter/core/InlineBlock.js @@ -0,0 +1,118 @@ +/* + * Copyright 2019 WeBank + * + * 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 tokenTypes from './tokenTypes'; + +const INLINE_MAX_LENGTH = 50; + +/** + * Bookkeeper for inline blocks. + * + * Inline blocks are parenthized expressions that are shorter than INLINE_MAX_LENGTH. + * These blocks are formatted on a single line, unlike longer parenthized + * expressions where open-parenthesis causes newline and increase of indentation. + */ +export default class InlineBlock { + /** + * + */ + constructor() { + this.level = 0; + } + + /** + * Begins inline block when lookahead through upcoming tokens determines + * that the block would be smaller than INLINE_MAX_LENGTH. + * @param {Object[]} tokens Array of all tokens + * @param {Number} index Current token position + */ + beginIfPossible(tokens, index) { + if (this.level === 0 && this.isInlineBlock(tokens, index)) { + this.level = 1; + } else if (this.level > 0) { + this.level++; + } else { + this.level = 0; + } + } + + /** + * Finishes current inline block. + * There might be several nested ones. + */ + end() { + this.level--; + } + + /** + * True when inside an inline block + * @return {Boolean} + */ + isActive() { + return this.level > 0; + } + + /** + * Check if this should be an inline parentheses block + * Examples are "NOW()", "COUNT(*)", "int(10)", key(`somecolumn`), DECIMAL(7,2) + * @param {*} tokens + * @param {*} index + * @return {*} + */ + isInlineBlock(tokens, index) { + let length = 0; + let level = 0; + + for (let i = index; i < tokens.length; i++) { + const token = tokens[i]; + length += token.value.length; + + // Overran max length + if (length > INLINE_MAX_LENGTH) { + return false; + } + + if (token.type === tokenTypes.OPEN_PAREN) { + level++; + } else if (token.type === tokenTypes.CLOSE_PAREN) { + level--; + if (level === 0) { + return true; + } + } + + if (this.isForbiddenToken(token)) { + return false; + } + } + return false; + } + + /** + * Reserved words that cause newlines, comments and semicolons + * are not allowed inside inline parentheses block + * @param {*} param0 + * @return {*} + */ + isForbiddenToken({ type, value }) { + return type === tokenTypes.RESERVED_TOPLEVEL || + type === tokenTypes.RESERVED_NEWLINE || + type === tokenTypes.COMMENT || + type === tokenTypes.BLOCK_COMMENT || + value === ';'; + } +} diff --git a/web/packages/editorLsp/sqlFormatter/core/Params.js b/web/packages/editorLsp/sqlFormatter/core/Params.js new file mode 100644 index 0000000000..5db46c8ad5 --- /dev/null +++ b/web/packages/editorLsp/sqlFormatter/core/Params.js @@ -0,0 +1,46 @@ +/* + * Copyright 2019 WeBank + * + * 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. + * + */ + +/** + * Handles placeholder replacement with given params. + */ +export default class Params { + /** + * @param {Object} params + */ + constructor(params) { + this.params = params; + this.index = 0; + } + + /** + * Returns param value that matches given placeholder with param key. + * @param {Object} token + * @param {String} token.key Placeholder key + * @param {String} token.value Placeholder value + * @return {String} param or token.value when params are missing + */ + get({ key, value }) { + if (!this.params) { + return value; + } + if (key) { + return this.params[key]; + } + return this.params[this.index++]; + } +} diff --git a/web/packages/editorLsp/sqlFormatter/core/Tokenizer.js b/web/packages/editorLsp/sqlFormatter/core/Tokenizer.js new file mode 100644 index 0000000000..8e48e1543b --- /dev/null +++ b/web/packages/editorLsp/sqlFormatter/core/Tokenizer.js @@ -0,0 +1,507 @@ +/* + * Copyright 2019 WeBank + * + * 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 isEmpty from 'lodash/isEmpty'; +import escapeRegExp from 'lodash/escapeRegExp'; +import tokenTypes from './tokenTypes'; + +/** + * + */ +export default class Tokenizer { + /** + * @param {Object} cfg + * @param {String[]} cfg.reservedWords Reserved words in SQL + * @param {String[]} cfg.reservedToplevelWords Words that are set to new line separately + * @param {String[]} cfg.reservedNewlineWords Words that are set to newline + * @param {String[]} cfg.stringTypes String types to enable: "", '', ``, [], N'' + * @param {String[]} cfg.openParens Opening parentheses to enable, like (, [ + * @param {String[]} cfg.closeParens Closing parentheses to enable, like ), ] + * @param {String[]} cfg.indexedPlaceholderTypes Prefixes for indexed placeholders, like ? + * @param {String[]} cfg.namedPlaceholderTypes Prefixes for named placeholders, like @ and : + * @param {String[]} cfg.lineCommentTypes Line comments to enable, like # and -- + * @param {String[]} cfg.specialWordChars Special chars that can be found inside of words, like @ and # + */ + constructor(cfg) { + this.WHITESPACE_REGEX = /^(\s+)/; + this.NUMBER_REGEX = /^((-\s*)?[0-9]+(\.[0-9]+)?|0x[0-9a-fA-F]+|0b[01]+)\b/; + this.OPERATOR_REGEX = /^(!=|<>|==|<=|>=|!<|!>|\|\||::|->>|->|~~\*|~~|!~~\*|!~~|~\*|!~\*|!~|.)/; + + this.BLOCK_COMMENT_REGEX = /^(\/\*[^]*?(?:\*\/|$))/; + this.PARAMS_REGEX = cfg.paramsPattern; + this.LINE_COMMENT_REGEX = this.createLineCommentRegex(cfg.lineCommentTypes); + + this.RESERVED_TOPLEVEL_REGEX = this.createReservedWordRegex(cfg.reservedToplevelWords); + this.RESERVED_NEWLINE_REGEX = this.createReservedWordRegex(cfg.reservedNewlineWords); + this.RESERVED_PLAIN_REGEX = this.createReservedWordRegex(cfg.reservedWords); + + this.WORD_REGEX = this.createWordRegex(cfg.specialWordChars); + this.STRING_REGEX = this.createStringRegex(cfg.stringTypes); + + this.OPEN_PAREN_REGEX = this.createParenRegex(cfg.openParens); + this.CLOSE_PAREN_REGEX = this.createParenRegex(cfg.closeParens); + + this.INDEXED_PLACEHOLDER_REGEX = this.createPlaceholderRegex(cfg.indexedPlaceholderTypes, '[0-9]*'); + this.IDENT_NAMED_PLACEHOLDER_REGEX = this.createPlaceholderRegex(cfg.namedPlaceholderTypes, '[a-zA-Z0-9._$]+'); + this.STRING_NAMED_PLACEHOLDER_REGEX = this.createPlaceholderRegex( + cfg.namedPlaceholderTypes, + this.createStringPattern(cfg.stringTypes) + ); + } + + /** + * + * @param {*} lineCommentTypes + * @return {*} + */ + createLineCommentRegex(lineCommentTypes) { + return new RegExp(`^((?:${lineCommentTypes.map((c) => escapeRegExp(c)).join('|')}).*?(?:\n|\r\n|$))`); + } + + /** + * + * @param {*} reservedWords + * @return {*} + */ + createReservedWordRegex(reservedWords) { + const reservedWordsPattern = reservedWords.join('|').replace(/ /g, '\\s+'); + return new RegExp(`^(${reservedWordsPattern})\\b`, 'i'); + } + + /** + * + * @param {*} specialChars + * @return {*} + */ + createWordRegex(specialChars = []) { + return new RegExp(`^([\\w${specialChars.join('')}]+)`); + } + + /** + * + * @param {*} stringTypes + * @return {*} + */ + createStringRegex(stringTypes) { + return new RegExp( + '^(' + this.createStringPattern(stringTypes) + ')' + ); + } + + /** + * This enables the following string patterns: + * 1.backtick quoted string using `` to escape + * 2.square bracket quoted string (SQL Server) using ]] to escape + * 3.double quoted string using "" or \" to escape + * 4.single quoted string using '' or \' to escape + * 5.national character quoted string using N'' or N\' to escape + * @param {*} stringTypes + * @return {*} + */ + createStringPattern(stringTypes) { + const patterns = { + '``': '((`[^`]*($|`))+)', + '[]': '((\\[[^\\]]*($|\\]))(\\][^\\]]*($|\\]))*)', + '""': '(("[^"\\\\]*(?:\\\\.[^"\\\\]*)*("|$))+)', + '\'\'': '((\'[^\'\\\\]*(?:\\\\.[^\'\\\\]*)*(\'|$))+)', + 'N\'\'': '((N\'[^N\'\\\\]*(?:\\\\.[^N\'\\\\]*)*(\'|$))+)', + }; + + return stringTypes.map((t) => patterns[t]).join('|'); + } + + /** + * + * @param {*} parens + * @return {*} + */ + createParenRegex(parens) { + return new RegExp( + '^(' + parens.map((p) => this.escapeParen(p)).join('|') + ')', + 'i' + ); + } + + /** + * + * @param {*} paren + * @return {*} + */ + escapeParen(paren) { + if (paren.length === 1) { + // A single punctuation character + return escapeRegExp(paren); + } else { + // longer word + return '\\b' + paren + '\\b'; + } + } + + /** + * + * @param {*} types + * @param {*} pattern + * @return {*} + */ + createPlaceholderRegex(types, pattern) { + if (isEmpty(types)) { + return false; + } + const typesRegex = types.map(escapeRegExp).join('|'); + + return new RegExp(`^((?:${typesRegex})(?:${pattern}))`); + } + + /** + * Takes a SQL string and breaks it into tokens. + * Each token is an object with type and value. + * + * @param {String} input The SQL string + * @return {Object[]} tokens An array of tokens. + * @return {String} token.type + * @return {String} token.value + */ + tokenize(input) { + const tokens = []; + let token; + + // Keep processing the string until it is empty + while (input.length) { + // Get the next token and the token type + token = this.getNextToken(input, token); + // Advance the string + input = input.substring(token.value.length); + + tokens.push(token); + } + return tokens; + } + + /** + * + * @param {*} input + * @param {*} previousToken + * @return {*} + */ + getNextToken(input, previousToken) { + return this.getWhitespaceToken(input) || + this.getCommentToken(input) || + this.getStringToken(input) || + this.getOpenParenToken(input) || + this.getCloseParenToken(input) || + this.getParamsToken(input) || + this.getPlaceholderToken(input) || + this.getNumberToken(input) || + this.getReservedWordToken(input, previousToken) || + this.getWordToken(input) || + this.getOperatorToken(input); + } + + /** + * + * @param {*} input + * @return {*} + */ + getWhitespaceToken(input) { + return this.getTokenOnFirstMatch({ + input, + type: tokenTypes.WHITESPACE, + regex: this.WHITESPACE_REGEX, + }); + } + + /** + * + * @param {*} input + * @return {*} + */ + getCommentToken(input) { + return this.getLineCommentToken(input) || this.getBlockCommentToken(input); + } + + /** + * + * @param {*} input + * @return {*} + */ + getLineCommentToken(input) { + return this.getTokenOnFirstMatch({ + input, + type: tokenTypes.LINE_COMMENT, + regex: this.LINE_COMMENT_REGEX, + }); + } + + /** + * + * @param {*} input + * @return {*} + */ + getBlockCommentToken(input) { + return this.getTokenOnFirstMatch({ + input, + type: tokenTypes.BLOCK_COMMENT, + regex: this.BLOCK_COMMENT_REGEX, + }); + } + + /** + * + * @param {*} input + * @return {*} + */ + getStringToken(input) { + return this.getTokenOnFirstMatch({ + input, + type: tokenTypes.STRING, + regex: this.STRING_REGEX, + }); + } + + /** + * + * @param {*} input + * @return {*} + */ + getOpenParenToken(input) { + return this.getTokenOnFirstMatch({ + input, + type: tokenTypes.OPEN_PAREN, + regex: this.OPEN_PAREN_REGEX, + }); + } + + /** + * + * @param {*} input + * @return {*} + */ + getCloseParenToken(input) { + return this.getTokenOnFirstMatch({ + input, + type: tokenTypes.CLOSE_PAREN, + regex: this.CLOSE_PAREN_REGEX, + }); + } + + getParamsToken(input) { + return this.getTokenOnFirstMatch({ + input, + type: tokenTypes.WORD, + regex: this.PARAMS_REGEX, + }); + } + + /** + * + * @param {*} input + * @return {*} + */ + getPlaceholderToken(input) { + return this.getIdentNamedPlaceholderToken(input) || + this.getStringNamedPlaceholderToken(input) || + this.getIndexedPlaceholderToken(input); + } + + /** + * + * @param {*} input + * @return {*} + */ + getIdentNamedPlaceholderToken(input) { + return this.getPlaceholderTokenWithKey({ + input, + regex: this.IDENT_NAMED_PLACEHOLDER_REGEX, + parseKey: (v) => v.slice(1), + }); + } + + /** + * + * @param {*} input + * @return {*} + */ + getStringNamedPlaceholderToken(input) { + return this.getPlaceholderTokenWithKey({ + input, + regex: this.STRING_NAMED_PLACEHOLDER_REGEX, + parseKey: (v) => this.getEscapedPlaceholderKey({ + key: v.slice(2, -1), + quoteChar: v.slice(-1) + }), + }); + } + + /** + * + * @param {*} input + * @return {*} + */ + getIndexedPlaceholderToken(input) { + return this.getPlaceholderTokenWithKey({ + input, + regex: this.INDEXED_PLACEHOLDER_REGEX, + parseKey: (v) => v.slice(1), + }); + } + + /** + * + * @param {*} param0 + * @return {*} + */ + getPlaceholderTokenWithKey({ + input, + regex, + parseKey + }) { + const token = this.getTokenOnFirstMatch({ + input, + regex, + type: tokenTypes.PLACEHOLDER + }); + if (token) { + token.key = parseKey(token.value); + } + return token; + } + + /** + * + * @param {*} param0 + * @return {*} + */ + getEscapedPlaceholderKey({ + key, + quoteChar + }) { + return key.replace(new RegExp(escapeRegExp('\\') + quoteChar, 'g'), quoteChar); + } + + /** + * Decimal, binary, or hex numbers + * @param {*} input + * @return {*} + */ + getNumberToken(input) { + return this.getTokenOnFirstMatch({ + input, + type: tokenTypes.NUMBER, + regex: this.NUMBER_REGEX, + }); + } + + /** + * Punctuation and symbols + * @param {*} input + * @return {*} + */ + getOperatorToken(input) { + return this.getTokenOnFirstMatch({ + input, + type: tokenTypes.OPERATOR, + regex: this.OPERATOR_REGEX, + }); + } + + /** + * + * @param {*} input + * @param {*} previousToken + * @return {*} + */ + getReservedWordToken(input, previousToken) { + // A reserved word cannot be preceded by a "." + // this makes it so in "mytable.from", "from" is not considered a reserved word + if (previousToken && previousToken.value && previousToken.value === '.') { + return; + } + return this.getToplevelReservedToken(input) || this.getNewlineReservedToken(input) || this.getPlainReservedToken(input); + } + + /** + * + * @param {*} input + * @return {*} + */ + getToplevelReservedToken(input) { + return this.getTokenOnFirstMatch({ + input, + type: tokenTypes.RESERVED_TOPLEVEL, + regex: this.RESERVED_TOPLEVEL_REGEX, + }); + } + + /** + * + * @param {*} input + * @return {*} + */ + getNewlineReservedToken(input) { + return this.getTokenOnFirstMatch({ + input, + type: tokenTypes.RESERVED_NEWLINE, + regex: this.RESERVED_NEWLINE_REGEX, + }); + } + + /** + * + * @param {*} input + * @return {*} + */ + getPlainReservedToken(input) { + return this.getTokenOnFirstMatch({ + input, + type: tokenTypes.RESERVED, + regex: this.RESERVED_PLAIN_REGEX, + }); + } + + /** + * + * @param {*} input + * @return {*} + */ + getWordToken(input) { + return this.getTokenOnFirstMatch({ + input, + type: tokenTypes.WORD, + regex: this.WORD_REGEX, + }); + } + + /** + * + * @param {*} param0 + * @return {*} + */ + getTokenOnFirstMatch({ + input, + type, + regex + }) { + const matches = input.match(regex); + + if (matches) { + return { + type, + value: matches[1] + }; + } + } +} diff --git a/web/packages/editorLsp/sqlFormatter/core/tokenTypes.js b/web/packages/editorLsp/sqlFormatter/core/tokenTypes.js new file mode 100644 index 0000000000..dedda2d748 --- /dev/null +++ b/web/packages/editorLsp/sqlFormatter/core/tokenTypes.js @@ -0,0 +1,35 @@ +/* + * Copyright 2019 WeBank + * + * 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. + * + */ + +/** + * Constants for token types + */ +export default { + WHITESPACE: 'whitespace', + WORD: 'word', + STRING: 'string', + RESERVED: 'reserved', + RESERVED_TOPLEVEL: 'reserved-toplevel', + RESERVED_NEWLINE: 'reserved-newline', + OPERATOR: 'operator', + OPEN_PAREN: 'open-paren', + CLOSE_PAREN: 'close-paren', + LINE_COMMENT: 'line-comment', + BLOCK_COMMENT: 'block-comment', + NUMBER: 'number', + PLACEHOLDER: 'placeholder', +}; diff --git a/web/packages/editorLsp/sqlFormatter/languages/Db2Formatter.js b/web/packages/editorLsp/sqlFormatter/languages/Db2Formatter.js new file mode 100644 index 0000000000..e06c8eeeb2 --- /dev/null +++ b/web/packages/editorLsp/sqlFormatter/languages/Db2Formatter.js @@ -0,0 +1,141 @@ +/* + * Copyright 2019 WeBank + * + * 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 Formatter from '../core/Formatter'; +import Tokenizer from '../core/Tokenizer'; + +const reservedWords = [ + 'ABS', 'ACTIVATE', 'ALIAS', 'ALL', 'ALLOCATE', 'ALLOW', 'ALTER', 'ANY', 'ARE', 'ARRAY', 'AS', 'ASC', + 'ASENSITIVE', 'ASSOCIATE', 'ASUTIME', 'ASYMMETRIC', 'AT', 'ATOMIC', 'ATTRIBUTES', 'AUDIT', 'AUTHORIZATION', 'AUX', 'AUXILIARY', 'AVG', + 'BEFORE', 'BEGIN', 'BETWEEN', 'BIGINT', 'BINARY', 'BLOB', 'BOOLEAN', 'BOTH', 'BUFFERPOOL', 'BY', + 'CACHE', 'CALL', 'CALLED', 'CAPTURE', 'CARDINALITY', 'CASCADED', 'CASE', 'CAST', 'CCSID', 'CEIL', 'CEILING', 'CHAR', 'CHARACTER', + 'CHARACTER_LENGTH', 'CHAR_LENGTH', 'CHECK', 'CLOB', 'CLONE', 'CLOSE', 'CLUSTER', 'COALESCE', 'COLLATE', 'COLLECT', 'COLLECTION', + 'COLLID', 'COLUMN', 'COMMENT', 'COMMIT', 'CONCAT', 'CONDITION', 'CONNECT', 'CONNECTION', 'CONSTRAINT', 'CONTAINS', 'CONTINUE', + 'CONVERT', 'CORR', 'CORRESPONDING', 'COUNT', 'COUNT_BIG', 'COVAR_POP', 'COVAR_SAMP', 'CREATE', 'CROSS', 'CUBE', 'CUME_DIST', 'CURRENT', + 'CURRENT_DATE', 'CURRENT_DEFAULT_TRANSFORM_GROUP', 'CURRENT_LC_CTYPE', 'CURRENT_PATH', 'CURRENT_ROLE', 'CURRENT_SCHEMA', + 'CURRENT_SERVER', 'CURRENT_TIME', 'CURRENT_TIMESTAMP', 'CURRENT_TIMEZONE', 'CURRENT_TRANSFORM_GROUP_FOR_TYPE', 'CURRENT_USER', 'CURSOR', + 'CYCLE', + 'DATA', 'DATABASE', 'DATAPARTITIONNAME', 'DATAPARTITIONNUM', 'DATE', 'DAY', 'DAYS', 'DB2GENERAL', 'DB2GENRL', 'DB2SQL', 'DBINFO', + 'DBPARTITIONNAME', 'DBPARTITIONNUM', 'DEALLOCATE', 'DEC', 'DECIMAL', 'DECLARE', 'DEFAULT', 'DEFAULTS', 'DEFINITION', 'DELETE', + 'DENSERANK', 'DENSE_RANK', 'DEREF', 'DESCRIBE', 'DESCRIPTOR', 'DETERMINISTIC', 'DIAGNOSTICS', 'DISABLE', 'DISALLOW', 'DISCONNECT', + 'DISTINCT', 'DO', 'DOCUMENT', 'DOUBLE', 'DROP', 'DSSIZE', 'DYNAMIC', + 'EACH', 'EDITPROC', 'ELEMENT', 'ELSE', 'ELSEIF', 'ENABLE', 'ENCODING', 'ENCRYPTION', 'END', 'END-EXEC', 'ENDING', 'ERASE', 'ESCAPE', + 'EVERY', 'EXCEPTION', 'EXCLUDING', 'EXCLUSIVE', 'EXEC', 'EXECUTE', 'EXISTS', 'EXIT', 'EXP', 'EXPLAIN', 'EXTENDED', 'EXTERNAL', + 'EXTRACT', + 'FALSE', 'FENCED', 'FETCH', 'FIELDPROC', 'FILE', 'FILTER', 'FINAL', 'FIRST', 'FLOAT', 'FLOOR', 'FOR', 'FOREIGN', 'FREE', 'FULL', + 'FUNCTION', 'FUSION', + 'GENERAL', 'GENERATED', 'GET', 'GLOBAL', 'GOTO', 'GRANT', 'GRAPHIC', 'GROUP', 'GROUPING', + 'HANDLER', 'HASH', 'HASHED_VALUE', 'HINT', 'HOLD', 'HOUR', 'HOURS', + 'IDENTITY', 'IF', 'IMMEDIATE', 'IN', 'INCLUDING', 'INCLUSIVE', 'INCREMENT', 'INDEX', 'INDICATOR', 'INDICATORS', 'INF', 'INFINITY', + 'INHERIT', 'INNER', 'INOUT', 'INSENSITIVE', 'INSERT', 'INT', 'INTEGER', 'INTEGRITY', 'INTERSECTION', 'INTERVAL', 'INTO', + 'IS', 'ISOBID', 'ISOLATION', 'ITERATE', + 'JAR', 'JAVA', + 'KEEP', 'KEY', + 'LABEL', 'LANGUAGE', 'LARGE', 'LATERAL', 'LC_CTYPE', 'LEADING', 'LEAVE', 'LEFT', 'LIKE', 'LINKTYPE', 'LN', 'LOCAL', + 'LOCALDATE', 'LOCALE', 'LOCALTIME', 'LOCALTIMESTAMP', 'LOCATOR', 'LOCATORS', 'LOCK', 'LOCKMAX', 'LOCKSIZE', 'LONG', 'LOOP', 'LOWER', + 'MAINTAINED', 'MATCH', 'MATERIALIZED', 'MAX', 'MAXVALUE', 'MEMBER', 'MERGE', 'METHOD', 'MICROSECOND', 'MICROSECONDS', 'MIN', 'MINUTE', + 'MINUTES', 'MINVALUE', 'MOD', 'MODE', 'MODIFIES', 'MODULE', 'MONTH', 'MONTHS', 'MULTISET', + 'NAN', 'NATIONAL', 'NATURAL', 'NCHAR', 'NCLOB', 'NEW', 'NEW_TABLE', 'NEXTVAL', 'NO', 'NOCACHE', 'NOCYCLE', 'NODENAME', 'NODENUMBER', + 'NOMAXVALUE', 'NOMINVALUE', 'NONE', 'NOORDER', 'NORMALIZE', 'NORMALIZED', 'NOT', 'NULL', 'NULLIF', 'NULLS', 'NUMERIC', 'NUMPARTS', + 'OBID', 'OCTET_LENGTH', 'OF', 'OFFSET', 'OLD', 'OLD_TABLE', 'ON', 'ONLY', 'OPEN', 'OPTIMIZATION', 'OPTIMIZE', 'OPTION', 'ORDER', + 'OUT', 'OUTER', 'OVER', 'OVERLAPS', 'OVERLAY', 'OVERRIDING', + 'PACKAGE', 'PADDED', 'PAGESIZE', 'PARAMETER', 'PART', 'PARTITION', 'PARTITIONED', 'PARTITIONING', 'PARTITIONS', 'PASSWORD', 'PATH', + 'PERCENTILE_CONT', 'PERCENTILE_DISC', 'PERCENT_RANK', 'PIECESIZE', 'PLAN', 'POSITION', 'POWER', 'PRECISION', 'PREPARE', 'PREVVAL', + 'PRIMARY', 'PRIQTY', 'PRIVILEGES', 'PROCEDURE', 'PROGRAM', 'PSID', 'PUBLIC', + 'QUERY', 'QUERYNO', + 'RANGE', 'RANK', 'READ', 'READS', 'REAL', 'RECOVERY', 'RECURSIVE', 'REF', 'REFERENCES', 'REFERENCING', 'REFRESH', 'REGR_AVGX', + 'REGR_AVGY', 'REGR_COUNT', 'REGR_INTERCEPT', 'REGR_R2', 'REGR_SLOPE', 'REGR_SXX', 'REGR_SXY', 'REGR_SYY', 'RELEASE', 'RENAME', 'REPEAT', + 'RESET', 'RESIGNAL', 'RESTART', 'RESTRICT', 'RESULT', 'RESULT_SET_LOCATOR', 'RETURN', 'RETURNS', 'REVOKE', 'RIGHT', 'ROLE', 'ROLLBACK', + 'ROLLUP', 'ROUND_CEILING', 'ROUND_DOWN', 'ROUND_FLOOR', 'ROUND_HALF_DOWN', 'ROUND_HALF_EVEN', 'ROUND_HALF_UP', 'ROUND_UP', 'ROUTINE', + 'ROW', 'ROWNUMBER', 'ROWS', 'ROWSET', 'ROW_NUMBER', 'RRN', 'RUN', + 'SAVEPOINT', 'SCHEMA', 'SCOPE', 'SCRATCHPAD', 'SCROLL', 'SEARCH', 'SECOND', 'SECONDS', 'SECQTY', 'SECURITY', 'SENSITIVE', + 'SEQUENCE', 'SESSION', 'SESSION_USER', 'SIGNAL', 'SIMILAR', 'SIMPLE', 'SMALLINT', 'SNAN', 'SOME', 'SOURCE', 'SPECIFIC', + 'SPECIFICTYPE', 'SQL', 'SQLEXCEPTION', 'SQLID', 'SQLSTATE', 'SQLWARNING', 'SQRT', 'STACKED', 'STANDARD', 'START', 'STARTING', + 'STATEMENT', 'STATIC', 'STATMENT', 'STAY', 'STDDEV_POP', 'STDDEV_SAMP', 'STOGROUP', 'STORES', 'STYLE', 'SUBMULTISET', 'SUBSTRING', + 'SUM', 'SUMMARY', 'SYMMETRIC', 'SYNONYM', 'SYSFUN', 'SYSIBM', 'SYSPROC', 'SYSTEM', 'SYSTEM_USER', + 'TABLE', 'TABLESAMPLE', 'TABLESPACE', 'THEN', 'TIME', 'TIMESTAMP', 'TIMEZONE_HOUR', 'TIMEZONE_MINUTE', 'TO', 'TRAILING', 'TRANSACTION', + 'TRANSLATE', 'TRANSLATION', 'TREAT', 'TRIGGER', 'TRIM', 'TRUE', 'TRUNCATE', 'TYPE', + 'UESCAPE', 'UNDO', 'UNIQUE', 'UNKNOWN', 'UNNEST', 'UNTIL', 'UPPER', 'USAGE', 'USER', 'USING', + 'VALIDPROC', 'VALUE', 'VARCHAR', 'VARIABLE', 'VARIANT', 'VARYING', 'VAR_POP', 'VAR_SAMP', 'VCAT', 'VERSION', 'VIEW', + 'VOLATILE', 'VOLUMES', 'WHEN', 'WHENEVER', 'WHILE', 'WIDTH_BUCKET', 'WINDOW', 'WITH', 'WITHIN', 'WITHOUT', 'WLM', 'WRITE', + 'XMLELEMENT', 'XMLEXISTS', 'XMLNAMESPACES', + 'YEAR', 'YEARS', +]; + +const reservedToplevelWords = [ + 'ADD', 'AFTER', 'ALTER COLUMN', 'ALTER TABLE', + 'DELETE FROM', + 'EXCEPT', + 'FETCH FIRST', 'FROM', + 'GROUP BY', 'GO', + 'HAVING', + 'INSERT INTO', 'INTERSECT', + 'LIMIT', + 'ORDER BY', + 'SELECT', 'SET CURRENT SCHEMA', 'SET SCHEMA', 'SET', + 'UNION ALL', 'UPDATE', + 'VALUES', + 'WHERE', +]; + +const reservedNewlineWords = [ + 'AND', + 'CROSS JOIN', + 'INNER JOIN', + 'JOIN', + 'LEFT JOIN', 'LEFT OUTER JOIN', + 'OR', 'OUTER JOIN', + 'RIGHT JOIN', 'RIGHT OUTER JOIN', +]; + +let tokenizer; + +/** + * + */ +export default class Db2Formatter { + /** + * @param {Object} cfg Different set of configurations + */ + constructor(cfg) { + this.cfg = cfg; + } + + /** + * Formats DB2 query to make it easier to read + * + * @param {String} query The DB2 query string + * @return {String} formatted string + */ + format(query) { + if (!tokenizer) { + tokenizer = new Tokenizer({ + reservedWords, + reservedToplevelWords, + reservedNewlineWords, + stringTypes: [`""`, '\'\'', '``', '[]'], + openParens: ['('], + closeParens: [')'], + indexedPlaceholderTypes: ['?'], + namedPlaceholderTypes: [':'], + lineCommentTypes: ['--'], + specialWordChars: ['#', '@'], + }); + } + return new Formatter(this.cfg, tokenizer).format(query); + } +} diff --git a/web/packages/editorLsp/sqlFormatter/languages/N1qlFormatter.js b/web/packages/editorLsp/sqlFormatter/languages/N1qlFormatter.js new file mode 100644 index 0000000000..cc1641c6fe --- /dev/null +++ b/web/packages/editorLsp/sqlFormatter/languages/N1qlFormatter.js @@ -0,0 +1,110 @@ +/* + * Copyright 2019 WeBank + * + * 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 Formatter from '../core/Formatter'; +import Tokenizer from '../core/Tokenizer'; + +const reservedWords = [ + 'ALL', 'ALTER', 'ANALYZE', 'AND', 'ANY', 'ARRAY', 'AS', 'ASC', + 'BEGIN', 'BETWEEN', 'BINARY', 'BOOLEAN', 'BREAK', 'BUCKET', 'BUILD', 'BY', + 'CALL', 'CASE', 'CAST', 'CLUSTER', 'COLLATE', 'COLLECTION', 'COMMIT', 'CONNECT', 'CONTINUE', 'CORRELATE', 'COVER', 'CREATE', + 'DATABASE', 'DATASET', 'DATASTORE', 'DECLARE', 'DECREMENT', 'DELETE', 'DERIVED', 'DESC', 'DESCRIBE', 'DISTINCT', 'DO', 'DROP', + 'EACH', 'ELEMENT', 'ELSE', 'END', 'EVERY', 'EXCEPT', 'EXCLUDE', 'EXECUTE', 'EXISTS', 'EXPLAIN', + 'FALSE', 'FETCH', 'FIRST', 'FLATTEN', 'FOR', 'FORCE', 'FROM', 'FUNCTION', + 'GRANT', 'GROUP', 'GSI', + 'HAVING', + 'IF', 'IGNORE', 'ILIKE', 'IN', 'INCLUDE', 'INCREMENT', 'INDEX', 'INFER', 'INLINE', 'INNER', 'INSERT', 'INTERSECT', 'INTO', 'IS', + 'JOIN', + 'KEY', 'KEYS', 'KEYSPACE', 'KNOWN', + 'LAST', 'LEFT', 'LET', 'LETTING', 'LIKE', 'LIMIT', 'LSM', + 'MAP', 'MAPPING', 'MATCHED', 'MATERIALIZED', 'MERGE', 'MINUS', 'MISSING', + 'NAMESPACE', 'NEST', 'NOT', 'NULL', 'NUMBER', + 'OBJECT', 'OFFSET', 'ON', 'OPTION', 'OR', 'ORDER', 'OUTER', 'OVER', + 'PARSE', 'PARTITION', 'PASSWORD', 'PATH', 'POOL', 'PREPARE', 'PRIMARY', 'PRIVATE', 'PRIVILEGE', 'PROCEDURE', 'PUBLIC', + 'RAW', 'REALM', 'REDUCE', 'RENAME', 'RETURN', 'RETURNING', 'REVOKE', 'RIGHT', 'ROLE', 'ROLLBACK', + 'SATISFIES', 'SCHEMA', 'SELECT', 'SELF', 'SEMI', 'SET', 'SHOW', 'SOME', 'START', 'STATISTICS', 'STRING', 'SYSTEM', + 'THEN', 'TO', 'TRANSACTION', 'TRIGGER', 'TRUE', 'TRUNCATE', + 'UNDER', 'UNION', 'UNIQUE', 'UNKNOWN', 'UNNEST', 'UNSET', 'UPDATE', 'UPSERT', 'USE', 'USER', 'USING', + 'VALIDATE', 'VALUE', 'VALUED', 'VALUES', 'VIA', 'VIEW', + 'WHEN', 'WHERE', 'WHILE', 'WITH', 'WITHIN', 'WORK', + 'XOR', +]; + +const reservedToplevelWords = [ + 'DELETE FROM', + 'EXCEPT ALL', 'EXCEPT', 'EXPLAIN DELETE FROM', 'EXPLAIN UPDATE', 'EXPLAIN UPSERT', + 'FROM', + 'GROUP BY', + 'HAVING', + 'INFER', 'INSERT INTO', 'INTERSECT ALL', 'INTERSECT', + 'LET', 'LIMIT', + 'MERGE', + 'NEST', + 'ORDER BY', + 'PREPARE', + 'SELECT', 'SET CURRENT SCHEMA', 'SET SCHEMA', 'SET', + 'UNION ALL', 'UNION', 'UNNEST', 'UPDATE', 'UPSERT', 'USE KEYS', + 'VALUES', + 'WHERE', +]; + +const reservedNewlineWords = [ + 'AND', + 'INNER JOIN', + 'JOIN', + 'LEFT JOIN', + 'LEFT OUTER JOIN', + 'OR', 'OUTER JOIN', + 'RIGHT JOIN', 'RIGHT OUTER JOIN', + 'XOR', +]; + +let tokenizer; + +/** + * + */ +export default class N1qlFormatter { + /** + * @param {Object} cfg Different set of configurations + */ + constructor(cfg) { + this.cfg = cfg; + } + + /** + * Format the whitespace in a N1QL string to make it easier to read + * + * @param {String} query The N1QL string + * @return {String} formatted string + */ + format(query) { + if (!tokenizer) { + tokenizer = new Tokenizer({ + reservedWords, + reservedToplevelWords, + reservedNewlineWords, + stringTypes: [`""`, '\'\'', '``'], + openParens: ['(', '[', '{'], + closeParens: [')', ']', '}'], + namedPlaceholderTypes: ['$'], + lineCommentTypes: ['#', '--'], + }); + } + return new Formatter(this.cfg, tokenizer).format(query); + } +} diff --git a/web/packages/editorLsp/sqlFormatter/languages/PlSqlFormatter.js b/web/packages/editorLsp/sqlFormatter/languages/PlSqlFormatter.js new file mode 100644 index 0000000000..3568a0d999 --- /dev/null +++ b/web/packages/editorLsp/sqlFormatter/languages/PlSqlFormatter.js @@ -0,0 +1,133 @@ +/* + * Copyright 2019 WeBank + * + * 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 Formatter from '../core/Formatter'; +import Tokenizer from '../core/Tokenizer'; + +const reservedWords = [ + 'A', 'ACCESSIBLE', 'AGENT', 'AGGREGATE', 'ALL', 'ALTER', 'ANY', 'ARRAY', 'AS', 'ASC', 'AT', 'ATTRIBUTE', 'AUTHID', 'AVG', + 'BETWEEN', 'BFILE_BASE', 'BINARY_INTEGER', 'BINARY', 'BLOB_BASE', 'BLOCK', 'BODY', 'BOOLEAN', 'BOTH', 'BOUND', + 'BULK', 'BY', 'BYTE', + 'C', 'CALL', 'CALLING', 'CASCADE', 'CASE', 'CHAR_BASE', 'CHAR', 'CHARACTER', 'CHARSET', 'CHARSETFORM', 'CHARSETID', + 'CHECK', 'CLOB_BASE', 'CLONE', 'CLOSE', 'CLUSTER', 'CLUSTERS', 'COALESCE', 'COLAUTH', 'COLLECT', 'COLUMNS', 'COMMENT', + 'COMMIT', 'COMMITTED', 'COMPILED', 'COMPRESS', 'CONNECT', 'CONSTANT', 'CONSTRUCTOR', 'CONTEXT', 'CONTINUE', 'CONVERT', + 'COUNT', 'CRASH', 'CREATE', 'CREDENTIAL', 'CURRENT', 'CURRVAL', 'CURSOR', 'CUSTOMDATUM', + 'DANGLING', 'DATA', 'DATE_BASE', 'DATE', 'DAY', 'DECIMAL', 'DEFAULT', 'DEFINE', 'DELETE', 'DESC', + 'DETERMINISTIC', 'DIRECTORY', 'DISTINCT', 'DO', 'DOUBLE', 'DROP', 'DURATION', + 'ELEMENT', 'ELSIF', 'EMPTY', 'ESCAPE', 'EXCEPTIONS', 'EXCLUSIVE', 'EXECUTE', 'EXISTS', + 'EXIT', 'EXTENDS', 'EXTERNAL', 'EXTRACT', + 'FALSE', 'FETCH', 'FINAL', 'FIRST', 'FIXED', 'FLOAT', 'FOR', 'FORALL', 'FORCE', 'FROM', 'FUNCTION', + 'GENERAL', 'GOTO', 'GRANT', 'GROUP', 'HASH', 'HEAP', 'HIDDEN', 'HOUR', + 'IDENTIFIED', 'IF', 'IMMEDIATE', 'IN', 'INCLUDING', 'INDEX', 'INDEXES', 'INDICATOR', 'INDICES', 'INFINITE', + 'INSTANTIABLE', 'INT', 'INTEGER', 'INTERFACE', 'INTERVAL', 'INTO', 'INVALIDATE', 'IS', 'ISOLATION', + 'JAVA', + 'LANGUAGE', 'LARGE', 'LEADING', 'LENGTH', 'LEVEL', 'LIBRARY', 'LIKE', 'LIKE2', 'LIKE4', 'LIKEC', 'LIMITED', 'LOCAL', + 'LOCK', 'LONG', + 'MAP', 'MAX', 'MAXLEN', 'MEMBER', 'MERGE', 'MIN', 'MINUS', 'MINUTE', 'MLSLABEL', 'MOD', 'MODE', 'MONTH', 'MULTISET', + 'NAME', 'NAN', 'NATIONAL', 'NATIVE', 'NATURAL', 'NATURALN', 'NCHAR', 'NEW', 'NEXTVAL', 'NOCOMPRESS', 'NOCOPY', 'NOT', + 'NOWAIT', 'NULL', 'NULLIF', 'NUMBER_BASE', 'NUMBER', + 'OBJECT', 'OCICOLL', 'OCIDATE', 'OCIDATETIME', 'OCIDURATION', 'OCIINTERVAL', 'OCILOBLOCATOR', 'OCINUMBER', 'OCIRAW', + 'OCIREF', 'OCIREFCURSOR', 'OCIROWID', 'OCISTRING', 'OCITYPE', 'OF', 'OLD', 'ON', 'ONLY', 'OPAQUE', 'OPEN', 'OPERATOR', + 'OPTION', 'ORACLE', 'ORADATA', 'ORDER', 'ORGANIZATION', 'ORLANY', 'ORLVARY', 'OTHERS', 'OUT', 'OVERLAPS', + 'OVERRIDING', + 'PACKAGE', 'PARALLEL_ENABLE', 'PARAMETER', 'PARAMETERS', 'PARENT', 'PARTITION', 'PASCAL', 'PCTFREE', 'PIPE', 'PIPELINED', + 'PLS_INTEGER', 'PLUGGABLE', 'POSITIVE', 'POSITIVEN', 'PRAGMA', 'PRECISION', 'PRIOR', 'PRIVATE', 'PROCEDURE', 'PUBLIC', + 'RAISE', 'RANGE', 'RAW', 'READ', 'REAL', 'RECORD', 'REF', 'REFERENCE', 'RELEASE', 'RELIES_ON', 'REM', 'REMAINDER', + 'RENAME', 'RESOURCE', 'RESULT_CACHE', 'RESULT', 'RETURN', 'RETURNING', 'REVERSE', 'REVOKE', 'ROLLBACK', 'ROW', 'ROWID', + 'ROWNUM', 'ROWTYPE', + 'SAMPLE', 'SAVE', 'SAVEPOINT', 'SB1', 'SB2', 'SB4', 'SECOND', 'SEGMENT', 'SELF', 'SEPARATE', 'SEQUENCE', + 'SERIALIZABLE', 'SHARE', 'SHORT', 'SIZE_T', 'SIZE', 'SMALLINT', 'SOME', 'SPACE', 'SPARSE', 'SQL', 'SQLCODE', + 'SQLDATA', 'SQLERRM', 'SQLNAME', 'SQLSTATE', 'STANDARD', 'START', 'STATIC', 'STDDEV', 'STORED', 'STRING', 'STRUCT', + 'STYLE', 'SUBMULTISET', 'SUBPARTITION', 'SUBSTITUTABLE', 'SUBTYPE', 'SUCCESSFUL', 'SUM', 'SYNONYM', 'SYSDATE', + 'TABAUTH', 'TABLE', 'TDO', 'THE', 'THEN', 'TIME', 'TIMESTAMP', 'TIMEZONE_ABBR', 'TIMEZONE_HOUR', 'TIMEZONE_MINUTE', + 'TIMEZONE_REGION', 'TO', 'TRAILING', 'TRANSACTION', 'TRANSACTIONAL', 'TRIGGER', 'TRUE', 'TRUSTED', 'TYPE', + 'UB1', 'UB2', 'UB4', 'UID', 'UNDER', 'UNIQUE', 'UNPLUG', 'UNSIGNED', 'UNTRUSTED', 'USE', 'USER', 'USING', + 'VALIDATE', 'VALIST', 'VALUE', 'VARCHAR', 'VARCHAR2', 'VARIABLE', 'VARIANCE', 'VARRAY', 'VARYING', 'VIEW', 'VIEWS', 'VOID', + 'WHENEVER', 'WHILE', 'WITH', 'WORK', 'WRAPPED', 'WRITE', + 'YEAR', + 'ZONE', +]; + +const reservedToplevelWords = [ + 'ADD', 'ALTER COLUMN', 'ALTER TABLE', + 'BEGIN', + 'CONNECT BY', + 'DECLARE', 'DELETE FROM', 'DELETE', + 'END', 'EXCEPT', 'EXCEPTION', + 'FETCH FIRST', 'FROM', + 'GROUP BY', + 'HAVING', + 'INSERT INTO', 'INSERT', 'INTERSECT', + 'LIMIT', 'LOOP', + 'MODIFY', + 'ORDER BY', + 'SELECT', 'SET CURRENT SCHEMA', 'SET SCHEMA', 'SET', 'START WITH', + 'UNION ALL', 'UNION', 'UPDATE', + 'VALUES', + 'WHERE', +]; + +const reservedNewlineWords = [ + 'AND', + 'CROSS APPLY', 'CROSS JOIN', + 'ELSE', 'END', + 'INNER JOIN', + 'JOIN', + 'LEFT JOIN', 'LEFT OUTER JOIN', + 'OR', 'OUTER APPLY', 'OUTER JOIN', + 'RIGHT JOIN', 'RIGHT OUTER JOIN', + 'WHEN', + 'XOR', +]; + +let tokenizer; + +/** + * + */ +export default class PlSqlFormatter { + /** + * @param {Object} cfg Different set of configurations + */ + constructor(cfg) { + this.cfg = cfg; + } + + /** + * Format the whitespace in a PL/SQL string to make it easier to read + * + * @param {String} query The PL/SQL string + * @return {String} formatted string + */ + format(query) { + if (!tokenizer) { + tokenizer = new Tokenizer({ + reservedWords, + reservedToplevelWords, + reservedNewlineWords, + stringTypes: [`""`, 'N\'\'', '\'\'', '``'], + openParens: ['(', 'CASE'], + closeParens: [')', 'END'], + indexedPlaceholderTypes: ['?'], + namedPlaceholderTypes: [':'], + lineCommentTypes: ['--'], + specialWordChars: ['_', '$', '#', '.', '@'], + }); + } + return new Formatter(this.cfg, tokenizer).format(query); + } +} diff --git a/web/packages/editorLsp/sqlFormatter/languages/StandardSqlFormatter.js b/web/packages/editorLsp/sqlFormatter/languages/StandardSqlFormatter.js new file mode 100644 index 0000000000..8c73941d2d --- /dev/null +++ b/web/packages/editorLsp/sqlFormatter/languages/StandardSqlFormatter.js @@ -0,0 +1,127 @@ +/* + * Copyright 2019 WeBank + * + * 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 Formatter from '../core/Formatter'; +import Tokenizer from '../core/Tokenizer'; + +const reservedWords = [ + 'ACCESSIBLE', 'ACTION', 'AGAINST', 'AGGREGATE', 'ALGORITHM', 'ALL', 'ALTER', 'ANALYSE', 'ANALYZE', 'AS', 'ASC', 'AUTOCOMMIT', + 'AUTO_INCREMENT', + 'BACKUP', 'BEGIN', 'BETWEEN', 'BINLOG', 'BOTH', + 'CASCADE', 'CASE', 'CHANGE', 'CHANGED', 'CHARACTER SET', 'CHARSET', 'CHECK', 'CHECKSUM', 'COLLATE', 'COLLATION', 'COLUMN', 'COLUMNS', + 'COMMENT', 'COMMIT', 'COMMITTED', 'COMPRESSED', 'CONCURRENT', 'CONSTRAINT', 'CONTAINS', 'CONVERT', 'CREATE', 'CROSS', + 'CURRENT_TIMESTAMP', + 'DATABASE', 'DATABASES', 'DAY', 'DAY_HOUR', 'DAY_MINUTE', 'DAY_SECOND', 'DEFAULT', 'DEFINER', 'DELAYED', 'DELETE', 'DESC', 'DESCRIBE', + 'DETERMINISTIC', 'DISTINCT', 'DISTINCTROW', 'DIV', 'DO', 'DROP', 'DUMPFILE', 'DUPLICATE', 'DYNAMIC', + 'ELSE', 'ENCLOSED', 'END', 'ENGINE', 'ENGINES', 'ENGINE_TYPE', 'ESCAPE', 'ESCAPED', 'EVENTS', 'EXEC', 'EXECUTE', 'EXISTS', 'EXPLAIN', + 'EXTENDED', + 'FAST', 'FETCH', 'FIELDS', 'FILE', 'FIRST', 'FIXED', 'FLUSH', 'FOR', 'FORCE', 'FOREIGN', 'FULL', 'FULLTEXT', 'FUNCTION', + 'GLOBAL', 'GRANT', 'GRANTS', 'GROUP_CONCAT', + 'HEAP', 'HIGH_PRIORITY', 'HOSTS', 'HOUR', 'HOUR_MINUTE', 'HOUR_SECOND', + 'IDENTIFIED', 'IF', 'IFNULL', 'IGNORE', 'IN', 'INDEX', 'INDEXES', 'INFILE', 'INSERT', 'INSERT_ID', 'INSERT_METHOD', 'INTERVAL', + 'INTO', 'INVOKER', 'IS', 'ISOLATION', + 'KEY', 'KEYS', 'KILL', + 'LAST_INSERT_ID', 'LEADING', 'LEVEL', 'LIKE', 'LINEAR', 'LINES', 'LOAD', 'LOCAL', 'LOCK', 'LOCKS', 'LOGS', 'LOW_PRIORITY', + 'MARIA', 'MASTER', 'MASTER_CONNECT_RETRY', 'MASTER_HOST', 'MASTER_LOG_FILE', 'MATCH', 'MAX_CONNECTIONS_PER_HOUR', + 'MAX_QUERIES_PER_HOUR', 'MAX_ROWS', 'MAX_UPDATES_PER_HOUR', 'MAX_USER_CONNECTIONS', 'MEDIUM', 'MERGE', 'MINUTE', 'MINUTE_SECOND', + 'MIN_ROWS', 'MODE', 'MODIFY', 'MONTH', 'MRG_MYISAM', 'MYISAM', + 'NAMES', 'NATURAL', 'NOT', 'NOW()', 'NULL', + 'OFFSET', 'ON DELETE', 'ON UPDATE', 'ON', 'ONLY', 'OPEN', 'OPTIMIZE', 'OPTION', 'OPTIONALLY', 'OUTFILE', + 'PACK_KEYS', 'PAGE', 'PARTIAL', 'PARTITION', 'PARTITIONS', 'PASSWORD', 'PRIMARY', 'PRIVILEGES', 'PROCEDURE', 'PROCESS', 'PROCESSLIST', + 'PURGE', + 'QUICK', + 'RAID0', 'RAID_CHUNKS', 'RAID_CHUNKSIZE', 'RAID_TYPE', 'RANGE', 'READ', 'READ_ONLY', 'READ_WRITE', 'REFERENCES', 'REGEXP', 'RELOAD', + 'RENAME', 'REPAIR', 'REPEATABLE', 'REPLACE', 'REPLICATION', 'RESET', 'RESTORE', 'RESTRICT', 'RETURN', 'RETURNS', 'REVOKE', 'RLIKE', + 'ROLLBACK', 'ROW', 'ROWS', 'ROW_FORMAT', + 'SECOND', 'SECURITY', 'SEPARATOR', 'SERIALIZABLE', 'SESSION', 'SHARE', 'SHOW', 'SHUTDOWN', 'SLAVE', 'SONAME', 'SOUNDS', 'SQL', + 'SQL_AUTO_IS_NULL', 'SQL_BIG_RESULT', 'SQL_BIG_SELECTS', 'SQL_BIG_TABLES', 'SQL_BUFFER_RESULT', 'SQL_CACHE', 'SQL_CALC_FOUND_ROWS', + 'SQL_LOG_BIN', 'SQL_LOG_OFF', 'SQL_LOG_UPDATE', 'SQL_LOW_PRIORITY_UPDATES', 'SQL_MAX_JOIN_SIZE', 'SQL_NO_CACHE', + 'SQL_QUOTE_SHOW_CREATE', 'SQL_SAFE_UPDATES', 'SQL_SELECT_LIMIT', 'SQL_SLAVE_SKIP_COUNTER', 'SQL_SMALL_RESULT', 'SQL_WARNINGS', + 'START', 'STARTING', 'STATUS', 'STOP', 'STORAGE', 'STRAIGHT_JOIN', 'STRING', 'STRIPED', 'SUPER', + 'TABLE', 'TABLES', 'TEMPORARY', 'TERMINATED', 'THEN', 'TO', 'TRAILING', 'TRANSACTIONAL', 'TRUE', 'TRUNCATE', 'TYPE', 'TYPES', + 'UNCOMMITTED', 'UNIQUE', 'UNLOCK', 'UNSIGNED', 'USAGE', 'USE', 'USING', + 'VARIABLES', 'VIEW', 'WHEN', 'WITH', 'WORK', 'WRITE', + 'YEAR_MONTH', +]; + +const reservedToplevelWords = [ + 'ADD', 'AFTER', 'ALTER COLUMN', 'ALTER TABLE', + 'DELETE FROM', + 'EXCEPT', + 'FETCH FIRST', 'FROM', + 'GROUP BY', 'GO', + 'HAVING', + 'INSERT INTO', 'INSERT', 'INTERSECT', + 'LIMIT', + 'MODIFY', + 'ORDER BY', + 'SELECT', 'SET CURRENT SCHEMA', 'SET SCHEMA', 'SET', + 'UNION ALL', 'UNION', 'UPDATE', + 'VALUES', + 'WHERE', +]; + +const reservedNewlineWords = [ + 'AND', + 'CROSS APPLY', 'CROSS JOIN', + 'ELSE', + 'INNER JOIN', + 'JOIN', + 'LEFT JOIN', 'LEFT OUTER JOIN', + 'OR', 'OUTER APPLY', 'OUTER JOIN', + 'RIGHT JOIN', 'RIGHT OUTER JOIN', + 'WHEN', + 'XOR', +]; + +let tokenizer; + +/** + * + */ +export default class StandardSqlFormatter { + /** + * @param {Object} cfg Different set of configurations + */ + constructor(cfg) { + this.cfg = cfg; + } + + /** + * Format the whitespace in a Standard SQL string to make it easier to read + * + * @param {String} query The Standard SQL string + * @return {String} formatted string + */ + format(query) { + if (!tokenizer) { + tokenizer = new Tokenizer({ + reservedWords, + reservedToplevelWords, + reservedNewlineWords, + stringTypes: [`""`, 'N\'\'', '\'\'', '``', '[]'], + openParens: ['(', 'CASE'], + closeParens: [')', 'END'], + indexedPlaceholderTypes: ['?'], + namedPlaceholderTypes: ['@', ':'], + lineCommentTypes: ['#', '--'], + paramsPattern: /^(\$\{[^\}\r\n]+\})/, + }); + } + return new Formatter(this.cfg, tokenizer).format(query); + } +} diff --git a/web/packages/editorLsp/sqlFormatter/sqlFormatter.js b/web/packages/editorLsp/sqlFormatter/sqlFormatter.js new file mode 100644 index 0000000000..d182f31be8 --- /dev/null +++ b/web/packages/editorLsp/sqlFormatter/sqlFormatter.js @@ -0,0 +1,51 @@ +/* + * Copyright 2019 WeBank + * + * 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 Db2Formatter from './languages/Db2Formatter'; +import N1qlFormatter from './languages/N1qlFormatter'; +import PlSqlFormatter from './languages/PlSqlFormatter'; +import StandardSqlFormatter from './languages/StandardSqlFormatter'; + +export default { + /** + * Format whitespaces in a query to make it easier to read. + * + * @param {String} query + * @param {Object} cfg + * @param {String} cfg.language Query language, default is Standard SQL + * @param {String} cfg.indent Characters used for indentation, default is " " (2 spaces) + * @param {Object} cfg.params Collection of params for placeholder replacement + * @return {String} + */ + format: (query, cfg) => { + cfg = cfg || {}; + + switch (cfg.language) { + case 'db2': + return new Db2Formatter(cfg).format(query); + case 'n1ql': + return new N1qlFormatter(cfg).format(query); + case 'pl/sql': + return new PlSqlFormatter(cfg).format(query); + case 'sql': + case undefined: + return new StandardSqlFormatter(cfg).format(query); + default: + throw Error(`Unsupported SQL dialect: ${cfg.language}`); + } + }, +}; diff --git a/web/packages/editorLsp/theme/defaultView.js b/web/packages/editorLsp/theme/defaultView.js new file mode 100644 index 0000000000..4bc3435571 --- /dev/null +++ b/web/packages/editorLsp/theme/defaultView.js @@ -0,0 +1,30 @@ +/* + * Copyright 2019 WeBank + * + * 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. + * + */ + +export default { + register(monaco) { + monaco.editor.defineTheme('defaultview', { + base: 'vs', + inherit: true, + rules: [{ background: '#FFFFFF' }], + colors: { + 'editor.lineHighlightBackground': '#ffffff', + 'editorGutter.background': '#f7f7f7', + }, + }); + }, +}; diff --git a/web/packages/editorLsp/theme/logView.js b/web/packages/editorLsp/theme/logView.js new file mode 100644 index 0000000000..1a1da0a6ce --- /dev/null +++ b/web/packages/editorLsp/theme/logView.js @@ -0,0 +1,36 @@ +/* + * Copyright 2019 WeBank + * + * 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. + * + */ + +export default { + register(monaco) { + monaco.editor.defineTheme('logview', { + base: 'vs', + inherit: true, + rules: [ + { token: 'log-info', foreground: '4b71ca' }, + { token: 'log-error', foreground: 'ff0000', fontStyle: 'bold' }, + { token: 'log-warn', foreground: 'FFA500' }, + { token: 'log-date', foreground: '008800' }, + { token: 'log-normal', foreground: '808080' }, + ], + colors: { + 'editor.lineHighlightBackground': '#ffffff', + 'editorGutter.background': '#f7f7f7', + }, + }); + }, +}; diff --git a/web/packages/editorLsp/util.js b/web/packages/editorLsp/util.js new file mode 100644 index 0000000000..689acc96f7 --- /dev/null +++ b/web/packages/editorLsp/util.js @@ -0,0 +1,72 @@ +/* + * Copyright 2019 WeBank + * + * 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 {filter, isFunction } from 'lodash'; + +/** + * + * @param {*} match 匹配到的文本 + * @param {*} proposals 需要过滤的列表 + * @param {*} fieldString 过滤条件中的字段名 + * @param {*} attachMatch 附加的过滤条件 + * @param {*} needSplit 是否需要对insertText进行截取 + * @return {*} + */ +const getReturnList = ({ match, proposals, fieldString, attachMatch, needSplit, position }, monaco) => { + if (!match || isFunction(match)) { + return; + } + let replacedStr = ''; + for (let i of match) { + const reg = /[~'!@#¥$%^&*()-+_=:]/g; + if (reg.test(i)) { + replacedStr += `\\${i}`; + } else { + replacedStr += i; + } + } + const regexp = new RegExp(`${replacedStr}`, 'i'); + let items = []; + if (attachMatch && !needSplit) { + items = filter(proposals, (it) => it[fieldString].startsWith(attachMatch) && regexp.test(it[fieldString])); + } else if (attachMatch && needSplit) { + // 这里是对例如create table和drop table的情况进行处理 + proposals.forEach((it) => { + if (regexp.test(it[fieldString]) && it.label.indexOf(attachMatch[1]) === 0) { + const text = it.insertText; + items.push({ + label: it.label, + documentation: it.documentation, + insertText: text.slice(text.indexOf(' ') + 1, text.length - 1), + detail: it.detail, + }); + } + }); + } else { + items = filter(proposals, (it) => regexp.test(it[fieldString])); + } + if (position && monaco) { + items.forEach( it => it.range = new monaco.Range(position.lineNumber, position.column - match.length , position.lineNumber, position.column)); + } + return { + isIncomplete: true, + suggestions: items.slice(0,200), // 太多建议数据无意义 + }; +} +export { + getReturnList, +}; diff --git a/web/packages/exts/open-source/README.md b/web/packages/exts/open-source/README.md new file mode 100644 index 0000000000..8f6c44f257 --- /dev/null +++ b/web/packages/exts/open-source/README.md @@ -0,0 +1,5 @@ +## 说明 + +此扩展包括以下功能: +- 工作流开发底部TAB面板复制历史 +- Scriptis脚本发布数据服务API \ No newline at end of file diff --git a/web/packages/exts/open-source/i18n/en.json b/web/packages/exts/open-source/i18n/en.json index c304e8793e..91243d12a1 100644 --- a/web/packages/exts/open-source/i18n/en.json +++ b/web/packages/exts/open-source/i18n/en.json @@ -71,6 +71,25 @@ "publishSuccess": "Publish Success" }, "tagPlaceholder": "tag,click enter add" + }, + "operateProxyTip": "Before entering, please select a proxy user or cancel", + "applyOperatePorxy": "Please apply for a proxy user first" + }, + "ext": { + "opensource": { + "selectservice": "Please select service", + "defalut": "Default value", + "detail": "Detailed description", + "longer1024": "No longer than 1024 characters!", + "rowlimit": "Too many rows, please query in batches!", + "outlimit": "Exceed the limit, please modify!!", + "cnanotnull": "Parameter cannot be null!", + "entercreate": "Tags, press enter to create", + "status": "Status", + "opreator": "Operator", + "copytime": "copy time", + "copyHistory": "Copy History", + "errmsg": "Error message" } } } diff --git a/web/packages/exts/open-source/i18n/zh.json b/web/packages/exts/open-source/i18n/zh.json index 8d2fc7f259..7057175a14 100644 --- a/web/packages/exts/open-source/i18n/zh.json +++ b/web/packages/exts/open-source/i18n/zh.json @@ -71,6 +71,25 @@ "publishSuccess": "发布成功" }, "tagPlaceholder": "标签,按 enter 创建" + }, + "operateProxyTip": "请选择代理用户进入或者取消", + "applyOperatePorxy": "请先申请实名用户允许的代理用户" + }, + "ext": { + "opensource": { + "selectservice": "请选择服务", + "defalut": "默认值", + "detail": "详细说明", + "longer1024": "不能超过1024个字符!", + "rowlimit": "行数过多请分批查询!", + "outlimit": "超出限制请修改!", + "cnanotnull": "参数值不能为空", + "entercreate": "标签,按 enter 创建", + "status": "状态", + "opreator": "复制人", + "copytime": "复制时间", + "copyHistory": "复制历史", + "errmsg": "错误信息" } } } diff --git a/web/packages/exts/open-source/index.js b/web/packages/exts/open-source/index.js index 5576ee7ff7..ca81ba104b 100644 --- a/web/packages/exts/open-source/index.js +++ b/web/packages/exts/open-source/index.js @@ -1,5 +1,10 @@ import ApiPublish from './scriptis/apiPublish/index.vue' - +import CopyHistory from './workflows/bottomTab/copyHistory.vue' +import createProxyModal from './proxyUser/index' +import i18n from '@dataspherestudio/shared/common/i18n' +import storage from '@dataspherestudio/shared/common/helper/storage' +import api from '@dataspherestudio/shared/common/service/api' +import API_PATH from '@dataspherestudio/shared/common/config/apiPath.js' /** * 插件绑定 */ @@ -17,4 +22,37 @@ export default function () { component: ApiPublish } }) + + // 登录后提示运维用户切换 + this.bindHook('after_login', async function ({homePageRes, context}) { + await api.fetch(`${API_PATH.WORKSPACE_PATH}workspaces/${homePageRes.workspaceId}`, 'get') + return api.fetch('/dss/framework/admin/globalLimits', {}, 'get').then((res) => { + let baseInfo = storage.get('baseInfo', 'local') + baseInfo = { + ...baseInfo, + dss: { + ...res.globalLimits + } + } + storage.set('baseInfo', baseInfo, 'local') + if (baseInfo.dss.proxyEnable) { + createProxyModal(homePageRes.homePageUrl, context) + } else { + context.$router.replace({path: homePageRes.homePageUrl}); + } + }) + }) + + // workflows: 工作流开发底部TAB面板复制历史 + this.bindHook('workflow_bottom_panel', function () { + return [ + { + name: i18n.t('message.ext.opensource.copyHistory'), + icon: 'md-paper-plane', + key: 'copyhistory', + component: CopyHistory + } + ] + }) + } diff --git a/web/packages/exts/open-source/proxyUser/index.js b/web/packages/exts/open-source/proxyUser/index.js new file mode 100644 index 0000000000..ceba0d4eb3 --- /dev/null +++ b/web/packages/exts/open-source/proxyUser/index.js @@ -0,0 +1,32 @@ +import Vue from 'vue' +import ProxyUserModal from './modal.vue' +import i18n from '@dataspherestudio/shared/common/i18n' + +const createProxyModal = (homePageRes, context) => { + let body = document.body; + let bindPhone = document.createElement('div') + bindPhone.setAttribute('id', 'proxy_modal_') + body.appendChild(bindPhone) + + return new Vue({ + i18n, + render: (h) => { + return h( + ProxyUserModal, + { + props: { + show: true, + canclose: false + }, + on: { + 'set-proxy': () => { + context.$router.replace({path: homePageRes}); + } + } + } + ) + } + }).$mount('#proxy_modal_') +} + +export default createProxyModal diff --git a/web/packages/exts/open-source/proxyUser/modal.vue b/web/packages/exts/open-source/proxyUser/modal.vue new file mode 100644 index 0000000000..884eb4b365 --- /dev/null +++ b/web/packages/exts/open-source/proxyUser/modal.vue @@ -0,0 +1,87 @@ + + diff --git a/web/packages/exts/open-source/scriptis/apiPublish/TagsInput/index.vue b/web/packages/exts/open-source/scriptis/apiPublish/TagsInput/index.vue index 9d28fb2390..11f8398d41 100644 --- a/web/packages/exts/open-source/scriptis/apiPublish/TagsInput/index.vue +++ b/web/packages/exts/open-source/scriptis/apiPublish/TagsInput/index.vue @@ -9,6 +9,7 @@ + + diff --git a/web/packages/exts/package.json b/web/packages/exts/package.json index b54d444fbb..a0ede29e14 100644 --- a/web/packages/exts/package.json +++ b/web/packages/exts/package.json @@ -1,7 +1,7 @@ { "name": "@dataspherestudio/exts", - "version": "1.1.4", + "version": "1.1.12", "dependencies": { - "@dataspherestudio/shared": "^1.1.4" + "@dataspherestudio/shared": "^1.1.12" } } diff --git a/web/packages/exts/sandbox/README.md b/web/packages/exts/sandbox/README.md new file mode 100644 index 0000000000..ef977fba09 --- /dev/null +++ b/web/packages/exts/sandbox/README.md @@ -0,0 +1,24 @@ +## 说明 + +此扩展包括以下功能,沙箱版本 +- 用户注册 +- 服务协议 +- 隐私政策 +- 工作空间首页顶部展示通知 + +沙箱版本打包时config.json 里exts需配置上此扩展及open-source + +```json + +"dss-plugin-sandbox": { + "module": "exts/sandbox/index.js", + "i18n": { + "en": "exts/sandbox/i18n/en.json", + "zh-CN": "exts/sandbox/i18n/zh.json" + }, + "options": null + } +``` + +## todo +packages里原来所有sandbox的部分移动到插件内 \ No newline at end of file diff --git a/web/packages/exts/sandbox/i18n/en.json b/web/packages/exts/sandbox/i18n/en.json new file mode 100644 index 0000000000..db7a0c3c26 --- /dev/null +++ b/web/packages/exts/sandbox/i18n/en.json @@ -0,0 +1,88 @@ +{ + "message": { + "workspace": { + "topNotice": { + "title": "Tips", + "content": "Dear Sandbox users: The sand environment is an open environment for testing, deploying all the components of the WeDataSphere open source version. Currently, only demos are allowed to run due to security restrictions, and users cannot add or modify test data. Please take care of the use of the sandbox environment." + } + }, + "register": { + "codeInfo": "Scan the code or email to get the invitation code. See the introduction on the right for details", + "invitationJoin": { + "welcome": "Scan the Qr Code below to join our DaraSphere Studio User Community!", + "community": "Datasphere stdio community group to stay up-to-date", + "join": "Scan the QR Code below to get in touch with our team members for invitation code and to join our community.", + "contact": "Contact team members for invitation code, scan code to add swarm robot", + "send": "Or send email to", + "tag": "for personalized assistance. Remember to include your company name, your position, and number of code needed.", + "thank": " If you need more than 5 codes, plaase state your reason." + }, + "registerTitle": "DataSphere Studio Trial Application", + "org": "Organization", + "invit": "Invitation Code", + "invitation": "Please enter your invitation code", + "organization": "Please enter your organization name", + "nam": "Name", + "name": "Please enter your name", + "pho": "Phone", + "phone": "Please enter your phone", + "ver": "Verification Code", + "veriCode": "Please enter your verification code", + "contact": "Contact Information", + "contactType": "Mailbox and wechat", + "application": "Immediate application", + "discover": { + "how": "How did you hear about us?", + "GitHub": "GitHub", + "OsChina": "OsChina/Gitee", + "search": "Search Engines", + "csdn": "CSDN", + "meeting": "Community", + "decomend": "Friend's/Colleague's recommendation" + }, + "des": "Requirement description", + "desc": "Please enter your business requirements, the maximum length is 160 characters", + "agree": { + "agr": "I agree to the", + "service": "Terms", + "and": "and", + "privacy": "Conditions" + }, + "errTip": { + "org": "Maximum length is 120 characters", + "phone": "Please input the correct mobile number", + "discovery": "Please select how you found DSS", + "desc": "Maximum length is 160 characters", + "contact": "Please enter your contact information", + "code": "Please enter the correct verification code", + "vericode": "Please enter the verification code", + "trim": "The input cannot have spaces", + "invitation": "The input format of verification code is incorrect", + "format": "The input format is incorrect, please re-enter" + }, + "tips": "Tip", + "success": "Successfully registered!", + "pass": "Your password is :", + "pa": "the last six digits of your mobile number", + "userNameTitle": "Your user name is :", + "userName": "cell-phone number", + "error": "Registration failed, please register again", + "watch": { + "welcome": "Follow our other open-source projects:", + "note": "follow ", + "and": "and ", + "star": "STAR ", + "project": "the following open source projects:", + "data": " DataSphereStudio - One-stop data application development management portal ", + "links": " Linkis - Computational middleware with multi engine", + "qualitis": "Qualitis - Heterogeneous data source & data quality management platform", + "Script": "Scriptis - Data development explores IDE tools", + "Visualis": "Visualis - BI tool for data visualization", + "Streamis": "Streamis - One-stop development management system for streaming application", + "Schedulis": "Schedulis - Workflow task scheduling system", + "Prophecis": "Prophecis - One-stop machine learning platform", + "Exchangis": "Exchangis - Data exchange platform" + } + } + } +} diff --git a/web/packages/exts/sandbox/i18n/zh.json b/web/packages/exts/sandbox/i18n/zh.json new file mode 100644 index 0000000000..3f6891226c --- /dev/null +++ b/web/packages/exts/sandbox/i18n/zh.json @@ -0,0 +1,90 @@ +{ + "message": { + "workspace": { + "topNotice": { + "title": "温馨提示", + "content": "沙箱环境属于开放体验环境,已部署WeDataSphere开源新版组件全部内容,但受安全限制仅开放指定DEMO的运行功能,且不能增改数据,敬请爱护使用。" + } + }, + "register": { + "codeInfo": "扫码或邮件获取邀请码,详见右边介绍", + "invitationJoin": { + "welcome": "欢迎扫码加入", + "community": "DataSphere Studio社区群了解最新动态", + "join": "请加入DataShpere Studio社区用户群,", + "contact": "联系团队成员获取邀请码,扫码添加加群机器人", + "send": "或发送邮件", + "tag": "邮件注明公司、职位及数量,如果申请的邀请码数量", + "thank": "超过5个请备注原因。DataSphere Studio团队感谢您的参与!" + }, + "inviation": "", + "remarks": "备注", + "registerTitle": "DataSphere Studio 试用申请", + "invit": "邀请码", + "invitation": "请输入您的邀请码", + "org": "组织", + "organization": "请输入您的组织名", + "nam": "姓名", + "name": "请输入您的姓名", + "pho": "手机", + "phone": "请填写联系人手机号码", + "ver": "验证码", + "veriCode": "请输入验证码", + "contact": "联系方式", + "contactType": "邮箱或者微信号", + "discover": { + "how": "如何发现DSS的", + "GitHub": "GitHub", + "OsChina": "OsChina/Gitee", + "search": "搜索引擎", + "csdn": "CSDN", + "meeting": "相关活动吸引", + "decomend": "朋友推荐" + }, + "des": "需求描述", + "desc": "请输入您的业务需求,最大长度为160个字符", + "agree": { + "agr": "同意并遵守", + "service": "《服务条款》", + "and": "和", + "privacy": "《隐私条款》" + }, + "application": "立即申请", + "errTip": { + "org": "最大长度为120个字符", + "phone": "请输入正确的手机号", + "discovery": "请选择您是如何发现DSS的", + "desc": "最大长度为160个字符", + "contact": "请输入您的联系方式", + "code": "请输入正确的验证码", + "vericode": "请输入验证码", + "trim": "输入的内容不能有空格", + "invitation": "邀请码格式不正确", + "format": "输入的格式有误,请重新输入" + }, + "tips": "提示", + "success": "注册成功!", + "pass": "您的密码是:", + "pa": "手机号后六位", + "userNameTitle": "您的用户名是:", + "userName": "手机号", + "error": "注册失败,请重新注册", + "watch": { + "welcome": "关于我们的其他开源项目:", + "note": "关注", + "and": "和", + "star": "STAR", + "project": "以下已开源项目: ", + "data": "DataSphereStudio——一站式数据应用开发管理门户", + "links": "Linkis——多引擎计算中间件", + "qualitis": "Qualitis——异构数据源数据质量管理平台", + "Script": "Scriptis——数据开发探索IDE工具", + "Visualis": "Visualis——数据可视化BI工具", + "Streamis": "Streamis一站式流式应用开发管理系统", + "Schedulis": "Schedulis——工作流任务调度系统", + "Prophecis": "Prophecis——一站式机器学习平台", + "Exchangis": "Exchangis——数据交换平台" + } + } + } +} diff --git a/web/packages/exts/sandbox/images/QR code.jpg b/web/packages/exts/sandbox/images/QR code.jpg new file mode 100644 index 0000000000..f85da5edab Binary files /dev/null and b/web/packages/exts/sandbox/images/QR code.jpg differ diff --git a/web/packages/exts/sandbox/index.js b/web/packages/exts/sandbox/index.js new file mode 100644 index 0000000000..0d15b56bef --- /dev/null +++ b/web/packages/exts/sandbox/index.js @@ -0,0 +1,90 @@ +import Vue from 'vue' +import i18n from '@dataspherestudio/shared/common/i18n' +import storage from "@dataspherestudio/shared/common/helper/storage" + + +const register = { + path: '/register', + name: 'register', + meta: { + title: 'Register', + publicPage: true, + keepAlive: true, + }, + component: () => + import ('./register/index.vue'), +} + +/** + * 插件绑定 + */ +export default function () { + // 增加注册页面 + this.bindHook('app_router_config', function (routes) { + routes.push(register) + }) + + // 登录页面增加注册链接 + // 工作空间首页增加通知 + this.bindHook('app_router_afterchange', function ({ to }) { + + setTimeout(function () { + if (to && to.path === '/login') { + let el = document.querySelector('.remember-user-name'); + let bindPhone = document.createElement('div') + bindPhone.setAttribute('id', 'reg_link_a') + el.appendChild(bindPhone) + return new Vue({ + render: (h) => { + return h( + 'a', { + style: { + color: '#2d8cf0', + position: 'absolute', + right: '30px' + }, + on: { + click: function () { + location.href = "#/register" + } + } + }, i18n.t('message.common.login.toRegister') + ) + } + }).$mount('#reg_link_a') + } + + if (to && to.path === '/workspaceHome') { + let headDiv = document.querySelector('.page-bgc-header'); + let bindPhone = document.createElement('div') + bindPhone.setAttribute('id', 'workspace_home_tip') + headDiv.appendChild(bindPhone) + const show = storage.get('workspace_top_notice') + return new Vue({ + render: (h) => { + return h( + 'Alert', { + style: { + display: show ? 'none' : 'block' + }, + props: { + type: 'warning', + 'show-icon': true, + closable: true, + }, + on: { + 'on-close': function () { + storage.set('workspace_top_notice', true) + } + } + }, [ + i18n.t('message.workspace.topNotice.content') + ] + ) + } + }).$mount('#workspace_home_tip') + } + }, 1000) + }) + +} diff --git a/web/packages/exts/sandbox/register/index.vue b/web/packages/exts/sandbox/register/index.vue new file mode 100644 index 0000000000..aca9212dde --- /dev/null +++ b/web/packages/exts/sandbox/register/index.vue @@ -0,0 +1,775 @@ + + + + diff --git a/web/packages/exts/sandbox/register/termsOfPrivacy.vue b/web/packages/exts/sandbox/register/termsOfPrivacy.vue new file mode 100644 index 0000000000..7b5c755136 --- /dev/null +++ b/web/packages/exts/sandbox/register/termsOfPrivacy.vue @@ -0,0 +1,52 @@ + + + diff --git a/web/packages/exts/sandbox/register/termsOfService.vue b/web/packages/exts/sandbox/register/termsOfService.vue new file mode 100644 index 0000000000..7d5c3a3791 --- /dev/null +++ b/web/packages/exts/sandbox/register/termsOfService.vue @@ -0,0 +1,60 @@ + + + diff --git a/web/packages/scriptis/assets/images/flink.svg b/web/packages/scriptis/assets/images/flink.svg index ebfbd55ef3..cc29a3e7d7 100644 --- a/web/packages/scriptis/assets/images/flink.svg +++ b/web/packages/scriptis/assets/images/flink.svg @@ -1,8564 +1 @@ -<<<<<<< HEAD -======= - - - - ->>>>>>> b5c29e41490d1200fbd6b32f407ea018def62aec diff --git a/web/packages/scriptis/assets/images/trino.svg b/web/packages/scriptis/assets/images/trino.svg new file mode 100644 index 0000000000..1c83e1efdc --- /dev/null +++ b/web/packages/scriptis/assets/images/trino.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/web/packages/scriptis/assets/styles/home.scss b/web/packages/scriptis/assets/styles/home.scss index dd04757638..689cb9118e 100644 --- a/web/packages/scriptis/assets/styles/home.scss +++ b/web/packages/scriptis/assets/styles/home.scss @@ -111,6 +111,13 @@ background-repeat: no-repeat; background-position: center; } + .fi-trino-sql { + background-image: url('../images/trino.svg'); + width: 12px; + background-size: contain; + background-repeat: no-repeat; + background-position: center; + } // @media screen and(max-height: 790px) { // .nav-list { // width: 30px; diff --git a/web/packages/scriptis/components/fileTree/index.vue b/web/packages/scriptis/components/fileTree/index.vue index 7e029a6ac4..332e5d9f2c 100644 --- a/web/packages/scriptis/components/fileTree/index.vue +++ b/web/packages/scriptis/components/fileTree/index.vue @@ -3,7 +3,6 @@ :class="{'is-empty': isEmpty}" class="we-file-tree"> -
暂无数据,请点击刷新重试!
+
{{ $t('message.scripts.nodataclick') }}{{ $t('message.scripts.retry') }}{{ $t('message.common.refresh') }}{{ $t('message.scripts.retry') }}
{}, @@ -146,11 +141,11 @@ export default { const reg = /^[\w\u4e00-\u9fa5]{1,200}$/; if (!reg.test(label)) { if (len === 0) { - msg = '文件夹名称不得为空!'; + msg = this.$t('message.scripts.folderempty'); } else if (len >= 200) { - msg = '文件夹名称超过200字符!'; + msg = this.$t('message.scripts.foldernameLen'); } else { - msg = '文件夹名称只支持大写、小写字母、数字、下划线和中文!'; + msg = this.$t('message.scripts.foldernamestyle'); } } } else { @@ -169,18 +164,18 @@ export default { const pointNum = label.split('.').length - 1; const reg = /^[\w\u4e00-\u9fa5]{1,200}\.[A-Za-z]+$/; const unReg = /[^\w\u4e00-\u9fa5]+/g; - const extReg = /\.(hql|sql)$/i; + const extAllowChange = ['.sql','.hql']; if (!reg.test(label)) { if (len === 0) { - msg = '脚本名称不得为空!'; + msg = this.$t('message.scripts.scriptNameEmpty'); } else if (!prefix.length) { - msg = '脚本前缀名不得为空!'; + msg = this.$t('message.scripts.scriptPrefix'); } else if (!isHaveSuffix) { - msg = '后缀名不得为空!'; + msg = this.$t('message.scripts.scriptSuffix'); } else if (pointNum > 1) { - msg = '存在非法字符"."!'; + msg = this.$t('message.scripts.illegal', {str: '"."'}); } else if (prefix.length >= 200) { - msg = '脚本前缀名称超过200字符!'; + msg = this.$t('message.scripts.Scriptprefix200 '); } else { let unRegString = ''; try { @@ -203,10 +198,10 @@ export default { 0, unRegString.length - 2 ); - msg = `存在非法字符"${unRegString}"`; + msg = this.$t('message.scripts.illegal', {str: `"${unRegString}"`}); } - } else if (newSuffix !== oldSuffix && !extReg.test(label)) { - msg = `该脚本类型(后缀名)不得修改!`; + } else if (newSuffix !== oldSuffix && (extAllowChange.indexOf(newSuffix) < 0 || extAllowChange.indexOf(oldSuffix) < 0)) { + msg = this.$t('message.scripts.cannotmodify'); } } return msg; diff --git a/web/packages/scriptis/components/importToHive/index.vue b/web/packages/scriptis/components/importToHive/index.vue index c3f4aee2bd..f0b9ebe03f 100644 --- a/web/packages/scriptis/components/importToHive/index.vue +++ b/web/packages/scriptis/components/importToHive/index.vue @@ -8,7 +8,7 @@ {{$t('message.scripts.importToHive.DRWJ')}} - {{isFullScreen?'取消全屏':'全屏展示'}} + {{isFullScreen?$t('message.scripts.cancelfullscreen'):$t('message.scripts.fullscreen')}}
@@ -154,8 +154,7 @@ - + :placeholder="$t('message.scripts.importToHive.SRSJBM')" />
@@ -165,12 +164,12 @@ prop="partitionValue" >
- {{ secondStep.partition }} + {{ secondStep.partition }} = + style="width: 50%" />
@@ -199,7 +198,7 @@ prop="partition" > + v-model="secondStep.partition" />
= + style="width: calc(100% - 30px);" /> @@ -586,7 +585,7 @@ export default { } }, async handleTbInput(val, cb) { - const activeTB = find(this.activeDB.children, (o) => o.value === val); + const activeTB = find(this.activeDB.children, (o) => o.value.toLowerCase() === val.toLowerCase()); if (!activeTB) { this.resetPartition(); return cb(); @@ -723,7 +722,7 @@ export default { let poptipValidate = { fieldName: { pattern: /^[a-zA-Z0-9_]+$/, - message: '字段名只支持大小写字母、数字和下划线', + message: this.$t('message.scripts.fieldnamestyle'), }, length: { pattern: /^[0-9]+$/, @@ -808,7 +807,7 @@ export default { if (this.secondStep.fields.length > 1) { this.secondStep.fields.splice(index, 1); } else if (this.secondStep.fields.length === 1) { - this.$Message.warning('无法删除所有的字段!'); + this.$Message.warning(this.$t('message.scripts.unableDel')); } }, // 在滚动字段列表时,需要对列表名称是否符合规范进行验证 diff --git a/web/packages/scriptis/components/newDialog/index.vue b/web/packages/scriptis/components/newDialog/index.vue index 04ce5d3075..ddfc93831f 100644 --- a/web/packages/scriptis/components/newDialog/index.vue +++ b/web/packages/scriptis/components/newDialog/index.vue @@ -5,7 +5,7 @@ {{ modalTitle }} - {{isFullScreen?'取消全屏':'全屏展示'}} + {{isFullScreen?this.$t('message.scripts.cancelfullscreen'):this.$t('message.scripts.fullscreen')}}
@@ -36,7 +36,7 @@ :placeholder="$t('message.scripts.newDialog.placeholder')" style="width: 360px;"> {{ ext }} - 脚本使用说明 + {{ $t('message.scripts.scriptUsage') }} import directoryDialog from '@dataspherestudio/shared/components/directoryDialog/index.vue'; import storage from '@dataspherestudio/shared/common/helper/storage'; +import i18n from '@dataspherestudio/shared/common/i18n'; export default { name: 'NewDialog', components: { @@ -145,7 +146,7 @@ export default { }, type: { type: String, - default: '文件夹', + default: i18n.t('message.scripts.folder'), }, node: { type: Object, @@ -206,7 +207,22 @@ export default { scriptHelpLink() { const baseinfo = storage.get("baseInfo", "local") || {} const item = this.scriptType.find((o) => o.scriptType === this.newForm.scriptType); - return item ? baseinfo[item.label+'UsageGuide'] || '' : '' + const scriptGuideMap = { + ".sql": "SqlUsageGuide", + ".hql": "HiveUsageGuide", + ".psql": "PrestoSqlUsageGuide", + ".tsql": "TrinoSqlUsageGuide", + ".fql": "FlinkSqlUsageGuide", + ".out": "StorageUsageGuide", + ".scala": "ScalaUsageGuide", + ".jdbc": "JdbcUsageGuide", + ".python": "PythonUsageGuide", + ".py": "PythonSparkUsageGuide", + ".r": "RUsageGuide", + ".sh": "ShellUsageGuide", + ".ngql": "NebulaUsageGuide", + } + return item ? baseinfo[scriptGuideMap[this.ext]] || '' : '' } }, methods: { diff --git a/web/packages/scriptis/i18n/common/en.json b/web/packages/scriptis/i18n/common/en.json index c9f6d4ef3e..cba86c8d3e 100644 --- a/web/packages/scriptis/i18n/common/en.json +++ b/web/packages/scriptis/i18n/common/en.json @@ -137,7 +137,8 @@ "partitionValue": "Partition value", "partitionAlias": "Partition alias", "fieldsNameRequire": "Column name cannot be empty", - "fieldsNameRule": "Only alphanumeric characters and underscore are allowed" + "fieldsNameRule": "Only alphanumeric characters and underscore are allowed", + "dupfields": "Invalid items found, Duplicate field:
{fields}" }, "tableDetails": { "BZBSX": "Table basic attributes", @@ -147,6 +148,7 @@ "TN": "Table namee", "BBM": "Table alias name", "FQM": "Partitioned table", + "compressTable":"Comprerss table", "CJYH": "Created By", "CJSI": "Created At", "ZHFWSJ": "Last Accessed At", @@ -357,6 +359,7 @@ "DH": "Comma(,)", "FH": "Semicolon(;)", "ZBF": "Tab(\\t)", + "SX": "Vertical bar(|)", "KG": "Space", "KZFC": "Empty string", "XZSJK": "Please choose database", @@ -500,6 +503,7 @@ "db": { "copyName": "Copy database name", "pasteName": "Copy database name & paste to script", + "dbinfo": "View database information", "listSort": "Sort by time", "liststringSort": "Sort by first character" }, @@ -588,7 +592,7 @@ "save": "Save", "saveAs": "Save As" }, - "tips": "What is Scriptis?\nScriptis is a one-stop interactive data analysis & exploration tool built by WeDataSphere at Webank. Using Linkis as its kernel, Scriptis provides functionalities including multiple computation engines(Spark, Hive, TiSpark...), Hive database management, resource management(such as Yarn resource and server resource), application management and user resource management(such as UDF and variables). \nScriptis?\n1. Select a directory in workspace, and create a sub directory; \n2. Right-click a directory => Create script; \n3. Select script type, for example: SQL, Pyspark, HQL, etc... \n4. Edit script, click & run, wait for result sets.", + "tips": "
Guidelines for Scriptis:
1. Set parameters: Set the queue and parameters on the management console
2. New Script: Right click to create the required script
3. Edit script: edit sql or command in the corresponding script file
4. Submit for execution: run the selected sql or command
5. Result query: query execution results and logs
Go to Quickly Use Scriptis for more detailed instructions
", "warning": { "noDBInfo": "Failed to request database/table information, word association in script may not be working well! Please refresh to retry!", "noUDF": "Failed to request UDF and Method information, word association in script may not be working well! Please refresh to retry!", @@ -621,6 +625,7 @@ }, "warning": { "running": "Script execution in progress, please hold on!", + "unchange": "No changes for save!", "invalidArgs": "Please check the correctness of the input of configuration parameters!", "emptyCode": "Cannot execute empty script!" } @@ -652,12 +657,14 @@ "title": "Code" }, "failedReason": { - "title": "Key information" + "title": "Key Information" }, "control": { "title": "Handle", "view": "View", - "download": "Download log" + "download": "Download log", + "noticeopen": "Enable Notice", + "noticeclose": "Clsoe Notice" } }, "success": { @@ -696,9 +703,7 @@ "progress": "Progress", "result": "Execution Result", "log": "Execution Log", - "history": "History", - "diagnosis": "Intelligent Diagnosis", - "resourceInfo": "Resource Info" + "history": "History" }, "notice": { "unsave": { @@ -739,7 +744,136 @@ "settingTitle": "Command line Arguments", "placeholder": "Please use space to separate multiple arguments" } - } + }, + "Username": "Username", + "downloadtime": "Download time", + "reset": "Reset", + "search": "Search", + "Date": "Date", + "user": "User", + "downloadpath": "Download path", + "proxyuser": "Current proxy user:", + "notsaved": "The code being edited is not saved, please check", + "confirmcheck": "Click confirm to check, Click cancel without saving, Continue to jump", + "folder": "Folder", + "view": "View", + "table": "Table", + "Partitionsize": "Partition size:", + "append": "Append", + "dbdetail": "Database details", + "Settings": "Settings", + "proxytip": "Before entering, please select a proxy user or cancel", + "applyporxy": "Please apply for a proxy user first", + "confirm": "Confirm", + "propmpt": "Prompt", + "largedatatip": " When the number of tables is large, the browser may freeze when writing scripts. You can choose to turn off the code completion function in the right-click menu of the editor.", + "hasopen": "There is an open script that does not support modification", + "upstreamTable": "Upstream staging table", + "dbinfs": "Database information", + "tableifs": "Table information", + "transfer": "Table Transfer", + "transferTitle": "Table Transfer", + "transferForm": { + "title": "Title", + "owner": "Table owner", + "admin": "Administrator", + "desc": "Description", + "required": "Required", + "max":"Maximum length({len})", + "ownerplaceholder": "Owner's English name of the enterprise WeChat of transfer", + "adminplaceholder": "Department administrator’s English name of the enterprise WeChat" + }, + "loadingwait": "Loading, please wait", + "loading": "Loading", + "running": "Running prompt", + "silent": "Start to run silent script", + "resultpropt": "Execution result prompt", + "solution": "Check solution", + "reporterror": "Report error", + "reported": "Error reported", + "optlimit": "Please do not operate frequently", + "optsuccess": "Successfully open the notification. You will receive the notification after the task is completed.", + "cancelOptSuccess": "Successfully closed the notification, you will not receive the task end notification", + "Permanent": "Permanent", + "validday": "Valid for today only", + "validweek": "Valid for one week", + "validmonth": "Valid for one month", + "validhalf": "Valid for six month", + "datastore": "Operation Data Store", + "warehouse": "Data Warehouse Details", + "warehouseservice": "Data Warehouse Service", + "appservice": "Application Data Service", + "writemany": "Write Once Read Many", + "curd": "CURD", + "multoverride": "Multiple override", + "once": "Write Once Read Occasionally", + "Metrics": "Metrics", + "Dimensions": "Dimensions", + "Yes": "Yes", + "No": "No", + "Field": "Field", + "Name": "Name", + "Type": "Type", + "busst": "Business standard", + "calcst": "Calculation standard", + "Formula": "Formula", + "Cancel": "Cancel", + "plsinputname": "Please input name", + "lengthlimit": "Out of length limit", + "max": "Maximum length", + "plsselect": "Please select", + "plsinputbusiness": "Please input business standard", + "inputfield": "Please input the field", + "dbname": "Database name", + "dbsize": "Database size", + "dbquota": "Database quota", + "tablenum": "Number of tables", + "dbinfotips": "Note: The database name has a one-day delay, the database size has an hour level delay, and the table quantity displayed in real time", + "dbdesc": "Description", + "plstablename": "Please input the table name", + "defaultsort": "Default sort order", + "ordersize": "Order by table size", + "ordercreatetime": "Order by create time", + "orderaccesstime": "Order by access time", + "owntable": "Tables I have access to", + "tablecreateby": "Tables I create", + "Search": "Search", + "copytbanme": "Copy table name", + "searchnow": "Real-time Search", + "realsearchTip": "Real time return of the existing tables in the current database. If the table has too much data, \nit may take some time to wait for the query results to return", + "batchdel": "Batch deletion", + "Serial": "Serial number", + "checkdelconfirm": "Please check again to confirm", + "selectfirst": "Please select!", + "tablename": "Table name", + "tbalias": "Table alias", + "createtime": "Create time", + "tablesize": "Table size", + "tbowner": "Table owner", + "ispartition": "Whether it is a partition table", + "iscompress": "Compressed", + "compressformat": "Compressed format", + "lastaccess": "Last access time", + "lastupdate": "Last update time", + "createtime:": "Create time:", + "nodata": "No data", + "retry": "Please retry!", + "nodataclick": "No data, please click", + "errorclick": "Service exception, please click", + "folderempty": "Folder name cannot be empty!", + "foldernameLen": "Folder name exceeds 200 characters!", + "foldernamestyle": "Folder names only accept uppercase,lowercase,digit,underscore and Chinese", + "scriptNameEmpty": "Script name cannot be empty!", + "scriptPrefix": "Script prefix name cannot be empty!", + "scriptSuffix": "Script suffix name cannot be empty!", + "illegal": "There are illegal characters {str}!", + "Scriptprefix200 ": "Script prefix name exceeds 200 characters!", + "cannotmodify": "Only hql and sql files mutual modification!", + "cancelfullscreen": "Cancel full screen", + "fullscreen": "Full screen", + "fieldnamestyle": "Filed names only accept uppercase, lowercase, digit and underscore", + "unableDel": "Unable to delete all fields!", + "scriptUsage": "Script usage instructions" } } } diff --git a/web/packages/scriptis/i18n/common/zh.json b/web/packages/scriptis/i18n/common/zh.json index 682c1b6517..8cb9846932 100644 --- a/web/packages/scriptis/i18n/common/zh.json +++ b/web/packages/scriptis/i18n/common/zh.json @@ -137,7 +137,8 @@ "partitionValue": "分区值", "partitionAlias": "分区别名", "fieldsNameRequire": "字段名称不能为空", - "fieldsNameRule": "仅支持字母数字和下划线" + "fieldsNameRule": "支持字母数字和下划线且不为纯数字", + "dupfields": "验证项未通过,字段重复:
{fields}" }, "tableDetails": { "BZBSX": "表基本属性", @@ -147,6 +148,7 @@ "TN": "表名", "BBM": "表别名", "FQM": "分区表", + "compressTable": "压缩表", "CJYH": "创建用户", "CJSI": "创建时间", "ZHFWSJ": "最后访问时间", @@ -357,6 +359,7 @@ "DH": "逗号(,)", "FH": "分号(;)", "ZBF": "制表符(\\t)", + "SX": "竖线(|)", "KG": "空格", "KZFC": "空字符串", "XZSJK": "请选择数据库", @@ -499,6 +502,7 @@ "db": { "copyName": "复制库名", "pasteName": "复制库名并粘贴至脚本", + "dbinfo": "查看库信息", "listSort": "表按时间排序", "liststringSort": "表按首字符排序" }, @@ -587,7 +591,7 @@ "save": "保存", "saveAs": "另存为" }, - "tips": "什么是Scriptis?\nScriptis是微众银行微数域(WeDataSphere)打造的一站式交互式数据探索分析工具,以任意桥(Linkis)做为内核,提供多种计算存储引擎(如Spark、Hive、TiSpark等)、Hive数据库管理功能、资源(如Yarn资源、服务器资源)管理、应用管理和各种用户资源(如UDF、变量等)管理的能力。\nScriptis?\n1. 选中工作空间的目录,创建文件夹;\n2. 右键某个文件夹 =>新建脚本;\n3. 选择脚本类型,如:SQL、Pyspark、HQL等;\n4. 书写脚本,点击执行,生成结果集。", + "tips": "
Scriptis使用指引:
1.设置参数:在管理台设置队列及参数
2.新建脚本:右键新建所需脚本
3.编辑脚本:在对应的脚本文件中编辑Sql或命令
4.提交执行:运行选中的sql或命令
5.结果查询:查询执行结果及日志
前往快速使用Scriptis了解更详细的使用指引
", "warning": { "noDBInfo": "未获取到数据库表信息,脚本联想词功能可能存在异常!可刷新重试!", "noUDF": "未获取到UDF和方法函数信息,脚本联想词功能可能存在异常!可刷新重试!", @@ -620,6 +624,7 @@ }, "warning": { "running": "脚本运行中,请稍候!", + "unchange": "未做修改,无需保存!", "invalidArgs": "请检查配置参数输入是否正确!", "emptyCode": "无法执行空代码!" } @@ -656,7 +661,9 @@ "control": { "title": "操作", "view": "查看", - "download": "日志下载" + "download": "日志下载", + "noticeopen": "开启通知", + "noticeclose": "关闭通知" } }, "success": { @@ -695,9 +702,7 @@ "progress": "进度", "result": "运行结果", "log": "运行日志", - "history": "历史", - "diagnosis": "智能诊断", - "resourceInfo": "资源信息" + "history": "历史" }, "notice": { "unsave": { @@ -738,7 +743,136 @@ "settingTitle": "命令行参数", "placeholder": "请使用空格分开多个参数" } - } + }, + "Username": "用户名", + "downloadtime": "下载时间", + "reset": "重置", + "search": "搜索", + "Date": "日期", + "user": "用户", + "downloadpath": "下载路径", + "proxyuser": "当前代理用户:", + "notsaved": "正在编辑的代码未保存,请先检查", + "confirmcheck": "点击确定去检查,点击取消不保存,继续跳转", + "folder": "文件夹", + "view": "视图", + "table": "表", + "Partitionsize": "分区大小:", + "append": "追加", + "dbdetail": "库详情", + "Settings": "设置", + "proxytip": "请选择代理用户进入或者取消", + "applyporxy": "请先申请实名用户允许的代理用户", + "confirm": "确定", + "propmpt": "提示", + "largedatatip": "库表数据量较大可能导致编写脚本时卡顿,可在编辑器右键菜单选择关闭库表联想解决", + "hasopen": "存在已打开脚本不支持修改", + "upstreamTable": "上游暂存表", + "dbinfs": "库基本信息", + "tableifs": "库表信息", + "transfer": "批量转移", + "transferTitle": "表属主转移", + "transferForm": { + "title": "申请单标题", + "owner": "表转移后属主", + "admin": "数据治理管理员", + "desc": "描述", + "required": "该项必填", + "max":"最大长度({len})", + "ownerplaceholder": "请准确填写表转移后属主的企业微信英文名", + "adminplaceholder": "请准确填写部门数据治理管理员的企业微信英文名" + }, + "loadingwait": "数据加载中,请耐心等待", + "loading": "数据正在加载中", + "running": "运行提示", + "silent": "开始执行静默脚本", + "resultpropt": "执行结果提示", + "solution": "查看解决方案", + "reporterror": "上报错误", + "reported": "错误已上报", + "optlimit": "请勿频繁操作", + "optsuccess": "成功开启通知,任务结束后您将收到通知。", + "cancelOptSuccess": "成功关闭通知,您将不会收到任务结束通知", + "Permanent": "永久", + "validday": "当天有效", + "validweek": "一周有效", + "validmonth": "一月有效", + "validhalf": "半年有效", + "datastore": "原始数据层", + "warehouse": "明细数据层", + "warehouseservice": "汇总数据层", + "appservice": "应用数据层", + "writemany": "一次写多次读", + "curd": "增删改查", + "multoverride": "多次覆盖写", + "once": "一次写偶尔读", + "Metrics": "指标", + "Dimensions": "维度", + "Yes": "是", + "No": "否", + "Field": "字段", + "Name": "名称", + "Type": "类型", + "busst": "业务口径", + "calcst": "计算口径", + "Formula": "计算公式", + "Cancel": "取消", + "plsinputname": "请输入名称", + "lengthlimit": "输入超长", + "max": "最大长度", + "plsselect": "请选择", + "plsinputbusiness": "请输入业务口径", + "inputfield": "请输入字段", + "dbname": "库名", + "dbsize": "库大小", + "dbquota": "库配额", + "tablenum": "表数量", + "dbinfotips": "说明:库名展示的数据存在一天的延时,库大小展示的数据存在小时级别的延时,表数量数据为实时展示", + "dbdesc": "备注", + "plstablename": "请输入表名", + "defaultsort": "默认排序", + "ordersize": "表大小排序", + "ordercreatetime": "创建时间排序", + "orderaccesstime": "访问时间排序", + "owntable": "我有权限的表", + "tablecreateby": "我创建的表", + "Search": "查询", + "copytbanme": "复制表名", + "searchnow": "实时查询", + "realsearchTip": "实时返回当前库下存在的表,表数据量过多时,\n可能需要一些时间等待查询结果返回", + "batchdel": "批量删除", + "Serial": "序号", + "checkdelconfirm": "请再次勾选进行确认", + "selectfirst": "请先选择", + "tablename": "表名", + "tbalias": "表别名", + "createtime": "创建时间", + "tablesize": "表大小", + "tbowner": "表属主", + "ispartition": "是否分区", + "iscompress": "是否压缩", + "compressformat": "压缩格式", + "lastaccess": "最近访问时间", + "lastupdate": "最近修改时间", + "createtime:": "创建时间:", + "nodata": "暂无数据", + "retry": "重试!", + "nodataclick": "暂无数据,请点击", + "errorclick": "后台异常,请点击", + "folderempty": "文件夹名称不得为空!", + "foldernameLen": "文件夹名称超过200字符!", + "foldernamestyle": "文件夹名称只支持大写、小写字母、数字、下划线和中文!", + "scriptNameEmpty": "脚本名称不得为空!", + "scriptPrefix": "脚本前缀名不得为空!", + "scriptSuffix": "后缀名不得为空!", + "illegal": "存在非法字符{str}!", + "Scriptprefix200 ": "脚本前缀名称超过200字符!", + "cannotmodify": "仅支持hql,sql互相修改", + "cancelfullscreen": "取消全屏", + "fullscreen": "全屏展示", + "fieldnamestyle": "字段名只支持大小写字母、数字和下划线", + "unableDel": "无法删除所有的字段!", + "scriptUsage": "脚本使用说明" } } } diff --git a/web/packages/scriptis/module/fnSidebar/fnSidebar.vue b/web/packages/scriptis/module/fnSidebar/fnSidebar.vue index 260bb236bb..60181971a2 100644 --- a/web/packages/scriptis/module/fnSidebar/fnSidebar.vue +++ b/web/packages/scriptis/module/fnSidebar/fnSidebar.vue @@ -156,7 +156,7 @@ export default { treeLoading: false, filterNode: () => {}, newDialog: { - type: '文件夹', + type: this.$t('message.scripts.folder'), isNew: true, node: {}, }, @@ -642,10 +642,13 @@ export default { return new Promise((resolve) => { if (this.treeLoading) return this.$Message.warning(this.$t('message.scripts.constants.warning.data')); let id = this.currentNode.data && this.currentNode.data.id; - let parent = this.currentNode.parent + let parent // 编辑或删除时要去刷新上一级目录 if (this.currentNode.isLeaf || type === 'edit' || type === 'delete') { + parent = this.currentNode.parent id = this.currentNode.parent.data.id; + } else { + parent = this.currentNode } if (type === 'new') { parent = this.currentNode @@ -655,7 +658,7 @@ export default { this.treeLoading = true; this.compLoading = true; api.fetch('/udf/list', { - type: parent.type || parent.data.type, + type: parent.type || parent.data.type || this.currentNode.data.type, treeId: id, category: this.FNTYPE, }).then((rst) => { diff --git a/web/packages/scriptis/module/footer/index.scss b/web/packages/scriptis/module/footer/index.scss index 92cfa8b4eb..5481fbfc79 100644 --- a/web/packages/scriptis/module/footer/index.scss +++ b/web/packages/scriptis/module/footer/index.scss @@ -42,6 +42,10 @@ visibility: visible; } } + .min.tool-btns { + height: 30px; + margin-bottom: 20px; + } .min_arrow, .show_arrow { font-size: 16px; padding: 5px; diff --git a/web/packages/scriptis/module/footer/index.vue b/web/packages/scriptis/module/footer/index.vue index 26208981a2..7c2f0401b7 100644 --- a/web/packages/scriptis/module/footer/index.vue +++ b/web/packages/scriptis/module/footer/index.vue @@ -1,6 +1,6 @@
- 因为您的结果集较大,为了更好的体验,点击查看结果集 + {{ $t('message.common.resulttip') }}{{ $t('message.common.viewSet') }}
@@ -125,7 +125,7 @@ show-sizer @on-change="change" @on-page-size-change="changeSize" /> - 前端只展示5000条数据 + {{ $t('message.common.only500') }} @@ -165,9 +165,6 @@ export default { getResultUrl: { type: String, defalut: `filesystem` - }, - comData: { - type: Object } }, mixins: [mixin], @@ -354,9 +351,11 @@ export default { } // 如果为整数型大小 if (['int', 'float', 'double', 'long', 'short', 'bigint', 'decimal'].includes(valueType.toLowerCase())) { - return strA - strB; } + if (['timestamp'].includes(valueType.toLowerCase()) && typeof strA === 'string') { + return strA.localeCompare(strB) + } const charAry = strA.split(''); for (const i in charAry) { if ((this.charCompare(strA[i], strB[i]) !== 0)) { diff --git a/web/packages/shared/components/consoleComponent/resultsExport.vue b/web/packages/shared/components/consoleComponent/resultsExport.vue index a744ad4fad..57c7beddc7 100644 --- a/web/packages/shared/components/consoleComponent/resultsExport.vue +++ b/web/packages/shared/components/consoleComponent/resultsExport.vue @@ -8,7 +8,7 @@ {{ $t('message.common.resultsExport.header') }} - {{isFullScreen?'取消全屏':'全屏展示'}} + {{isFullScreen?this.$t('message.common.cancelFullScreen'):this.$t('message.common.fullScreen')}}
diff --git a/web/packages/shared/components/consoleComponent/steps.vue b/web/packages/shared/components/consoleComponent/steps.vue index d83b895a56..9ee3a3e062 100644 --- a/web/packages/shared/components/consoleComponent/steps.vue +++ b/web/packages/shared/components/consoleComponent/steps.vue @@ -13,7 +13,20 @@ class="we-steps-child" v-for="(child, index) in renderList" :key="index"> - + + + {{ child.label }}({{percent}}%) + + {{ p }}

{{ child.label }} + >{{ child.label }} +
+ {{$t('message.common.process')}} + {{ costTime }} +
@@ -67,6 +85,14 @@ export default { type: Array, default: () => [], }, + percent: { + type: Number, + default: 0 + }, + costTime: { + type: String, + default: '0 second' + } }, data() { return { @@ -236,7 +262,7 @@ export default { .we-steps { .we-steps-row-first { width: 100%; - padding: 20px 0 10px; + padding: 15px 0; display: flex; align-items: center; .we-steps-title { @@ -306,6 +332,9 @@ export default { } } } + .progress-costtime { + color: $success-color; + } } diff --git a/web/packages/shared/components/consoleComponent/toolbar.vue b/web/packages/shared/components/consoleComponent/toolbar.vue index b2ea1b4579..13064e7ae7 100644 --- a/web/packages/shared/components/consoleComponent/toolbar.vue +++ b/web/packages/shared/components/consoleComponent/toolbar.vue @@ -17,23 +17,22 @@ @event-from-ext="eventFromExt" /> -
  • +
  • + placement="right">
    {{ $t('message.common.download') }}
    - + {{ $t('message.common.toolbar.format') }} - +
  • CSV @@ -48,10 +47,10 @@
    - + {{ $t('message.common.toolbar.coding') }} - +
    UTF-8 @@ -65,10 +64,10 @@
    - + {{$t('message.common.toolbar.replace')}} - +
    NULL @@ -81,18 +80,37 @@ +
    + + {{$t('message.common.toolbar.selectsplit')}} + + + + + {{ item.label }} + + + +
    - + {{$t('message.common.toolbar.downloadMode')}} - + {{$t('message.common.toolbar.all')}}
    - + {{$t('message.common.toolbar.autoformat')}} - +
    @@ -185,7 +203,7 @@ export default { type: String, defalut: `filesystem` }, - comData: { + work: { type: Object }, resultType: { @@ -194,16 +212,8 @@ export default { } }, data() { - // let appItem = 'linkis' - // if (this.$route.name === 'Workflow') { - // appItem = 'workflow' - // } else if(this.$route.name === 'ServicesExecute') { - // appItem = 'apiService' - // } - let rsDownload = true // todo return { activeTool: 'table', - rsDownload, popup: { download: false, export: false, @@ -214,12 +224,20 @@ export default { format: '1', coding: '1', nullValue: '1', + splitChar: '1', }, isIconLabelShow: true, iconSize: 14, allDownload: false, // 是否下载全部结果集 autoFormat: false, - extComponents + extComponents, + splitList: [ + { label: this.$t('message.scripts.hiveTableExport.DH'), value: '1' }, + { label: this.$t('message.scripts.hiveTableExport.FH'), value: '2' }, + { label: this.$t('message.scripts.hiveTableExport.ZBF'), value: '3' }, + { label: this.$t('message.scripts.hiveTableExport.KG'), value: '4' }, + { label: this.$t('message.scripts.hiveTableExport.SX'), value: '5' }, + ] }; }, computed: { @@ -230,7 +248,7 @@ export default { let isScriptis = this.$route.name === 'Home' || (this.$route.name === 'results' && this.$route.query.from === 'Home') return { export: this.baseinfo.exportResEnable !== false && isScriptis && this.activeTool === 'table' && this.resultType === '2', - download: this.activeTool === 'table' && this.rsDownload, + download: this.activeTool === 'table' && this.baseinfo.downloadResEnable !== false, } } }, @@ -245,6 +263,7 @@ export default { format: '1', coding: '1', nullValue: '1', + splitChar: '1' } } if (type === 'export') { @@ -266,8 +285,8 @@ export default { }, downloadConfirm() { this.$Modal.confirm({ - title: '提示', - content: '当前进行数据下载,可能涉及敏感数据,请保障数据的安全', + title: this.$t('message.common.Prompt'), + content: this.$t('message.common.safetips'), onOk: async () => { const splitor = this.download.format === '1' ? 'csv' : 'xlsx'; const charset = this.download.coding === '1' ? 'utf-8' : 'gbk'; @@ -290,16 +309,17 @@ export default { let querys = 'path=' + temPath + '&charset=' + charset + '&outputFileType=' + splitor + '&nullValue=' + nullValue + '&outputFileName=' + filename + '&autoFormat=' + this.autoFormat; // 如果是api执行页获取结果集,需要带上taskId if(this.getResultUrl !== 'filesystem') { - querys += `&taskId=${this.comData.taskID}` + querys += `&taskId=${this.work.taskID}` + } + const splitChar = [',',';','\t',' ','|'][this.download.splitChar - 1] || ',' + if(this.download.format == 1) { + querys += `&csvSeparator=${encodeURIComponent(splitChar)}` } let url = `${window.location.protocol}//${window.location.host}/api/rest_j/v1/${apiPath}?` + querys - - window.console.log(url, 'DOWNLOD-URL') - // 下载之前条用心跳接口确认是否登录 await api.fetch('/user/heartbeat', 'get'); // 下载记录日志 - api.fetch('/dss/framework/admin/audit/script/download/save', { + api.fetch('/dss/scriptis/audit/download/save', { creator: this.baseinfo.username, tenant: this.baseinfo.proxyUserName, path: temPath, @@ -357,17 +377,6 @@ export default { left: 40px; margin-left: -$toolbarWidth; @include bg-color($light-base-color, $dark-menu-base-color); - .we-poptip { - padding: 12px; - line-height: 28px; - .confirm { - margin-top: 10px; - } - .title { - margin: 10px 0; - text-align: center; - } - } .we-toolbar { width: 100%; height: 100%; @@ -391,4 +400,9 @@ export default { } } + diff --git a/web/packages/shared/components/deleteDialog/index.vue b/web/packages/shared/components/deleteDialog/index.vue index bf024807ef..fbd7c05289 100644 --- a/web/packages/shared/components/deleteDialog/index.vue +++ b/web/packages/shared/components/deleteDialog/index.vue @@ -7,13 +7,13 @@ slot="header" class="delete-modal-header"> - {{ label }}{{$t('message.common.deleteDialog.waring')}} + {{ label }} {{$t('message.common.deleteDialog.waring')}}

    - 您真的确定{{$t('message.common.deleteDialog.action', {label:label})}} + {{$t('message.common.deleteDialog.confirm')}} {{$t('message.common.deleteDialog.action', {label:label})}} {{ name }} - {{ type }}嘛? + {{ type }}?

    diff --git a/web/packages/shared/components/directoryDialog/show.vue b/web/packages/shared/components/directoryDialog/show.vue index bd044aff43..1c98fda4e5 100644 --- a/web/packages/shared/components/directoryDialog/show.vue +++ b/web/packages/shared/components/directoryDialog/show.vue @@ -8,7 +8,7 @@ {{ title }} - {{isFullScreen?'取消全屏':'全屏展示'}} + {{isFullScreen?this.$t('message.common.cancelFullScreen'):this.$t('message.common.fullScreen')}}
    diff --git a/web/packages/shared/components/dynamicForm/index.vue b/web/packages/shared/components/dynamicForm/index.vue index e2ed30c120..f14a0c04f1 100644 --- a/web/packages/shared/components/dynamicForm/index.vue +++ b/web/packages/shared/components/dynamicForm/index.vue @@ -2,7 +2,7 @@
    @@ -45,7 +45,7 @@ - +
    + diff --git a/web/packages/shared/components/panel/panel.vue b/web/packages/shared/components/panel/panel.vue index 99b071c57d..bdababb26c 100644 --- a/web/packages/shared/components/panel/panel.vue +++ b/web/packages/shared/components/panel/panel.vue @@ -266,7 +266,6 @@ export default { this.currentX = null; this.currentY = null; this.currentChange = null; - this.$emit('on-resized'); }); this.requestFrameIds.push(AnimationFrameId); } diff --git a/web/packages/shared/components/projectForm/index.vue b/web/packages/shared/components/projectForm/index.vue index e7aa460888..1a167a75c7 100644 --- a/web/packages/shared/components/projectForm/index.vue +++ b/web/packages/shared/components/projectForm/index.vue @@ -3,8 +3,8 @@ v-model="ProjectShow" :title=" actionType === 'add' - ? $t('message.workflow.projectDetail.createProject') - : $t('message.workflow.projectDetail.editorProject') + ? $t('message.common.projectDetail.createProject') + : $t('message.common.projectDetail.editorProject') " :closable="false" > @@ -14,10 +14,10 @@ ref="projectForm" :model="projectDataCurrent" :rules="formValid" - v-if="ProjectShow" + class="project_form" > + + + -
    +
    + + diff --git a/web/packages/sparketl/components/InputStringArray/index.vue b/web/packages/sparketl/components/InputStringArray/index.vue new file mode 100644 index 0000000000..85b78d86f2 --- /dev/null +++ b/web/packages/sparketl/components/InputStringArray/index.vue @@ -0,0 +1,36 @@ + + + + + + + diff --git a/web/packages/sparketl/components/InputStringMap/index.vue b/web/packages/sparketl/components/InputStringMap/index.vue new file mode 100644 index 0000000000..1f60bb0914 --- /dev/null +++ b/web/packages/sparketl/components/InputStringMap/index.vue @@ -0,0 +1,39 @@ + + + + + diff --git a/web/packages/sparketl/components/Sink/FileSink.vue b/web/packages/sparketl/components/Sink/FileSink.vue new file mode 100644 index 0000000000..890843c754 --- /dev/null +++ b/web/packages/sparketl/components/Sink/FileSink.vue @@ -0,0 +1,130 @@ + + + + diff --git a/web/packages/sparketl/components/Sink/HiveSink.vue b/web/packages/sparketl/components/Sink/HiveSink.vue new file mode 100644 index 0000000000..b5878d0d23 --- /dev/null +++ b/web/packages/sparketl/components/Sink/HiveSink.vue @@ -0,0 +1,149 @@ + + + + diff --git a/web/packages/sparketl/components/Sink/JdbcSink.vue b/web/packages/sparketl/components/Sink/JdbcSink.vue new file mode 100644 index 0000000000..807fa663e9 --- /dev/null +++ b/web/packages/sparketl/components/Sink/JdbcSink.vue @@ -0,0 +1,187 @@ + + + + diff --git a/web/packages/sparketl/components/Sink/ManagedJdbcSink.vue b/web/packages/sparketl/components/Sink/ManagedJdbcSink.vue new file mode 100644 index 0000000000..126e27c203 --- /dev/null +++ b/web/packages/sparketl/components/Sink/ManagedJdbcSink.vue @@ -0,0 +1,164 @@ + + + + diff --git a/web/packages/sparketl/components/Source/FileSource.vue b/web/packages/sparketl/components/Source/FileSource.vue new file mode 100644 index 0000000000..f3c9bc1587 --- /dev/null +++ b/web/packages/sparketl/components/Source/FileSource.vue @@ -0,0 +1,110 @@ + + + + diff --git a/web/packages/sparketl/components/Source/JdbcSource.vue b/web/packages/sparketl/components/Source/JdbcSource.vue new file mode 100644 index 0000000000..3dbfff2e11 --- /dev/null +++ b/web/packages/sparketl/components/Source/JdbcSource.vue @@ -0,0 +1,138 @@ + + + + diff --git a/web/packages/sparketl/components/Source/ManagedJdbcSource.vue b/web/packages/sparketl/components/Source/ManagedJdbcSource.vue new file mode 100644 index 0000000000..908937bec6 --- /dev/null +++ b/web/packages/sparketl/components/Source/ManagedJdbcSource.vue @@ -0,0 +1,104 @@ + + + + diff --git a/web/packages/sparketl/components/Transformation/SqlTransformation.vue b/web/packages/sparketl/components/Transformation/SqlTransformation.vue new file mode 100644 index 0000000000..c6ed56a518 --- /dev/null +++ b/web/packages/sparketl/components/Transformation/SqlTransformation.vue @@ -0,0 +1,88 @@ + + + + diff --git a/web/packages/sparketl/components/TypedPlugin.vue b/web/packages/sparketl/components/TypedPlugin.vue new file mode 100644 index 0000000000..4e692456e6 --- /dev/null +++ b/web/packages/sparketl/components/TypedPlugin.vue @@ -0,0 +1,90 @@ + + + + + diff --git a/web/packages/sparketl/package.json b/web/packages/sparketl/package.json new file mode 100644 index 0000000000..d8a0b45035 --- /dev/null +++ b/web/packages/sparketl/package.json @@ -0,0 +1,9 @@ +{ + "name": "@dataspherestudio/sparketl", + "version": "1.0.0", + "dependencies": { + "@codemirror/lang-json": "^6.0.1", + "codemirror": "^5.65.13", + "vue-codemirror": "^4.0.6" + } +} diff --git a/web/packages/sparketl/pages/home.vue b/web/packages/sparketl/pages/home.vue new file mode 100644 index 0000000000..35d0d71d85 --- /dev/null +++ b/web/packages/sparketl/pages/home.vue @@ -0,0 +1,221 @@ + + + + + + + diff --git a/web/packages/sparketl/router/index.js b/web/packages/sparketl/router/index.js new file mode 100644 index 0000000000..00d393cd51 --- /dev/null +++ b/web/packages/sparketl/router/index.js @@ -0,0 +1,10 @@ +const routes = [ + { + path: "sparketl", + name: "Sparketl", + component: () => import("../pages/home.vue"), + }, +]; + + +export default routes; diff --git a/web/packages/sparketl/utils/BaseConstant.js b/web/packages/sparketl/utils/BaseConstant.js new file mode 100644 index 0000000000..a68b12e35d --- /dev/null +++ b/web/packages/sparketl/utils/BaseConstant.js @@ -0,0 +1,482 @@ +const getBaseSource = () => ({ + variables: {}, + resultTable: "", + persist: false, + storageLevel: "MEMORY_AND_DISK", + options: {}, +}); + +const getBaseTransformation = () => ({ + variables: {}, + sourceTable: "", + resultTable: "", + persist: false, + storageLevel: "MEMORY_AND_DISK", +}); + +const getBaseSink = () => ({ + variables: {}, + sourceTable: "", + sourceQuery: "", + numPartitions: 0, + options: {}, +}); + +const PluginModels = { + source: { + jdbc: () => ({ + type: "source", + name: "jdbc", + config: { + ...getBaseSource(), + url: "", + driver: "", + user: "", + password: "", + query: "", + }, + }), + managed_jdbc: () => ({ + type: "source", + name: "managed_jdbc", + config: { + ...getBaseSource(), + datasource: "", + query: "", + }, + }), + file: () => ({ + type: "source", + name: "file", + config: { + ...getBaseSource(), + path: "", + serializer: "parquet", + columnNames: [], + }, + }), + }, + transformation: { + sql: () => ({ + name: "sql", + type: "transformation", + config: { + ...getBaseTransformation(), + sql: "", + }, + }), + }, + sink: { + hive: () => ({ + type: "sink", + name: "hive", + config: { + ...getBaseSink(), + targetDatabase: "", + targetTable: "", + saveMode: "overwrite", + strongCheck: false, + writeAsFile: false, + }, + }), + jdbc: () => ({ + type: "sink", + name: "jdbc", + config: { + ...getBaseSink(), + url: "", + driver: "", + user: "", + password: "", + targetDatabase: "", + targetTable: "", + saveMode: "overwrite", + preQueries: [], + }, + }), + managed_jdbc: () => ({ + type: "sink", + name: "managed_jdbc", + config: { + ...getBaseSink(), + targetDatasource: "", + targetDatabase: "", + targetTable: "", + saveMode: "overwrite", + preQueries: [], + }, + }), + file: () => ({ + type: "sink", + name: "file", + config: { + ...getBaseSink(), + path: "", + serializer: "parquet", + saveMode: "overwrite", + partitionBy: [], + }, + }), + }, +}; + +const FileSerializers = ["parquet", "orc", "csv", "text", "json"]; + +const Layout = { + cols: { + small: { xs: 24, sm: 24, md: 12, lg: 12, xl: 8 }, + }, + labelCols: { + small: { xs: 24, sm: 8, md: 10, lg: 10, xl: 9 }, + medium: {}, + large: { xs: 24, sm: 8, md: 5, lg: 5, xl: 3 }, + }, + wrapperColsWithLabel: { + small: { xs: 24, sm: 8, md: 10, lg: 10, xl: 9 }, + medium: {}, + large: { + xs: { span: 24, offset: 0 }, + sm: { span: 16, offset: 8 }, + md: { span: 19, offset: 5 }, + lg: { span: 19, offset: 5 }, + xl: { span: 21, offset: 3 }, + }, + }, +}; + +const StorageLevels = [ + { value: "NONE", label: "NONE" }, + { value: "DISK_ONLY", label: "DISK_ONLY" }, + { value: "DISK_ONLY_2", label: "DISK_ONLY_2" }, + { value: "MEMORY_ONLY", label: "MEMORY_ONLY" }, + { value: "MEMORY_ONLY_2", label: "MEMORY_ONLY_2" }, + { value: "MEMORY_ONLY_SER", label: "MEMORY_ONLY_SER" }, + { value: "MEMORY_ONLY_SER_2", label: "MEMORY_ONLY_SER_2" }, + { value: "MEMORY_AND_DISK", label: "MEMORY_AND_DISK" }, + { value: "MEMORY_AND_DISK_2", label: "MEMORY_AND_DISK_2" }, + { value: "MEMORY_AND_DISK_SER", label: "MEMORY_AND_DISK_SER" }, + { value: "MEMORY_AND_DISK_SER_2", label: "MEMORY_AND_DISK_SER_2" }, + { value: "OFF_HEAP", label: "OFF_HEAP" }, +]; + +const Databases = [ + { + databaseName: "ods", + }, + { + databaseName: "dw", + }, + { + databaseName: "dm", + }, +]; +const Datasource = ["hive:1000", "sr:2000"]; + +const SampleConfig = { + variables: {}, + sources: [ + { + name: "managed_jdbc", + config: { + variables: { + dt: "2022-01-01", + }, + resultTable: "source_table_01", + persist: true, + storageLevel: "MEMORY_AND_DISK", + options: { + connectionCollation: "utf8mb4_unicode_ci", + }, + datasource: "test_mysql", + query: "select * from test_mysql_db.test_tb where id > 0", + }, + }, + { + name: "jdbc", + config: { + variables: { + dt: "2022-01-01", + }, + resultTable: "source_table_02", + persist: false, + storageLevel: "MEMORY_AND_DISK", + options: { + connectionCollation: "utf8mb4_unicode_ci", + }, + url: "jdbc:mysql://localhost:3306/maple_datasource?useUnicode=true&characterEncoding=UTF-8&autoReconnect=true&rewriteBatchedStatements=true&useServerPrepStmts=true&allowMultiQueries=true&zeroDateTimeBehavior=convertToNull&useSSL=false&serverTimezone=Asia/Shanghai", + driver: "com.mysql.cj.jdbc.Driver", + user: "xi_root", + password: "123456", + query: "select * from test_mysql_db.test_tb where id > 0", + }, + }, + { + name: "file", + config: { + variables: { + dt: "2022-01-01", + }, + resultTable: "source_table_03", + persist: false, + storageLevel: "MEMORY_AND_DISK", + options: {}, + path: "hdfs:///data/tmp_file/${dt}", + serializer: "csv", + columnNames: ["id", "name", "address"], + }, + }, + ], + transformations: [ + { + name: "sql", + config: { + variables: { + dt: "2022-01-01", + }, + sourceTable: "", + resultTable: "transform_table_01", + persist: false, + storageLevel: "MEMORY_AND_DISK", + sql: "select st03.name, st03.address, dtt.test\nfrom source_table_03 st03 join dw.test_tb dtt on st03.id=dtt.tid", + }, + }, + ], + sinks: [ + { + name: "hive", + config: { + variables: { + dt: "2022-01-01", + }, + sourceTable: "transform_table_01", + sourceQuery: "", + numPartitions: 10, + options: {}, + targetDatabase: "dm", + targetTable: "dm_test_table", + saveMode: "overwrite", + strongCheck: false, + writeAsFile: false, + }, + }, + { + name: "jdbc", + config: { + variables: { + dt: "2022-01-01", + }, + sourceTable: "source_table_01", + sourceQuery: "", + numPartitions: 0, + options: { + connectionCollation: "utf8mb4_unicode_ci", + isolationLevel: "NONE", + batchsize: "5000", + }, + url: "jdbc:mysql://localhost:3306/maple_datasource?useUnicode=true&characterEncoding=UTF-8&autoReconnect=true&rewriteBatchedStatements=true&useServerPrepStmts=true&allowMultiQueries=true&zeroDateTimeBehavior=convertToNull&useSSL=false&serverTimezone=Asia/Shanghai", + driver: "com.mysql.cj.jdbc.Driver", + user: "xi_root", + password: "123456", + targetDatabase: "test_mysql_db", + targetTable: "test_mysql_db_table_0001", + saveMode: "overwrite", + preQueries: ["delete from test_mysql_db.test_mysql_db_table_0001"], + }, + }, + { + name: "managed_jdbc", + config: { + variables: { + dt: "2022-01-01", + }, + sourceTable: "", + sourceQuery: "select * from source_table_01", + numPartitions: 0, + options: {}, + targetDatasource: "test_postgresql", + targetDatabase: "postgresql_test_db", + targetTable: "postgresql_test_tb", + saveMode: "overwrite", + preQueries: ["delete from postgresql_test_tb where id > 0"], + }, + }, + { + name: "file", + config: { + variables: { + dt: "2022-01-01", + }, + sourceTable: "source_table_03", + sourceQuery: "", + numPartitions: 0, + options: {}, + path: "hdfs:///data/tmp_file/${dt}", + serializer: "parquet", + saveMode: "overwrite", + partitionBy: ["type"], + }, + }, + ], +}; + +const SampleArrayConfig = { + variables: {}, + plugins: [ + { + type: "source", + name: "managed_jdbc", + config: { + variables: { + dt: "2022-01-01", + }, + resultTable: "source_table_01", + persist: true, + storageLevel: "MEMORY_AND_DISK", + options: { + connectionCollation: "utf8mb4_unicode_ci", + }, + datasource: "test_mysql", + query: "select * from test_mysql_db.test_tb where id > 0", + }, + }, + { + type: "source", + name: "jdbc", + config: { + variables: { + dt: "2022-01-01", + }, + resultTable: "source_table_02", + persist: false, + storageLevel: "MEMORY_AND_DISK", + options: { + connectionCollation: "utf8mb4_unicode_ci", + }, + url: "jdbc:mysql://localhost:3306/maple_datasource?useUnicode=true&characterEncoding=UTF-8&autoReconnect=true&rewriteBatchedStatements=true&useServerPrepStmts=true&allowMultiQueries=true&zeroDateTimeBehavior=convertToNull&useSSL=false&serverTimezone=Asia/Shanghai", + driver: "com.mysql.cj.jdbc.Driver", + user: "xi_root", + password: "123456", + query: "select * from test_mysql_db.test_tb where id > 0", + }, + }, + { + type: "source", + name: "file", + config: { + variables: { + dt: "2022-01-01", + }, + resultTable: "source_table_03", + persist: false, + storageLevel: "MEMORY_AND_DISK", + options: {}, + path: "hdfs:///data/tmp_file/${dt}", + serializer: "csv", + columnNames: ["id", "name", "address"], + }, + }, + { + type: "source", + name: "sql", + config: { + variables: { + dt: "2022-01-01", + }, + sourceTable: "", + resultTable: "transform_table_01", + persist: false, + storageLevel: "MEMORY_AND_DISK", + sql: "select st03.name, st03.address, dtt.test\nfrom source_table_03 st03 join dw.test_tb dtt on st03.id=dtt.tid", + }, + }, + { + type: "sink", + name: "hive", + config: { + variables: { + dt: "2022-01-01", + }, + sourceTable: "transform_table_01", + sourceQuery: "", + numPartitions: 10, + options: {}, + targetDatabase: "dm", + targetTable: "dm_test_table", + saveMode: "overwrite", + strongCheck: false, + writeAsFile: false, + }, + }, + { + type: "sink", + name: "jdbc", + config: { + variables: { + dt: "2022-01-01", + }, + sourceTable: "source_table_01", + sourceQuery: "", + numPartitions: 0, + options: { + connectionCollation: "utf8mb4_unicode_ci", + isolationLevel: "NONE", + batchsize: "5000", + }, + url: "jdbc:mysql://localhost:3306/maple_datasource?useUnicode=true&characterEncoding=UTF-8&autoReconnect=true&rewriteBatchedStatements=true&useServerPrepStmts=true&allowMultiQueries=true&zeroDateTimeBehavior=convertToNull&useSSL=false&serverTimezone=Asia/Shanghai", + driver: "com.mysql.cj.jdbc.Driver", + user: "xi_root", + password: "123456", + targetDatabase: "test_mysql_db", + targetTable: "test_mysql_db_table_0001", + saveMode: "overwrite", + preQueries: ["delete from test_mysql_db.test_mysql_db_table_0001"], + }, + }, + { + type: "sink", + name: "managed_jdbc", + config: { + variables: { + dt: "2022-01-01", + }, + sourceTable: "", + sourceQuery: "select * from source_table_01", + numPartitions: 0, + options: {}, + targetDatasource: "test_postgresql", + targetDatabase: "postgresql_test_db", + targetTable: "postgresql_test_tb", + saveMode: "overwrite", + preQueries: ["delete from postgresql_test_tb where id > 0"], + }, + }, + { + type: "sink", + name: "file", + config: { + variables: { + dt: "2022-01-01", + }, + sourceTable: "source_table_03", + sourceQuery: "", + numPartitions: 0, + options: {}, + path: "hdfs:///data/tmp_file/${dt}", + serializer: "parquet", + saveMode: "overwrite", + partitionBy: ["type"], + }, + }, + ], +}; + +export { + PluginModels, + FileSerializers, + Layout, + Databases, + StorageLevels, + SampleConfig, + SampleArrayConfig, + Datasource, +}; diff --git a/web/packages/sparketl/utils/i18n/zh.json b/web/packages/sparketl/utils/i18n/zh.json new file mode 100644 index 0000000000..c91be64861 --- /dev/null +++ b/web/packages/sparketl/utils/i18n/zh.json @@ -0,0 +1,45 @@ +{ + "plugin" : { + "registryTableName": "注册表名", + "outputTableName": "输出表名", + "parameter": "参数", + "variable": "变量", + "saveMode": "写入模式", + "storageLevel": "缓存级别", + "enableCache": "开启缓存", + "open": "开启", + "close": "关闭", + "driver": "驱动类名", + "jdbcUrl": "jdbc Url", + "querySql": "查询语句", + "username": "用户名", + "password": "密码", + "sql": "sql", + "append": "追加", + "cover": "覆盖", + "sourceSql": "来源语句", + "sourceTable": "来源表", + "partitions": "分区数", + "partitionsColumn": "分区字段", + "filePriority": "文件优先", + "strongCheck": "强校验", + "targetDatabase": "目标库", + "targetTable": "目标表", + "datasource": "数据源", + "preQueries": "预执行SQL", + "addPreQueries": "添加预执行SQL", + "fileFormat": "文件格式", + "path": "文件路径", + "delete": "删除", + "columnNames": "字段名", + "enter": "请输入", + "common": { + "globalVariable": "全局变量", + "addPlugin": "添加插件", + "addPath": "写入路径", + "outTableName": "输出表名", + "registryTableName": "注册表名" + } + } + +} \ No newline at end of file diff --git a/web/packages/sparketl/utils/index.js b/web/packages/sparketl/utils/index.js new file mode 100644 index 0000000000..987f2558d4 --- /dev/null +++ b/web/packages/sparketl/utils/index.js @@ -0,0 +1,8 @@ +export const toJson = (s, separator) => { + let json = {}; + s.split(separator).forEach((line) => { + let option = line.split("=").map((item) => item.trim()); + if (option[0] !== "") json[option[0]] = option[1] || ""; + }); + return json; +}; diff --git a/web/packages/workflows/assets/styles/workflow.scss b/web/packages/workflows/assets/styles/workflow.scss index 26337da5b4..77ded9282c 100644 --- a/web/packages/workflows/assets/styles/workflow.scss +++ b/web/packages/workflows/assets/styles/workflow.scss @@ -112,7 +112,6 @@ overflow: hidden; &-top { display: flex; - // height: 65px; width: 250px; padding: 0 12px; flex-direction: column; @@ -122,15 +121,24 @@ justify-content: space-between; align-items: center; &-txt { - font-size: 14px; - color: rgba(0, 0, 0, 0.65); text-align: right; font-weight: 600; } + .projcat-icon, .sort-icon { + margin-right: 7px; + } &-icon { cursor: pointer; font-size: 16px; + display: flex; + align-self: start; + .icon { + font-size: 16px; + &:hover { + color: $link-color !important; + } + } } } &-b { diff --git a/web/packages/workflows/i18n/common/en.json b/web/packages/workflows/i18n/common/en.json index 0edcd2bbd2..dacc3b3d36 100644 --- a/web/packages/workflows/i18n/common/en.json +++ b/web/packages/workflows/i18n/common/en.json @@ -2,6 +2,7 @@ "message": { "workflow": { "downloadTolocal": "Successfully downloaded, please check from local download directory!", + "downloadrequest": "Download request send Successfully, please check from local download directory!", "script": "Script", "name": "Name", "detail": "Details", @@ -53,6 +54,7 @@ "synchronousPublishing": "Synchronous Publishing", "exportWorkflow": "Export Workflow", "ok": "OK", + "showVersionDiff": "Compare", "requesting": "Interface is requesting...", "cancel": "Cancel", "newCreate": "ADD", @@ -109,7 +111,8 @@ "value": "value", "attributeName": "Please enter attribute name", "propertyValue": "Please enter value", - "search": "search" + "search": "search", + "comp": "Components" }, "tabs": { "progress": "Progress", @@ -156,95 +159,6 @@ "GJZ": "Open source co-construction is in progress, so stay tuned!", "unChange": "Not change, do not need to publish" }, - "projectDetail": { - "WDGZL": "My personal workflow", - "overTime": "Over Time", - "workflowRunFail": "Workflow run fail", - "workflowRunSuccess": "Workflow run success", - "workflowRunCanceled": "Workflow run canceled", - "workflowRunOvertime": "Workflow run overtime", - "workflowPublishOvertime": "Workflow publishing overtime", - "jumpRule": "Only root workflow jump is supported! ", - "WCYDXM": "My cooperative project", - "SRMCSS": "Please enter a name to seatch", - "XJ": "Add", - "validateName": "Name cannot contain current Workspace name and user name!", - "editPermissions": "Edit Permissions", - "usersAllowedToEdit": "Users Allowed To Edit", - "viewPermissions": "View Permissions", - "usersAllowedToView": "Users Allowed To View", - "publishPermissions": "Publish Permissions", - "userAllowedPublish": "Users Allowed To Publish", - "devProcess": "Development process", - "orchestratorMode": "Orchestrator mode", - "GZL": "Workflow", - "HZXM": "Projects", - "projectName": "Project Name", - "inputProjectName": "Input Project Name", - "gotoScriptis": "Visit Scriptis", - "createProject": "Create Project", - "editorProject": "Edit Project", - "search": "Search Project", - "deleteProject": "Delete Project", - "confirmDeleteProject": "Confirm to Delete Project", - "enterName": "Please enter the name", - "sortUpdateTime": "Order by Updated Time", - "sortName": "Order by Name", - "product": "Product", - "selectProduct": "Please choose a product", - "appArea": "Appplication Domain", - "selectAppArea": "Please choose an application domain", - "business": "Business", - "app": "Appplication", - "addBusiness": "New Business", - "tips": "{app_name} Introduction:\n{app_name} is a self-developed gateway for data application development of ctyun Luban. It is built on top of Linkis data middleware and easily integrates various applications/components on the upper-level of data platforms, making data application development easier and concise.\nAs a one-stop data application development gateway for data development, data visualization, data governance and scheduling, {app_name} is keen on covering the whole lifecycle of data application development, including scenarios of data importing, data cleansing, data visualization, data governance and data exporting. \n Create a project, and enjoy your journey of data application development! ", - "updteTime": "Updated At", - "nameUnrepeatable": "Duplicated name!", - "eidtorProjectSuccess": "Successfully updated project{name}!", - "deleteTitle": "Force delete", - "projectCopy": "Copy project", - "projectPublish": "Publish project", - "projectVersion": "Project version", - "projectDesc": "Project description", - "pleaseInputProjectDesc": "Please enter project description", - "name": "Name", - "projectClassify": "Project Category", - "nameLength": "The name length should not be greater than", - "validNameDesc": "Must start with alphabetic characters, followed by alphanumeric, underscore(_) and Chinese characters!", - "selectVersion": "Please choose a version", - "resourceUpload": "Drag file here, or Click to Upload", - "fileExist": "File already exists, please choose other files or directories!", - "uploadNameWar": "Invalid file name, must be alphanumeric, underscore(_), hyphen(-) and Chinese characters and end by suffix!", - "uploadLimit": "File to upload cannot exceed 100M!", - "uploadSuccess": "Successfully uploaded file {name}!", - "uploadOver": "File size exceeds limit!", - "resourceDeleteSuccess": "Successfully deleted resource file {name}!", - "copyNotice": "Resource name copied to clipboard", - "rollBack": "Rollback", - "publishDetail": "Publish Details", - "version": "Version", - "status": "Status", - "published": "Published", - "unpublish": "Not Published", - "updator": "Published By", - "comment": "Version Comment", - "uptateTime": "Updated At", - "action": "Aciton", - "creator": "Published By", - "publishData": "Publish Info", - "publishTime": "Published At", - "pubDelete": "Delete Version", - "confirmDelete": "Confirm to delete project version {version}", - "addVersion": "New version", - "newVersion": "Confirm to set {version} as the latest version", - "init": "Init", - "run": "Run", - "publishSuccess": "Successfully published", - "publishFailed": "Publish failed", - "infoTitle": "Welcome", - "noData": "No data please add", - "pleaseSelect": "please select" - }, "process": { "index": { "DTMS": "Map mode", @@ -293,6 +207,10 @@ "bindView": "Bind front node", "associate": "Import script", "relySelect": "Select dependency", + "upstreamLevelOne": "First level upstream", + "downstreamLevelOne": "First level downstream", + "upAllLevel": "All upstream", + "downAllLevel": "All downstream", "allDelete": "Batch delete", "console": "Open console", "createSubFlow": "Create sub workflow", @@ -337,6 +255,7 @@ "readonlyNoCopy": "This node is on read-only status, copy is not supported!", "noPaste": "This node is on read-only status, paste is not supported!", "firstCopy": "Please copy a node first!", + "namelength": "Node name length exceeds 150!", "noDelete": "This node is on read-only status, delete is not supported!", "createdSuccess": "Successfully created!", "arguments": { @@ -359,7 +278,6 @@ "inputeNodeName": "Please enter the node name", "nodeType": "Node Type", "nodeDesc": "Node Description:", - "inputeNodeDesc": "Please enter the node description", "SXXX": "Attribute information:", "XZRMBXXJZ": "Please choose the sending mechanism of RMB message", "XZSJLY": "Please choose the data source", @@ -383,7 +301,6 @@ "filter": "Filter", "executeUser": "Executor", "QTXZQJSON": "Please enter the correct json format!", - "JDMCBNYGZL": "Workflow name + underscore are not allowed", "CZBNSRZW": "Chinese characters are not allowed", "QTXYLSJMC": "Please enter the name of data dependency", "CD500ZF": "Length between 1 to 500 characters", @@ -474,14 +391,111 @@ "cancel": "Cancel", "timeout": "Timeout on publishing orchestrator {name}!", "publishNotice": "Publishing will close the opened orchestrator, please save it!" - } + }, + "CopyWrokflow": "Copy workflow", + "Original": "Original workflow", + "CopyProj": "Copy to project", + "Copied": "Copied workflow", + "Node": "Node suffix", + "WorkflowSameName": "Workflow nodes with same name in the same project will cause the workflow publishing failure. Please consider the node suffix name before filling in. If the publishing fails due to the same name problem, the user needs to manually modify the corresponding workflow node name. ", + "autosuffix": "Auto add suffix for nodes", + "Cancel": "Cancel", + "PleaseInputName": "Please input name", + "selcProj": "Please choose project", + "Copyreq": "Copy request sent successfully", + "CopySucceed": "Copy workflow successfully", + "CopyFailed": "Copy workflow failed", + "Project": "Project", + "AllProj": "All projects", + "Individual": "Individual projects", + "ShareProj": "Share projects", + "Simultaneously": "Simultaneously delete all third-party system projects", + "Essential": "Essential attributes", + "unlock": "Unlock", + "unlockflow": "Unlock", + "unlockflowtitle": "Confirm unlock workflow", + "unlocksuccess": "Unlock successfully", + "unlocktip": "Unlock workflow", + "CreateUser": "Create user:", + "CreateTime": "Create time:", + "LastModifyUser": "Last modified user:", + "LastModifyTime": "Last modified time:", + "ImportType": "Import type:", + "zipfile": "From zip file", + "hdfs": "From HDFS", + "Zip": "Zip file path:", + "Upload": "Upload Zip file:", + "zippath": "Please input zip file hdfs path", + "typerequired": "Import type is required", + "fileempty": "Import file is required", + "Path": "Path is required", + "importflow": "Import workflow", + "Nopermission": "No ops permission", + "Configuration": "Configuration", + "Delete": "Delete", + "Copy": "Copy", + "Show": "Show version", + "Name:": "Name:", + "Description:": "Description:", + "Simplified": "Simplified Chinese", + "Loading": "Loading", + "wait": "Please wait", + "Streaming": "Streaming computing center operation steps", + "Import": "Import", + "confitEnv": "配置资源及生产环境", + "Configure": "Configure Flink parameters and run", + "Workflow": "Workflow", + "Scripttask": "Script task", + "SelectRun": "Select and run", + "Rerun": "Rerun", + "Prompt": "Prompt", + "Copying": "Copying,Not allowed to edit the script", + "Save": "Save script", + "Manually": "Manually save", + "AddNode": "Add node", + "BatchDel": "Batch deletion of sub workflows is not supported!", + "PleaseSelectNode": "Please select the node to be executed", + "Saving": "Saving", + "request": "request sent", + "repeate": "Please do not click repeatedly", + "Scheduled": "Scheduled", + "Running": "Running", + "ExecuteSuccess": "Execute successfully", + "ExecuteFailed": "Execute failed", + "Skip": "Skip", + "Publishwork": "Publish workflow", + "Export": "Export", + "General": "General settings", + "Addworkflow": "Add workflow", + "Success": "Success", + "Development": "Development center", + "Noopen": "No open workflow, Click the button below to add", + "Add": "Add project", + "Recent": "Recent", + "Createworkflow": "Create workflow", + "CancelFullScreen": "Cancel full screen", + "Full": "Full screen", + "Log": "Log", + "Fold": "Fold", + "workflowcat": "workflow category", + "Create": "Create", + "Edit": "Edit", + "WorkflowCategory": "Workflow category:", + "inputCategory": "Please input workflow category", + "Myworkflow": "My workflow", + "WorkflowMy": "Workflow I participate in", + "workflowdesc": "workflow category description:", + "Confirm": "Confirm", + "namelength": "Length of name cannot be longer than {num}", + "inputStyle": "Must start with a letter or Chinese, and only accept letters, digits, underscores and Chinese!", + "Publishfailed": "Publish failed" }, "orchestratorModes": { "menu": "Menu", "setting": "Setting", "createOrchestrator": "Create orchestrator", - "deleteOrchestrator": "Delete orchestrator modes", - "confirmDeleteOrchestrator": "Confirm deletion of orchestrator mode:", + "deleteOrchestrator": "Delete orchestrator", + "confirmDeleteOrchestrator": "Confirm deletion of orchestrator:", "orchestratorPublish": "orchestrator publish", "publishLoading": "There is already an orchestrator in publishing, please try again later", "modeifyOrchestrator": "Modify orchesstrator", @@ -505,4 +519,3 @@ } } } - diff --git a/web/packages/workflows/i18n/common/zh.json b/web/packages/workflows/i18n/common/zh.json index a81e4afe85..f4011f49d3 100644 --- a/web/packages/workflows/i18n/common/zh.json +++ b/web/packages/workflows/i18n/common/zh.json @@ -2,6 +2,7 @@ "message": { "workflow": { "downloadTolocal": "下载成功,请到本地的download文件夹查看!", + "downloadrequest": "请求已发出,下载完成后,请到本地download文件夹查看", "script": "脚本", "name": "名称", "detail": "详情", @@ -53,6 +54,7 @@ "historicalVersion": "历史版本", "exportWorkflow": "工作流导出", "ok": "确认", + "showVersionDiff": "版本比对", "requesting": "接口正在请求中...", "cancel": "取消", "newCreate": "新建", @@ -109,7 +111,8 @@ "value": "值", "attributeName": "请输入属性名", "propertyValue": "请输入属性值", - "search": "搜索" + "search": "搜索", + "comp": "组件" }, "tabs": { "progress": "进度", @@ -155,95 +158,6 @@ "comingSoon": "尚未开放,敬请期待!", "unChange": "工作流无变化,不需要发布" }, - "projectDetail": { - "WDGZL": "我的工作流", - "overTime": "超时", - "workflowRunFail": "工作流执行失败!", - "workflowRunSuccess": "工作流执行成功!", - "workflowRunCanceled": "工作流执行已取消!", - "workflowRunOvertime": "工作流执行超时!", - "workflowPublishOvertime": "工作流发布超时!", - "jumpRule": "只支持本工程的根工作流可跳转!", - "WCYDXM": "我参与的项目", - "SRMCSS": "请输入名称搜索", - "XJ": "新建", - "validateName": "名字不能包含当前工作空间名或用户名", - "editPermissions": "编辑权限", - "usersAllowedToEdit": "请选择可编辑用户", - "viewPermissions": "查看权限", - "usersAllowedToView": "请选择可查看用户", - "publishPermissions": "发布权限", - "userAllowedPublish": "请选择可发布用户", - "devProcess": "开发流程", - "orchestratorMode": "工作流模式", - "GZL": "工作流", - "HZXM": "合作项目", - "projectName": "项目名称", - "inputProjectName": "请输入项目名称", - "gotoScriptis": "进入Scriptis", - "createProject": "创建项目", - "editorProject": "修改项目", - "search": "搜索工程", - "deleteProject": "删除项目", - "confirmDeleteProject": "确认删除项目", - "enterName": "请输入名称", - "sortUpdateTime": "按修改时间排序", - "sortName": "按名称排序", - "product": "产品", - "selectProduct": "请选择产品", - "appArea": "应用领域", - "selectAppArea": "请选择应用领域", - "business": "业务", - "app": "应用", - "addBusiness": "新增业务", - "tips": "{app_name}介绍:\n{app_name}是微众银行开源自研的数据应用开发门户,基于Linkis数据中间件构建,轻松整合数据平台上层各应用组件,让数据应用开发变得简洁又易用。\n{app_name}作为一站式数据研发、可视化、数据治理和调度的数据应用开发门户,定位为闭环涵盖数据应用的全流程,满足从数据导入、数据清洗、数据加工、数据可视化展现、数据治理到数据输出的数据应用全生命周期开发场景。\n创建一个工程,开始您的数据应用开发之旅吧!", - "updteTime": "修改时间", - "nameUnrepeatable": "名称不可重复!", - "eidtorProjectSuccess": "修改项目{name}成功!", - "deleteTitle": "强制删除", - "projectCopy": "工程复制", - "projectPublish": "工程发布", - "projectVersion": "工程版本", - "projectDesc": "项目描述", - "pleaseInputProjectDesc": "请输入项目描述", - "name": "名称", - "projectClassify": "工程分类", - "nameLength": "名称长度不能大于", - "validNameDesc": "必须以字母开头,且只支持字母、数字、下划线和中文!", - "selectVersion": "请选择版本号", - "resourceUpload": "将文件拖到此处,或点击上传", - "fileExist": "文件已存在,请选择其它文件或选择其它文件夹!", - "uploadNameWar": "文件名称不合法,仅支持以字母、数字、中文、下划线、中短线且带后缀的命名!", - "uploadLimit": "上传文件不超过100M!", - "uploadSuccess": "文件{name}上传成功!", - "uploadOver": "文件大小超出限额!", - "resourceDeleteSuccess": "资源文件{name}已被成功删除!", - "copyNotice": "资源名称已复制至粘贴板", - "rollBack": "回滚", - "publishDetail": "发布详情", - "version": "版本", - "status": "状态", - "published": "已发布", - "unpublish": "未发布", - "updator": "发布者", - "comment": "版本注释", - "uptateTime": "更新时间", - "action": "操作", - "creator": "发布者", - "publishData": "发布信息", - "publishTime": "发布时间", - "pubDelete": "版本删除", - "confirmDelete": "确认删除工程版本{version}", - "addVersion": "新建版本", - "newVersion": "确认将{version}设为最新版本", - "init": "初始化", - "run": "运行", - "publishSuccess": "发布成功", - "publishFailed": "发布失败", - "infoTitle": "欢迎信息", - "noData": "暂无数据请添加", - "pleaseSelect": "请选择" - }, "process": { "index": { "DTMS": "地图模式", @@ -293,6 +207,10 @@ "inputBindView": "请选择一个View进行绑定", "associate": "导入脚本", "relySelect": "依赖选择", + "upstreamLevelOne": "一级上游", + "downstreamLevelOne": "一级下游", + "upAllLevel": "所有上游", + "downAllLevel": "所有下游", "console": "管理台", "allDelete": "批量删除", "createSubFlow": "创建子工作流", @@ -337,6 +255,7 @@ "readonlyNoCopy": "节点只读状态,节点不支持复制!", "noPaste": "节点只读状态,节点不支持粘贴!", "firstCopy": "请先复制节点!", + "namelength": "节点名称长度超过150!", "noDelete": "节点只读状态,节点不支持删除!", "createdSuccess": "创建成功!", "arguments": { @@ -359,7 +278,6 @@ "inputeNodeName": "请填写节点名", "nodeType": "节点类型", "nodeDesc": "节点描述:", - "inputeNodeDesc": "请填写节点描述", "SXXX": "属性信息:", "XZRMBXXJZ": "请选择RMB消息发送机制", "XZSJLY": "请选择数据来源", @@ -383,7 +301,6 @@ "filter": "过滤条件", "executeUser": "执行用户", "QTXZQJSON": "请填写正确的json格式!", - "JDMCBNYGZL": "节点名称不能以工作流名加下划线", "CZBNSRZW": "此值不能输入中文", "QTXYLSJMC": "请填写依赖数据名称", "CD500ZF": "长度在1到500个字符", @@ -475,14 +392,111 @@ "cancel": "关闭", "timeout": "工作流{name}发布超时!", "publishNotice": "发布将会关闭已打开的工作流,请注意保存!" - } + }, + "CopyWrokflow": "复制工作流", + "Original": "被复制工作流", + "CopyProj": "复制到项目", + "Copied": "复制后工作流名", + "Node": "节点后缀", + "WorkflowSameName": "同一项目中工作流节点重名会导致工作流发布失败,请考虑好节点后缀名再填写,若因重名问题导致发布失败,用户需手动修改对应工作流节点名称", + "autosuffix": "填写后将自动为工作流节点名添加该后缀", + "Cancel": "取消", + "PleaseInputName": "请输入名称", + "selcProj": "请选择项目", + "Copyreq": "复制请求发送成功", + "CopySucceed": "复制成功", + "CopyFailed": "复制失败", + "Project": "项目", + "AllProj": "所有项目", + "Individual": "个人项目", + "ShareProj": "共享项目", + "Simultaneously": "同步删除所有第三方系统的工程", + "Essential": "基础属性", + "unlock": "解锁", + "unlockflow": "确认解锁工作流", + "unlockflowtitle": "解锁工作流", + "unlocksuccess": "解锁成功", + "unlocktip": "注意:当前工作流已被 {name} 锁定,强制解锁工作流,会导致 {name} 已编辑未保存的内容无法保存,请与 {name} 确认后再解锁工作流", + "CreateUser": "创建用户:", + "CreateTime": "创建时间:", + "LastModifyUser": "最近修改用户:", + "LastModifyTime": "最近修改时间:", + "ImportType": "导入方式:", + "zipfile": "从Zip包导入", + "hdfs": "从HDFS导入", + "Zip": "Zip包路径:", + "Upload": "Zip包上传:", + "zippath": "请输入zip包的HDFS路径", + "typerequired": "导入方式必填", + "fileempty": "导入文件必填", + "Path": "路径必填", + "importflow": "导入工作流", + "Nopermission": "无运维权限", + "Configuration": "配置", + "Delete": "删除", + "Copy": "复制", + "Show": "查看版本", + "Name:": "名称:", + "Description:": "描述:", + "Simplified": "简体中文", + "Loading": "加载中", + "wait": "请稍后", + "Streaming": "流式计算中心操作步骤", + "Import": "导入", + "confitEnv": "配置资源及生产环境", + "Configure": "配置Flink参数并运行", + "Workflow": "工作流", + "Scripttask": "脚本任务", + "SelectRun": "选中执行", + "Rerun": "失败重跑", + "Prompt": "提示", + "Copying": "复制过程中,不允许编辑工作流", + "Save": "保存脚本", + "Manually": "手动保存", + "AddNode": "新增节点", + "BatchDel": "子工作流不支持批量删除!", + "PleaseSelectNode": "请先选择需要执行的节点", + "Saving": "执行保存", + "request": "请求已发出", + "repeate": "请勿重复点击", + "Scheduled": "等待执行", + "Running": "执行中", + "ExecuteSuccess": "执行成功", + "ExecuteFailed": "执行失败", + "Skip": "跳过", + "Publishwork": "发布工作流", + "Export": "导出", + "General": "通用设置", + "Addworkflow": "添加工作流", + "Success": "操作成功", + "Development": "开发中心", + "Noopen": "无打开的工作流,可点击下方按钮添加", + "Add": "添加项目", + "Recent": "最近", + "Createworkflow": "新建工作流", + "CancelFullScreen": "取消全屏", + "Full": "全屏展示", + "Log": "日志", + "Fold": "收起", + "workflowcat": "工作流分类", + "Create": "新建", + "Edit": "修改", + "WorkflowCategory": "工作流分类名:", + "inputCategory": "请输入分类名", + "Myworkflow": "我的工作流", + "WorkflowMy": "我参与的工作流", + "workflowdesc": "工作流分类描述:", + "Confirm": "确认", + "numnamelength": "名称长度不能大于{num}", + "inputStyle": "必须以字母或中文开头,且只支持字母、数字、下划线和中文!", + "Publishfailed": "发布失败" }, "orchestratorModes": { "menu": "菜单", "setting": "设置", "createOrchestrator": "新建工作流", - "deleteOrchestrator": "删除工作流模式", - "confirmDeleteOrchestrator": "确认删除工作流模式:", + "deleteOrchestrator": "删除工作流", + "confirmDeleteOrchestrator": "确认删除工作流:", "orchestratorPublish": "工作流发布", "publishLoading": "已有工作流在发布,稍后再试", "modeifyOrchestrator": "修改工作流", diff --git a/web/packages/workflows/module/common/iframe/index.vue b/web/packages/workflows/module/common/iframe/index.vue index 3305342ce3..8613098053 100644 --- a/web/packages/workflows/module/common/iframe/index.vue +++ b/web/packages/workflows/module/common/iframe/index.vue @@ -17,6 +17,9 @@ export default { type: [String, Object], default: null }, + query: { + type: Object + }, url: { type: String, default: '' @@ -40,13 +43,10 @@ export default { }, methods: { getUrl() { - let workspaceId = this.$route.query.workspaceId; - let workFlowLists = JSON.parse(sessionStorage.getItem(`work_flow_lists_${workspaceId}`)) || []; - const item = workFlowLists.find(it => (it.query && it.query.flowId == this.$route.query.flowId)) const params = { nodeType: this.node.type, projectID: +this.$route.query.projectID, - flowId: item && item.query.appId, + flowId: this.query.appId, params: { ...this.node.jobContent, title: this.node.title, diff --git a/web/packages/workflows/module/common/index.js b/web/packages/workflows/module/common/index.js index 33f9e1dc51..e4f82fa0c2 100644 --- a/web/packages/workflows/module/common/index.js +++ b/web/packages/workflows/module/common/index.js @@ -23,6 +23,5 @@ export default { component: { iframe: () => import('./iframe/index.vue'), publish: () => import('./projectPublishForm.vue'), - workflowContentItem: () => import('./workflowContentItem') }, }; diff --git a/web/packages/workflows/module/common/voidPage/void.vue b/web/packages/workflows/module/common/voidPage/void.vue index f8ddb7a7b0..0e3e00a699 100644 --- a/web/packages/workflows/module/common/voidPage/void.vue +++ b/web/packages/workflows/module/common/voidPage/void.vue @@ -2,30 +2,30 @@
    - 开发中心 + {{ $t('message.workflow.Development') }}
    - 无打开的工作流,可点击下方按钮添加 + {{ $t('message.workflow.Noopen') }}
    - + {{ $t('message.workflow.Addworkflow') }}
    - 最近 + {{ $t('message.workflow.Recent') }}
    \ No newline at end of file diff --git a/web/packages/workflows/module/common/workflowContentItem/img/product.svg b/web/packages/workflows/module/common/workflowContentItem/img/product.svg deleted file mode 100644 index b2dd3abdfe..0000000000 --- a/web/packages/workflows/module/common/workflowContentItem/img/product.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/web/packages/workflows/module/common/workflowContentItem/img/project_bg.png b/web/packages/workflows/module/common/workflowContentItem/img/project_bg.png deleted file mode 100644 index fe51e3c945..0000000000 Binary files a/web/packages/workflows/module/common/workflowContentItem/img/project_bg.png and /dev/null differ diff --git a/web/packages/workflows/module/common/workflowContentItem/index.scss b/web/packages/workflows/module/common/workflowContentItem/index.scss deleted file mode 100644 index 46c10477ea..0000000000 --- a/web/packages/workflows/module/common/workflowContentItem/index.scss +++ /dev/null @@ -1,254 +0,0 @@ -/*! - * Copyright 2019 WeBank - * - * 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 '@dataspherestudio/shared/common/style/variables.scss'; -.workflow-item { - // overflow: hidden; -} -.content-item { - margin: 2px 0px 25px 0px; - .project-list-ul { - padding: 10px 20px; - display: flex; - justify-content: space-around; - align-items: center; - flex-wrap: wrap; - } - .project-item { - height: 142px; - margin-right: 25px; - margin-bottom: 25px; - // background: #fff; - @include bg-color($workspace-body-bg-color, $dark-workspace-body-bg-color); - max-width: 430px; - min-width: 320px; - border-radius: 2px; - border: 1px solid #dcdee2; - @include border-color($border-color-base, $dark-border-color-base); - padding: 20px; - cursor: pointer; - &:hover { - box-shadow: 0 2px 12px 0 $shadow-color; - } - .project-add { - display: flex; - height: 100%; - align-items: center; - justify-content: center; - @include font-color($light-text-color, $dark-text-color); - span { - font-size: 20px; - } - } - .project-main { - height: 100%; - .top-bar { - width: 100%; - display: flex; - justify-content: space-between; - align-items: center; - /deep/.ivu-btn { - @include bg-color($workspace-body-bg-color, $dark-workspace-body-bg-color); - @include border-color($border-color-base, $dark-border-color-base); - @include font-color($light-text-color, $dark-text-color); - } - .project-title { - flex: 1; - font-size: $font-size-large; - font-weight: 600; - white-space: nowrap; - text-overflow: ellipsis; - overflow: hidden; - - // color: $text-title-color; - @include font-color($workspace-title-color, $dark-workspace-title-color); - letter-spacing: 0; - } - .menu-bar { - position: relative; - flex-basis: 40px; - margin-right: 8px; - .menu-list { - display: none; - position: absolute; - bottom: 25px; - left: -8px; - border-radius: 4px; - padding: 5px 0; - box-shadow: 0 1px 6px rgba(0,0,0,.2); - z-index: 999; - // background-color: #fff; - @include bg-color($menu-list-bg-color, $dark-menu-list-bg-color); - cursor: pointer; - .list-item { - width: 60px; - padding: 5px 8px; - text-align: center; - @include font-color($light-text-color, $dark-text-color); - &:hover { - // background-color: #f3f3f3; - @include bg-color($hover-color-base, $dark-hover-color-base); - } - } - } - &:hover { - .menu-list { - display: inline; - } - } - } - } - .mid-bar { - width: 100%; - white-space: nowrap; - text-overflow: ellipsis; - overflow: hidden; - font-size: $font-size-14; - // color: rgba(0,0,0,0.5); - @include font-color($light-text-desc-color, $dark-text-desc-color); - margin: 10px 0; - } - .bottom-bar { - display: flex; - justify-content: flex-start; - align-items: center; - width: 100%; - .tag-item { - color: $text-desc-color; - @include font-color($light-text-color, $dark-text-color); - padding: 2px 10px; - margin-right: 10px; - border-radius: 14px; - @include bg-color(#F3F3F3, $dark-base-color); - border: 1px solid $border-color-base; - @include border-color($border-color-base, $dark-border-color-base); - white-space: nowrap; - text-overflow: ellipsis; - overflow: hidden; - - } - } - } - .process-bar { - position: absolute; - left: 0; - bottom: -20px; - } - } - .project-header { - text-align: center; - font-size: $font-size-large; - line-height: 102px; - border: 1px dashed #dcdee2; - // background: #F8F9FC; - @include border-color($border-color-base, $dark-border-color-base); - @include bg-color($workflow-body-bg-color, $dark-workflow-body-bg-color); - .add-icon { - margin-top: -2px; - margin-right: 5px; - } - } -} - -.item-header { - font-size: $font-size-base; - margin: 10px 25px; - font-weight: bold; - padding-left: 5px; - border-left: 3px solid $primary-color; - @include border-color($primary-color, $dark-primary-color); -} - -.page-bar { - padding: 0 30px; - margin-bottom: 30px; - @include font-color($light-text-desc-color, $dark-text-desc-color); -} - -.no-data { - text-align: center; - padding: 20px; - @include font-color($light-text-color, $dark-text-color); -} - -.queue-manager-status { - position: $relative; - width: 100%; - height: 10px; - display: flex; - align-items: center; - font-size: 10px; - border-radius: 10px; - background-image: linear-gradient(60deg, transparent 0rem, transparent 0.8rem, rgb(88, 175, 251) 0.8rem, rgb(88, 175, 251) 1.6rem, transparent 1.6rem, transparent 2.4rem, rgb(88, 175, 251) 2.4rem); - background-size: 21px 27px; - box-shadow: 1px 1px 5px rgba(88, 175, 251, .6); - -webkit-animation: process 800ms infinite linear; - animation: process 800ms infinite linear; - background-color: $background-color-base; - &:after { - content: ''; - position: absolute; - top: 0; - left: 0; - right: 0; - bottom: 0; - height: 100%; - border-radius: 10px; - background-image: linear-gradient(to bottom, rgba(88, 175, 251, .4), rgba(88, 175, 251, .4) 15%, transparent 60%, rgba(88, 175, 251, .4)); - } - .queue-manager-status-busy { - position: $absolute; - left: 0; - background: $success-color; - height: 100%; - border-radius: 10px; - z-index: 1; - } - .queue-manager-status-idle { - background: $success-color; - height: 100%; - border-radius: 10px; - position: $absolute; - z-index: 0; - } - .queue-manager-status-label { - position: $absolute; - right: 6px; - color: $tooltip-color; - } -} - - -/* 动画 */ - -@-webkit-keyframes process { - 0% { - background-position: 0 0; - } - 100% { - background-position: 20px 0; - } -} - -@keyframes process { - 0% { - background-position: 0 0; - } - 100% { - background-position: 20px 0; - } -} diff --git a/web/packages/workflows/module/common/workflowContentItem/index.vue b/web/packages/workflows/module/common/workflowContentItem/index.vue deleted file mode 100644 index de1918c2b8..0000000000 --- a/web/packages/workflows/module/common/workflowContentItem/index.vue +++ /dev/null @@ -1,205 +0,0 @@ - - - diff --git a/web/packages/workflows/module/header/index.vue b/web/packages/workflows/module/header/index.vue index 4a39aa86bf..7031be4632 100644 --- a/web/packages/workflows/module/header/index.vue +++ b/web/packages/workflows/module/header/index.vue @@ -60,7 +60,7 @@ export default { sysVersion: process.env.VUE_APP_VERSION, isUserMenuShow: false, userName: "", - isSandbox: process.env.NODE_ENV === "sandbox", + isSandbox: this.$APP_CONF.isSandbox, }; }, mixins: [mixin], diff --git a/web/packages/workflows/module/header/userMenu.vue b/web/packages/workflows/module/header/userMenu.vue index 33b04601ad..27d26a52a7 100644 --- a/web/packages/workflows/module/header/userMenu.vue +++ b/web/packages/workflows/module/header/userMenu.vue @@ -37,7 +37,7 @@ export default { }, { id: 'changeLang', - name: localStorage.getItem('locale') === 'zh-CN' ? 'English' : '简体中文', + name: localStorage.getItem('locale') === 'zh-CN' ? 'English' : this.$t('message.workflow.Simplified'), icon: 'md-repeat', }, { id: 'logout', @@ -56,7 +56,7 @@ export default { this.clearCache(); break; case 'logout': - this.getRunningJob(); + this.logout(); break; case 'changeLang': this.changeLang(); @@ -81,9 +81,6 @@ export default { }, }); }, - getRunningJob() { - this.logout(); - }, logout() { api.fetch('/user/logout', {}).then(() => { this.$emit('clear-session'); diff --git a/web/packages/workflows/module/indexedDB/index.js b/web/packages/workflows/module/indexedDB/index.js index 21be67cf3f..06715e1aff 100644 --- a/web/packages/workflows/module/indexedDB/index.js +++ b/web/packages/workflows/module/indexedDB/index.js @@ -1,29 +1,10 @@ import { db } from '@dataspherestudio/shared/common/service/db/index.js'; -import project from '@/workflows/service/db/project.js'; import node from '@/workflows/service/db/node.js'; export default { name: 'workflowIndexedDB', events: [], methods: { - async getProjectCache({projectID, key, cb}) { - const result = await project.getProjectCache({projectID, key}); - cb(result); - }, - - addProjectCache({projectID, value}) { - project.addProjectCache({projectID, value}); - }, - - updateProjectCache({projectID, key, value, isDeep}) { - project.updateProjectCache({projectID, key, value, isDeep}); - }, - removeProjectCache({projectID, key}) { - project.removeProjectCache({projectID, key}); - }, - clearProjectCache() { - db.db.project.clear(); - }, async getNodeCache({nodeId, key, cb}) { const result = await node.getNodeCache({nodeId, key}); cb(result); @@ -43,4 +24,4 @@ export default { db.db.node.clear(); } }, -}; \ No newline at end of file +}; diff --git a/web/packages/workflows/module/innerIframe/emptyGuide.vue b/web/packages/workflows/module/innerIframe/emptyGuide.vue index 2409a41c22..ff41c8208c 100644 --- a/web/packages/workflows/module/innerIframe/emptyGuide.vue +++ b/web/packages/workflows/module/innerIframe/emptyGuide.vue @@ -1,6 +1,6 @@ + {{ $t('message.common.Loading') }}
    + diff --git a/web/packages/workflows/module/process/component/console.vue b/web/packages/workflows/module/process/component/console.vue index 5ae507f5a2..362036615d 100644 --- a/web/packages/workflows/module/process/component/console.vue +++ b/web/packages/workflows/module/process/component/console.vue @@ -1,7 +1,9 @@
    ").append(content).append("
    ").append(content).append("
    ").append(emailContent.getContent).append("