Compare commits
4 Commits
v6.5.0
...
developmen
Author | SHA1 | Date | |
---|---|---|---|
b4e5dd74d9 | |||
64746b0aae | |||
16ae77c804 | |||
481c14f066 |
@ -1,5 +1,5 @@
|
|||||||
name: Build
|
name: Build
|
||||||
run-name: Running Lint Check and Licence checker on Pull Request
|
run-name: Running Lint Check
|
||||||
on: [pull_request]
|
on: [pull_request]
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
@ -18,11 +18,8 @@ jobs:
|
|||||||
env:
|
env:
|
||||||
NPMRC: ${{ secrets.NPMRC}}
|
NPMRC: ${{ secrets.NPMRC}}
|
||||||
|
|
||||||
- name: Lint check
|
- run: npm run lint:check
|
||||||
run: npm run lint:check
|
- run: |
|
||||||
|
|
||||||
- name: Licence checker
|
|
||||||
run: |
|
|
||||||
cd client
|
cd client
|
||||||
npm ci
|
npm ci
|
||||||
npm run license-checker
|
npm run license-checker
|
226
.gitea/workflows/development-test.yaml
Normal file
226
.gitea/workflows/development-test.yaml
Normal file
@ -0,0 +1,226 @@
|
|||||||
|
name: Test
|
||||||
|
run-name: Building and testing development branch
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- development
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
Build-production-and-ng-test:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v3
|
||||||
|
- uses: actions/setup-node@v3
|
||||||
|
with:
|
||||||
|
node-version: 18
|
||||||
|
|
||||||
|
- name: Write .npmrc file
|
||||||
|
run: |
|
||||||
|
touch client/.npmrc
|
||||||
|
echo '${{ secrets.NPMRC}}' > client/.npmrc
|
||||||
|
|
||||||
|
- name: Install Chrome for Angular tests
|
||||||
|
run: |
|
||||||
|
apt-get update
|
||||||
|
wget https://dl.google.com/linux/direct/google-chrome-stable_current_amd64.deb
|
||||||
|
apt install -y ./google-chrome*.deb;
|
||||||
|
export CHROME_BIN=/usr/bin/google-chrome
|
||||||
|
|
||||||
|
- name: Write cypress credentials
|
||||||
|
run: echo "$CYPRESS_CREDS" > ./client/cypress.env.json
|
||||||
|
shell: bash
|
||||||
|
env:
|
||||||
|
CYPRESS_CREDS: ${{ secrets.CYPRESS_CREDS }}
|
||||||
|
|
||||||
|
- name: Install dependencies
|
||||||
|
run: npm ci
|
||||||
|
|
||||||
|
- name: Check audit
|
||||||
|
# Audit should fail and stop the CI if critical vulnerability found
|
||||||
|
run: |
|
||||||
|
npm audit --audit-level=critical --omit=dev
|
||||||
|
cd ./sas
|
||||||
|
npm audit --audit-level=critical --omit=dev
|
||||||
|
cd ../client
|
||||||
|
npm audit --audit-level=critical --omit=dev
|
||||||
|
|
||||||
|
- name: Angular Tests
|
||||||
|
run: |
|
||||||
|
cd client
|
||||||
|
npm test -- --no-watch --no-progress --browsers=ChromeHeadlessCI
|
||||||
|
|
||||||
|
- name: Angular Production Build
|
||||||
|
run: |
|
||||||
|
cd client
|
||||||
|
npm run postinstall
|
||||||
|
npm run build
|
||||||
|
|
||||||
|
Build-and-test-development:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v3
|
||||||
|
- uses: actions/setup-node@v3
|
||||||
|
with:
|
||||||
|
node-version: 18
|
||||||
|
|
||||||
|
- name: Write .npmrc file
|
||||||
|
run: |
|
||||||
|
touch client/.npmrc
|
||||||
|
echo '${{ secrets.NPMRC}}' > client/.npmrc
|
||||||
|
|
||||||
|
- run: apt-get update
|
||||||
|
- run: wget https://dl.google.com/linux/direct/google-chrome-stable_current_amd64.deb
|
||||||
|
- run: apt install -y ./google-chrome*.deb;
|
||||||
|
- run: export CHROME_BIN=/usr/bin/google-chrome
|
||||||
|
- run: apt-get update -y
|
||||||
|
- run: apt-get -y install libgtk2.0-0 libgtk-3-0 libgbm-dev libnotify-dev libgconf-2-4 libnss3 libxss1 libasound2 libxtst6 xauth xvfb
|
||||||
|
- run: apt -y install jq
|
||||||
|
|
||||||
|
- name: Write cypress credentials
|
||||||
|
run: echo "$CYPRESS_CREDS" > ./client/cypress.env.json
|
||||||
|
shell: bash
|
||||||
|
env:
|
||||||
|
CYPRESS_CREDS: ${{ secrets.CYPRESS_CREDS }}
|
||||||
|
|
||||||
|
- name: Install dependencies
|
||||||
|
run: npm ci
|
||||||
|
|
||||||
|
# Install pm2 and prepare SASJS server
|
||||||
|
- run: npm i -g pm2
|
||||||
|
- run: curl -L https://github.com/sasjs/server/releases/latest/download/linux.zip > linux.zip
|
||||||
|
- run: unzip linux.zip
|
||||||
|
- run: touch .env
|
||||||
|
- run: echo RUN_TIMES=js >> .env
|
||||||
|
- run: echo NODE_PATH=node >> .env
|
||||||
|
- run: echo CORS=enable >> .env
|
||||||
|
- run: echo WHITELIST=http://localhost:4200 >> .env
|
||||||
|
- run: cat .env
|
||||||
|
- run: pm2 start api-linux --wait-ready
|
||||||
|
|
||||||
|
- name: Deploy mocked services
|
||||||
|
run: |
|
||||||
|
cd ./sas/mocks/sasjs
|
||||||
|
npm install -g @sasjs/cli
|
||||||
|
npm install -g replace-in-files-cli
|
||||||
|
sasjs cbd -t server-ci
|
||||||
|
# sasjs request services/admin/makedata -t server-ci -d ./deploy/makeData4GL.json -c ./deploy/requestConfig.json -o ./output.json
|
||||||
|
|
||||||
|
- name: Install ZIP
|
||||||
|
run: |
|
||||||
|
apt-get update
|
||||||
|
apt-get install zip
|
||||||
|
|
||||||
|
- name: Prepare and run frontend and cypress
|
||||||
|
run: |
|
||||||
|
cd ./client
|
||||||
|
mv ./cypress.env.example.json ./cypress.env.json
|
||||||
|
replace-in-files --regex='"username".*' --replacement='"username":"'${{ secrets.CYPRESS_USERNAME_SASJS }}'",' ./cypress.env.json
|
||||||
|
replace-in-files --regex='"password".*' --replacement='"password":"'${{ secrets.CYPRESS_PWD_SASJS }}'" ' ./cypress.env.json
|
||||||
|
cat ./cypress.env.json
|
||||||
|
npm run postinstall
|
||||||
|
# Prepare index.html to SASJS local
|
||||||
|
replace-in-files --regex='serverUrl=".*?"' --replacement='serverUrl="http://localhost:5000"' ./src/index.html
|
||||||
|
replace-in-files --regex='appLoc=".*?"' --replacement='appLoc="/Public/app/devtest"' ./src/index.html
|
||||||
|
replace-in-files --regex='serverType=".*?"' --replacement='serverType="SASJS"' ./src/index.html
|
||||||
|
replace-in-files --regex='"hosturl".*' --replacement='hosturl:"http://localhost:4200",' ./cypress.config.ts
|
||||||
|
cat ./cypress.config.ts
|
||||||
|
# Start frontend and run cypress
|
||||||
|
npm start & npx wait-on http://localhost:4200 && npx cypress run --browser chrome --spec "cypress/e2e/liveness.cy.ts,cypress/e2e/editor.cy.ts,cypress/e2e/excel.cy.ts,cypress/e2e/filtering.cy.ts,cypress/e2e/licensing.cy.ts"
|
||||||
|
|
||||||
|
- name: Zip Cypress videos
|
||||||
|
if: always()
|
||||||
|
run: |
|
||||||
|
zip -r cypress-videos ./client/cypress/videos
|
||||||
|
|
||||||
|
- name: Cypress videos artifacts
|
||||||
|
uses: actions/upload-artifact@v3
|
||||||
|
with:
|
||||||
|
name: cypress-videos.zip
|
||||||
|
path: cypress-videos.zip
|
||||||
|
|
||||||
|
Build-and-test-development-latest-adapter:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v3
|
||||||
|
- uses: actions/setup-node@v3
|
||||||
|
with:
|
||||||
|
node-version: 18
|
||||||
|
|
||||||
|
- name: Write .npmrc file
|
||||||
|
run: echo "$NPMRC" > client/.npmrc
|
||||||
|
shell: bash
|
||||||
|
env:
|
||||||
|
NPMRC: ${{ secrets.NPMRC}}
|
||||||
|
|
||||||
|
- run: apt-get update
|
||||||
|
- run: wget https://dl.google.com/linux/direct/google-chrome-stable_current_amd64.deb
|
||||||
|
- run: apt install -y ./google-chrome*.deb;
|
||||||
|
- run: export CHROME_BIN=/usr/bin/google-chrome
|
||||||
|
- run: apt-get update -y
|
||||||
|
- run: apt-get -y install libgtk2.0-0 libgtk-3-0 libgbm-dev libnotify-dev libgconf-2-4 libnss3 libxss1 libasound2 libxtst6 xauth xvfb
|
||||||
|
- run: apt -y install jq
|
||||||
|
|
||||||
|
- name: Write cypress credentials
|
||||||
|
run: echo "$CYPRESS_CREDS" > ./client/cypress.env.json
|
||||||
|
shell: bash
|
||||||
|
env:
|
||||||
|
CYPRESS_CREDS: ${{ secrets.CYPRESS_CREDS }}
|
||||||
|
|
||||||
|
- name: Install dependencies
|
||||||
|
run: npm ci
|
||||||
|
|
||||||
|
# Install pm2 and prepare SASJS server
|
||||||
|
- run: npm i -g pm2
|
||||||
|
- run: curl -L https://github.com/sasjs/server/releases/latest/download/linux.zip > linux.zip
|
||||||
|
- run: unzip linux.zip
|
||||||
|
- run: touch .env
|
||||||
|
- run: echo RUN_TIMES=js >> .env
|
||||||
|
- run: echo NODE_PATH=node >> .env
|
||||||
|
- run: echo CORS=enable >> .env
|
||||||
|
- run: echo WHITELIST=http://localhost:4200 >> .env
|
||||||
|
- run: cat .env
|
||||||
|
- run: pm2 start api-linux --wait-ready
|
||||||
|
|
||||||
|
- name: Deploy mocked services
|
||||||
|
run: |
|
||||||
|
cd ./sas/mocks/sasjs
|
||||||
|
npm install -g @sasjs/cli
|
||||||
|
npm install -g replace-in-files-cli
|
||||||
|
sasjs cbd -t server-ci
|
||||||
|
|
||||||
|
- name: Install ZIP
|
||||||
|
run: |
|
||||||
|
apt-get update
|
||||||
|
apt-get install zip
|
||||||
|
|
||||||
|
- name: Prepare and run frontend and cypress
|
||||||
|
run: |
|
||||||
|
cd ./client
|
||||||
|
mv ./cypress.env.example.json ./cypress.env.json
|
||||||
|
replace-in-files --regex='"username".*' --replacement='"username":"'${{ secrets.CYPRESS_USERNAME_SASJS }}'",' ./cypress.env.json
|
||||||
|
replace-in-files --regex='"password".*' --replacement='"password":"'${{ secrets.CYPRESS_PWD_SASJS }}'" ' ./cypress.env.json
|
||||||
|
cat ./cypress.env.json
|
||||||
|
npm run postinstall
|
||||||
|
npm install @sasjs/adapter@latest
|
||||||
|
# Prepare index.html to SASJS local
|
||||||
|
replace-in-files --regex='serverUrl=".*?"' --replacement='serverUrl="http://localhost:5000"' ./src/index.html
|
||||||
|
replace-in-files --regex='appLoc=".*?"' --replacement='appLoc="/Public/app/devtest"' ./src/index.html
|
||||||
|
replace-in-files --regex='serverType=".*?"' --replacement='serverType="SASJS"' ./src/index.html
|
||||||
|
replace-in-files --regex='"hosturl".*' --replacement='hosturl:"http://localhost:4200",' ./cypress.config.ts
|
||||||
|
cat ./cypress.config.ts
|
||||||
|
# Start frontend and run cypress
|
||||||
|
npm start & npx wait-on http://localhost:4200 && npx cypress run --browser chrome --spec "cypress/e2e/liveness.cy.ts,cypress/e2e/editor.cy.ts,cypress/e2e/excel.cy.ts,cypress/e2e/filtering.cy.ts,cypress/e2e/licensing.cy.ts"
|
||||||
|
|
||||||
|
- name: Zip Cypress videos
|
||||||
|
if: always()
|
||||||
|
run: |
|
||||||
|
zip -r cypress-videos ./client/cypress/videos
|
||||||
|
|
||||||
|
- name: Cypress videos artifacts
|
||||||
|
uses: actions/upload-artifact@v3
|
||||||
|
with:
|
||||||
|
name: cypress-videos-latest-adapter.zip
|
||||||
|
path: cypress-videos.zip
|
@ -1,156 +1,18 @@
|
|||||||
name: Release
|
name: Release
|
||||||
run-name: Testing and Releasing DC
|
run-name: Releasing DC
|
||||||
on:
|
on:
|
||||||
push:
|
push:
|
||||||
branches:
|
branches:
|
||||||
- main
|
- main
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
Build-production-and-ng-test:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v3
|
|
||||||
- uses: actions/setup-node@v3
|
|
||||||
with:
|
|
||||||
node-version: 18
|
|
||||||
|
|
||||||
- name: Write .npmrc file
|
|
||||||
run: |
|
|
||||||
touch client/.npmrc
|
|
||||||
echo '${{ secrets.NPMRC}}' > client/.npmrc
|
|
||||||
|
|
||||||
- name: Install Chrome for Angular tests
|
|
||||||
run: |
|
|
||||||
apt-get update
|
|
||||||
wget https://dl.google.com/linux/direct/google-chrome-stable_current_amd64.deb
|
|
||||||
apt install -y ./google-chrome*.deb;
|
|
||||||
export CHROME_BIN=/usr/bin/google-chrome
|
|
||||||
|
|
||||||
- name: Write cypress credentials
|
|
||||||
run: echo "$CYPRESS_CREDS" > ./client/cypress.env.json
|
|
||||||
shell: bash
|
|
||||||
env:
|
|
||||||
CYPRESS_CREDS: ${{ secrets.CYPRESS_CREDS }}
|
|
||||||
|
|
||||||
- name: Install dependencies
|
|
||||||
run: npm ci
|
|
||||||
|
|
||||||
- name: Check audit
|
|
||||||
# Audit should fail and stop the CI if critical vulnerability found
|
|
||||||
run: |
|
|
||||||
npm audit --audit-level=critical --omit=dev
|
|
||||||
cd ./sas
|
|
||||||
npm audit --audit-level=critical --omit=dev
|
|
||||||
cd ../client
|
|
||||||
npm audit --audit-level=critical --omit=dev
|
|
||||||
|
|
||||||
- name: Angular Tests
|
|
||||||
run: |
|
|
||||||
cd client
|
|
||||||
npm test -- --no-watch --no-progress --browsers=ChromeHeadlessCI
|
|
||||||
|
|
||||||
- name: Angular Production Build
|
|
||||||
run: |
|
|
||||||
cd client
|
|
||||||
npm run postinstall
|
|
||||||
npm run build
|
|
||||||
|
|
||||||
Build-and-test-development:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
needs: Build-production-and-ng-test
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v3
|
|
||||||
- uses: actions/setup-node@v3
|
|
||||||
with:
|
|
||||||
node-version: 18
|
|
||||||
|
|
||||||
- name: Write .npmrc file
|
|
||||||
run: |
|
|
||||||
touch client/.npmrc
|
|
||||||
echo '${{ secrets.NPMRC}}' > client/.npmrc
|
|
||||||
|
|
||||||
- run: apt-get update
|
|
||||||
- run: wget https://dl.google.com/linux/direct/google-chrome-stable_current_amd64.deb
|
|
||||||
- run: apt install -y ./google-chrome*.deb;
|
|
||||||
- run: export CHROME_BIN=/usr/bin/google-chrome
|
|
||||||
- run: apt-get update -y
|
|
||||||
- run: apt-get -y install libgtk2.0-0 libgtk-3-0 libgbm-dev libnotify-dev libgconf-2-4 libnss3 libxss1 libasound2 libxtst6 xauth xvfb
|
|
||||||
- run: apt -y install jq
|
|
||||||
|
|
||||||
- name: Write cypress credentials
|
|
||||||
run: echo "$CYPRESS_CREDS" > ./client/cypress.env.json
|
|
||||||
shell: bash
|
|
||||||
env:
|
|
||||||
CYPRESS_CREDS: ${{ secrets.CYPRESS_CREDS }}
|
|
||||||
|
|
||||||
- name: Install dependencies
|
|
||||||
run: npm ci
|
|
||||||
|
|
||||||
# Install pm2 and prepare SASJS server
|
|
||||||
- run: npm i -g pm2
|
|
||||||
- run: curl -L https://github.com/sasjs/server/releases/latest/download/linux.zip > linux.zip
|
|
||||||
- run: unzip linux.zip
|
|
||||||
- run: touch .env
|
|
||||||
- run: echo RUN_TIMES=js >> .env
|
|
||||||
- run: echo NODE_PATH=node >> .env
|
|
||||||
- run: echo CORS=enable >> .env
|
|
||||||
- run: echo WHITELIST=http://localhost:4200 >> .env
|
|
||||||
- run: cat .env
|
|
||||||
- run: pm2 start api-linux --wait-ready
|
|
||||||
|
|
||||||
- name: Deploy mocked services
|
|
||||||
run: |
|
|
||||||
cd ./sas/mocks/sasjs
|
|
||||||
npm install -g @sasjs/cli
|
|
||||||
npm install -g replace-in-files-cli
|
|
||||||
sasjs cbd -t server-ci
|
|
||||||
# sasjs request services/admin/makedata -t server-ci -d ./deploy/makeData4GL.json -c ./deploy/requestConfig.json -o ./output.json
|
|
||||||
|
|
||||||
- name: Install ZIP
|
|
||||||
run: |
|
|
||||||
apt-get update
|
|
||||||
apt-get install zip
|
|
||||||
|
|
||||||
- name: Prepare and run frontend and cypress
|
|
||||||
run: |
|
|
||||||
cd ./client
|
|
||||||
mv ./cypress.env.example.json ./cypress.env.json
|
|
||||||
replace-in-files --regex='"username".*' --replacement='"username":"'${{ secrets.CYPRESS_USERNAME_SASJS }}'",' ./cypress.env.json
|
|
||||||
replace-in-files --regex='"password".*' --replacement='"password":"'${{ secrets.CYPRESS_PWD_SASJS }}'" ' ./cypress.env.json
|
|
||||||
cat ./cypress.env.json
|
|
||||||
npm run postinstall
|
|
||||||
# Prepare index.html to SASJS local
|
|
||||||
replace-in-files --regex='serverUrl=".*?"' --replacement='serverUrl="http://localhost:5000"' ./src/index.html
|
|
||||||
replace-in-files --regex='appLoc=".*?"' --replacement='appLoc="/Public/app/devtest"' ./src/index.html
|
|
||||||
replace-in-files --regex='serverType=".*?"' --replacement='serverType="SASJS"' ./src/index.html
|
|
||||||
replace-in-files --regex='"hosturl".*' --replacement='hosturl:"http://localhost:4200",' ./cypress.config.ts
|
|
||||||
cat ./cypress.config.ts
|
|
||||||
# Start frontend and run cypress
|
|
||||||
npm start & npx wait-on http://localhost:4200 && npx cypress run --browser chrome --spec "cypress/e2e/liveness.cy.ts,cypress/e2e/editor.cy.ts,cypress/e2e/excel.cy.ts,cypress/e2e/filtering.cy.ts,cypress/e2e/licensing.cy.ts"
|
|
||||||
|
|
||||||
- name: Zip Cypress videos
|
|
||||||
if: always()
|
|
||||||
run: |
|
|
||||||
zip -r cypress-videos ./client/cypress/videos
|
|
||||||
|
|
||||||
- name: Add cypress videos artifacts
|
|
||||||
if: always()
|
|
||||||
uses: actions/upload-artifact@v3
|
|
||||||
with:
|
|
||||||
name: cypress-videos.zip
|
|
||||||
path: cypress-videos.zip
|
|
||||||
|
|
||||||
release:
|
release:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
needs: [Build-production-and-ng-test, Build-and-test-development]
|
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v3
|
||||||
- uses: actions/setup-node@v3
|
- uses: actions/setup-node@v3
|
||||||
with:
|
with:
|
||||||
node-version: 20
|
node-version: 18
|
||||||
|
|
||||||
- name: Write .npmrc file
|
- name: Write .npmrc file
|
||||||
run: |
|
run: |
|
||||||
@ -168,9 +30,6 @@ jobs:
|
|||||||
npm i -g @sasjs/cli
|
npm i -g @sasjs/cli
|
||||||
# jq is used to parse the release JSON
|
# jq is used to parse the release JSON
|
||||||
apt-get install jq -y
|
apt-get install jq -y
|
||||||
# doxygen is used for the SASJS docs
|
|
||||||
apt-get update
|
|
||||||
apt-get install doxygen -y
|
|
||||||
|
|
||||||
- name: Create Empty Release (assets are posted later)
|
- name: Create Empty Release (assets are posted later)
|
||||||
run: |
|
run: |
|
||||||
@ -243,12 +102,6 @@ jobs:
|
|||||||
npm run compodoc:build
|
npm run compodoc:build
|
||||||
surfer put --token ${{ secrets.TSDOC_TOKEN }} --server webdoc.datacontroller.io documentation/* /
|
surfer put --token ${{ secrets.TSDOC_TOKEN }} --server webdoc.datacontroller.io documentation/* /
|
||||||
|
|
||||||
- name: Release code.datacontroller.io
|
|
||||||
run: |
|
|
||||||
cd sas
|
|
||||||
sasjs doc
|
|
||||||
surfer put --token ${{ secrets.CODE_DATACONTROLLER_IO }} --server code.datacontroller.io sasjsbuild/sasdocs/* /
|
|
||||||
|
|
||||||
- name: Upload assets to release
|
- name: Upload assets to release
|
||||||
run: |
|
run: |
|
||||||
RELEASE_ID=`curl -k 'https://git.datacontroller.io/api/v1/repos/dc/dc/releases/latest?access_token=${{ secrets.RELEASE_TOKEN }}' | jq -r '.id'`
|
RELEASE_ID=`curl -k 'https://git.datacontroller.io/api/v1/repos/dc/dc/releases/latest?access_token=${{ secrets.RELEASE_TOKEN }}' | jq -r '.id'`
|
||||||
|
11
.vscode/settings.json
vendored
11
.vscode/settings.json
vendored
@ -1,19 +1,18 @@
|
|||||||
{
|
{
|
||||||
"cSpell.words": [
|
"cSpell.words": [
|
||||||
"Licence",
|
|
||||||
"SYSERRORTEXT",
|
"SYSERRORTEXT",
|
||||||
"SYSWARNINGTEXT",
|
"SYSWARNINGTEXT"
|
||||||
"xlmaprules",
|
],
|
||||||
"xlmaps"
|
"editor.rulers": [
|
||||||
|
80
|
||||||
],
|
],
|
||||||
"editor.rulers": [80],
|
|
||||||
"files.trimTrailingWhitespace": true,
|
"files.trimTrailingWhitespace": true,
|
||||||
"[markdown]": {
|
"[markdown]": {
|
||||||
"files.trimTrailingWhitespace": false
|
"files.trimTrailingWhitespace": false
|
||||||
},
|
},
|
||||||
"workbench.colorCustomizations": {
|
"workbench.colorCustomizations": {
|
||||||
"titleBar.activeForeground": "#ebe8e8",
|
"titleBar.activeForeground": "#ebe8e8",
|
||||||
"titleBar.activeBackground": "#95ff0053"
|
"titleBar.activeBackground": "#95ff0053",
|
||||||
},
|
},
|
||||||
"terminal.integrated.wordSeparators": " ()[]{}',\"`─‘’"
|
"terminal.integrated.wordSeparators": " ()[]{}',\"`─‘’"
|
||||||
}
|
}
|
94
CHANGELOG.md
94
CHANGELOG.md
@ -1,97 +1,3 @@
|
|||||||
# [6.5.0](https://git.datacontroller.io/dc/dc/compare/v6.4.0...v6.5.0) (2024-01-26)
|
|
||||||
|
|
||||||
|
|
||||||
### Features
|
|
||||||
|
|
||||||
* filtering by reference to Variables as well as Values ([6eb1aa8](https://git.datacontroller.io/dc/dc/commit/6eb1aa85d29294d63e6af377e622fbed7fd1fab8))
|
|
||||||
|
|
||||||
# [6.4.0](https://git.datacontroller.io/dc/dc/compare/v6.3.1...v6.4.0) (2024-01-24)
|
|
||||||
|
|
||||||
|
|
||||||
### Bug Fixes
|
|
||||||
|
|
||||||
* add dcLib to globals ([5d93346](https://git.datacontroller.io/dc/dc/commit/5d93346b52eda27c2829770e96686a713296d373))
|
|
||||||
* add service to get xlmap rules and fixed interface name ([9ffa30a](https://git.datacontroller.io/dc/dc/commit/9ffa30ab747f5b62acbd452431a5e6e440afcb80))
|
|
||||||
* increasing length of mpe_excel_map cols to ([2d4d068](https://git.datacontroller.io/dc/dc/commit/2d4d068413dcdac98581f08939e74bde65b73428))
|
|
||||||
* providing info on mapids to FE ([fd94945](https://git.datacontroller.io/dc/dc/commit/fd94945466c1a797ddc89815258a65624a9cb0cf))
|
|
||||||
* removing tables from EDIT menu that are in xlmaps ([9550ae4](https://git.datacontroller.io/dc/dc/commit/9550ae4d1154a0272f8a2427ac9d2afdfd699c96))
|
|
||||||
* removing XLMAP_TARGETLIBDS from mpe_xlmaps_rules table ([93702c6](https://git.datacontroller.io/dc/dc/commit/93702c63dc280cdba1e46f0fd8fe0deaec879611))
|
|
||||||
* renaming TABLE macvar to LOAD_REF in postdata.sas ([01915a2](https://git.datacontroller.io/dc/dc/commit/01915a2db9a4dfb94e4e8213e2c32181da36d349))
|
|
||||||
* reverting xlmap in getdata change ([2d6e747](https://git.datacontroller.io/dc/dc/commit/2d6e747db9b84e9fb0dfcf9102a2f7dd2cb51891))
|
|
||||||
* update edit tab to load ([516e5a2](https://git.datacontroller.io/dc/dc/commit/516e5a206216f79ab1dce9f4eab0d31115743160))
|
|
||||||
|
|
||||||
|
|
||||||
### Features
|
|
||||||
|
|
||||||
* adding ability to define the target table for excel maps ([c86fba9](https://git.datacontroller.io/dc/dc/commit/c86fba9dc75ddc6033132f469ad1c31b9131b12e))
|
|
||||||
* adding ismap attribute to getdata response (and fixing test) ([2702bb3](https://git.datacontroller.io/dc/dc/commit/2702bb3c84c45903def1aa2b8cc20a6dd080281b))
|
|
||||||
* Complex Excel Uploads ([cf19381](https://git.datacontroller.io/dc/dc/commit/cf193810606f287b8d6f864c4eb64d43c5ab5f3c)), closes [#69](https://git.datacontroller.io/dc/dc/issues/69)
|
|
||||||
* Create Tables / Files dropdown under load tab ([b473b19](https://git.datacontroller.io/dc/dc/commit/b473b198a61f468dff74cd8e64692e7847084a80))
|
|
||||||
* display list of maps in sidebar ([5aec024](https://git.datacontroller.io/dc/dc/commit/5aec0242429942f8a989b5fb79f8d3865e9de01a))
|
|
||||||
* implemented the logic for xlmap component ([50696bb](https://git.datacontroller.io/dc/dc/commit/50696bb926dd00472db65a008771a4b6352871be))
|
|
||||||
* model changes for [#69](https://git.datacontroller.io/dc/dc/issues/69) ([271543a](https://git.datacontroller.io/dc/dc/commit/271543a446a2116718f99f0540e3cd911f9f5fe7))
|
|
||||||
* new getxlmaps service to return rules for a particular xlmap_id ([56264ec](https://git.datacontroller.io/dc/dc/commit/56264ecc6908bf6c8e3e666dfeba7068d6195df8))
|
|
||||||
* validating the excel map after stage (adding load-ref) ([a485c3b](https://git.datacontroller.io/dc/dc/commit/a485c3b78724a36f7bacb264fb02140cc62d6512))
|
|
||||||
|
|
||||||
## [6.3.1](https://git.datacontroller.io/dc/dc/compare/v6.3.0...v6.3.1) (2024-01-01)
|
|
||||||
|
|
||||||
|
|
||||||
### Bug Fixes
|
|
||||||
|
|
||||||
* enabling excel uploads to tables with retained keys, also adding more validation to MPE_TABLES updates ([3efccc4](https://git.datacontroller.io/dc/dc/commit/3efccc4cf3752763d049836724f2491c287f65db))
|
|
||||||
|
|
||||||
# [6.3.0](https://git.datacontroller.io/dc/dc/compare/v6.2.8...v6.3.0) (2023-12-04)
|
|
||||||
|
|
||||||
|
|
||||||
### Features
|
|
||||||
|
|
||||||
* viewer row handle ([dadac4f](https://git.datacontroller.io/dc/dc/commit/dadac4f13f85b5446198b6340cad28844defc94d))
|
|
||||||
|
|
||||||
## [6.2.8](https://git.datacontroller.io/dc/dc/compare/v6.2.7...v6.2.8) (2023-12-04)
|
|
||||||
|
|
||||||
|
|
||||||
### Bug Fixes
|
|
||||||
|
|
||||||
* bumping sasjs/core to fix mp_loadformat issue ([a1d308e](https://git.datacontroller.io/dc/dc/commit/a1d308ea078786b27bf7ec940d018fc657d4c398))
|
|
||||||
* new logic for -fc suffix. Closes [#63](https://git.datacontroller.io/dc/dc/issues/63) ([5579db0](https://git.datacontroller.io/dc/dc/commit/5579db0eafc668b1bc310099b7cc3062e0598fc4))
|
|
||||||
|
|
||||||
## [6.2.7](https://git.datacontroller.io/dc/dc/compare/v6.2.6...v6.2.7) (2023-11-09)
|
|
||||||
|
|
||||||
|
|
||||||
### Bug Fixes
|
|
||||||
|
|
||||||
* **audit:** updated crypto-js (hashing rows in dynamic cell validation) ([a7aa42a](https://git.datacontroller.io/dc/dc/commit/a7aa42a59b71597399924b8d2d06010c806321f3))
|
|
||||||
* missing dependency and avoiding label length limit issue ([91f128c](https://git.datacontroller.io/dc/dc/commit/91f128c2fead1e4f72267d689e67f49ec9a2ab35))
|
|
||||||
|
|
||||||
## [6.2.6](https://git.datacontroller.io/dc/dc/compare/v6.2.5...v6.2.6) (2023-10-18)
|
|
||||||
|
|
||||||
|
|
||||||
### Bug Fixes
|
|
||||||
|
|
||||||
* bumping core to address mm_assigndirectlib issue ([c27cdab](https://git.datacontroller.io/dc/dc/commit/c27cdab3fccbde814a29424d0344173a73ea816c))
|
|
||||||
|
|
||||||
## [6.2.5](https://git.datacontroller.io/dc/dc/compare/v6.2.4...v6.2.5) (2023-10-17)
|
|
||||||
|
|
||||||
|
|
||||||
### Bug Fixes
|
|
||||||
|
|
||||||
* enabling AUTHDOMAIN in MM_ASSIGNDIRECTLIB ([008b45a](https://git.datacontroller.io/dc/dc/commit/008b45ad175ec0e6026f5ef3bc210470226e328f))
|
|
||||||
|
|
||||||
## [6.2.4](https://git.datacontroller.io/dc/dc/compare/v6.2.3...v6.2.4) (2023-10-16)
|
|
||||||
|
|
||||||
|
|
||||||
### Bug Fixes
|
|
||||||
|
|
||||||
* Enable display of metadata-only tables. Closes [#56](https://git.datacontroller.io/dc/dc/issues/56) ([f3e82b4](https://git.datacontroller.io/dc/dc/commit/f3e82b4ee2a9c1c851f812ac60e9eaf05f91a0f9))
|
|
||||||
|
|
||||||
## [6.2.3](https://git.datacontroller.io/dc/dc/compare/v6.2.2...v6.2.3) (2023-10-12)
|
|
||||||
|
|
||||||
|
|
||||||
### Bug Fixes
|
|
||||||
|
|
||||||
* bumping core library to avoid non-ascii char in mp_validatecols.sas. [#50](https://git.datacontroller.io/dc/dc/issues/50) ([11b06f6](https://git.datacontroller.io/dc/dc/commit/11b06f6416300b6d70b1570c415d5a5c004976db))
|
|
||||||
* removing copyright symbol from mpe_alerts macro. [#50](https://git.datacontroller.io/dc/dc/issues/50) ([adb7eb7](https://git.datacontroller.io/dc/dc/commit/adb7eb77550c68a2dab15a6ff358129820e9b612))
|
|
||||||
|
|
||||||
## [6.2.2](https://git.datacontroller.io/dc/dc/compare/v6.2.1...v6.2.2) (2023-10-09)
|
## [6.2.2](https://git.datacontroller.io/dc/dc/compare/v6.2.1...v6.2.2) (2023-10-09)
|
||||||
|
|
||||||
|
|
||||||
|
@ -10,7 +10,7 @@ const check = (cwd) => {
|
|||||||
onlyAllow:
|
onlyAllow:
|
||||||
'AFLv2.1;Apache 2.0;Apache-2.0;Apache*;Artistic-2.0;0BSD;BSD*;BSD-2-Clause;BSD-3-Clause;CC0-1.0;CC-BY-3.0;CC-BY-4.0;ISC;MIT;MPL-2.0;ODC-By-1.0;Python-2.0;Unlicense;',
|
'AFLv2.1;Apache 2.0;Apache-2.0;Apache*;Artistic-2.0;0BSD;BSD*;BSD-2-Clause;BSD-3-Clause;CC0-1.0;CC-BY-3.0;CC-BY-4.0;ISC;MIT;MPL-2.0;ODC-By-1.0;Python-2.0;Unlicense;',
|
||||||
excludePackages:
|
excludePackages:
|
||||||
'@cds/city@1.1.0;@handsontable/angular@13.1.0;handsontable@13.1.0;hyperformula@2.5.0;jackspeak@2.2.0;path-scurry@1.7.0'
|
'@cds/city@1.1.0;@handsontable/angular@13.1.0;handsontable@13.1.0;hyperformula@2.6.0;jackspeak@2.2.0;path-scurry@1.7.0'
|
||||||
},
|
},
|
||||||
(error, json) => {
|
(error, json) => {
|
||||||
if (error) {
|
if (error) {
|
||||||
|
17558
client/package-lock.json
generated
17558
client/package-lock.json
generated
File diff suppressed because it is too large
Load Diff
@ -57,7 +57,7 @@
|
|||||||
"base64-arraybuffer": "^0.2.0",
|
"base64-arraybuffer": "^0.2.0",
|
||||||
"buffer": "^5.4.3",
|
"buffer": "^5.4.3",
|
||||||
"crypto-browserify": "3.12.0",
|
"crypto-browserify": "3.12.0",
|
||||||
"crypto-js": "^4.2.0",
|
"crypto-js": "^3.3.0",
|
||||||
"d3-graphviz": "^5.0.2",
|
"d3-graphviz": "^5.0.2",
|
||||||
"fs-extra": "^7.0.1",
|
"fs-extra": "^7.0.1",
|
||||||
"handsontable": "^13.1.0",
|
"handsontable": "^13.1.0",
|
||||||
@ -93,7 +93,7 @@
|
|||||||
"@compodoc/compodoc": "^1.1.21",
|
"@compodoc/compodoc": "^1.1.21",
|
||||||
"@cypress/webpack-preprocessor": "^5.17.1",
|
"@cypress/webpack-preprocessor": "^5.17.1",
|
||||||
"@types/core-js": "^2.5.5",
|
"@types/core-js": "^2.5.5",
|
||||||
"@types/crypto-js": "^4.2.1",
|
"@types/crypto-js": "^4.0.1",
|
||||||
"@types/es6-shim": "^0.31.39",
|
"@types/es6-shim": "^0.31.39",
|
||||||
"@types/jasmine": "~3.6.0",
|
"@types/jasmine": "~3.6.0",
|
||||||
"@types/lodash-es": "^4.17.3",
|
"@types/lodash-es": "^4.17.3",
|
||||||
|
@ -37,12 +37,6 @@ export const initFilter: { filter: FilterCache } = {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface XLMapListItem {
|
|
||||||
id: string
|
|
||||||
description: string
|
|
||||||
targetDS: string
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Cached filtering values across whole app (editor, viewer, viewboxes)
|
* Cached filtering values across whole app (editor, viewer, viewboxes)
|
||||||
* Cached lineage libraries, tables
|
* Cached lineage libraries, tables
|
||||||
@ -52,8 +46,6 @@ export interface XLMapListItem {
|
|||||||
*/
|
*/
|
||||||
export const globals: {
|
export const globals: {
|
||||||
rootParam: string
|
rootParam: string
|
||||||
dcLib: string
|
|
||||||
xlmaps: XLMapListItem[]
|
|
||||||
editor: any
|
editor: any
|
||||||
viewer: any
|
viewer: any
|
||||||
viewboxes: ViewboxCache
|
viewboxes: ViewboxCache
|
||||||
@ -65,13 +57,11 @@ export const globals: {
|
|||||||
[key: string]: any
|
[key: string]: any
|
||||||
} = {
|
} = {
|
||||||
rootParam: <string>'',
|
rootParam: <string>'',
|
||||||
dcLib: '',
|
|
||||||
xlmaps: [],
|
|
||||||
editor: {
|
editor: {
|
||||||
startupSet: <boolean>false,
|
startupSet: <boolean>false,
|
||||||
treeNodeLibraries: <any[] | null>[],
|
treeNodeLibraries: <any[] | null>[],
|
||||||
libsAndTables: <any[]>[],
|
libsAndTables: <any[]>[],
|
||||||
libraries: <string[] | undefined>[],
|
libraries: <String[] | undefined>[],
|
||||||
library: <string>'',
|
library: <string>'',
|
||||||
table: <string>'',
|
table: <string>'',
|
||||||
filter: <FilterCache>{
|
filter: <FilterCache>{
|
||||||
|
@ -168,7 +168,7 @@
|
|||||||
</button>
|
</button>
|
||||||
<clr-dropdown-menu *clrIfOpen clrPosition="bottom-left">
|
<clr-dropdown-menu *clrIfOpen clrPosition="bottom-left">
|
||||||
<a [routerLink]="['/view']" clrDropdownItem>VIEW</a>
|
<a [routerLink]="['/view']" clrDropdownItem>VIEW</a>
|
||||||
<a [routerLink]="['/home']" clrDropdownItem>LOAD</a>
|
<a [routerLink]="['/home']" clrDropdownItem>EDIT</a>
|
||||||
<a [routerLink]="['/review/submitted']" clrDropdownItem>REVIEW</a>
|
<a [routerLink]="['/review/submitted']" clrDropdownItem>REVIEW</a>
|
||||||
</clr-dropdown-menu>
|
</clr-dropdown-menu>
|
||||||
</clr-dropdown>
|
</clr-dropdown>
|
||||||
@ -189,7 +189,7 @@
|
|||||||
router.url.includes('edit-record') ||
|
router.url.includes('edit-record') ||
|
||||||
router.url.includes('home')
|
router.url.includes('home')
|
||||||
"
|
"
|
||||||
>LOAD</a
|
>EDIT</a
|
||||||
>
|
>
|
||||||
<a
|
<a
|
||||||
[routerLink]="['/review/submitted']"
|
[routerLink]="['/review/submitted']"
|
||||||
|
@ -4,19 +4,19 @@
|
|||||||
* The full license information can be found in LICENSE in the root directory of this project.
|
* The full license information can be found in LICENSE in the root directory of this project.
|
||||||
*/
|
*/
|
||||||
import { ModuleWithProviders } from '@angular/core'
|
import { ModuleWithProviders } from '@angular/core'
|
||||||
import { RouterModule, Routes } from '@angular/router'
|
import { Routes, RouterModule } from '@angular/router'
|
||||||
|
|
||||||
|
import { HomeComponent } from './home/home.component'
|
||||||
import { NotFoundComponent } from './not-found/not-found.component'
|
import { NotFoundComponent } from './not-found/not-found.component'
|
||||||
|
|
||||||
import { DeployModule } from './deploy/deploy.module'
|
|
||||||
import { EditorModule } from './editor/editor.module'
|
|
||||||
import { HomeModule } from './home/home.module'
|
|
||||||
import { LicensingModule } from './licensing/licensing.module'
|
|
||||||
import { ReviewModule } from './review/review.module'
|
|
||||||
import { ReviewRouteComponent } from './routes/review-route/review-route.component'
|
import { ReviewRouteComponent } from './routes/review-route/review-route.component'
|
||||||
import { StageModule } from './stage/stage.module'
|
import { StageModule } from './stage/stage.module'
|
||||||
import { SystemModule } from './system/system.module'
|
import { EditorModule } from './editor/editor.module'
|
||||||
import { ViewerModule } from './viewer/viewer.module'
|
import { ViewerModule } from './viewer/viewer.module'
|
||||||
|
import { ReviewModule } from './review/review.module'
|
||||||
|
import { DeployModule } from './deploy/deploy.module'
|
||||||
|
import { LicensingModule } from './licensing/licensing.module'
|
||||||
|
import { SystemModule } from './system/system.module'
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defining routes
|
* Defining routes
|
||||||
@ -45,7 +45,7 @@ export const ROUTES: Routes = [
|
|||||||
path: 'licensing',
|
path: 'licensing',
|
||||||
loadChildren: () => LicensingModule
|
loadChildren: () => LicensingModule
|
||||||
},
|
},
|
||||||
{ path: 'home', loadChildren: () => HomeModule },
|
{ path: 'home', component: HomeComponent },
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
* Load editor module with subroutes
|
* Load editor module with subroutes
|
||||||
|
@ -280,7 +280,7 @@
|
|||||||
licenceState.value.editor_rows_allowed === 1
|
licenceState.value.editor_rows_allowed === 1
|
||||||
? 'row'
|
? 'row'
|
||||||
: 'rows'
|
: 'rows'
|
||||||
}}, contact support@datacontroller.io</span
|
}}, contact support@datacontroller.io</span
|
||||||
>
|
>
|
||||||
</clr-tooltip-content>
|
</clr-tooltip-content>
|
||||||
</clr-tooltip>
|
</clr-tooltip>
|
||||||
@ -417,7 +417,7 @@
|
|||||||
licenceState.value.editor_rows_allowed === 1
|
licenceState.value.editor_rows_allowed === 1
|
||||||
? 'row'
|
? 'row'
|
||||||
: 'rows'
|
: 'rows'
|
||||||
}}, contact support@datacontroller.io</span
|
}}, contact support@datacontroller.io</span
|
||||||
>
|
>
|
||||||
</clr-tooltip-content>
|
</clr-tooltip-content>
|
||||||
</clr-tooltip>
|
</clr-tooltip>
|
||||||
@ -467,7 +467,7 @@
|
|||||||
: 'rows'
|
: 'rows'
|
||||||
}}
|
}}
|
||||||
will be submitted. To remove the restriction, contact
|
will be submitted. To remove the restriction, contact
|
||||||
support@datacontroller.io</span
|
support@datacontroller.io</span
|
||||||
>
|
>
|
||||||
<div *ngIf="tableTrue" class="clr-offset-md-2 clr-col-md-8">
|
<div *ngIf="tableTrue" class="clr-offset-md-2 clr-col-md-8">
|
||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
@ -528,7 +528,7 @@
|
|||||||
Due to current licence, only
|
Due to current licence, only
|
||||||
{{ licenceState.value.submit_rows_limit }} rows in a file will
|
{{ licenceState.value.submit_rows_limit }} rows in a file will
|
||||||
be submitted. To remove the restriction, contact
|
be submitted. To remove the restriction, contact
|
||||||
support@datacontroller.io
|
support@datacontroller.io
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
<div class="modal-footer">
|
<div class="modal-footer">
|
||||||
|
@ -38,7 +38,7 @@ import { HotTableInterface } from '../models/HotTable.interface'
|
|||||||
import {
|
import {
|
||||||
$DataFormats,
|
$DataFormats,
|
||||||
DSMeta,
|
DSMeta,
|
||||||
EditorsGetDataServiceResponse
|
EditorsGetdataServiceResponse
|
||||||
} from '../models/sas/editors-getdata.model'
|
} from '../models/sas/editors-getdata.model'
|
||||||
import { DataFormat } from '../models/sas/common/DateFormat'
|
import { DataFormat } from '../models/sas/common/DateFormat'
|
||||||
import SheetInfo from '../models/SheetInfo'
|
import SheetInfo from '../models/SheetInfo'
|
||||||
@ -2964,7 +2964,7 @@ export class EditorComponent implements OnInit, AfterViewInit {
|
|||||||
|
|
||||||
await this.sasStoreService
|
await this.sasStoreService
|
||||||
.callService(myParams, 'SASControlTable', 'editors/getdata', this.libds)
|
.callService(myParams, 'SASControlTable', 'editors/getdata', this.libds)
|
||||||
.then((res: EditorsGetDataServiceResponse) => {
|
.then((res: EditorsGetdataServiceResponse) => {
|
||||||
this.initSetup(res)
|
this.initSetup(res)
|
||||||
})
|
})
|
||||||
.catch((err: any) => {
|
.catch((err: any) => {
|
||||||
@ -2976,7 +2976,7 @@ export class EditorComponent implements OnInit, AfterViewInit {
|
|||||||
|
|
||||||
ngAfterViewInit() {}
|
ngAfterViewInit() {}
|
||||||
|
|
||||||
initSetup(response: EditorsGetDataServiceResponse) {
|
initSetup(response: EditorsGetdataServiceResponse) {
|
||||||
this.hotInstance = this.hotRegisterer.getInstance('hotInstance')
|
this.hotInstance = this.hotRegisterer.getInstance('hotInstance')
|
||||||
|
|
||||||
if (this.getdataError) return
|
if (this.getdataError) return
|
||||||
|
@ -1,23 +0,0 @@
|
|||||||
import { NgModule } from '@angular/core'
|
|
||||||
import { RouterModule, Routes } from '@angular/router'
|
|
||||||
import { HomeRouteComponent } from '../routes/home-route/home-route.component'
|
|
||||||
import { HomeComponent } from './home.component'
|
|
||||||
import { XLMapModule } from '../xlmap/xlmap.module'
|
|
||||||
|
|
||||||
const routes: Routes = [
|
|
||||||
{
|
|
||||||
path: '',
|
|
||||||
component: HomeRouteComponent,
|
|
||||||
children: [
|
|
||||||
{ path: '', pathMatch: 'full', redirectTo: 'tables' },
|
|
||||||
{ path: 'tables', component: HomeComponent },
|
|
||||||
{ path: 'files', loadChildren: () => XLMapModule }
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
|
|
||||||
@NgModule({
|
|
||||||
imports: [RouterModule.forChild(routes)],
|
|
||||||
exports: [RouterModule]
|
|
||||||
})
|
|
||||||
export class HomeRoutingModule {}
|
|
@ -100,7 +100,7 @@
|
|||||||
*clrIfOpen
|
*clrIfOpen
|
||||||
>
|
>
|
||||||
<span *ngIf="tableLocked">
|
<span *ngIf="tableLocked">
|
||||||
To unlock all tables, contact support@datacontroller.io
|
To unlock all tables, contact support@datacontroller.io
|
||||||
</span>
|
</span>
|
||||||
</clr-tooltip-content>
|
</clr-tooltip-content>
|
||||||
</clr-tooltip>
|
</clr-tooltip>
|
||||||
|
@ -1,18 +1,15 @@
|
|||||||
import { CommonModule } from '@angular/common'
|
|
||||||
import { NgModule } from '@angular/core'
|
import { NgModule } from '@angular/core'
|
||||||
import { FormsModule } from '@angular/forms'
|
import { CommonModule } from '@angular/common'
|
||||||
import { ClarityModule } from '@clr/angular'
|
|
||||||
import { AppSharedModule } from '../app-shared.module'
|
|
||||||
import { DirectivesModule } from '../directives/directives.module'
|
|
||||||
import { HomeRouteComponent } from '../routes/home-route/home-route.component'
|
|
||||||
import { DcTreeModule } from '../shared/dc-tree/dc-tree.module'
|
|
||||||
import { HomeRoutingModule } from './home-routing.module'
|
|
||||||
import { HomeComponent } from './home.component'
|
import { HomeComponent } from './home.component'
|
||||||
|
import { ClarityModule } from '@clr/angular'
|
||||||
|
import { FormsModule } from '@angular/forms'
|
||||||
|
import { AppSharedModule } from '../app-shared.module'
|
||||||
|
import { DcTreeModule } from '../shared/dc-tree/dc-tree.module'
|
||||||
|
import { DirectivesModule } from '../directives/directives.module'
|
||||||
|
|
||||||
@NgModule({
|
@NgModule({
|
||||||
declarations: [HomeComponent, HomeRouteComponent],
|
declarations: [HomeComponent],
|
||||||
imports: [
|
imports: [
|
||||||
HomeRoutingModule,
|
|
||||||
FormsModule,
|
FormsModule,
|
||||||
ClarityModule,
|
ClarityModule,
|
||||||
AppSharedModule,
|
AppSharedModule,
|
||||||
|
@ -656,7 +656,8 @@ export class LineageComponent {
|
|||||||
this.flatdata = res.flatdata
|
this.flatdata = res.flatdata
|
||||||
|
|
||||||
if (this.libraryList) {
|
if (this.libraryList) {
|
||||||
let libraryToSelect = this.libraryList.find((library: any) =>
|
let libraryToSelect = this.libraryList.find(
|
||||||
|
(library: any) =>
|
||||||
res.info[0]?.LIBURI?.toUpperCase()?.includes(
|
res.info[0]?.LIBURI?.toUpperCase()?.includes(
|
||||||
library?.LIBRARYID?.toUpperCase()
|
library?.LIBRARYID?.toUpperCase()
|
||||||
)
|
)
|
||||||
|
@ -6,7 +6,6 @@ export interface FilterClause {
|
|||||||
operators: string[]
|
operators: string[]
|
||||||
type: string
|
type: string
|
||||||
value: any
|
value: any
|
||||||
valueVariable: boolean
|
|
||||||
values: { formatted: string; unformatted: any }[]
|
values: { formatted: string; unformatted: any }[]
|
||||||
variable: string
|
variable: string
|
||||||
}
|
}
|
||||||
|
@ -4,12 +4,12 @@ import { DQData, SASParam } from '../TableData'
|
|||||||
import { BaseSASResponse } from './common/BaseSASResponse'
|
import { BaseSASResponse } from './common/BaseSASResponse'
|
||||||
import { DataFormat } from './common/DateFormat'
|
import { DataFormat } from './common/DateFormat'
|
||||||
|
|
||||||
export interface EditorsGetDataServiceResponse {
|
export interface EditorsGetdataServiceResponse {
|
||||||
data: EditorsGetDataSASResponse
|
data: EditorsGetdataSASResponse
|
||||||
libds: string
|
libds: string
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface EditorsGetDataSASResponse extends BaseSASResponse {
|
export interface EditorsGetdataSASResponse extends BaseSASResponse {
|
||||||
$sasdata: $DataFormats
|
$sasdata: $DataFormats
|
||||||
sasdata: Sasdata[]
|
sasdata: Sasdata[]
|
||||||
sasparams: SASParam[]
|
sasparams: SASParam[]
|
||||||
|
@ -413,10 +413,7 @@
|
|||||||
>
|
>
|
||||||
<app-soft-select
|
<app-soft-select
|
||||||
label="Value"
|
label="Value"
|
||||||
[secondLabel]="'Variable'"
|
|
||||||
[emitOnlySelected]="query.valueVariable"
|
|
||||||
[inputId]="'vals_' + queryIndex + '_' + clauseIndex"
|
[inputId]="'vals_' + queryIndex + '_' + clauseIndex"
|
||||||
(selectedLabelChange)="selectedLabelChange($event, query)"
|
|
||||||
[(value)]="query.value"
|
[(value)]="query.value"
|
||||||
[enableLoadMore]="query.nobs > query.values.length"
|
[enableLoadMore]="query.nobs > query.values.length"
|
||||||
(onInputEvent)="
|
(onInputEvent)="
|
||||||
@ -426,19 +423,9 @@
|
|||||||
onAutocompleteLoadingMore($event, query.variable, queryIndex, clauseIndex)
|
onAutocompleteLoadingMore($event, query.variable, queryIndex, clauseIndex)
|
||||||
"
|
"
|
||||||
>
|
>
|
||||||
<div *ngIf="!query.valueVariable">
|
|
||||||
<option [value]="column.unformatted" *ngFor="let column of query.values">
|
<option [value]="column.unformatted" *ngFor="let column of query.values">
|
||||||
{{ column.formatted.trim() }}
|
{{ column.formatted.trim() }}
|
||||||
</option>
|
</option>
|
||||||
</div>
|
|
||||||
|
|
||||||
<div *ngIf="query.valueVariable">
|
|
||||||
<ng-container *ngFor="let column of cols">
|
|
||||||
<option [value]="column.NAME" *ngIf="column.TYPE === query.type">
|
|
||||||
{{ column.NAME }}
|
|
||||||
</option>
|
|
||||||
</ng-container>
|
|
||||||
</div>
|
|
||||||
</app-soft-select>
|
</app-soft-select>
|
||||||
</ng-template>
|
</ng-template>
|
||||||
|
|
||||||
|
@ -95,7 +95,6 @@ export class QueryComponent
|
|||||||
variable: null,
|
variable: null,
|
||||||
operator: null,
|
operator: null,
|
||||||
value: null,
|
value: null,
|
||||||
valueVariable: false,
|
|
||||||
startrow: 0,
|
startrow: 0,
|
||||||
rows: 0,
|
rows: 0,
|
||||||
nobs: 0,
|
nobs: 0,
|
||||||
@ -194,20 +193,6 @@ export class QueryComponent
|
|||||||
*/
|
*/
|
||||||
usePickersChange() {
|
usePickersChange() {
|
||||||
this.queryDateTime = []
|
this.queryDateTime = []
|
||||||
if (this.usePickers) {
|
|
||||||
this.clauses.queryObj.forEach((queryObj: any) => {
|
|
||||||
queryObj.elements.forEach((element: any) => {
|
|
||||||
const isDateOrTime = ['DATETIME', 'TIME', 'DATE'].includes(
|
|
||||||
element.ddtype
|
|
||||||
)
|
|
||||||
|
|
||||||
if (isDateOrTime && element.valueVariable) {
|
|
||||||
element.value = ''
|
|
||||||
element.valueVariable = false
|
|
||||||
}
|
|
||||||
})
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -268,6 +253,8 @@ export class QueryComponent
|
|||||||
get(globals, objPath).filter.libds = this.libds
|
get(globals, objPath).filter.libds = this.libds
|
||||||
}
|
}
|
||||||
get(globals, objPath).filter.clauses = this.clauses
|
get(globals, objPath).filter.clauses = this.clauses
|
||||||
|
|
||||||
|
console.log('globals', globals)
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -763,12 +750,6 @@ export class QueryComponent
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
public selectedLabelChange(label: string, query: any) {
|
|
||||||
query.valueVariable = label === 'Variable'
|
|
||||||
query.value = ''
|
|
||||||
this.whereClauseFn()
|
|
||||||
}
|
|
||||||
|
|
||||||
public variableInputChange(
|
public variableInputChange(
|
||||||
queryVariable: any,
|
queryVariable: any,
|
||||||
index: number,
|
index: number,
|
||||||
|
@ -72,7 +72,7 @@
|
|||||||
>
|
>
|
||||||
To unlock more than
|
To unlock more than
|
||||||
{{ licenceState.value.history_rows_allowed }} records, contact
|
{{ licenceState.value.history_rows_allowed }} records, contact
|
||||||
support@datacontroller.io
|
support@datacontroller.io
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
@ -1 +0,0 @@
|
|||||||
<router-outlet></router-outlet>
|
|
@ -1,17 +0,0 @@
|
|||||||
import { Component, OnInit, OnDestroy } from '@angular/core'
|
|
||||||
|
|
||||||
@Component({
|
|
||||||
selector: 'app-home-route',
|
|
||||||
templateUrl: './home-route.component.html',
|
|
||||||
styleUrls: ['./home-route.component.scss'],
|
|
||||||
host: {
|
|
||||||
class: 'content-container'
|
|
||||||
}
|
|
||||||
})
|
|
||||||
export class HomeRouteComponent implements OnInit, OnDestroy {
|
|
||||||
constructor() {}
|
|
||||||
|
|
||||||
ngOnInit() {}
|
|
||||||
|
|
||||||
ngOnDestroy() {}
|
|
||||||
}
|
|
@ -1 +0,0 @@
|
|||||||
<router-outlet></router-outlet>
|
|
@ -1,17 +0,0 @@
|
|||||||
import { Component, OnInit, OnDestroy } from '@angular/core'
|
|
||||||
|
|
||||||
@Component({
|
|
||||||
selector: 'app-xlmap-route',
|
|
||||||
templateUrl: './xlmap-route.component.html',
|
|
||||||
styleUrls: ['./xlmap-route.component.scss'],
|
|
||||||
host: {
|
|
||||||
class: 'content-container'
|
|
||||||
}
|
|
||||||
})
|
|
||||||
export class XLMapRouteComponent implements OnInit, OnDestroy {
|
|
||||||
constructor() {}
|
|
||||||
|
|
||||||
ngOnInit() {}
|
|
||||||
|
|
||||||
ngOnDestroy() {}
|
|
||||||
}
|
|
@ -74,7 +74,6 @@ export class AppService {
|
|||||||
missingProps.push('Globvars')
|
missingProps.push('Globvars')
|
||||||
if (!res.sasdatasets) missingProps.push('Sasdatasets')
|
if (!res.sasdatasets) missingProps.push('Sasdatasets')
|
||||||
if (!res.saslibs) missingProps.push('Saslibs')
|
if (!res.saslibs) missingProps.push('Saslibs')
|
||||||
if (!res.xlmaps) missingProps.push('XLMaps')
|
|
||||||
|
|
||||||
if (missingProps.length > 0) {
|
if (missingProps.length > 0) {
|
||||||
startupServiceError = true
|
startupServiceError = true
|
||||||
@ -136,17 +135,10 @@ export class AppService {
|
|||||||
globals.editor.libsAndTables = libsAndTables
|
globals.editor.libsAndTables = libsAndTables
|
||||||
}
|
}
|
||||||
|
|
||||||
globals.xlmaps = res.xlmaps.map((xlmap: any) => ({
|
|
||||||
id: xlmap[0],
|
|
||||||
description: xlmap[1],
|
|
||||||
targetDS: xlmap[2]
|
|
||||||
}))
|
|
||||||
globals.editor.treeNodeLibraries = treeNodeLibraries
|
globals.editor.treeNodeLibraries = treeNodeLibraries
|
||||||
globals.editor.libraries = libraries
|
globals.editor.libraries = libraries
|
||||||
globals.editor.startupSet = true
|
globals.editor.startupSet = true
|
||||||
|
|
||||||
globals.dcLib = res.globvars[0].DCLIB
|
|
||||||
|
|
||||||
await this.licenceService.activation(res)
|
await this.licenceService.activation(res)
|
||||||
})
|
})
|
||||||
.catch((err: any) => {
|
.catch((err: any) => {
|
||||||
|
@ -10,8 +10,8 @@ import { globals } from '../_globals'
|
|||||||
import { FilterClause, FilterGroup, FilterQuery } from '../models/FilterQuery'
|
import { FilterClause, FilterGroup, FilterQuery } from '../models/FilterQuery'
|
||||||
import {
|
import {
|
||||||
$DataFormats,
|
$DataFormats,
|
||||||
EditorsGetDataSASResponse,
|
EditorsGetdataSASResponse,
|
||||||
EditorsGetDataServiceResponse
|
EditorsGetdataServiceResponse
|
||||||
} from '../models/sas/editors-getdata.model'
|
} from '../models/sas/editors-getdata.model'
|
||||||
import { LoggerService } from './logger.service'
|
import { LoggerService } from './logger.service'
|
||||||
import { isSpecialMissing } from '@sasjs/utils/input/validators'
|
import { isSpecialMissing } from '@sasjs/utils/input/validators'
|
||||||
@ -57,13 +57,13 @@ export class SasStoreService {
|
|||||||
libds: string
|
libds: string
|
||||||
) {
|
) {
|
||||||
this.libds = libds
|
this.libds = libds
|
||||||
const tables: any = {}
|
let tables: any = {}
|
||||||
tables[tableName] = [tableData]
|
tables[tableName] = [tableData]
|
||||||
const res: EditorsGetDataSASResponse = await this.sasService.request(
|
let res: EditorsGetdataSASResponse = await this.sasService.request(
|
||||||
program,
|
program,
|
||||||
tables
|
tables
|
||||||
)
|
)
|
||||||
const response: EditorsGetDataServiceResponse = {
|
let response: EditorsGetdataServiceResponse = {
|
||||||
data: res,
|
data: res,
|
||||||
libds: this.libds
|
libds: this.libds
|
||||||
}
|
}
|
||||||
@ -209,14 +209,6 @@ export class SasStoreService {
|
|||||||
return res
|
return res
|
||||||
}
|
}
|
||||||
|
|
||||||
public async getXLMapRules(id: string) {
|
|
||||||
const tables = {
|
|
||||||
getxlmaps_in: [{ XLMAP_ID: id }]
|
|
||||||
}
|
|
||||||
const res: any = await this.sasService.request('editors/getxlmaps', tables)
|
|
||||||
return res
|
|
||||||
}
|
|
||||||
|
|
||||||
public async getDetails(tableData: any, tableName: string, program: string) {
|
public async getDetails(tableData: any, tableName: string, program: string) {
|
||||||
let tables: any = {}
|
let tables: any = {}
|
||||||
tables[tableName] = [tableData]
|
tables[tableName] = [tableData]
|
||||||
@ -416,18 +408,14 @@ export class SasStoreService {
|
|||||||
for (let index = 0; index < clauses.queryObj.length; index++) {
|
for (let index = 0; index < clauses.queryObj.length; index++) {
|
||||||
let string = ''
|
let string = ''
|
||||||
let clause = clauses.queryObj[index]
|
let clause = clauses.queryObj[index]
|
||||||
|
|
||||||
for (let ind = 0; ind < clause.elements.length; ind++) {
|
for (let ind = 0; ind < clause.elements.length; ind++) {
|
||||||
let query = clause.elements[ind]
|
let query = clause.elements[ind]
|
||||||
|
|
||||||
if (ind < clause.elements.length - 1) {
|
if (ind < clause.elements.length - 1) {
|
||||||
opr = clause.clauseLogic
|
opr = clause.clauseLogic
|
||||||
} else {
|
} else {
|
||||||
opr = ''
|
opr = ''
|
||||||
}
|
}
|
||||||
|
|
||||||
let val: any
|
let val: any
|
||||||
|
|
||||||
for (let k = 0; k < query.values.length; k++) {
|
for (let k = 0; k < query.values.length; k++) {
|
||||||
if (
|
if (
|
||||||
typeof query.value === 'string' &&
|
typeof query.value === 'string' &&
|
||||||
@ -498,8 +486,6 @@ export class SasStoreService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
let type = query.type
|
let type = query.type
|
||||||
//if the value is variable, omit quotes in the 'where' string
|
|
||||||
const isValueVariable = query.valueVariable
|
|
||||||
let variable = query.variable === null ? '' : query.variable
|
let variable = query.variable === null ? '' : query.variable
|
||||||
let oper = query.operator === null ? '' : query.operator
|
let oper = query.operator === null ? '' : query.operator
|
||||||
// let value = val === null ? "''" : val;
|
// let value = val === null ? "''" : val;
|
||||||
@ -513,14 +499,10 @@ export class SasStoreService {
|
|||||||
if (type === 'char' && oper !== 'IN' && oper !== 'NOT IN') {
|
if (type === 'char' && oper !== 'IN' && oper !== 'NOT IN') {
|
||||||
if (typeof value === 'undefined') {
|
if (typeof value === 'undefined') {
|
||||||
value = ''
|
value = ''
|
||||||
}
|
value = " '" + value + "' "
|
||||||
|
|
||||||
if (isValueVariable) {
|
|
||||||
value = ' ' + value + ' ' //without quotes, with spaces
|
|
||||||
} else {
|
} else {
|
||||||
value = " '" + value + "' " //with quotes and spaces
|
value = " '" + value + "' "
|
||||||
}
|
}
|
||||||
|
|
||||||
string = string + ' ' + variable + ' ' + oper + value + opr
|
string = string + ' ' + variable + ' ' + oper + value + opr
|
||||||
} else {
|
} else {
|
||||||
if (type === 'num' && typeof value === 'undefined') {
|
if (type === 'num' && typeof value === 'undefined') {
|
||||||
@ -614,7 +596,7 @@ export class SasStoreService {
|
|||||||
rawValue = '.'
|
rawValue = '.'
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if (filterClause.type === 'char' && !filterClause.valueVariable) {
|
if (filterClause.type === 'char') {
|
||||||
rawValue = `'${filterClause.value.replace(/'/g, "''")}'`
|
rawValue = `'${filterClause.value.replace(/'/g, "''")}'`
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2,5 +2,5 @@
|
|||||||
[ngClass]="classes"
|
[ngClass]="classes"
|
||||||
[class.unset]="classes !== ''"
|
[class.unset]="classes !== ''"
|
||||||
href="mailto:support@datacontroller.io?subject=Licence"
|
href="mailto:support@datacontroller.io?subject=Licence"
|
||||||
>support@datacontroller.io</a
|
>support@datacontroller.io</a
|
||||||
>
|
>
|
||||||
|
@ -106,7 +106,7 @@
|
|||||||
*clrIfOpen
|
*clrIfOpen
|
||||||
>
|
>
|
||||||
<span *ngIf="tableLocked">
|
<span *ngIf="tableLocked">
|
||||||
To unlock all tables, contact support@datacontroller.io
|
To unlock all tables, contact support@datacontroller.io
|
||||||
</span>
|
</span>
|
||||||
</clr-tooltip-content>
|
</clr-tooltip-content>
|
||||||
|
|
||||||
|
@ -107,29 +107,7 @@
|
|||||||
</clr-tab-content>
|
</clr-tab-content>
|
||||||
</clr-tab>
|
</clr-tab>
|
||||||
</clr-tabs>
|
</clr-tabs>
|
||||||
|
<p *ngIf="isMainRoute('home')" class="page-title">Edit</p>
|
||||||
<div
|
|
||||||
*ngIf="isMainRoute('home')"
|
|
||||||
class="d-flex justify-content-center sub-dropdown"
|
|
||||||
>
|
|
||||||
<clr-dropdown>
|
|
||||||
<button class="dropdown-toggle btn btn-link" clrDropdownTrigger>
|
|
||||||
{{ getSubPage() }}
|
|
||||||
<clr-icon shape="caret down"></clr-icon>
|
|
||||||
</button>
|
|
||||||
<clr-dropdown-menu *clrIfOpen>
|
|
||||||
<a
|
|
||||||
clrVerticalNavLink
|
|
||||||
routerLink="/home/tables"
|
|
||||||
routerLinkActive="active"
|
|
||||||
>Tables</a
|
|
||||||
>
|
|
||||||
<a clrVerticalNavLink routerLink="/home/files" routerLinkActive="active"
|
|
||||||
>Files</a
|
|
||||||
>
|
|
||||||
</clr-dropdown-menu>
|
|
||||||
</clr-dropdown>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="nav-divider"></div>
|
<div class="nav-divider"></div>
|
||||||
|
|
||||||
|
@ -1,22 +1,4 @@
|
|||||||
<label
|
<label *ngIf="label" class="clr-control-label">{{ label }}</label>
|
||||||
*ngIf="label"
|
|
||||||
[class.secondLabelActive]="secondLabel && secondLabel.length > 0"
|
|
||||||
class="clr-control-label"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
(click)="onChangeLabel('first')"
|
|
||||||
[class.value-type-selected]="labelSelected === 'first'"
|
|
||||||
>{{ label }}</span
|
|
||||||
>
|
|
||||||
<ng-container *ngIf="secondLabel">
|
|
||||||
/
|
|
||||||
<span
|
|
||||||
(click)="onChangeLabel('second')"
|
|
||||||
[class.value-type-selected]="labelSelected === 'second'"
|
|
||||||
>{{ secondLabel }}</span
|
|
||||||
>
|
|
||||||
</ng-container>
|
|
||||||
</label>
|
|
||||||
<ng-container [ngSwitch]="type">
|
<ng-container [ngSwitch]="type">
|
||||||
<ng-container *ngSwitchCase="'date'">
|
<ng-container *ngSwitchCase="'date'">
|
||||||
<clr-date-container>
|
<clr-date-container>
|
||||||
|
@ -29,11 +29,3 @@ clr-date-container {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
label.secondLabelActive span {
|
|
||||||
&:not(.value-type-selected) {
|
|
||||||
text-decoration: line-through;
|
|
||||||
cursor: pointer;
|
|
||||||
opacity: 0.6;
|
|
||||||
}
|
|
||||||
}
|
|
@ -18,7 +18,6 @@ import { OnLoadingMoreEvent } from '../autocomplete/autocomplete.component'
|
|||||||
export class SoftSelectComponent implements OnInit, OnChanges {
|
export class SoftSelectComponent implements OnInit, OnChanges {
|
||||||
@Input() inputId: string = ''
|
@Input() inputId: string = ''
|
||||||
@Input() label: string | undefined
|
@Input() label: string | undefined
|
||||||
@Input() secondLabel: string | undefined
|
|
||||||
@Input() value: Date | string | null = ''
|
@Input() value: Date | string | null = ''
|
||||||
@Input() disabled: boolean = false
|
@Input() disabled: boolean = false
|
||||||
@Input() type: string = 'text'
|
@Input() type: string = 'text'
|
||||||
@ -31,25 +30,21 @@ export class SoftSelectComponent implements OnInit, OnChanges {
|
|||||||
@Output() focusinInput: EventEmitter<any> = new EventEmitter()
|
@Output() focusinInput: EventEmitter<any> = new EventEmitter()
|
||||||
@Output() onAutocompleteLoadingMore: EventEmitter<OnLoadingMoreEvent> =
|
@Output() onAutocompleteLoadingMore: EventEmitter<OnLoadingMoreEvent> =
|
||||||
new EventEmitter()
|
new EventEmitter()
|
||||||
@Output() selectedLabelChange: EventEmitter<string> = new EventEmitter()
|
|
||||||
|
|
||||||
@ViewChild('input') inputElement: any
|
@ViewChild('input') inputElement: any
|
||||||
|
|
||||||
temp: Date | string | null = ''
|
temp: Date | string | null = ''
|
||||||
inputFocused: boolean = false
|
inputFocused: boolean = false
|
||||||
|
|
||||||
labelSelected: LabelTypes = 'first'
|
|
||||||
|
|
||||||
constructor() {}
|
constructor() {}
|
||||||
|
|
||||||
ngOnChanges(changes: SimpleChanges): void {
|
ngOnChanges(changes: SimpleChanges): void {
|
||||||
if (
|
if (
|
||||||
changes.value &&
|
changes.value &&
|
||||||
changes.value.currentValue !== changes.value.previousValue
|
changes.value.currentValue !== changes.value.previousValue
|
||||||
) {
|
)
|
||||||
this.valueChange.emit(changes.value.currentValue)
|
this.valueChange.emit(changes.value.currentValue)
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
ngOnInit(): void {}
|
ngOnInit(): void {}
|
||||||
|
|
||||||
@ -90,14 +85,4 @@ export class SoftSelectComponent implements OnInit, OnChanges {
|
|||||||
onFocusinInput(event: any) {
|
onFocusinInput(event: any) {
|
||||||
this.focusinInput.emit(event)
|
this.focusinInput.emit(event)
|
||||||
}
|
}
|
||||||
|
|
||||||
onChangeLabel(label: LabelTypes) {
|
|
||||||
this.labelSelected = label
|
|
||||||
|
|
||||||
const selectedLabelText = label === 'first' ? this.label : this.secondLabel
|
|
||||||
|
|
||||||
this.selectedLabelChange.emit(selectedLabelText)
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
export type LabelTypes = 'first' | 'second'
|
|
||||||
|
@ -13,7 +13,7 @@
|
|||||||
class="licence-notice"
|
class="licence-notice"
|
||||||
>To unlock more then {{ licenceState.value.viewbox_limit }}
|
>To unlock more then {{ licenceState.value.viewbox_limit }}
|
||||||
{{ licenceState.value.viewbox_limit === 1 ? 'viewbox' : 'viewboxes' }},
|
{{ licenceState.value.viewbox_limit === 1 ? 'viewbox' : 'viewboxes' }},
|
||||||
contact support@datacontroller.io</span
|
contact support@datacontroller.io</span
|
||||||
>
|
>
|
||||||
</h3>
|
</h3>
|
||||||
|
|
||||||
|
@ -7,7 +7,6 @@ import { EventService } from '../services/event.service'
|
|||||||
import { AppService } from '../services/app.service'
|
import { AppService } from '../services/app.service'
|
||||||
import { HotTableInterface } from '../models/HotTable.interface'
|
import { HotTableInterface } from '../models/HotTable.interface'
|
||||||
import { LicenceService } from '../services/licence.service'
|
import { LicenceService } from '../services/licence.service'
|
||||||
import { globals } from '../_globals'
|
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'app-stage',
|
selector: 'app-stage',
|
||||||
@ -56,16 +55,8 @@ export class StageComponent implements OnInit {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public goBack() {
|
public goBack() {
|
||||||
const xlmap = globals.xlmaps.find(
|
|
||||||
(xlmap) => xlmap.targetDS === this.tableDetails.BASE_TABLE
|
|
||||||
)
|
|
||||||
if (xlmap) {
|
|
||||||
const id = this.hotTable.data[0].XLMAP_ID
|
|
||||||
this.route.navigateByUrl('/home/files/' + id)
|
|
||||||
} else {
|
|
||||||
this.route.navigateByUrl('/editor/' + this.tableDetails.BASE_TABLE)
|
this.route.navigateByUrl('/editor/' + this.tableDetails.BASE_TABLE)
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
public download(id: any) {
|
public download(id: any) {
|
||||||
let sasjsConfig = this.sasService.getSasjsConfig()
|
let sasjsConfig = this.sasService.getSasjsConfig()
|
||||||
|
@ -105,7 +105,7 @@
|
|||||||
*clrIfOpen
|
*clrIfOpen
|
||||||
>
|
>
|
||||||
<span *ngIf="tableLocked">
|
<span *ngIf="tableLocked">
|
||||||
To unlock all tables, contact support@datacontroller.io
|
To unlock all tables, contact support@datacontroller.io
|
||||||
</span>
|
</span>
|
||||||
</clr-tooltip-content>
|
</clr-tooltip-content>
|
||||||
</clr-tooltip>
|
</clr-tooltip>
|
||||||
@ -630,9 +630,6 @@
|
|||||||
[cells]="hotTable.cells"
|
[cells]="hotTable.cells"
|
||||||
[maxRows]="hotTable.maxRows"
|
[maxRows]="hotTable.maxRows"
|
||||||
[manualColumnResize]="true"
|
[manualColumnResize]="true"
|
||||||
[rowHeaders]="hotTable.rowHeaders"
|
|
||||||
[rowHeaderWidth]="hotTable.rowHeaderWidth"
|
|
||||||
[rowHeights]="hotTable.rowHeights"
|
|
||||||
[licenseKey]="hotTable.licenseKey"
|
[licenseKey]="hotTable.licenseKey"
|
||||||
>
|
>
|
||||||
</hot-table>
|
</hot-table>
|
||||||
|
@ -108,11 +108,6 @@ export class ViewerComponent implements AfterContentInit, AfterViewInit {
|
|||||||
settings: {},
|
settings: {},
|
||||||
afterGetColHeader: undefined,
|
afterGetColHeader: undefined,
|
||||||
licenseKey: undefined,
|
licenseKey: undefined,
|
||||||
rowHeaders: (index: number) => {
|
|
||||||
return ' '
|
|
||||||
},
|
|
||||||
rowHeaderWidth: 15,
|
|
||||||
rowHeights: 20,
|
|
||||||
contextMenu: ['copy_with_column_headers', 'copy_column_headers_only'],
|
contextMenu: ['copy_with_column_headers', 'copy_column_headers_only'],
|
||||||
copyPaste: {
|
copyPaste: {
|
||||||
copyColumnHeaders: true,
|
copyColumnHeaders: true,
|
||||||
|
@ -1,159 +0,0 @@
|
|||||||
import {
|
|
||||||
extractRowAndCol,
|
|
||||||
getCellAddress,
|
|
||||||
getFinishingCell,
|
|
||||||
isBlankRow
|
|
||||||
} from '../utils/xl.utils'
|
|
||||||
|
|
||||||
describe('isBlankRow', () => {
|
|
||||||
it('should return true for a blank row', () => {
|
|
||||||
const blankRow = { __rowNum__: 1 }
|
|
||||||
expect(isBlankRow(blankRow)).toBeTrue()
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should return false for a non-blank row', () => {
|
|
||||||
const nonBlankRow = {
|
|
||||||
B: 3,
|
|
||||||
C: 'some value',
|
|
||||||
D: -203
|
|
||||||
}
|
|
||||||
expect(isBlankRow(nonBlankRow)).toBeFalse()
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
describe('extractRowAndCol', () => {
|
|
||||||
it('should extract row and column from "MATCH F R[2]C[0]: CASH BALANCE"', () => {
|
|
||||||
const input = 'MATCH F R[2]C[0]: CASH BALANCE'
|
|
||||||
const result = extractRowAndCol(input)
|
|
||||||
expect(result).toEqual({ row: 2, column: 0 })
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should extract row and column from "RELATIVE R[10]C[6]"', () => {
|
|
||||||
const input = 'RELATIVE R[10]C[6]'
|
|
||||||
const result = extractRowAndCol(input)
|
|
||||||
expect(result).toEqual({ row: 10, column: 6 })
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should return null for invalid input', () => {
|
|
||||||
const invalidInput = 'INVALID INPUT'
|
|
||||||
const result = extractRowAndCol(invalidInput)
|
|
||||||
expect(result).toBeNull()
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
describe('getCellAddress', () => {
|
|
||||||
const arrayOfObjects = [
|
|
||||||
{ A: 'valueA1', B: 'valueB1' },
|
|
||||||
{ A: 'valueA2', B: 'valueB2' }
|
|
||||||
]
|
|
||||||
|
|
||||||
it('should convert "ABSOLUTE D8" to A1-style address', () => {
|
|
||||||
const input = 'ABSOLUTE D8'
|
|
||||||
const result = getCellAddress(input, arrayOfObjects)
|
|
||||||
expect(result).toBe('D8')
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should convert "RELATIVE R[10]C[6]" to A1-style address', () => {
|
|
||||||
const input = 'RELATIVE R[10]C[6]'
|
|
||||||
const result = getCellAddress(input, arrayOfObjects)
|
|
||||||
expect(result).toBe('F10')
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should convert "MATCH 1 R[0]C[0]:valueA1" to A1-style address', () => {
|
|
||||||
const input = 'MATCH 1 R[0]C[0]:valueA1'
|
|
||||||
const result = getCellAddress(input, arrayOfObjects)
|
|
||||||
expect(result).toBe('A1')
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should convert "MATCH A R[0]C[0]:valueA1" to A1-style address', () => {
|
|
||||||
const input = 'MATCH A R[0]C[0]:valueA1'
|
|
||||||
const result = getCellAddress(input, arrayOfObjects)
|
|
||||||
expect(result).toBe('A1')
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should convert "MATCH 1 R[1]C[0]:valueA1" to A1-style address', () => {
|
|
||||||
const input = 'MATCH 1 R[1]C[0]:valueA1'
|
|
||||||
const result = getCellAddress(input, arrayOfObjects)
|
|
||||||
expect(result).toBe('A2')
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should convert "MATCH A R[0]C[1]:valueA1" to A1-style address', () => {
|
|
||||||
const input = 'MATCH A R[0]C[1]:valueA1'
|
|
||||||
const result = getCellAddress(input, arrayOfObjects)
|
|
||||||
expect(result).toBe('B1')
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should convert "MATCH 1 R[1]C[1]:valueA1" to A1-style address', () => {
|
|
||||||
const input = 'MATCH 1 R[1]C[1]:valueA1'
|
|
||||||
const result = getCellAddress(input, arrayOfObjects)
|
|
||||||
expect(result).toBe('B2')
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should convert "MATCH A R[1]C[1]:valueA1" to A1-style address', () => {
|
|
||||||
const input = 'MATCH A R[1]C[1]:valueA1'
|
|
||||||
const result = getCellAddress(input, arrayOfObjects)
|
|
||||||
expect(result).toBe('B2')
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
describe('getFinishingCell', () => {
|
|
||||||
const arrayOfObjects = [
|
|
||||||
{ A: 'valueA1', B: 'valueB1' },
|
|
||||||
{ A: 'valueA2', B: 'valueB2' },
|
|
||||||
{ A: 'valueA3', B: 'valueB3' },
|
|
||||||
{ B: 'valueB4' },
|
|
||||||
{ A: 'valueA5' },
|
|
||||||
{ A: 'valueA6', B: 'valueB6' },
|
|
||||||
{},
|
|
||||||
{ A: 'valueA8' }
|
|
||||||
]
|
|
||||||
|
|
||||||
it('should return the start cell if finish is an empty string', () => {
|
|
||||||
const start = 'A1'
|
|
||||||
const finish = ''
|
|
||||||
const result = getFinishingCell(start, finish, arrayOfObjects)
|
|
||||||
expect(result).toBe(start)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should convert "ABSOLUTE D8" to A1-style address', () => {
|
|
||||||
const start = 'A1'
|
|
||||||
const finish = 'ABSOLUTE D8'
|
|
||||||
const result = getFinishingCell(start, finish, arrayOfObjects)
|
|
||||||
expect(result).toBe('D8')
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should convert "RELATIVE R[2]C[1]" to A1-style address', () => {
|
|
||||||
const start = 'A1'
|
|
||||||
const finish = 'RELATIVE R[2]C[1]'
|
|
||||||
const result = getFinishingCell(start, finish, arrayOfObjects)
|
|
||||||
expect(result).toBe('B3')
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should convert "MATCH A R[0]C[1]:valueA1" to A1-style address', () => {
|
|
||||||
const start = 'A1'
|
|
||||||
const finish = 'MATCH A R[0]C[1]:valueA1'
|
|
||||||
const result = getFinishingCell(start, finish, arrayOfObjects)
|
|
||||||
expect(result).toBe('B1')
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should convert "MATCH 1 R[4]C[0]:valueB1" to A1-style address', () => {
|
|
||||||
const start = 'A1'
|
|
||||||
const finish = 'MATCH 1 R[4]C[0]:valueB1'
|
|
||||||
const result = getFinishingCell(start, finish, arrayOfObjects)
|
|
||||||
expect(result).toBe('B5')
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should convert "LASTDOWN" to A1-style address of the last non-blank cell in column A', () => {
|
|
||||||
const start = 'A1'
|
|
||||||
const finish = 'LASTDOWN'
|
|
||||||
const result = getFinishingCell(start, finish, arrayOfObjects)
|
|
||||||
expect(result).toBe('A3')
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should convert "BLANKROW" to A1-style address of the last row with blank cells', () => {
|
|
||||||
const start = 'A1'
|
|
||||||
const finish = 'BLANKROW'
|
|
||||||
const result = getFinishingCell(start, finish, arrayOfObjects)
|
|
||||||
expect(result).toBe('B6')
|
|
||||||
})
|
|
||||||
})
|
|
@ -1,31 +0,0 @@
|
|||||||
export const blobToFile = (blob: Blob, fileName: string): File => {
|
|
||||||
const file = new File([blob], fileName, {
|
|
||||||
lastModified: new Date().getTime()
|
|
||||||
})
|
|
||||||
return file
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Convert an array of bytes (Uint8Array) to a binary string.
|
|
||||||
* @param {Uint8Array} res - The array of bytes to convert.
|
|
||||||
* @returns {string} The binary string representation of the array of bytes.
|
|
||||||
*/
|
|
||||||
export const byteArrayToBinaryString = (res: Uint8Array): string => {
|
|
||||||
// Create a Uint8Array from the input array (if it's not already)
|
|
||||||
const bytes = new Uint8Array(res)
|
|
||||||
|
|
||||||
// Initialize an empty string to store the binary representation
|
|
||||||
let binary = ''
|
|
||||||
|
|
||||||
// Get the length of the byte array
|
|
||||||
const length = bytes.byteLength
|
|
||||||
|
|
||||||
// Iterate through each byte in the array
|
|
||||||
for (let i = 0; i < length; i++) {
|
|
||||||
// Convert each byte to its binary representation and append to the string
|
|
||||||
binary += String.fromCharCode(bytes[i])
|
|
||||||
}
|
|
||||||
|
|
||||||
// Return the binary string
|
|
||||||
return binary
|
|
||||||
}
|
|
@ -1,225 +0,0 @@
|
|||||||
import * as XLSX from '@sheet/crypto'
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Checks if an excel row is blank or not
|
|
||||||
*
|
|
||||||
* @param row object is of shape {[key: string]: any}
|
|
||||||
*/
|
|
||||||
export const isBlankRow = (row: any) => {
|
|
||||||
for (const key in row) {
|
|
||||||
if (key !== '__rowNum__') {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Extracts row and column number from xlmap rule.
|
|
||||||
*
|
|
||||||
* Input string should be in form of
|
|
||||||
* either "MATCH F R[2]C[0]: CASH BALANCE" or "RELATIVE R[10]C[6]"
|
|
||||||
*/
|
|
||||||
export const extractRowAndCol = (str: string) => {
|
|
||||||
// Regular expression to match and capture the values inside square brackets
|
|
||||||
const regex = /R\[(\d+)\]C\[(\d+)\]/
|
|
||||||
|
|
||||||
// Match the regular expression against the input string
|
|
||||||
const match = str.match(regex)
|
|
||||||
|
|
||||||
if (!match) {
|
|
||||||
return null
|
|
||||||
}
|
|
||||||
|
|
||||||
// Extract values from the match groups
|
|
||||||
const row = parseInt(match[1], 10)
|
|
||||||
const column = parseInt(match[2], 10)
|
|
||||||
|
|
||||||
return {
|
|
||||||
row,
|
|
||||||
column
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Generate an A1-Style excel cell address from xlmap rule.
|
|
||||||
*
|
|
||||||
* Expect "ABSOLUTE D8" or "RELATIVE R[10]C[6]" or
|
|
||||||
* "MATCH C R[0]C[4]:Common Equity Tier 1 (CET1)" kinds of string as rule input
|
|
||||||
*/
|
|
||||||
export const getCellAddress = (rule: string, arrayOfObjects: any[]) => {
|
|
||||||
if (rule.startsWith('ABSOLUTE ')) {
|
|
||||||
rule = rule.replace('ABSOLUTE ', '')
|
|
||||||
}
|
|
||||||
|
|
||||||
if (rule.startsWith('RELATIVE ')) {
|
|
||||||
const rowAndCol = extractRowAndCol(rule)
|
|
||||||
|
|
||||||
if (rowAndCol) {
|
|
||||||
const { row, column } = rowAndCol
|
|
||||||
|
|
||||||
// Generate an A1-Style address string from a SheetJS cell address
|
|
||||||
// Spreadsheet applications typically display ordinal row numbers,
|
|
||||||
// where 1 is the first row, 2 is the second row, etc. The numbering starts at 1.
|
|
||||||
// SheetJS follows JavaScript counting conventions,
|
|
||||||
// where 0 is the first row, 1 is the second row, etc. The numbering starts at 0.
|
|
||||||
// Therefore, we have to subtract 1 from row and column to match SheetJS indexing convention
|
|
||||||
rule = XLSX.utils.encode_cell({ r: row - 1, c: column - 1 })
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (rule.startsWith('MATCH ')) {
|
|
||||||
let targetValue = ''
|
|
||||||
|
|
||||||
// using a regular expression to match "C[x]:" and extract the value after it
|
|
||||||
const match = rule.match(/C\[\d+\]:(.+)/)
|
|
||||||
|
|
||||||
// Check if there is a match
|
|
||||||
if (match) {
|
|
||||||
// Extract the value after "C[x]:"
|
|
||||||
targetValue = match[1]
|
|
||||||
}
|
|
||||||
|
|
||||||
// Split the string by spaces to get target row/column
|
|
||||||
const splittedArray = rule.split(' ')
|
|
||||||
|
|
||||||
// Extract the second word
|
|
||||||
const secondWord = splittedArray[1]
|
|
||||||
|
|
||||||
let targetColumn = ''
|
|
||||||
let targetRow = -1
|
|
||||||
let cellAddress = ''
|
|
||||||
|
|
||||||
// Check if the secondWord is a number
|
|
||||||
if (!isNaN(Number(secondWord))) {
|
|
||||||
targetRow = parseInt(secondWord)
|
|
||||||
} else {
|
|
||||||
targetColumn = secondWord
|
|
||||||
}
|
|
||||||
|
|
||||||
if (targetRow !== -1) {
|
|
||||||
// sheetJS index starts from 0,
|
|
||||||
// therefore, decremented 1 to make it correct row address for js array
|
|
||||||
const row = arrayOfObjects[targetRow - 1]
|
|
||||||
for (const col in row) {
|
|
||||||
if (col !== '__rowNum__' && row[col] === targetValue) {
|
|
||||||
cellAddress = col + targetRow
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
for (let i = 0; i < arrayOfObjects.length; i++) {
|
|
||||||
const row = arrayOfObjects[i]
|
|
||||||
if (row[targetColumn] === targetValue) {
|
|
||||||
// sheetJS index starts from 0,
|
|
||||||
// therefore, incremented 1 to make it correct row address
|
|
||||||
const rowIndex = i + 1
|
|
||||||
cellAddress = targetColumn + rowIndex
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Converts A1 cell address to 0-indexed form
|
|
||||||
const matchedCellAddress = XLSX.utils.decode_cell(cellAddress)
|
|
||||||
|
|
||||||
// extract number of rows and columns that we have to move
|
|
||||||
// from matched cell to reach target cell
|
|
||||||
const rowAndCol = extractRowAndCol(rule)
|
|
||||||
|
|
||||||
if (rowAndCol) {
|
|
||||||
const { row, column } = rowAndCol
|
|
||||||
|
|
||||||
// Converts 0-indexed cell address to A1 form
|
|
||||||
rule = XLSX.utils.encode_cell({
|
|
||||||
r: matchedCellAddress.r + row,
|
|
||||||
c: matchedCellAddress.c + column
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return rule
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Generate an A1-Style excel cell address for last cell
|
|
||||||
*
|
|
||||||
* @param start A1 style excel cell address
|
|
||||||
* @param finish XLMAP_FINISH attribute of xlmap rule
|
|
||||||
* @param arrayOfObjects an array of row objects
|
|
||||||
* @returns
|
|
||||||
*/
|
|
||||||
export const getFinishingCell = (
|
|
||||||
start: string,
|
|
||||||
finish: string,
|
|
||||||
arrayOfObjects: any[]
|
|
||||||
) => {
|
|
||||||
// in this case an individual cell would be extracted
|
|
||||||
if (finish === '') {
|
|
||||||
return start
|
|
||||||
}
|
|
||||||
|
|
||||||
if (finish.startsWith('ABSOLUTE ')) {
|
|
||||||
finish = finish.replace('ABSOLUTE ', '')
|
|
||||||
}
|
|
||||||
|
|
||||||
if (finish.startsWith('RELATIVE ')) {
|
|
||||||
const rowAndCol = extractRowAndCol(finish)
|
|
||||||
if (rowAndCol) {
|
|
||||||
const { row, column } = rowAndCol
|
|
||||||
|
|
||||||
const { r, c } = XLSX.utils.decode_cell(start)
|
|
||||||
|
|
||||||
// finish is relative to starting point
|
|
||||||
// therefore, we need to add extracted row and columns
|
|
||||||
// in starting cell address to get actual finishing cell
|
|
||||||
finish = XLSX.utils.encode_cell({ r: r + row, c: c + column })
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (finish.startsWith('MATCH ')) {
|
|
||||||
finish = getCellAddress(finish, arrayOfObjects)
|
|
||||||
}
|
|
||||||
|
|
||||||
if (finish === 'LASTDOWN') {
|
|
||||||
const { r, c } = XLSX.utils.decode_cell(start)
|
|
||||||
const colName = XLSX.utils.encode_col(c)
|
|
||||||
let lastNonBlank = r
|
|
||||||
for (let i = r + 1; i < arrayOfObjects.length; i++) {
|
|
||||||
const row = arrayOfObjects[i]
|
|
||||||
if (!row[colName]) {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
lastNonBlank = i
|
|
||||||
}
|
|
||||||
finish = colName + (lastNonBlank + 1) // excel numbering starts from 1. So incremented 1 to 0 based index
|
|
||||||
}
|
|
||||||
|
|
||||||
if (finish === 'BLANKROW') {
|
|
||||||
const { r } = XLSX.utils.decode_cell(start)
|
|
||||||
let lastNonBlankRow = r
|
|
||||||
for (let i = r + 1; i < arrayOfObjects.length; i++) {
|
|
||||||
const row = arrayOfObjects[i]
|
|
||||||
if (isBlankRow(row)) {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
lastNonBlankRow = i
|
|
||||||
}
|
|
||||||
const row = arrayOfObjects[lastNonBlankRow]
|
|
||||||
|
|
||||||
// Get the keys of the object (excluding '__rowNum__')
|
|
||||||
const keys = Object.keys(row).filter((key) => key !== '__rowNum__')
|
|
||||||
|
|
||||||
// Finding last column in a row
|
|
||||||
// Find the key with the highest alphanumeric value (assumes keys are letters)
|
|
||||||
const lastColumn = keys.reduce(
|
|
||||||
(maxKey, currentKey) => (currentKey > maxKey ? currentKey : maxKey),
|
|
||||||
''
|
|
||||||
)
|
|
||||||
|
|
||||||
// make finishing cell address in A1 style
|
|
||||||
finish = lastColumn + (lastNonBlankRow + 1) // excel numbering starts from 1. So incremented 1 to 0 based index
|
|
||||||
}
|
|
||||||
|
|
||||||
return finish
|
|
||||||
}
|
|
@ -1,22 +0,0 @@
|
|||||||
import { NgModule } from '@angular/core'
|
|
||||||
import { RouterModule, Routes } from '@angular/router'
|
|
||||||
|
|
||||||
import { XLMapComponent } from '../xlmap/xlmap.component'
|
|
||||||
import { XLMapRouteComponent } from '../routes/xlmap-route/xlmap-route.component'
|
|
||||||
|
|
||||||
const routes: Routes = [
|
|
||||||
{
|
|
||||||
path: '',
|
|
||||||
component: XLMapRouteComponent,
|
|
||||||
children: [
|
|
||||||
{ path: '', component: XLMapComponent },
|
|
||||||
{ path: ':id', component: XLMapComponent }
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
|
|
||||||
@NgModule({
|
|
||||||
imports: [RouterModule.forChild(routes)],
|
|
||||||
exports: [RouterModule]
|
|
||||||
})
|
|
||||||
export class XLMapRoutingModule {}
|
|
@ -1,252 +0,0 @@
|
|||||||
<app-sidebar>
|
|
||||||
<div *ngIf="xlmapsLoading" class="my-10-mx-auto text-center">
|
|
||||||
<clr-spinner clrMedium></clr-spinner>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<clr-tree>
|
|
||||||
<clr-tree-node class="search-node">
|
|
||||||
<div class="tree-search-wrapper">
|
|
||||||
<input
|
|
||||||
clrInput
|
|
||||||
#searchXLMapTreeInput
|
|
||||||
placeholder="Filter by Id"
|
|
||||||
name="input"
|
|
||||||
[(ngModel)]="searchString"
|
|
||||||
(keyup)="xlmapListOnFilter()"
|
|
||||||
autocomplete="off"
|
|
||||||
/>
|
|
||||||
<clr-icon
|
|
||||||
*ngIf="searchXLMapTreeInput.value.length < 1"
|
|
||||||
shape="search"
|
|
||||||
></clr-icon>
|
|
||||||
<clr-icon
|
|
||||||
*ngIf="searchXLMapTreeInput.value.length > 0"
|
|
||||||
(click)="searchString = ''; xlmapListOnFilter()"
|
|
||||||
shape="times"
|
|
||||||
></clr-icon>
|
|
||||||
</div>
|
|
||||||
</clr-tree-node>
|
|
||||||
|
|
||||||
<ng-container *ngFor="let xlmap of xlmaps">
|
|
||||||
<clr-tree-node>
|
|
||||||
<button
|
|
||||||
(click)="xlmapOnClick(xlmap)"
|
|
||||||
class="clr-treenode-link"
|
|
||||||
[class.table-active]="isActiveXLMap(xlmap.id)"
|
|
||||||
>
|
|
||||||
<clr-icon shape="file"></clr-icon>
|
|
||||||
{{ xlmap.id }}
|
|
||||||
</button>
|
|
||||||
</clr-tree-node>
|
|
||||||
</ng-container>
|
|
||||||
</clr-tree>
|
|
||||||
</app-sidebar>
|
|
||||||
|
|
||||||
<div class="content-area">
|
|
||||||
<div *ngIf="!selectedXLMap" class="no-table-selected">
|
|
||||||
<clr-icon
|
|
||||||
shape="warning-standard"
|
|
||||||
size="60"
|
|
||||||
class="is-info icon-dc-fill"
|
|
||||||
></clr-icon>
|
|
||||||
<h3 *ngIf="xlmaps.length > 0" class="text-center color-gray">
|
|
||||||
Please select a map
|
|
||||||
</h3>
|
|
||||||
<h3 *ngIf="xlmaps.length < 1" class="text-center color-gray">
|
|
||||||
No excel map is found
|
|
||||||
</h3>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="loadingSpinner" *ngIf="isLoading">
|
|
||||||
<span class="spinner"> Loading... </span>
|
|
||||||
<div>
|
|
||||||
<h4>{{ isLoadingDesc }}</h4>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div
|
|
||||||
appDragNdrop
|
|
||||||
(fileDraggedOver)="onShowUploadModal()"
|
|
||||||
class="card h-100 d-flex clr-flex-column"
|
|
||||||
*ngIf="!isLoading && selectedXLMap"
|
|
||||||
>
|
|
||||||
<clr-tabs>
|
|
||||||
<clr-tab>
|
|
||||||
<button clrTabLink (click)="selectedTab = TabsEnum.Rules">Rules</button>
|
|
||||||
<clr-tab-content *clrIfActive="selectedTab === TabsEnum.Rules">
|
|
||||||
</clr-tab-content>
|
|
||||||
</clr-tab>
|
|
||||||
<clr-tab>
|
|
||||||
<button clrTabLink (click)="selectedTab = TabsEnum.Data">Data</button>
|
|
||||||
<clr-tab-content *clrIfActive="selectedTab === TabsEnum.Data">
|
|
||||||
</clr-tab-content>
|
|
||||||
</clr-tab>
|
|
||||||
</clr-tabs>
|
|
||||||
|
|
||||||
<ng-container *ngTemplateOutlet="actionButtons"></ng-container>
|
|
||||||
|
|
||||||
<div class="clr-row m-0 mb-10-i viewerTitle">
|
|
||||||
<h3 class="d-flex clr-col-12 clr-justify-content-center mt-5-i">
|
|
||||||
{{ selectedXLMap.id }}
|
|
||||||
</h3>
|
|
||||||
<i class="d-flex clr-col-12 clr-justify-content-center mt-5-i">{{
|
|
||||||
selectedXLMap.description
|
|
||||||
}}</i>
|
|
||||||
<h5 class="d-flex clr-col-12 clr-justify-content-center mt-5-i">
|
|
||||||
Rules Source:
|
|
||||||
<a class="ml-10" [routerLink]="'/view/data/' + rulesSource">
|
|
||||||
{{ rulesSource }}
|
|
||||||
</a>
|
|
||||||
</h5>
|
|
||||||
<h5 class="d-flex clr-col-12 clr-justify-content-center mt-5-i">
|
|
||||||
Target dataset:
|
|
||||||
<a class="ml-10" [routerLink]="'/view/data/' + selectedXLMap.targetDS">
|
|
||||||
{{ selectedXLMap.targetDS }}
|
|
||||||
</a>
|
|
||||||
</h5>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="clr-flex-1">
|
|
||||||
<hot-table
|
|
||||||
hotId="hotInstance"
|
|
||||||
id="hot-table"
|
|
||||||
[multiColumnSorting]="true"
|
|
||||||
[viewportRowRenderingOffset]="50"
|
|
||||||
[data]="selectedTab === TabsEnum.Rules ? xlmapRules : xlData"
|
|
||||||
[colHeaders]="
|
|
||||||
selectedTab === TabsEnum.Rules ? xlmapRulesHeaders : xlUploadHeader
|
|
||||||
"
|
|
||||||
[columns]="
|
|
||||||
selectedTab === TabsEnum.Rules ? xlmapRulesColumns : xlUploadColumns
|
|
||||||
"
|
|
||||||
[filters]="true"
|
|
||||||
[height]="'100%'"
|
|
||||||
stretchH="all"
|
|
||||||
[modifyColWidth]="maxWidthChecker"
|
|
||||||
[cells]="getCellConfiguration"
|
|
||||||
[maxRows]="hotTableMaxRows"
|
|
||||||
[manualColumnResize]="true"
|
|
||||||
[rowHeaders]="rowHeaders"
|
|
||||||
[rowHeaderWidth]="15"
|
|
||||||
[rowHeights]="20"
|
|
||||||
[licenseKey]="hotTableLicenseKey"
|
|
||||||
>
|
|
||||||
</hot-table>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<clr-modal
|
|
||||||
appFileDrop
|
|
||||||
(fileOver)="fileOverBase($event)"
|
|
||||||
(fileDrop)="getFileDesc($event, true)"
|
|
||||||
[uploader]="uploader"
|
|
||||||
[clrModalSize]="'xl'"
|
|
||||||
[clrModalStaticBackdrop]="false"
|
|
||||||
[clrModalClosable]="true"
|
|
||||||
[(clrModalOpen)]="showUploadModal"
|
|
||||||
class="relative"
|
|
||||||
>
|
|
||||||
<h3 class="modal-title">Upload File</h3>
|
|
||||||
<div class="modal-body">
|
|
||||||
<div class="drop-area">
|
|
||||||
<span>Drop file anywhere to upload!</span>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="clr-col-md-12">
|
|
||||||
<div class="clr-row card-block mt-15 d-flex justify-content-between">
|
|
||||||
<div class="clr-col-md-3 filterBtn">
|
|
||||||
<span class="filterBtn w-100">
|
|
||||||
<label
|
|
||||||
for="file-upload"
|
|
||||||
class="btn btn-sm btn-outline profile-buttons w-100"
|
|
||||||
>
|
|
||||||
Browse
|
|
||||||
</label>
|
|
||||||
</span>
|
|
||||||
<input
|
|
||||||
hidden
|
|
||||||
#fileUploadInput
|
|
||||||
id="file-upload"
|
|
||||||
type="file"
|
|
||||||
appFileSelect
|
|
||||||
[uploader]="uploader"
|
|
||||||
(change)="getFileDesc($event)"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</clr-modal>
|
|
||||||
|
|
||||||
<clr-modal [(clrModalOpen)]="submitLimitNotice">
|
|
||||||
<h3 class="modal-title">Notice</h3>
|
|
||||||
<div class="modal-body">
|
|
||||||
<p class="m-0">
|
|
||||||
Due to current licence, only
|
|
||||||
{{ licenceState.value.submit_rows_limit }} rows in a file will be
|
|
||||||
submitted. To remove the restriction, contact
|
|
||||||
support@datacontroller.io
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
<div class="modal-footer">
|
|
||||||
<button
|
|
||||||
type="button"
|
|
||||||
class="btn btn-sm btn-primary"
|
|
||||||
(click)="submitLimitNotice = false"
|
|
||||||
>
|
|
||||||
Cancel
|
|
||||||
</button>
|
|
||||||
<button
|
|
||||||
type="button"
|
|
||||||
class="btn btn-sm btn-primary"
|
|
||||||
(click)="submit(); submitLimitNotice = false"
|
|
||||||
>
|
|
||||||
Submit
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
</clr-modal>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<ng-template #actionButtons>
|
|
||||||
<div class="clr-row m-0 clr-justify-content-center">
|
|
||||||
<div
|
|
||||||
*ngIf="status === StatusEnum.ReadyToUpload"
|
|
||||||
class="d-flex clr-justify-content-center clr-col-12 clr-col-lg-4"
|
|
||||||
>
|
|
||||||
<button
|
|
||||||
type="button"
|
|
||||||
class="btn btn-sm btn-success btn-block mr-0"
|
|
||||||
(click)="onShowUploadModal()"
|
|
||||||
>
|
|
||||||
<clr-icon shape="upload"></clr-icon>
|
|
||||||
<span>Upload</span>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
*ngIf="status === StatusEnum.ReadyToSubmit"
|
|
||||||
class="d-flex clr-justify-content-center clr-col-12 clr-col-lg-4"
|
|
||||||
>
|
|
||||||
<button
|
|
||||||
type="button"
|
|
||||||
class="btn btn-sm btn-success btn-block mr-0"
|
|
||||||
(click)="submitExcel()"
|
|
||||||
>
|
|
||||||
<clr-icon shape="upload"></clr-icon>
|
|
||||||
<span>Submit</span>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
*ngIf="status === StatusEnum.ReadyToSubmit"
|
|
||||||
class="d-flex clr-justify-content-center clr-col-12 clr-col-lg-4"
|
|
||||||
>
|
|
||||||
<button
|
|
||||||
type="button"
|
|
||||||
class="btn btn-sm btn-outline-danger btn-block mr-0"
|
|
||||||
(click)="discardExtractedData()"
|
|
||||||
>
|
|
||||||
<clr-icon shape="times"></clr-icon>
|
|
||||||
<span>Discard</span>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</ng-template>
|
|
@ -1,77 +0,0 @@
|
|||||||
.card {
|
|
||||||
margin-top: 0;
|
|
||||||
|
|
||||||
flex: 1;
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
}
|
|
||||||
|
|
||||||
clr-tree-node button {
|
|
||||||
white-space: nowrap;
|
|
||||||
}
|
|
||||||
|
|
||||||
.no-table-selected {
|
|
||||||
position: relative;
|
|
||||||
}
|
|
||||||
|
|
||||||
.header-row {
|
|
||||||
.title-col {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
}
|
|
||||||
.options-col {
|
|
||||||
display: flex;
|
|
||||||
justify-content: flex-end;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.sw {
|
|
||||||
margin: 1rem 0rem 0.5rem 1rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.viewerTitle {
|
|
||||||
text-align: center;
|
|
||||||
}
|
|
||||||
|
|
||||||
.cardFlex {
|
|
||||||
display: flex;
|
|
||||||
justify-content: center;
|
|
||||||
}
|
|
||||||
|
|
||||||
.content-area {
|
|
||||||
padding: 0.5rem !important;
|
|
||||||
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
}
|
|
||||||
|
|
||||||
hot-table {
|
|
||||||
::ng-deep {
|
|
||||||
.primaryKeyHeaderStyle {
|
|
||||||
background: #306b006e;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.drop-area {
|
|
||||||
position: fixed;
|
|
||||||
top: 0;
|
|
||||||
left: 0;
|
|
||||||
bottom: 0;
|
|
||||||
right: 0;
|
|
||||||
|
|
||||||
display: flex;
|
|
||||||
justify-content: center;
|
|
||||||
|
|
||||||
margin: 1px;
|
|
||||||
|
|
||||||
border: 2px dashed #fff;
|
|
||||||
|
|
||||||
z-index: -1;
|
|
||||||
|
|
||||||
span {
|
|
||||||
font-size: 20px;
|
|
||||||
margin-top: 20px;
|
|
||||||
color: #fff;
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,485 +0,0 @@
|
|||||||
import {
|
|
||||||
AfterContentInit,
|
|
||||||
AfterViewInit,
|
|
||||||
Component,
|
|
||||||
ElementRef,
|
|
||||||
HostBinding,
|
|
||||||
OnInit,
|
|
||||||
QueryList,
|
|
||||||
ViewChildren
|
|
||||||
} from '@angular/core'
|
|
||||||
import { ActivatedRoute, Router } from '@angular/router'
|
|
||||||
import { UploadFile } from '@sasjs/adapter'
|
|
||||||
import * as XLSX from '@sheet/crypto'
|
|
||||||
import { XLMapListItem, globals } from '../_globals'
|
|
||||||
import { FileUploader } from '../models/FileUploader.class'
|
|
||||||
import {
|
|
||||||
EventService,
|
|
||||||
LicenceService,
|
|
||||||
LoggerService,
|
|
||||||
SasService,
|
|
||||||
SasStoreService
|
|
||||||
} from '../services'
|
|
||||||
import { getCellAddress, getFinishingCell } from './utils/xl.utils'
|
|
||||||
import { blobToFile, byteArrayToBinaryString } from './utils/file.utils'
|
|
||||||
|
|
||||||
interface XLMapRule {
|
|
||||||
XLMAP_ID: string
|
|
||||||
XLMAP_SHEET: string
|
|
||||||
XLMAP_RANGE_ID: string
|
|
||||||
XLMAP_START: string
|
|
||||||
XLMAP_FINISH: string
|
|
||||||
}
|
|
||||||
|
|
||||||
interface XLUploadEntry {
|
|
||||||
LOAD_REF: string
|
|
||||||
XLMAP_ID: string
|
|
||||||
XLMAP_RANGE_ID: string
|
|
||||||
ROW_NO: number
|
|
||||||
COL_NO: number
|
|
||||||
VALUE_TXT: string
|
|
||||||
}
|
|
||||||
|
|
||||||
enum Status {
|
|
||||||
NoMapSelected,
|
|
||||||
FetchingRules,
|
|
||||||
ReadyToUpload,
|
|
||||||
ExtractingData,
|
|
||||||
ReadyToSubmit,
|
|
||||||
SubmittingExtractedData,
|
|
||||||
Submitting
|
|
||||||
}
|
|
||||||
|
|
||||||
enum Tabs {
|
|
||||||
Rules,
|
|
||||||
Data
|
|
||||||
}
|
|
||||||
|
|
||||||
@Component({
|
|
||||||
selector: 'app-xlmap',
|
|
||||||
templateUrl: './xlmap.component.html',
|
|
||||||
styleUrls: ['./xlmap.component.scss']
|
|
||||||
})
|
|
||||||
export class XLMapComponent implements AfterContentInit, AfterViewInit, OnInit {
|
|
||||||
@HostBinding('class.content-container') contentContainerClass = true
|
|
||||||
@ViewChildren('fileUploadInput')
|
|
||||||
fileUploadInputCompList: QueryList<ElementRef> = new QueryList()
|
|
||||||
|
|
||||||
StatusEnum = Status
|
|
||||||
TabsEnum = Tabs
|
|
||||||
|
|
||||||
public selectedTab = Tabs.Rules
|
|
||||||
public rulesSource = globals.dcLib + '.MPE_XLMAP_RULES'
|
|
||||||
|
|
||||||
public xlmaps: XLMapListItem[] = []
|
|
||||||
public selectedXLMap: XLMapListItem | undefined = undefined
|
|
||||||
public searchString = ''
|
|
||||||
public xlmapsLoading = true
|
|
||||||
public isLoading = false
|
|
||||||
public isLoadingDesc = ''
|
|
||||||
public status = Status.NoMapSelected
|
|
||||||
|
|
||||||
public xlmapRulesHeaders = [
|
|
||||||
'XLMAP_SHEET',
|
|
||||||
'XLMAP_RANGE_ID',
|
|
||||||
'XLMAP_START',
|
|
||||||
'XLMAP_FINISH'
|
|
||||||
]
|
|
||||||
public xlmapRulesColumns = [
|
|
||||||
{
|
|
||||||
data: 'XLMAP_SHEET'
|
|
||||||
},
|
|
||||||
{
|
|
||||||
data: 'XLMAP_RANGE_ID'
|
|
||||||
},
|
|
||||||
|
|
||||||
{
|
|
||||||
data: 'XLMAP_START'
|
|
||||||
},
|
|
||||||
{
|
|
||||||
data: 'XLMAP_FINISH'
|
|
||||||
}
|
|
||||||
]
|
|
||||||
public xlmapRules: XLMapRule[] = []
|
|
||||||
|
|
||||||
public xlUploadHeader = ['XLMAP_RANGE_ID', 'ROW_NO', 'COL_NO', 'VALUE_TXT']
|
|
||||||
public xlUploadColumns = [
|
|
||||||
{
|
|
||||||
data: 'XLMAP_RANGE_ID'
|
|
||||||
},
|
|
||||||
{
|
|
||||||
data: 'ROW_NO'
|
|
||||||
},
|
|
||||||
{
|
|
||||||
data: 'COL_NO'
|
|
||||||
},
|
|
||||||
{
|
|
||||||
data: 'VALUE_TXT'
|
|
||||||
}
|
|
||||||
]
|
|
||||||
public xlData: XLUploadEntry[] = []
|
|
||||||
|
|
||||||
public showUploadModal = false
|
|
||||||
public hasBaseDropZoneOver = false
|
|
||||||
public filename = ''
|
|
||||||
public submitLimitNotice = false
|
|
||||||
|
|
||||||
public uploader: FileUploader = new FileUploader()
|
|
||||||
|
|
||||||
public licenceState = this.licenceService.licenceState
|
|
||||||
|
|
||||||
public hotTableLicenseKey: string | undefined = undefined
|
|
||||||
public hotTableMaxRows =
|
|
||||||
this.licenceState.value.viewer_rows_allowed || Infinity
|
|
||||||
|
|
||||||
constructor(
|
|
||||||
private eventService: EventService,
|
|
||||||
private licenceService: LicenceService,
|
|
||||||
private loggerService: LoggerService,
|
|
||||||
private route: ActivatedRoute,
|
|
||||||
private router: Router,
|
|
||||||
private sasStoreService: SasStoreService,
|
|
||||||
private sasService: SasService
|
|
||||||
) {}
|
|
||||||
|
|
||||||
public xlmapOnClick(xlmap: XLMapListItem) {
|
|
||||||
if (xlmap.id !== this.selectedXLMap?.id) {
|
|
||||||
this.selectedXLMap = xlmap
|
|
||||||
this.xlData = []
|
|
||||||
this.filename = ''
|
|
||||||
this.uploader.queue = []
|
|
||||||
if (this.fileUploadInputCompList.first) {
|
|
||||||
this.fileUploadInputCompList.first.nativeElement.value = ''
|
|
||||||
}
|
|
||||||
this.selectedTab = Tabs.Rules
|
|
||||||
this.viewXLMapRules()
|
|
||||||
this.router.navigateByUrl('/home/files/' + xlmap.id)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public xlmapListOnFilter() {
|
|
||||||
if (this.searchString.length > 0) {
|
|
||||||
const array: XLMapListItem[] = globals.xlmaps
|
|
||||||
this.xlmaps = array.filter((item) =>
|
|
||||||
item.id.toLowerCase().includes(this.searchString.toLowerCase())
|
|
||||||
)
|
|
||||||
} else {
|
|
||||||
this.xlmaps = globals.xlmaps
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public isActiveXLMap(id: string) {
|
|
||||||
return this.selectedXLMap?.id === id
|
|
||||||
}
|
|
||||||
|
|
||||||
public maxWidthChecker(width: any, col: any) {
|
|
||||||
if (width > 200) return 200
|
|
||||||
else return width
|
|
||||||
}
|
|
||||||
|
|
||||||
public getCellConfiguration() {
|
|
||||||
return { readOnly: true }
|
|
||||||
}
|
|
||||||
|
|
||||||
public rowHeaders() {
|
|
||||||
return ' '
|
|
||||||
}
|
|
||||||
|
|
||||||
public onShowUploadModal() {
|
|
||||||
this.showUploadModal = true
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Called by FileDropDirective
|
|
||||||
* @param e true if file is dragged over the drop zone
|
|
||||||
*/
|
|
||||||
public fileOverBase(e: boolean): void {
|
|
||||||
this.hasBaseDropZoneOver = e
|
|
||||||
}
|
|
||||||
|
|
||||||
public getFileDesc(event: any, dropped = false) {
|
|
||||||
const file = dropped ? event[0] : event.target.files[0]
|
|
||||||
|
|
||||||
if (!file) return
|
|
||||||
|
|
||||||
const filename = file.name
|
|
||||||
this.filename = filename
|
|
||||||
|
|
||||||
const fileType = filename.slice(
|
|
||||||
filename.lastIndexOf('.') + 1,
|
|
||||||
filename.lastIndexOf('.') + 4
|
|
||||||
)
|
|
||||||
|
|
||||||
if (fileType.toLowerCase() === 'xls') {
|
|
||||||
this.showUploadModal = false
|
|
||||||
this.isLoading = true
|
|
||||||
this.isLoadingDesc = 'Extracting Data'
|
|
||||||
this.status = Status.ExtractingData
|
|
||||||
|
|
||||||
const reader = new FileReader()
|
|
||||||
reader.onload = async (theFile: any) => {
|
|
||||||
/* read workbook */
|
|
||||||
const bstr = byteArrayToBinaryString(theFile.target.result)
|
|
||||||
let wb: XLSX.WorkBook | undefined = undefined
|
|
||||||
|
|
||||||
const xlsxOptions: XLSX.ParsingOptions = {
|
|
||||||
type: 'binary',
|
|
||||||
cellDates: false,
|
|
||||||
cellFormula: true,
|
|
||||||
cellStyles: true,
|
|
||||||
cellNF: false,
|
|
||||||
cellText: false
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
wb = XLSX.read(bstr, {
|
|
||||||
...xlsxOptions
|
|
||||||
})
|
|
||||||
} catch (err: any) {
|
|
||||||
this.eventService.showAbortModal(
|
|
||||||
null,
|
|
||||||
err,
|
|
||||||
undefined,
|
|
||||||
'Error reading file'
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!wb) {
|
|
||||||
this.isLoading = false
|
|
||||||
this.isLoadingDesc = ''
|
|
||||||
this.status = Status.ReadyToUpload
|
|
||||||
this.uploader.queue.pop()
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
this.extractData(wb)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
reader.readAsArrayBuffer(file)
|
|
||||||
} else {
|
|
||||||
this.isLoading = false
|
|
||||||
this.isLoadingDesc = ''
|
|
||||||
this.status = Status.ReadyToUpload
|
|
||||||
this.showUploadModal = true
|
|
||||||
this.uploader.queue.pop()
|
|
||||||
|
|
||||||
const abortMsg =
|
|
||||||
'Invalid file type "<b>' +
|
|
||||||
this.filename +
|
|
||||||
'</b>". Please upload excel file.'
|
|
||||||
this.eventService.showAbortModal(null, abortMsg)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public discardExtractedData() {
|
|
||||||
this.isLoading = false
|
|
||||||
this.isLoadingDesc = ''
|
|
||||||
this.status = Status.ReadyToUpload
|
|
||||||
this.xlData = []
|
|
||||||
this.selectedTab = Tabs.Rules
|
|
||||||
this.filename = ''
|
|
||||||
this.uploader.queue = []
|
|
||||||
if (this.fileUploadInputCompList.first) {
|
|
||||||
this.fileUploadInputCompList.first.nativeElement.value = ''
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Submits attached excel file that is in preview mode
|
|
||||||
*/
|
|
||||||
public submitExcel() {
|
|
||||||
if (this.licenceState.value.submit_rows_limit !== Infinity) {
|
|
||||||
this.submitLimitNotice = true
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
this.submit()
|
|
||||||
}
|
|
||||||
|
|
||||||
public submit() {
|
|
||||||
if (!this.selectedXLMap || !this.xlData.length) return
|
|
||||||
|
|
||||||
this.status = Status.Submitting
|
|
||||||
this.isLoading = true
|
|
||||||
this.isLoadingDesc = 'Submitting extracted data'
|
|
||||||
|
|
||||||
const filesToUpload: UploadFile[] = []
|
|
||||||
|
|
||||||
for (const file of this.uploader.queue) {
|
|
||||||
filesToUpload.push({
|
|
||||||
file: file,
|
|
||||||
fileName: file.name
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
const csvContent =
|
|
||||||
Object.keys(this.xlData[0]).join(',') +
|
|
||||||
'\n' +
|
|
||||||
this.xlData
|
|
||||||
.slice(0, this.licenceState.value.submit_rows_limit)
|
|
||||||
.map((row: any) => Object.values(row).join(','))
|
|
||||||
.join('\n')
|
|
||||||
|
|
||||||
const blob = new Blob([csvContent], { type: 'application/csv' })
|
|
||||||
const file: File = blobToFile(blob, this.filename + '.csv')
|
|
||||||
|
|
||||||
filesToUpload.push({
|
|
||||||
file: file,
|
|
||||||
fileName: file.name
|
|
||||||
})
|
|
||||||
|
|
||||||
const uploadUrl = 'services/editors/loadfile'
|
|
||||||
this.sasService
|
|
||||||
.uploadFile(uploadUrl, filesToUpload, {
|
|
||||||
table: this.selectedXLMap.targetDS
|
|
||||||
})
|
|
||||||
.then((res: any) => {
|
|
||||||
if (res.sasjsAbort) {
|
|
||||||
const abortRes = res
|
|
||||||
const abortMsg = abortRes.sasjsAbort[0].MSG
|
|
||||||
const macMsg = abortRes.sasjsAbort[0].MAC
|
|
||||||
|
|
||||||
this.eventService.showAbortModal('', abortMsg, {
|
|
||||||
SYSWARNINGTEXT: abortRes.SYSWARNINGTEXT,
|
|
||||||
SYSERRORTEXT: abortRes.SYSERRORTEXT,
|
|
||||||
MAC: macMsg
|
|
||||||
})
|
|
||||||
} else if (res.sasparams) {
|
|
||||||
const params = res.sasparams[0]
|
|
||||||
const tableId = params.DSID
|
|
||||||
this.router.navigateByUrl('/stage/' + tableId)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.catch((err: any) => {
|
|
||||||
this.eventService.catchResponseError('file upload', err)
|
|
||||||
})
|
|
||||||
.finally(() => {
|
|
||||||
this.status = Status.ReadyToSubmit
|
|
||||||
this.isLoading = false
|
|
||||||
this.isLoadingDesc = ''
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
public extractData(wb: XLSX.WorkBook) {
|
|
||||||
const extractedData: XLUploadEntry[] = []
|
|
||||||
|
|
||||||
this.xlmapRules.forEach((rule) => {
|
|
||||||
let sheetName = rule.XLMAP_SHEET
|
|
||||||
// if sheet name is not an absolute name rather an index string like /1, /2, etc
|
|
||||||
// we extract the index and find absolute sheet name for specified index
|
|
||||||
if (sheetName.startsWith('/')) {
|
|
||||||
const temp = sheetName.split('/')[1]
|
|
||||||
const sheetIndex = parseInt(temp) - 1
|
|
||||||
sheetName = wb.SheetNames[sheetIndex]
|
|
||||||
}
|
|
||||||
|
|
||||||
const sheet = wb.Sheets[sheetName]
|
|
||||||
|
|
||||||
const arrayOfObjects = <any[]>XLSX.utils.sheet_to_json(sheet, {
|
|
||||||
raw: true,
|
|
||||||
header: 'A',
|
|
||||||
blankrows: true
|
|
||||||
})
|
|
||||||
|
|
||||||
const start = getCellAddress(rule.XLMAP_START, arrayOfObjects)
|
|
||||||
const finish = getFinishingCell(start, rule.XLMAP_FINISH, arrayOfObjects)
|
|
||||||
|
|
||||||
const range = `${start}:${finish}`
|
|
||||||
|
|
||||||
const rangedData = <any[]>XLSX.utils.sheet_to_json(sheet, {
|
|
||||||
raw: true,
|
|
||||||
range: range,
|
|
||||||
header: 'A',
|
|
||||||
blankrows: true
|
|
||||||
})
|
|
||||||
|
|
||||||
for (let i = 0; i < rangedData.length; i++) {
|
|
||||||
const row = rangedData[i]
|
|
||||||
// Get the keys of the object (excluding '__rowNum__')
|
|
||||||
const keys = Object.keys(row).filter((key) => key !== '__rowNum__')
|
|
||||||
|
|
||||||
for (let j = 0; j < keys.length; j++) {
|
|
||||||
const key = keys[j]
|
|
||||||
const val = row[key]
|
|
||||||
|
|
||||||
// in excel's R1C1 notation indexing starts from 1 but in JS it starts from 0
|
|
||||||
// therefore, we'll have to add 1 to rows and cols
|
|
||||||
extractedData.push({
|
|
||||||
LOAD_REF: '0',
|
|
||||||
XLMAP_ID: rule.XLMAP_ID,
|
|
||||||
XLMAP_RANGE_ID: rule.XLMAP_RANGE_ID,
|
|
||||||
ROW_NO: i + 1,
|
|
||||||
COL_NO: j + 1,
|
|
||||||
VALUE_TXT: val
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
this.status = Status.ReadyToSubmit
|
|
||||||
this.isLoading = false
|
|
||||||
this.isLoadingDesc = ''
|
|
||||||
|
|
||||||
this.xlData = extractedData
|
|
||||||
this.selectedTab = Tabs.Data
|
|
||||||
}
|
|
||||||
|
|
||||||
async viewXLMapRules() {
|
|
||||||
if (!this.selectedXLMap) return
|
|
||||||
|
|
||||||
this.isLoading = true
|
|
||||||
this.isLoadingDesc = 'Loading excel rules'
|
|
||||||
this.status = Status.FetchingRules
|
|
||||||
|
|
||||||
await this.sasStoreService
|
|
||||||
.getXLMapRules(this.selectedXLMap.id)
|
|
||||||
.then((res) => {
|
|
||||||
this.xlmapRules = res.xlmaprules
|
|
||||||
this.status = Status.ReadyToUpload
|
|
||||||
})
|
|
||||||
.catch((err) => {
|
|
||||||
this.loggerService.error(err)
|
|
||||||
})
|
|
||||||
|
|
||||||
this.isLoading = false
|
|
||||||
this.isLoadingDesc = ''
|
|
||||||
}
|
|
||||||
|
|
||||||
private load() {
|
|
||||||
this.xlmaps = globals.xlmaps
|
|
||||||
this.xlmapsLoading = false
|
|
||||||
|
|
||||||
const id = this.route.snapshot.params['id']
|
|
||||||
|
|
||||||
if (id) {
|
|
||||||
const xlmapListItem = this.xlmaps.find((item) => item.id === id)
|
|
||||||
if (xlmapListItem) {
|
|
||||||
this.selectedXLMap = xlmapListItem
|
|
||||||
this.viewXLMapRules()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
ngOnInit() {
|
|
||||||
this.licenceService.hot_license_key.subscribe(
|
|
||||||
(hot_license_key: string | undefined) => {
|
|
||||||
this.hotTableLicenseKey = hot_license_key
|
|
||||||
}
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
ngAfterViewInit() {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
ngAfterContentInit(): void {
|
|
||||||
if (globals.editor.startupSet) {
|
|
||||||
this.load()
|
|
||||||
} else {
|
|
||||||
this.eventService.onStartupDataLoaded.subscribe(() => {
|
|
||||||
this.load()
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,31 +0,0 @@
|
|||||||
import { CommonModule } from '@angular/common'
|
|
||||||
import { NgModule } from '@angular/core'
|
|
||||||
import { FormsModule } from '@angular/forms'
|
|
||||||
import { ClarityModule } from '@clr/angular'
|
|
||||||
import { HotTableModule } from '@handsontable/angular'
|
|
||||||
import { registerAllModules } from 'handsontable/registry'
|
|
||||||
import { AppSharedModule } from '../app-shared.module'
|
|
||||||
import { DirectivesModule } from '../directives/directives.module'
|
|
||||||
import { XLMapRouteComponent } from '../routes/xlmap-route/xlmap-route.component'
|
|
||||||
import { DcTreeModule } from '../shared/dc-tree/dc-tree.module'
|
|
||||||
import { XLMapRoutingModule } from './xlmap-routing.module'
|
|
||||||
import { XLMapComponent } from './xlmap.component'
|
|
||||||
|
|
||||||
// register Handsontable's modules
|
|
||||||
registerAllModules()
|
|
||||||
|
|
||||||
@NgModule({
|
|
||||||
declarations: [XLMapRouteComponent, XLMapComponent],
|
|
||||||
imports: [
|
|
||||||
HotTableModule,
|
|
||||||
XLMapRoutingModule,
|
|
||||||
FormsModule,
|
|
||||||
ClarityModule,
|
|
||||||
AppSharedModule,
|
|
||||||
CommonModule,
|
|
||||||
DcTreeModule,
|
|
||||||
DirectivesModule
|
|
||||||
],
|
|
||||||
exports: [XLMapComponent]
|
|
||||||
})
|
|
||||||
export class XLMapModule {}
|
|
@ -1,16 +1,15 @@
|
|||||||
/* You can add global styles to this file, and also import other style files */
|
/* You can add global styles to this file, and also import other style files */
|
||||||
@import '~handsontable/dist/handsontable.full.css';
|
@import '~handsontable/dist/handsontable.full.css';
|
||||||
|
|
||||||
@import '~@clr/ui/clr-ui.min.css';
|
@import "~@clr/ui/clr-ui.min.css";
|
||||||
@import '~@clr/icons/clr-icons.min.css';
|
@import "~@clr/icons/clr-icons.min.css";
|
||||||
|
|
||||||
@font-face{
|
@font-face{
|
||||||
font-family: text-security-disc;
|
font-family: text-security-disc;
|
||||||
src: url('https://raw.githubusercontent.com/noppa/text-security/master/dist/text-security-disc.woff');
|
src: url("https://raw.githubusercontent.com/noppa/text-security/master/dist/text-security-disc.woff");
|
||||||
}
|
}
|
||||||
|
|
||||||
body,
|
body, html {
|
||||||
html {
|
|
||||||
font-weight: 400!important;
|
font-weight: 400!important;
|
||||||
|
|
||||||
padding: 0;
|
padding: 0;
|
||||||
@ -42,14 +41,14 @@ button {
|
|||||||
.line{
|
.line{
|
||||||
position:absolute;
|
position:absolute;
|
||||||
opacity: 0.4;
|
opacity: 0.4;
|
||||||
background: #73d544;
|
background:#73D544;
|
||||||
width:150%;
|
width:150%;
|
||||||
height:5px;
|
height:5px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.subline{
|
.subline{
|
||||||
position:absolute;
|
position:absolute;
|
||||||
background: #73d544;
|
background:#73D544;
|
||||||
height:5px;
|
height:5px;
|
||||||
}
|
}
|
||||||
.inc{
|
.inc{
|
||||||
@ -60,24 +59,12 @@ button {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@keyframes increase {
|
@keyframes increase {
|
||||||
from {
|
from { left: -5%; width: 5%; }
|
||||||
left: -5%;
|
to { left: 130%; width: 100%;}
|
||||||
width: 5%;
|
|
||||||
}
|
|
||||||
to {
|
|
||||||
left: 130%;
|
|
||||||
width: 100%;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
@keyframes decrease {
|
@keyframes decrease {
|
||||||
from {
|
from { left: -80%; width: 80%; }
|
||||||
left: -80%;
|
to { left: 110%; width: 10%;}
|
||||||
width: 80%;
|
|
||||||
}
|
|
||||||
to {
|
|
||||||
left: 110%;
|
|
||||||
width: 10%;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
// Custo loading spinner end
|
// Custo loading spinner end
|
||||||
|
|
||||||
@ -289,10 +276,6 @@ button {
|
|||||||
margin-bottom: 10px;
|
margin-bottom: 10px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.mb-10-i {
|
|
||||||
margin-bottom: 10px !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
.mb-20 {
|
.mb-20 {
|
||||||
margin-bottom: 20px;
|
margin-bottom: 20px;
|
||||||
}
|
}
|
||||||
@ -338,11 +321,11 @@ button {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.color-dark-gray {
|
.color-dark-gray {
|
||||||
color: #495967;
|
color: #495967
|
||||||
}
|
}
|
||||||
|
|
||||||
.color-darker-gray{
|
.color-darker-gray{
|
||||||
color: #314351;
|
color: #314351
|
||||||
}
|
}
|
||||||
|
|
||||||
.color-white {
|
.color-white {
|
||||||
@ -350,7 +333,7 @@ button {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.color-white-i {
|
.color-white-i {
|
||||||
color: white !important;
|
color: white !important
|
||||||
}
|
}
|
||||||
|
|
||||||
.color-green {
|
.color-green {
|
||||||
@ -358,15 +341,15 @@ button {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.color-dc-green {
|
.color-dc-green {
|
||||||
color: #81b440;
|
color: #81b440
|
||||||
}
|
}
|
||||||
|
|
||||||
.color-red {
|
.color-red {
|
||||||
color: #e45454;
|
color: #e45454
|
||||||
}
|
}
|
||||||
|
|
||||||
.color-orange {
|
.color-orange {
|
||||||
color: #e67e22;
|
color: #E67E22;
|
||||||
}
|
}
|
||||||
|
|
||||||
.color-blue {
|
.color-blue {
|
||||||
@ -374,7 +357,7 @@ button {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.color-yellow {
|
.color-yellow {
|
||||||
color: #f1c40f;
|
color: #f1c40f
|
||||||
}
|
}
|
||||||
|
|
||||||
.cursor-pointer {
|
.cursor-pointer {
|
||||||
@ -518,7 +501,7 @@ button {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.z-index-highest {
|
.z-index-highest {
|
||||||
z-index: 10000000;
|
z-index: 10000000
|
||||||
}
|
}
|
||||||
|
|
||||||
.vertical-align-middle {
|
.vertical-align-middle {
|
||||||
@ -541,15 +524,14 @@ button {
|
|||||||
z-index: 10000!important;
|
z-index: 10000!important;
|
||||||
}
|
}
|
||||||
|
|
||||||
.progress,
|
.progress, .progress-static {
|
||||||
.progress-static {
|
|
||||||
background-color: #f5f6fe;
|
background-color: #f5f6fe;
|
||||||
border-radius: 0;
|
border-radius: 0;
|
||||||
font-size: inherit;
|
font-size: inherit;
|
||||||
height: 6px;
|
height: 6px;
|
||||||
margin: 0;
|
margin: 0;
|
||||||
max-height: 0.583333rem;
|
max-height: .583333rem;
|
||||||
min-height: 0.166667rem;
|
min-height: .166667rem;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
display: block;
|
display: block;
|
||||||
width: calc(100% - 63px);
|
width: calc(100% - 63px);
|
||||||
@ -558,8 +540,8 @@ button {
|
|||||||
.progress.loop:after {
|
.progress.loop:after {
|
||||||
-webkit-animation: clr-progress-looper 1.5s ease-in-out infinite;
|
-webkit-animation: clr-progress-looper 1.5s ease-in-out infinite;
|
||||||
animation: clr-progress-looper 1.5s ease-in-out infinite;
|
animation: clr-progress-looper 1.5s ease-in-out infinite;
|
||||||
content: ' ';
|
content: " ";
|
||||||
top: 0.166667rem;
|
top: .166667rem;
|
||||||
bottom: 0;
|
bottom: 0;
|
||||||
left: 0;
|
left: 0;
|
||||||
position: absolute;
|
position: absolute;
|
||||||
@ -588,7 +570,7 @@ button {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.alert-app-level.alert-danger {
|
.alert-app-level.alert-danger {
|
||||||
background: #d94b2e;
|
background: #D94B2E;
|
||||||
color: #fff;
|
color: #fff;
|
||||||
border: none;
|
border: none;
|
||||||
}
|
}
|
||||||
@ -615,9 +597,7 @@ button {
|
|||||||
background: #d8e3e9;
|
background: #d8e3e9;
|
||||||
}
|
}
|
||||||
|
|
||||||
clr-select-container .clr-control-container,
|
clr-select-container .clr-control-container, clr-select-container .clr-control-container .clr-select-wrapper, clr-select-container select {
|
||||||
clr-select-container .clr-control-container .clr-select-wrapper,
|
|
||||||
clr-select-container select {
|
|
||||||
width: 100%;
|
width: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -625,8 +605,7 @@ tbody {
|
|||||||
font-weight: 400;
|
font-weight: 400;
|
||||||
}
|
}
|
||||||
|
|
||||||
h3,
|
h3, h4 {
|
||||||
h4 {
|
|
||||||
color: #585858;
|
color: #585858;
|
||||||
font-weight: 400;
|
font-weight: 400;
|
||||||
letter-spacing: normal;
|
letter-spacing: normal;
|
||||||
@ -636,8 +615,7 @@ h4 {
|
|||||||
/* text-transform: uppercase; */
|
/* text-transform: uppercase; */
|
||||||
}
|
}
|
||||||
|
|
||||||
h1,
|
h1, h2 {
|
||||||
h2 {
|
|
||||||
color: #585858;
|
color: #585858;
|
||||||
font-weight: 400;
|
font-weight: 400;
|
||||||
/* font-family: Metropolis,Avenir Next,Helvetica Neue,Arial,sans-serif; */
|
/* font-family: Metropolis,Avenir Next,Helvetica Neue,Arial,sans-serif; */
|
||||||
@ -652,18 +630,16 @@ clr-icon.is-info {
|
|||||||
fill: #80b441;
|
fill: #80b441;
|
||||||
}
|
}
|
||||||
|
|
||||||
.datagrid-host,
|
.datagrid-host, .datagrid-overlay-wrapper {
|
||||||
.datagrid-overlay-wrapper {
|
|
||||||
display: -webkit-box;
|
display: -webkit-box;
|
||||||
display: -ms-flexbox;
|
display: -ms-flexbox;
|
||||||
display: -webkit-box!important;
|
display: -webkit-box!important;
|
||||||
-webkit-box-direction: normal;
|
-webkit-box-direction: normal;
|
||||||
}
|
}
|
||||||
|
|
||||||
.btn.btn-danger,
|
.btn.btn-danger, .btn.btn-warning {
|
||||||
.btn.btn-warning {
|
|
||||||
border-color: #ef4f2e;
|
border-color: #ef4f2e;
|
||||||
background-color: #d94b2e;
|
background-color: #D94B2E;
|
||||||
color: #fff;
|
color: #fff;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -855,6 +831,7 @@ clr-icon.is-info {
|
|||||||
font-family: monospace;
|
font-family: monospace;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#graph svg {
|
#graph svg {
|
||||||
height: 100%;
|
height: 100%;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
@ -883,6 +860,7 @@ clr-icon.is-info {
|
|||||||
z-index: 100;
|
z-index: 100;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
.nav-tree > clr-tree-node.clr-expanded {
|
.nav-tree > clr-tree-node.clr-expanded {
|
||||||
display: inline-block !important;
|
display: inline-block !important;
|
||||||
}
|
}
|
||||||
@ -978,8 +956,7 @@ input::-ms-clear {
|
|||||||
overflow: hidden !important;
|
overflow: hidden !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
.clr-treenode-content .clr-icon,
|
.clr-treenode-content .clr-icon, .clr-treenode-content clr-icon {
|
||||||
.clr-treenode-content clr-icon {
|
|
||||||
min-width: 16px;
|
min-width: 16px;
|
||||||
min-height: 16px;
|
min-height: 16px;
|
||||||
}
|
}
|
||||||
@ -1048,8 +1025,7 @@ hr.light {
|
|||||||
position: relative;
|
position: relative;
|
||||||
min-width: 170px;
|
min-width: 170px;
|
||||||
|
|
||||||
clr-icon,
|
clr-icon, .spinner {
|
||||||
.spinner {
|
|
||||||
position: absolute;
|
position: absolute;
|
||||||
right: 19px;
|
right: 19px;
|
||||||
top: 0px;
|
top: 0px;
|
||||||
@ -1087,7 +1063,7 @@ hr.light {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Firefox */
|
/* Firefox */
|
||||||
input[type='number'] {
|
input[type=number] {
|
||||||
-moz-appearance: textfield;
|
-moz-appearance: textfield;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
28
package-lock.json
generated
28
package-lock.json
generated
@ -1,12 +1,12 @@
|
|||||||
{
|
{
|
||||||
"name": "dcfrontend",
|
"name": "dcfrontend",
|
||||||
"version": "6.3.0",
|
"version": "6.2.2",
|
||||||
"lockfileVersion": 2,
|
"lockfileVersion": 2,
|
||||||
"requires": true,
|
"requires": true,
|
||||||
"packages": {
|
"packages": {
|
||||||
"": {
|
"": {
|
||||||
"name": "dcfrontend",
|
"name": "dcfrontend",
|
||||||
"version": "6.3.0",
|
"version": "6.2.2",
|
||||||
"hasInstallScript": true,
|
"hasInstallScript": true,
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@saithodev/semantic-release-gitea": "^2.1.0",
|
"@saithodev/semantic-release-gitea": "^2.1.0",
|
||||||
@ -15,7 +15,8 @@
|
|||||||
"@semantic-release/git": "^10.0.1",
|
"@semantic-release/git": "^10.0.1",
|
||||||
"@semantic-release/npm": "11.0.0",
|
"@semantic-release/npm": "11.0.0",
|
||||||
"@semantic-release/release-notes-generator": "^11.0.4",
|
"@semantic-release/release-notes-generator": "^11.0.4",
|
||||||
"commit-and-tag-version": "^11.2.2"
|
"commit-and-tag-version": "^11.2.2",
|
||||||
|
"prettier": "3.0.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@babel/code-frame": {
|
"node_modules/@babel/code-frame": {
|
||||||
@ -6877,6 +6878,21 @@
|
|||||||
"node": ">=0.10.0"
|
"node": ">=0.10.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/prettier": {
|
||||||
|
"version": "3.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/prettier/-/prettier-3.0.0.tgz",
|
||||||
|
"integrity": "sha512-zBf5eHpwHOGPC47h0zrPyNn+eAEIdEzfywMoYn2XPi0P44Zp0tSq64rq0xAREh4auw2cJZHo9QUob+NqCQky4g==",
|
||||||
|
"dev": true,
|
||||||
|
"bin": {
|
||||||
|
"prettier": "bin/prettier.cjs"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=14"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"url": "https://github.com/prettier/prettier?sponsor=1"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/process-nextick-args": {
|
"node_modules/process-nextick-args": {
|
||||||
"version": "2.0.1",
|
"version": "2.0.1",
|
||||||
"resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz",
|
"resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz",
|
||||||
@ -12599,6 +12615,12 @@
|
|||||||
"integrity": "sha512-udgsAY+fTnvv7kI7aaxbqwWNb0AHiB0qBO89PZKPkoTmGOgdbrHDKD+0B2X4uTfJ/FT1R09r9gTsjUjNJotuog==",
|
"integrity": "sha512-udgsAY+fTnvv7kI7aaxbqwWNb0AHiB0qBO89PZKPkoTmGOgdbrHDKD+0B2X4uTfJ/FT1R09r9gTsjUjNJotuog==",
|
||||||
"dev": true
|
"dev": true
|
||||||
},
|
},
|
||||||
|
"prettier": {
|
||||||
|
"version": "3.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/prettier/-/prettier-3.0.0.tgz",
|
||||||
|
"integrity": "sha512-zBf5eHpwHOGPC47h0zrPyNn+eAEIdEzfywMoYn2XPi0P44Zp0tSq64rq0xAREh4auw2cJZHo9QUob+NqCQky4g==",
|
||||||
|
"dev": true
|
||||||
|
},
|
||||||
"process-nextick-args": {
|
"process-nextick-args": {
|
||||||
"version": "2.0.1",
|
"version": "2.0.1",
|
||||||
"resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz",
|
"resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz",
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "dcfrontend",
|
"name": "dcfrontend",
|
||||||
"version": "6.5.0",
|
"version": "6.2.2",
|
||||||
"description": "Data Controller",
|
"description": "Data Controller",
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@saithodev/semantic-release-gitea": "^2.1.0",
|
"@saithodev/semantic-release-gitea": "^2.1.0",
|
||||||
@ -9,7 +9,8 @@
|
|||||||
"@semantic-release/npm": "11.0.0",
|
"@semantic-release/npm": "11.0.0",
|
||||||
"@semantic-release/git": "^10.0.1",
|
"@semantic-release/git": "^10.0.1",
|
||||||
"@semantic-release/release-notes-generator": "^11.0.4",
|
"@semantic-release/release-notes-generator": "^11.0.4",
|
||||||
"commit-and-tag-version": "^11.2.2"
|
"commit-and-tag-version": "^11.2.2",
|
||||||
|
"prettier": "3.0.0"
|
||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"install": "cd client && npm i && cd ../sas && npm i",
|
"install": "cd client && npm i && cd ../sas && npm i",
|
||||||
|
@ -83,12 +83,6 @@ _webout = `{"SYSDATE" : "26SEP22"
|
|||||||
"DC_RESTRICT_EDITRECORD": "NO"
|
"DC_RESTRICT_EDITRECORD": "NO"
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
,"xlmaps":
|
|
||||||
[
|
|
||||||
["BASEL-CR2" ,"" ,"DC695588.MPE_XLMAP_DATA" ]
|
|
||||||
,["BASEL-KM1" ,"Basel 3 Key Metrics report" ,"DC695588.MPE_XLMAP_DATA" ]
|
|
||||||
,["SAMPLE" ,"" ,"DC695588.MPE_XLMAP_DATA" ]
|
|
||||||
]
|
|
||||||
,"_DEBUG" : ""
|
,"_DEBUG" : ""
|
||||||
,"_METAUSER": "sasdemo@SAS"
|
,"_METAUSER": "sasdemo@SAS"
|
||||||
,"_METAPERSON": "sasdemo"
|
,"_METAPERSON": "sasdemo"
|
||||||
|
29
sas/package-lock.json
generated
29
sas/package-lock.json
generated
@ -7,7 +7,7 @@
|
|||||||
"name": "dc-sas",
|
"name": "dc-sas",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@sasjs/cli": "^4.11.1",
|
"@sasjs/cli": "^4.11.1",
|
||||||
"@sasjs/core": "^4.49.0"
|
"@sasjs/core": "^4.48.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@coolaj86/urequest": {
|
"node_modules/@coolaj86/urequest": {
|
||||||
@ -116,9 +116,9 @@
|
|||||||
"integrity": "sha512-Grwydm5GxBsYk238PZw41XPjXVVQ9vWcvfZ06L2P0bQbvK0sGn7l69JA7H5MGr3QcaLpiD4Kg70cAh7PgE+JOw=="
|
"integrity": "sha512-Grwydm5GxBsYk238PZw41XPjXVVQ9vWcvfZ06L2P0bQbvK0sGn7l69JA7H5MGr3QcaLpiD4Kg70cAh7PgE+JOw=="
|
||||||
},
|
},
|
||||||
"node_modules/@sasjs/core": {
|
"node_modules/@sasjs/core": {
|
||||||
"version": "4.49.0",
|
"version": "4.48.0",
|
||||||
"resolved": "https://registry.npmjs.org/@sasjs/core/-/core-4.49.0.tgz",
|
"resolved": "https://registry.npmjs.org/@sasjs/core/-/core-4.48.0.tgz",
|
||||||
"integrity": "sha512-hp3Hb4DkT6FmowyNHTOvSlgmSObW9WeuTJj+TQlwPgnBo59mAB4XFUnUaYSA+7ghvsHqUZf1OP2eSYqmnN5swQ=="
|
"integrity": "sha512-KaAvfTPv1UrP0I1fREDYjkfa7FRM9+yCseGXGLYylD30oH7BBOwLc7o/BkhRjjDvrBFoiJMjAOVKULhmkHz9zQ=="
|
||||||
},
|
},
|
||||||
"node_modules/@sasjs/lint": {
|
"node_modules/@sasjs/lint": {
|
||||||
"version": "2.3.1",
|
"version": "2.3.1",
|
||||||
@ -229,12 +229,6 @@
|
|||||||
"@types/node": "*"
|
"@types/node": "*"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@types/tough-cookie": {
|
|
||||||
"version": "4.0.5",
|
|
||||||
"resolved": "https://registry.npmjs.org/@types/tough-cookie/-/tough-cookie-4.0.5.tgz",
|
|
||||||
"integrity": "sha512-/Ad8+nIOV7Rl++6f1BdKxFSMgmoqEoYbHRpPcx3JEfv8VRsQe9Z4mCXeJBzxs7mbHY/XOZZuXlRNfhpVPbs6ZA==",
|
|
||||||
"peer": true
|
|
||||||
},
|
|
||||||
"node_modules/abab": {
|
"node_modules/abab": {
|
||||||
"version": "2.0.6",
|
"version": "2.0.6",
|
||||||
"resolved": "https://registry.npmjs.org/abab/-/abab-2.0.6.tgz",
|
"resolved": "https://registry.npmjs.org/abab/-/abab-2.0.6.tgz",
|
||||||
@ -1834,9 +1828,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"@sasjs/core": {
|
"@sasjs/core": {
|
||||||
"version": "4.49.0",
|
"version": "4.48.0",
|
||||||
"resolved": "https://registry.npmjs.org/@sasjs/core/-/core-4.49.0.tgz",
|
"resolved": "https://registry.npmjs.org/@sasjs/core/-/core-4.48.0.tgz",
|
||||||
"integrity": "sha512-hp3Hb4DkT6FmowyNHTOvSlgmSObW9WeuTJj+TQlwPgnBo59mAB4XFUnUaYSA+7ghvsHqUZf1OP2eSYqmnN5swQ=="
|
"integrity": "sha512-KaAvfTPv1UrP0I1fREDYjkfa7FRM9+yCseGXGLYylD30oH7BBOwLc7o/BkhRjjDvrBFoiJMjAOVKULhmkHz9zQ=="
|
||||||
},
|
},
|
||||||
"@sasjs/lint": {
|
"@sasjs/lint": {
|
||||||
"version": "2.3.1",
|
"version": "2.3.1",
|
||||||
@ -1933,12 +1927,6 @@
|
|||||||
"@types/node": "*"
|
"@types/node": "*"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"@types/tough-cookie": {
|
|
||||||
"version": "4.0.5",
|
|
||||||
"resolved": "https://registry.npmjs.org/@types/tough-cookie/-/tough-cookie-4.0.5.tgz",
|
|
||||||
"integrity": "sha512-/Ad8+nIOV7Rl++6f1BdKxFSMgmoqEoYbHRpPcx3JEfv8VRsQe9Z4mCXeJBzxs7mbHY/XOZZuXlRNfhpVPbs6ZA==",
|
|
||||||
"peer": true
|
|
||||||
},
|
|
||||||
"abab": {
|
"abab": {
|
||||||
"version": "2.0.6",
|
"version": "2.0.6",
|
||||||
"resolved": "https://registry.npmjs.org/abab/-/abab-2.0.6.tgz",
|
"resolved": "https://registry.npmjs.org/abab/-/abab-2.0.6.tgz",
|
||||||
@ -2965,8 +2953,7 @@
|
|||||||
"ws": {
|
"ws": {
|
||||||
"version": "8.13.0",
|
"version": "8.13.0",
|
||||||
"resolved": "https://registry.npmjs.org/ws/-/ws-8.13.0.tgz",
|
"resolved": "https://registry.npmjs.org/ws/-/ws-8.13.0.tgz",
|
||||||
"integrity": "sha512-x9vcZYTrFPC7aSIbj7sRCYo7L/Xb8Iy+pW0ng0wt2vCJv7M9HOMy0UoN3rr+IFC7hb7vXoqS+P9ktyLLLhO+LA==",
|
"integrity": "sha512-x9vcZYTrFPC7aSIbj7sRCYo7L/Xb8Iy+pW0ng0wt2vCJv7M9HOMy0UoN3rr+IFC7hb7vXoqS+P9ktyLLLhO+LA=="
|
||||||
"requires": {}
|
|
||||||
},
|
},
|
||||||
"xml": {
|
"xml": {
|
||||||
"version": "1.0.1",
|
"version": "1.0.1",
|
||||||
|
@ -14,8 +14,7 @@
|
|||||||
"sas9e": "sasjs request services/admin/makedata -d deploy/makeDataSas9.json -t sas9 ",
|
"sas9e": "sasjs request services/admin/makedata -d deploy/makeDataSas9.json -t sas9 ",
|
||||||
"sas9f": "sasjs request services/admin/refreshtablelineage -t sas9 ",
|
"sas9f": "sasjs request services/admin/refreshtablelineage -t sas9 ",
|
||||||
"sas9g": "sasjs request services/admin/refreshcatalog -t sas9",
|
"sas9g": "sasjs request services/admin/refreshcatalog -t sas9",
|
||||||
"4gl": "npm run cpfavicon && sasjs cbd -t 4gl && npm run 4glmakedata",
|
"4gl": "npm run cpfavicon && sasjs cbd -t 4gl && sasjs request services/admin/makedata -d deploy/makeData4GL.json -l sasjsresults/makedata_4gl.log -o sasjsresults/makedata_4gl.json -t 4gl",
|
||||||
"4glmakedata": "sasjs request services/admin/makedata -d deploy/makeData4GL.json -l sasjsresults/makedata_4gl.log -o sasjsresults/makedata_4gl.json -t 4gl",
|
|
||||||
"server": "npm run cpfavicon && sasjs cbd -t server && npm run serverdata",
|
"server": "npm run cpfavicon && sasjs cbd -t server && npm run serverdata",
|
||||||
"server-mihajlo": "npm run cpfavicon && sasjs cbd -t server-mihajlo && npm run serverdata-mihajlo",
|
"server-mihajlo": "npm run cpfavicon && sasjs cbd -t server-mihajlo && npm run serverdata-mihajlo",
|
||||||
"serverdata-mihajlo": "sasjs request services/admin/makedata -d deploy/makeDataServer.json -l sasjsresults/makedata_server.log -o sasjsresults/makedata_server.json -t server-mihajlo",
|
"serverdata-mihajlo": "sasjs request services/admin/makedata -d deploy/makeDataServer.json -l sasjsresults/makedata_server.log -o sasjsresults/makedata_server.json -t server-mihajlo",
|
||||||
@ -29,6 +28,6 @@
|
|||||||
"private": true,
|
"private": true,
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@sasjs/cli": "^4.11.1",
|
"@sasjs/cli": "^4.11.1",
|
||||||
"@sasjs/core": "^4.49.0"
|
"@sasjs/core": "^4.48.0"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,18 +0,0 @@
|
|||||||
/**
|
|
||||||
@file
|
|
||||||
@brief DDL for MPE_XLMAP_DATA
|
|
||||||
|
|
||||||
@version 9.3
|
|
||||||
@author 4GL Apps Ltd
|
|
||||||
@copyright 4GL Apps Ltd
|
|
||||||
**/
|
|
||||||
|
|
||||||
create table &curlib..MPE_XLMAP_DATA(
|
|
||||||
LOAD_REF char(32) not null,
|
|
||||||
XLMAP_ID char(32) not null,
|
|
||||||
XLMAP_RANGE_ID char(32) not null,
|
|
||||||
ROW_NO num not null,
|
|
||||||
COL_NO num not null,
|
|
||||||
VALUE_TXT char(4000),
|
|
||||||
constraint pk_MPE_XLMAP_DATA
|
|
||||||
primary key(LOAD_REF, XLMAP_ID, XLMAP_RANGE_ID, ROW_NO, COL_NO));
|
|
@ -1,17 +0,0 @@
|
|||||||
/**
|
|
||||||
@file
|
|
||||||
@brief DDL for mpe_xlmap_info
|
|
||||||
|
|
||||||
@version 9.3
|
|
||||||
@author 4GL Apps Ltd
|
|
||||||
@copyright 4GL Apps Ltd
|
|
||||||
**/
|
|
||||||
|
|
||||||
create table &curlib..mpe_xlmap_info(
|
|
||||||
tx_from num not null,
|
|
||||||
XLMAP_ID char(32) not null,
|
|
||||||
XLMAP_DESCRIPTION char(1000) not null,
|
|
||||||
XLMAP_TARGETLIBDS char(41) not null,
|
|
||||||
tx_to num not null,
|
|
||||||
constraint pk_mpe_xlmap_info
|
|
||||||
primary key(tx_from,XLMAP_ID));
|
|
@ -1,19 +0,0 @@
|
|||||||
/**
|
|
||||||
@file
|
|
||||||
@brief DDL for mpe_xlmap_rules
|
|
||||||
|
|
||||||
@version 9.3
|
|
||||||
@author 4GL Apps Ltd
|
|
||||||
@copyright 4GL Apps Ltd
|
|
||||||
**/
|
|
||||||
|
|
||||||
create table &curlib..mpe_xlmap_rules(
|
|
||||||
tx_from num not null,
|
|
||||||
XLMAP_ID char(32) not null,
|
|
||||||
XLMAP_RANGE_ID char(32) not null,
|
|
||||||
XLMAP_SHEET char(32) not null,
|
|
||||||
XLMAP_START char(1000) not null,
|
|
||||||
XLMAP_FINISH char(1000),
|
|
||||||
tx_to num not null,
|
|
||||||
constraint pk_mpe_xlmap_rules
|
|
||||||
primary key(tx_from,XLMAP_ID,XLMAP_RANGE_ID));
|
|
@ -1,44 +0,0 @@
|
|||||||
/**
|
|
||||||
@file
|
|
||||||
@brief migration script to move from v5 to v6.5 of data controller
|
|
||||||
|
|
||||||
**/
|
|
||||||
|
|
||||||
%let dclib=YOURDCLIB;
|
|
||||||
|
|
||||||
libname &dclib "/your/dc/path";
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Change 1
|
|
||||||
* New MPE_SUBMIT table
|
|
||||||
*/
|
|
||||||
proc sql;
|
|
||||||
create table &dclib..mpe_xlmap_rules(
|
|
||||||
tx_from num not null,
|
|
||||||
XLMAP_ID char(32) not null,
|
|
||||||
XLMAP_RANGE_ID char(32) not null,
|
|
||||||
XLMAP_SHEET char(32) not null,
|
|
||||||
XLMAP_START char(1000) not null,
|
|
||||||
XLMAP_FINISH char(1000),
|
|
||||||
tx_to num not null,
|
|
||||||
constraint pk_mpe_xlmap_rules
|
|
||||||
primary key(tx_from,XLMAP_ID,XLMAP_RANGE_ID));
|
|
||||||
|
|
||||||
create table &dclib..MPE_XLMAP_DATA(
|
|
||||||
LOAD_REF char(32) not null,
|
|
||||||
XLMAP_ID char(32) not null,
|
|
||||||
XLMAP_RANGE_ID char(32) not null,
|
|
||||||
ROW_NO num not null,
|
|
||||||
COL_NO num not null,
|
|
||||||
VALUE_TXT char(4000),
|
|
||||||
constraint pk_MPE_XLMAP_DATA
|
|
||||||
primary key(LOAD_REF, XLMAP_ID, XLMAP_RANGE_ID, ROW_NO, COL_NO));
|
|
||||||
|
|
||||||
create table &dclib..mpe_xlmap_info(
|
|
||||||
tx_from num not null,
|
|
||||||
XLMAP_ID char(32) not null,
|
|
||||||
XLMAP_DESCRIPTION char(1000) not null,
|
|
||||||
XLMAP_TARGETLIBDS char(41) not null,
|
|
||||||
tx_to num not null,
|
|
||||||
constraint pk_mpe_xlmap_info
|
|
||||||
primary key(tx_from,XLMAP_ID));
|
|
@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
This site contains the SAS code used in Data Controller for SAS. The pages were generated using [`sasjs doc`](https://cli.sasjs.io/doc).
|
This site contains the SAS code used in Data Controller for SAS. The pages were generated using [`sasjs doc`](https://cli.sasjs.io/doc).
|
||||||
|
|
||||||
You can download Data Controller from [here](https://git.datacontroller.io/dc/dc/releases).
|
You can download Data Controller from [here](https://4gl.uk/dcdeploy).
|
||||||
|
|
||||||
The main website is [https://datacontroller.io](https://datacontroller.io) and the user guide is [here](https://docs.datacontroller.io).
|
The main website is [https://datacontroller.io](https://datacontroller.io) and the user guide is [here](https://docs.datacontroller.io).
|
||||||
|
|
||||||
|
@ -24,7 +24,6 @@
|
|||||||
@li mp_lockanytable.sas
|
@li mp_lockanytable.sas
|
||||||
@li mpe_accesscheck.sas
|
@li mpe_accesscheck.sas
|
||||||
@li mpe_alerts.sas
|
@li mpe_alerts.sas
|
||||||
@li mpe_xlmapvalidate.sas
|
|
||||||
@li mpe_loadfail.sas
|
@li mpe_loadfail.sas
|
||||||
@li mpe_runhook.sas
|
@li mpe_runhook.sas
|
||||||
|
|
||||||
@ -451,7 +450,7 @@ run;
|
|||||||
%do i=1 %to %sysfunc(countw(&pk));
|
%do i=1 %to %sysfunc(countw(&pk));
|
||||||
%let iWord=%scan(&pk,&i);
|
%let iWord=%scan(&pk,&i);
|
||||||
call symputx('duplist',symget('duplist')!!
|
call symputx('duplist',symget('duplist')!!
|
||||||
" &iWord="!!cats(&iWord));
|
" &iWord="!!trim(&iWord));
|
||||||
%end;
|
%end;
|
||||||
run;
|
run;
|
||||||
%let msg=This upload contains duplicates on the Primary Key columns %trim(
|
%let msg=This upload contains duplicates on the Primary Key columns %trim(
|
||||||
@ -473,10 +472,6 @@ run;
|
|||||||
%return;
|
%return;
|
||||||
%end;
|
%end;
|
||||||
|
|
||||||
/* If a Complex Excel Upload, needs to have the load ref added to the table */
|
|
||||||
%mpe_xlmapvalidate(&mperef,work.staging_ds,&mpelib,&orig_libds)
|
|
||||||
|
|
||||||
/* Run the Post Edit Hook prior to creation of staging folder */
|
|
||||||
%mpe_runhook(POST_EDIT_HOOK)
|
%mpe_runhook(POST_EDIT_HOOK)
|
||||||
|
|
||||||
/* stop if err */
|
/* stop if err */
|
||||||
|
@ -269,210 +269,6 @@ insert into &lib..mpe_datadictionary set
|
|||||||
,DD_SENSITIVITY="Low"
|
,DD_SENSITIVITY="Low"
|
||||||
,tx_to='31DEC5999:23:59:59'dt;
|
,tx_to='31DEC5999:23:59:59'dt;
|
||||||
|
|
||||||
/**
|
|
||||||
* mpe_xlmap_info
|
|
||||||
*/
|
|
||||||
insert into &lib..mpe_xlmap_info set
|
|
||||||
tx_from=0
|
|
||||||
,tx_to='31DEC5999:23:59:59'dt
|
|
||||||
,xlmap_id='BASEL-KM1'
|
|
||||||
,xlmap_description='Basel 3 Key Metrics report'
|
|
||||||
,XLMAP_TARGETLIBDS="&lib..MPE_XLMAP_DATA";
|
|
||||||
|
|
||||||
/**
|
|
||||||
* mpe_xlmap_rules
|
|
||||||
*/
|
|
||||||
insert into &lib..mpe_xlmap_rules set
|
|
||||||
tx_from=0
|
|
||||||
,tx_to='31DEC5999:23:59:59'dt
|
|
||||||
,xlmap_id='BASEL-KM1'
|
|
||||||
,xlmap_range_id='KM1:a'
|
|
||||||
,xlmap_sheet='KM1'
|
|
||||||
,xlmap_start='MATCH 4 R[2]C[0]:a';
|
|
||||||
insert into &lib..mpe_xlmap_rules set
|
|
||||||
tx_from=0
|
|
||||||
,tx_to='31DEC5999:23:59:59'dt
|
|
||||||
,xlmap_id='BASEL-KM1'
|
|
||||||
,xlmap_range_id='KM1:b'
|
|
||||||
,xlmap_sheet='KM1'
|
|
||||||
,xlmap_start='MATCH 4 R[2]C[0]:b';
|
|
||||||
insert into &lib..mpe_xlmap_rules set
|
|
||||||
tx_from=0
|
|
||||||
,tx_to='31DEC5999:23:59:59'dt
|
|
||||||
,xlmap_id='BASEL-KM1'
|
|
||||||
,xlmap_range_id='KM1:c'
|
|
||||||
,xlmap_sheet='KM1'
|
|
||||||
,xlmap_start='MATCH 4 R[2]C[0]:c';
|
|
||||||
insert into &lib..mpe_xlmap_rules set
|
|
||||||
tx_from=0
|
|
||||||
,tx_to='31DEC5999:23:59:59'dt
|
|
||||||
,xlmap_id='BASEL-KM1'
|
|
||||||
,xlmap_range_id='KM1:d'
|
|
||||||
,xlmap_sheet='KM1'
|
|
||||||
,xlmap_start='MATCH 4 R[2]C[0]:d';
|
|
||||||
insert into &lib..mpe_xlmap_rules set
|
|
||||||
tx_from=0
|
|
||||||
,tx_to='31DEC5999:23:59:59'dt
|
|
||||||
,xlmap_id='BASEL-KM1'
|
|
||||||
,xlmap_range_id='KM1:e'
|
|
||||||
,xlmap_sheet='KM1'
|
|
||||||
,xlmap_start='MATCH 4 R[2]C[0]:e';
|
|
||||||
insert into &lib..mpe_xlmap_rules set
|
|
||||||
tx_from=0
|
|
||||||
,tx_to='31DEC5999:23:59:59'dt
|
|
||||||
,xlmap_id='BASEL-KM1'
|
|
||||||
,xlmap_range_id='KM1:f'
|
|
||||||
,xlmap_sheet='KM1'
|
|
||||||
,xlmap_start='MATCH 4 R[2]C[0]:f';
|
|
||||||
insert into &lib..mpe_xlmap_rules set
|
|
||||||
tx_from=0
|
|
||||||
,tx_to='31DEC5999:23:59:59'dt
|
|
||||||
,xlmap_id='BASEL-KM1'
|
|
||||||
,xlmap_range_id='KM1:1/a'
|
|
||||||
,xlmap_sheet='KM1'
|
|
||||||
,xlmap_start='MATCH C R[0]C[1]:Common Equity Tier 1 (CET1)';
|
|
||||||
insert into &lib..mpe_xlmap_rules set
|
|
||||||
tx_from=0
|
|
||||||
,tx_to='31DEC5999:23:59:59'dt
|
|
||||||
,xlmap_id='BASEL-KM1'
|
|
||||||
,xlmap_range_id='KM1:1/b'
|
|
||||||
,xlmap_sheet='KM1'
|
|
||||||
,xlmap_start='MATCH C R[0]C[2]:Common Equity Tier 1 (CET1)';
|
|
||||||
insert into &lib..mpe_xlmap_rules set
|
|
||||||
tx_from=0
|
|
||||||
,tx_to='31DEC5999:23:59:59'dt
|
|
||||||
,xlmap_id='BASEL-KM1'
|
|
||||||
,xlmap_range_id='KM1:1/c'
|
|
||||||
,xlmap_sheet='KM1'
|
|
||||||
,xlmap_start='MATCH C R[0]C[3]:Common Equity Tier 1 (CET1)';
|
|
||||||
insert into &lib..mpe_xlmap_rules set
|
|
||||||
tx_from=0
|
|
||||||
,tx_to='31DEC5999:23:59:59'dt
|
|
||||||
,xlmap_id='BASEL-KM1'
|
|
||||||
,xlmap_range_id='KM1:1/d'
|
|
||||||
,xlmap_sheet='KM1'
|
|
||||||
,xlmap_start='MATCH C R[0]C[4]:Common Equity Tier 1 (CET1)';
|
|
||||||
insert into &lib..mpe_xlmap_rules set
|
|
||||||
tx_from=0
|
|
||||||
,tx_to='31DEC5999:23:59:59'dt
|
|
||||||
,xlmap_id='BASEL-KM1'
|
|
||||||
,xlmap_range_id='KM1:1/e'
|
|
||||||
,xlmap_sheet='KM1'
|
|
||||||
,xlmap_start='MATCH C R[0]C[5]:Common Equity Tier 1 (CET1)';
|
|
||||||
insert into &lib..mpe_xlmap_rules set
|
|
||||||
tx_from=0
|
|
||||||
,tx_to='31DEC5999:23:59:59'dt
|
|
||||||
,xlmap_id='BASEL-KM1'
|
|
||||||
,xlmap_range_id='KM1:1/f'
|
|
||||||
,xlmap_sheet='KM1'
|
|
||||||
,xlmap_start='MATCH C R[0]C[6]:Common Equity Tier 1 (CET1)';
|
|
||||||
insert into &lib..mpe_xlmap_rules set
|
|
||||||
tx_from=0
|
|
||||||
,tx_to='31DEC5999:23:59:59'dt
|
|
||||||
,xlmap_id='BASEL-KM1'
|
|
||||||
,xlmap_range_id='KM1:1a/e'
|
|
||||||
,xlmap_sheet='KM1'
|
|
||||||
,xlmap_start='MATCH C R[1]C[5]:Common Equity Tier 1 (CET1)';
|
|
||||||
insert into &lib..mpe_xlmap_rules set
|
|
||||||
tx_from=0
|
|
||||||
,tx_to='31DEC5999:23:59:59'dt
|
|
||||||
,xlmap_id='BASEL-KM1'
|
|
||||||
,xlmap_range_id='KM1:1a/f'
|
|
||||||
,xlmap_sheet='KM1'
|
|
||||||
,xlmap_start='MATCH C R[1]C[6]:Common Equity Tier 1 (CET1)';
|
|
||||||
insert into &lib..mpe_xlmap_rules set
|
|
||||||
tx_from=0
|
|
||||||
,tx_to='31DEC5999:23:59:59'dt
|
|
||||||
,xlmap_id='BASEL-KM1'
|
|
||||||
,xlmap_range_id='KM1:2/a'
|
|
||||||
,xlmap_sheet='KM1'
|
|
||||||
,xlmap_start='ABSOLUTE D10';
|
|
||||||
insert into &lib..mpe_xlmap_rules set
|
|
||||||
tx_from=0
|
|
||||||
,tx_to='31DEC5999:23:59:59'dt
|
|
||||||
,xlmap_id='BASEL-KM1'
|
|
||||||
,xlmap_range_id='KM1:2/b'
|
|
||||||
,xlmap_sheet='/3'
|
|
||||||
,xlmap_start='ABSOLUTE E10';
|
|
||||||
insert into &lib..mpe_xlmap_rules set
|
|
||||||
tx_from=0
|
|
||||||
,tx_to='31DEC5999:23:59:59'dt
|
|
||||||
,xlmap_id='BASEL-KM1'
|
|
||||||
,xlmap_range_id='KM1:2/c'
|
|
||||||
,xlmap_sheet='/3'
|
|
||||||
,xlmap_start='RELATIVE R[10]C[6]';
|
|
||||||
insert into &lib..mpe_xlmap_rules set
|
|
||||||
tx_from=0
|
|
||||||
,tx_to='31DEC5999:23:59:59'dt
|
|
||||||
,xlmap_id='BASEL-KM1'
|
|
||||||
,xlmap_range_id='KM1:2/d'
|
|
||||||
,xlmap_sheet='/3'
|
|
||||||
,xlmap_start='RELATIVE R[10]C[8]';
|
|
||||||
insert into &lib..mpe_xlmap_rules set
|
|
||||||
tx_from=0
|
|
||||||
,tx_to='31DEC5999:23:59:59'dt
|
|
||||||
,xlmap_id='BASEL-KM1'
|
|
||||||
,xlmap_range_id='KM1:2/e'
|
|
||||||
,xlmap_sheet='/3'
|
|
||||||
,xlmap_start='RELATIVE R[10]C[9]';
|
|
||||||
insert into &lib..mpe_xlmap_rules set
|
|
||||||
tx_from=0
|
|
||||||
,tx_to='31DEC5999:23:59:59'dt
|
|
||||||
,xlmap_id='BASEL-KM1'
|
|
||||||
,xlmap_range_id='KM1:2/f'
|
|
||||||
,xlmap_sheet='/3'
|
|
||||||
,xlmap_start='RELATIVE R[10]C[10]';
|
|
||||||
insert into &lib..mpe_xlmap_rules set
|
|
||||||
tx_from=0
|
|
||||||
,tx_to='31DEC5999:23:59:59'dt
|
|
||||||
,xlmap_id='BASEL-KM1'
|
|
||||||
,xlmap_range_id='KM1:2a'
|
|
||||||
,xlmap_sheet='KM1'
|
|
||||||
,xlmap_start='ABSOLUTE H11'
|
|
||||||
,xlmap_finish='RELATIVE R[0]C[1]';
|
|
||||||
insert into &lib..mpe_xlmap_rules set
|
|
||||||
tx_from=0
|
|
||||||
,tx_to='31DEC5999:23:59:59'dt
|
|
||||||
,xlmap_id='BASEL-KM1'
|
|
||||||
,xlmap_range_id='KM1:3'
|
|
||||||
,xlmap_sheet='KM1'
|
|
||||||
,xlmap_start='RELATIVE R[12]C[4]'
|
|
||||||
,xlmap_finish='ABSOLUTE I13';
|
|
||||||
insert into &lib..mpe_xlmap_rules set
|
|
||||||
tx_from=0
|
|
||||||
,tx_to='31DEC5999:23:59:59'dt
|
|
||||||
,xlmap_id='BASEL-CR2'
|
|
||||||
,xlmap_range_id='CR2-sec1'
|
|
||||||
,xlmap_sheet='CR2'
|
|
||||||
,xlmap_start='ABSOLUTE D8'
|
|
||||||
,xlmap_finish='BLANKROW';
|
|
||||||
insert into &lib..mpe_xlmap_rules set
|
|
||||||
tx_from=0
|
|
||||||
,tx_to='31DEC5999:23:59:59'dt
|
|
||||||
,xlmap_id='BASEL-CR2'
|
|
||||||
,xlmap_range_id='CR2-sec2'
|
|
||||||
,xlmap_sheet='CR2'
|
|
||||||
,xlmap_start='ABSOLUTE D18'
|
|
||||||
,xlmap_finish='LASTDOWN';
|
|
||||||
insert into &lib..mpe_xlmap_rules set
|
|
||||||
tx_from=0
|
|
||||||
,tx_to='31DEC5999:23:59:59'dt
|
|
||||||
,xlmap_id='SAMPLE'
|
|
||||||
,xlmap_range_id='header'
|
|
||||||
,xlmap_sheet='/1'
|
|
||||||
,xlmap_start='ABSOLUTE B3'
|
|
||||||
,xlmap_finish='ABSOLUTE B8';
|
|
||||||
insert into &lib..mpe_xlmap_rules set
|
|
||||||
tx_from=0
|
|
||||||
,tx_to='31DEC5999:23:59:59'dt
|
|
||||||
,xlmap_id='SAMPLE'
|
|
||||||
,xlmap_range_id='data'
|
|
||||||
,xlmap_sheet='/1'
|
|
||||||
,xlmap_start='ABSOLUTE B13'
|
|
||||||
,xlmap_finish='ABSOLUTE E16';
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* MPE_GROUPS
|
* MPE_GROUPS
|
||||||
*/
|
*/
|
||||||
@ -1185,42 +981,6 @@ insert into &lib..mpe_selectbox set
|
|||||||
,notes='Docs: https://docs.datacontroller.io/column-level-security'
|
,notes='Docs: https://docs.datacontroller.io/column-level-security'
|
||||||
,post_edit_hook='services/hooks/mpe_column_level_security_postedit'
|
,post_edit_hook='services/hooks/mpe_column_level_security_postedit'
|
||||||
;
|
;
|
||||||
insert into &lib..mpe_tables
|
|
||||||
set tx_from=0
|
|
||||||
,tx_to='31DEC5999:23:59:59'dt
|
|
||||||
,libref="&lib"
|
|
||||||
,dsn='MPE_XLMAP_INFO'
|
|
||||||
,num_of_approvals_required=1
|
|
||||||
,loadtype='TXTEMPORAL'
|
|
||||||
,var_txfrom='TX_FROM'
|
|
||||||
,var_txto='TX_TO'
|
|
||||||
,buskey='XLMAP_ID'
|
|
||||||
,notes='Docs: https://docs.datacontroller.io/complex-excel-uploads'
|
|
||||||
,post_edit_hook='services/hooks/mpe_xlmap_info_postedit'
|
|
||||||
;
|
|
||||||
insert into &lib..mpe_tables
|
|
||||||
set tx_from=0
|
|
||||||
,tx_to='31DEC5999:23:59:59'dt
|
|
||||||
,libref="&lib"
|
|
||||||
,dsn='MPE_XLMAP_RULES'
|
|
||||||
,num_of_approvals_required=1
|
|
||||||
,loadtype='TXTEMPORAL'
|
|
||||||
,var_txfrom='TX_FROM'
|
|
||||||
,var_txto='TX_TO'
|
|
||||||
,buskey='XLMAP_ID XLMAP_RANGE_ID'
|
|
||||||
,notes='Docs: https://docs.datacontroller.io/complex-excel-uploads'
|
|
||||||
,post_edit_hook='services/hooks/mpe_xlmap_rules_postedit'
|
|
||||||
;
|
|
||||||
insert into &lib..mpe_tables
|
|
||||||
set tx_from=0
|
|
||||||
,tx_to='31DEC5999:23:59:59'dt
|
|
||||||
,libref="&lib"
|
|
||||||
,dsn='MPE_XLMAP_DATA'
|
|
||||||
,num_of_approvals_required=1
|
|
||||||
,loadtype='UPDATE'
|
|
||||||
,buskey='LOAD_REF XLMAP_ID XLMAP_RANGE_ID ROW_NO COL_NO'
|
|
||||||
,notes='Docs: https://docs.datacontroller.io/complex-excel-uploads'
|
|
||||||
;
|
|
||||||
insert into &lib..mpe_tables
|
insert into &lib..mpe_tables
|
||||||
set tx_from=0
|
set tx_from=0
|
||||||
,tx_to='31DEC5999:23:59:59'dt
|
,tx_to='31DEC5999:23:59:59'dt
|
||||||
@ -1493,27 +1253,6 @@ insert into &lib..MPE_VALIDATIONS set
|
|||||||
,rule_value="services/validations/mpe_alerts.alert_lib"
|
,rule_value="services/validations/mpe_alerts.alert_lib"
|
||||||
,rule_active=1
|
,rule_active=1
|
||||||
,tx_to='31DEC5999:23:59:59'dt;
|
,tx_to='31DEC5999:23:59:59'dt;
|
||||||
|
|
||||||
insert into &lib..MPE_VALIDATIONS set
|
|
||||||
tx_from=0
|
|
||||||
,base_lib="&lib"
|
|
||||||
,base_ds="MPE_XLMAP_INFO"
|
|
||||||
,base_col="XLMAP_ID"
|
|
||||||
,rule_type='CASE'
|
|
||||||
,rule_value='UPCASE'
|
|
||||||
,rule_active=1
|
|
||||||
,tx_to='31DEC5999:23:59:59'dt;
|
|
||||||
|
|
||||||
insert into &lib..MPE_VALIDATIONS set
|
|
||||||
tx_from=0
|
|
||||||
,base_lib="&lib"
|
|
||||||
,base_ds="MPE_XLMAP_RULES"
|
|
||||||
,base_col="XLMAP_ID"
|
|
||||||
,rule_type='CASE'
|
|
||||||
,rule_value='UPCASE'
|
|
||||||
,rule_active=1
|
|
||||||
,tx_to='31DEC5999:23:59:59'dt;
|
|
||||||
|
|
||||||
insert into &lib..MPE_VALIDATIONS set
|
insert into &lib..MPE_VALIDATIONS set
|
||||||
tx_from=0
|
tx_from=0
|
||||||
,base_lib="&lib"
|
,base_lib="&lib"
|
||||||
@ -1901,16 +1640,6 @@ insert into &lib..MPE_VALIDATIONS set
|
|||||||
,rule_value='1'
|
,rule_value='1'
|
||||||
,rule_active=1
|
,rule_active=1
|
||||||
,tx_to='31DEC5999:23:59:59'dt;
|
,tx_to='31DEC5999:23:59:59'dt;
|
||||||
insert into &lib..MPE_VALIDATIONS set
|
|
||||||
tx_from=0
|
|
||||||
,base_lib="&lib"
|
|
||||||
,base_ds="MPE_XLMAP_INFO"
|
|
||||||
,base_col="XLMAP_ID"
|
|
||||||
,rule_type='SOFTSELECT'
|
|
||||||
,rule_value="&lib..MPE_XLMAP_RULES.XLMAP_ID"
|
|
||||||
,rule_active=1
|
|
||||||
,tx_to='31DEC5999:23:59:59'dt;
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* MPE_X_TEST
|
* MPE_X_TEST
|
||||||
|
@ -268,55 +268,6 @@ proc datasets lib=&lib noprint;
|
|||||||
pk_mpe_excel_config=(tx_to xl_libref xl_table xl_column)
|
pk_mpe_excel_config=(tx_to xl_libref xl_table xl_column)
|
||||||
/nomiss unique;
|
/nomiss unique;
|
||||||
quit;
|
quit;
|
||||||
|
|
||||||
proc sql;
|
|
||||||
create table &lib..MPE_XLMAP_DATA(
|
|
||||||
LOAD_REF char(32) ¬null,
|
|
||||||
XLMAP_ID char(32) ¬null,
|
|
||||||
XLMAP_RANGE_ID char(32) ¬null,
|
|
||||||
ROW_NO num ¬null,
|
|
||||||
COL_NO num ¬null,
|
|
||||||
VALUE_TXT char(4000)
|
|
||||||
);quit;
|
|
||||||
proc datasets lib=&lib noprint;
|
|
||||||
modify MPE_XLMAP_DATA;
|
|
||||||
index create
|
|
||||||
pk_MPE_XLMAP_DATA=(load_ref xlmap_id xlmap_range_id row_no col_no)
|
|
||||||
/nomiss unique;
|
|
||||||
quit;
|
|
||||||
|
|
||||||
proc sql;
|
|
||||||
create table &lib..mpe_xlmap_info(
|
|
||||||
tx_from num ¬null,
|
|
||||||
tx_to num ¬null,
|
|
||||||
XLMAP_ID char(32) ¬null,
|
|
||||||
XLMAP_DESCRIPTION char(1000) ¬null,
|
|
||||||
XLMAP_TARGETLIBDS char(41) ¬null
|
|
||||||
);quit;
|
|
||||||
proc datasets lib=&lib noprint;
|
|
||||||
modify mpe_xlmap_info;
|
|
||||||
index create
|
|
||||||
pk_mpe_xlmap_info=(tx_to xlmap_id)
|
|
||||||
/nomiss unique;
|
|
||||||
quit;
|
|
||||||
|
|
||||||
proc sql;
|
|
||||||
create table &lib..mpe_xlmap_rules(
|
|
||||||
tx_from num ¬null,
|
|
||||||
tx_to num ¬null,
|
|
||||||
XLMAP_ID char(32) ¬null,
|
|
||||||
XLMAP_RANGE_ID char(32) ¬null,
|
|
||||||
XLMAP_SHEET char(32) ¬null,
|
|
||||||
XLMAP_START char(1000) ¬null,
|
|
||||||
XLMAP_FINISH char(1000)
|
|
||||||
);quit;
|
|
||||||
proc datasets lib=&lib noprint;
|
|
||||||
modify mpe_xlmap_rules;
|
|
||||||
index create
|
|
||||||
pk_mpe_xlmap_rules=(tx_to xlmap_id xlmap_range_id)
|
|
||||||
/nomiss unique;
|
|
||||||
quit;
|
|
||||||
|
|
||||||
proc sql;
|
proc sql;
|
||||||
create table &lib..mpe_filteranytable(
|
create table &lib..mpe_filteranytable(
|
||||||
filter_rk num ¬null,
|
filter_rk num ¬null,
|
||||||
|
@ -1,28 +0,0 @@
|
|||||||
/**
|
|
||||||
@file
|
|
||||||
@brief Validates excel map structure and adds load ref
|
|
||||||
@details Used in staging, prior to the post edit hook
|
|
||||||
|
|
||||||
@version 9.2
|
|
||||||
@author 4GL Apps Ltd
|
|
||||||
@copyright 4GL Apps Ltd. This code may only be used within Data Controller
|
|
||||||
and may not be re-distributed or re-sold without the express permission of
|
|
||||||
4GL Apps Ltd.
|
|
||||||
**/
|
|
||||||
|
|
||||||
%macro mpe_xlmapvalidate(mperef,inds,dclib,tgtds);
|
|
||||||
|
|
||||||
%local ismap;
|
|
||||||
proc sql noprint;
|
|
||||||
select count(*) into: ismap
|
|
||||||
from &dclib..mpe_xlmap_info
|
|
||||||
where XLMAP_TARGETLIBDS="&tgtds" and &dc_dttmtfmt. le TX_TO ;
|
|
||||||
|
|
||||||
%if "&tgtds"="&dclib..MPE_XLMAP_DATA" or &ismap>0 %then %do;
|
|
||||||
data &inds;
|
|
||||||
set &inds;
|
|
||||||
LOAD_REF="&mperef";
|
|
||||||
run;
|
|
||||||
%end;
|
|
||||||
|
|
||||||
%mend mpe_xlmapvalidate;
|
|
@ -1,59 +0,0 @@
|
|||||||
/**
|
|
||||||
@file
|
|
||||||
@brief Testing mpe_xlmapvalidate macro
|
|
||||||
|
|
||||||
<h4> SAS Macros </h4>
|
|
||||||
@li mpe_xlmapvalidate.sas
|
|
||||||
@li mp_assert.sas
|
|
||||||
@li mp_assertscope.sas
|
|
||||||
|
|
||||||
<h4> SAS Includes </h4>
|
|
||||||
@li mpe_xlmap_data.ddl ul
|
|
||||||
|
|
||||||
@author 4GL Apps Ltd
|
|
||||||
@copyright 4GL Apps Ltd. This code may only be used within Data Controller
|
|
||||||
and may not be re-distributed or re-sold without the express permission of
|
|
||||||
4GL Apps Ltd.
|
|
||||||
|
|
||||||
**/
|
|
||||||
|
|
||||||
/* create the table */
|
|
||||||
%let curlib=work;
|
|
||||||
proc sql;
|
|
||||||
%inc ul;
|
|
||||||
|
|
||||||
data work.test1;
|
|
||||||
if 0 then set work.MPE_XLMAP_DATA;
|
|
||||||
LOAD_REF='0';
|
|
||||||
XLMAP_ID='Sample';
|
|
||||||
XLMAP_RANGE_ID='Range 1';
|
|
||||||
ROW_NO=1;
|
|
||||||
COL_NO=2;
|
|
||||||
VALUE_TXT='something';
|
|
||||||
run;
|
|
||||||
|
|
||||||
%mp_assertscope(SNAPSHOT)
|
|
||||||
%mpe_xlmapvalidate(DCTEST1,work.test1,&dclib,NOT.MAP)
|
|
||||||
%mp_assertscope(COMPARE,
|
|
||||||
desc=Checking macro variables against previous snapshot
|
|
||||||
)
|
|
||||||
|
|
||||||
data _null_;
|
|
||||||
set work.test1;
|
|
||||||
call symputx('test1',load_ref);
|
|
||||||
run;
|
|
||||||
%mp_assert(
|
|
||||||
iftrue=(&test1=0),
|
|
||||||
desc=Checking load ref was not applied
|
|
||||||
)
|
|
||||||
|
|
||||||
%mpe_xlmapvalidate(DCTEST2,work.test1,&dclib,&dclib..MPE_XLMAP_DATA)
|
|
||||||
|
|
||||||
data _null_;
|
|
||||||
set work.test1;
|
|
||||||
call symputx('test2',load_ref);
|
|
||||||
run;
|
|
||||||
%mp_assert(
|
|
||||||
iftrue=(&test2=DCTEST2),
|
|
||||||
desc=Checking load ref was applied for default case
|
|
||||||
)
|
|
@ -207,7 +207,7 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "4gl",
|
"name": "4gl",
|
||||||
"serverUrl": "https://sas.4gl.io",
|
"serverUrl": "https://sas9.4gl.io",
|
||||||
"serverType": "SASJS",
|
"serverType": "SASJS",
|
||||||
"httpsAgentOptions": {
|
"httpsAgentOptions": {
|
||||||
"allowInsecureRequests": false
|
"allowInsecureRequests": false
|
||||||
|
@ -56,12 +56,12 @@
|
|||||||
data _null_;
|
data _null_;
|
||||||
set work.sascontroltable;
|
set work.sascontroltable;
|
||||||
call symputx('ACTION',ACTION);
|
call symputx('ACTION',ACTION);
|
||||||
call symputx('LOAD_REF',TABLE);
|
call symputx('TABLE',TABLE);
|
||||||
/* DIFFTIME is when the DIFF was generated on the frontend */
|
/* DIFFTIME is when the DIFF was generated on the frontend */
|
||||||
call symputx('DIFFTIME',DIFFTIME);
|
call symputx('DIFFTIME',DIFFTIME);
|
||||||
run;
|
run;
|
||||||
|
|
||||||
%global action is_err err_msg msg;
|
%global action is_err err_msg;
|
||||||
%let is_err=0;
|
%let is_err=0;
|
||||||
|
|
||||||
%let user=%mf_getuser();
|
%let user=%mf_getuser();
|
||||||
@ -80,7 +80,7 @@ RUN;
|
|||||||
%let isfmtcat=0;
|
%let isfmtcat=0;
|
||||||
data APPROVE1;
|
data APPROVE1;
|
||||||
set &mpelib..mpe_submit;
|
set &mpelib..mpe_submit;
|
||||||
where TABLE_ID="&LOAD_REF";
|
where TABLE_ID="&TABLE";
|
||||||
/* fetch mpe_submit data */
|
/* fetch mpe_submit data */
|
||||||
libds=cats(base_lib,'.',base_ds);
|
libds=cats(base_lib,'.',base_ds);
|
||||||
REVIEWED_ON=put(reviewed_on_dttm,datetime19.);
|
REVIEWED_ON=put(reviewed_on_dttm,datetime19.);
|
||||||
@ -115,9 +115,9 @@ run;
|
|||||||
)
|
)
|
||||||
|
|
||||||
%mp_abort(
|
%mp_abort(
|
||||||
iftrue=(%mf_verifymacvars(difftime orig_libds libds load_ref)=0)
|
iftrue=(%mf_verifymacvars(difftime orig_libds libds table)=0)
|
||||||
,mac=&_program
|
,mac=&_program
|
||||||
,msg=%str(Missing: difftime orig_libds libds load_ref)
|
,msg=%str(Missing: difftime orig_libds libds table)
|
||||||
)
|
)
|
||||||
|
|
||||||
/* security checks */
|
/* security checks */
|
||||||
@ -186,7 +186,7 @@ run;
|
|||||||
%let prev_upload_check=1;
|
%let prev_upload_check=1;
|
||||||
proc sql;
|
proc sql;
|
||||||
select count(*) into: prev_upload_check from &mpelib..mpe_review
|
select count(*) into: prev_upload_check from &mpelib..mpe_review
|
||||||
where TABLE_ID="&LOAD_REF" and REVIEWED_BY_NM="&user"
|
where TABLE_ID="&TABLE" and REVIEWED_BY_NM="&user"
|
||||||
and REVIEW_STATUS_ID ne "SUBMITTED";
|
and REVIEW_STATUS_ID ne "SUBMITTED";
|
||||||
%let authcheck=%mf_getattrn(work.authAPP,NLOBS);
|
%let authcheck=%mf_getattrn(work.authAPP,NLOBS);
|
||||||
%if &authcheck=0 or &prev_upload_check=1 %then %do;
|
%if &authcheck=0 or &prev_upload_check=1 %then %do;
|
||||||
@ -233,7 +233,7 @@ run;
|
|||||||
%else %let oldloc=%qsysfunc(getoption(LOG));
|
%else %let oldloc=%qsysfunc(getoption(LOG));
|
||||||
%if %length(&oldloc)>0 %then %do;
|
%if %length(&oldloc)>0 %then %do;
|
||||||
proc printto
|
proc printto
|
||||||
log="&mpelocapprovals/&LOAD_REF/approval.log";
|
log="&mpelocapprovals/&TABLE/approval.log";
|
||||||
run;
|
run;
|
||||||
data _null_;
|
data _null_;
|
||||||
if _n_=1 then do;
|
if _n_=1 then do;
|
||||||
@ -247,7 +247,7 @@ run;
|
|||||||
%end;
|
%end;
|
||||||
%else %do;
|
%else %do;
|
||||||
proc printto
|
proc printto
|
||||||
log="&mpelocapprovals/&LOAD_REF/approval.log";
|
log="&mpelocapprovals/&TABLE/approval.log";
|
||||||
run;
|
run;
|
||||||
%end;
|
%end;
|
||||||
|
|
||||||
@ -285,11 +285,11 @@ select PRE_APPROVE_HOOK, POST_APPROVE_HOOK, LOADTYPE, var_txfrom, var_txto
|
|||||||
,msg=%str(Missing: mpelocapprovals orig_libds)
|
,msg=%str(Missing: mpelocapprovals orig_libds)
|
||||||
)
|
)
|
||||||
|
|
||||||
/* get dataset from approvals location (has same name as load_ref) */
|
/* get dataset from approvals location */
|
||||||
%let tmplib=%mf_getuniquelibref();
|
%let tmplib=%mf_getuniquelibref();
|
||||||
libname &tmplib "&mpelocapprovals/&LOAD_REF";
|
libname &tmplib "&mpelocapprovals/&TABLE";
|
||||||
data STAGING_DS;
|
data STAGING_DS;
|
||||||
set &tmplib..&LOAD_REF;
|
set &tmplib..&TABLE;
|
||||||
run;
|
run;
|
||||||
|
|
||||||
%mp_abort(iftrue= (&syscc ne 0)
|
%mp_abort(iftrue= (&syscc ne 0)
|
||||||
@ -313,7 +313,7 @@ run;
|
|||||||
%let apprno=%eval(&num_of_approvals_required-&num_of_approvals_remaining+1);
|
%let apprno=%eval(&num_of_approvals_required-&num_of_approvals_remaining+1);
|
||||||
data work.append_review;
|
data work.append_review;
|
||||||
if 0 then set &mpelib..mpe_review;
|
if 0 then set &mpelib..mpe_review;
|
||||||
TABLE_ID="&LOAD_REF";
|
TABLE_ID="&TABLE";
|
||||||
BASE_TABLE="&orig_libds";
|
BASE_TABLE="&orig_libds";
|
||||||
REVIEW_STATUS_ID="APPROVED";
|
REVIEW_STATUS_ID="APPROVED";
|
||||||
REVIEWED_BY_NM="&user";
|
REVIEWED_BY_NM="&user";
|
||||||
@ -323,7 +323,7 @@ run;
|
|||||||
stop;
|
stop;
|
||||||
run;
|
run;
|
||||||
%mp_lockanytable(LOCK,
|
%mp_lockanytable(LOCK,
|
||||||
lib=&mpelib,ds=mpe_review,ref=%str(&LOAD_REF Approval),
|
lib=&mpelib,ds=mpe_review,ref=%str(&table Approval),
|
||||||
ctl_ds=&mpelib..mpe_lockanytable
|
ctl_ds=&mpelib..mpe_lockanytable
|
||||||
)
|
)
|
||||||
proc append base=&mpelib..mpe_review data=work.append_review;
|
proc append base=&mpelib..mpe_review data=work.append_review;
|
||||||
@ -335,7 +335,7 @@ run;
|
|||||||
|
|
||||||
/* update mpe_submit table */
|
/* update mpe_submit table */
|
||||||
%mp_lockanytable(LOCK,
|
%mp_lockanytable(LOCK,
|
||||||
lib=&mpelib,ds=mpe_submit,ref=%str(&LOAD_REF Approval),
|
lib=&mpelib,ds=mpe_submit,ref=%str(&table Approval),
|
||||||
ctl_ds=&mpelib..mpe_lockanytable
|
ctl_ds=&mpelib..mpe_lockanytable
|
||||||
)
|
)
|
||||||
proc sql;
|
proc sql;
|
||||||
@ -343,7 +343,7 @@ run;
|
|||||||
set num_of_approvals_remaining=&num_of_approvals_remaining-1,
|
set num_of_approvals_remaining=&num_of_approvals_remaining-1,
|
||||||
reviewed_by_nm="&user",
|
reviewed_by_nm="&user",
|
||||||
reviewed_on_dttm=&sastime
|
reviewed_on_dttm=&sastime
|
||||||
where table_id="&LOAD_REF";
|
where table_id="&table";
|
||||||
%mp_lockanytable(UNLOCK,
|
%mp_lockanytable(UNLOCK,
|
||||||
lib=&mpelib,ds=mpe_submit,
|
lib=&mpelib,ds=mpe_submit,
|
||||||
ctl_ds=&mpelib..mpe_lockanytable
|
ctl_ds=&mpelib..mpe_lockanytable
|
||||||
@ -369,7 +369,7 @@ run;
|
|||||||
)
|
)
|
||||||
%mpe_targetloader(libds=&orig_libds
|
%mpe_targetloader(libds=&orig_libds
|
||||||
,now= &sastime
|
,now= &sastime
|
||||||
,etlsource=&LOAD_REF
|
,etlsource=&TABLE
|
||||||
,STAGING_DS=STAGING_DS
|
,STAGING_DS=STAGING_DS
|
||||||
,dclib=&mpelib
|
,dclib=&mpelib
|
||||||
%if &action=APPROVE_TABLE %then %do;
|
%if &action=APPROVE_TABLE %then %do;
|
||||||
@ -405,7 +405,7 @@ run;
|
|||||||
proc sql noprint;
|
proc sql noprint;
|
||||||
select max(processed_dttm)-1 format=datetime19. into: tstamp
|
select max(processed_dttm)-1 format=datetime19. into: tstamp
|
||||||
from &mpelib..mpe_dataloads
|
from &mpelib..mpe_dataloads
|
||||||
where libref="&libref" and dsn="&ds" and ETLSOURCE="&LOAD_REF";
|
where libref="&libref" and dsn="&ds" and ETLSOURCE="&TABLE";
|
||||||
quit;
|
quit;
|
||||||
%if &tstamp=. %then %let tstamp=%sysfunc(datetime(),datetime19.);
|
%if &tstamp=. %then %let tstamp=%sysfunc(datetime(),datetime19.);
|
||||||
|
|
||||||
@ -498,7 +498,7 @@ run;
|
|||||||
else if _____orig then _____status='ORIGINAL';
|
else if _____orig then _____status='ORIGINAL';
|
||||||
run;
|
run;
|
||||||
proc export data=TEMPDIFFS dbms=csv replace
|
proc export data=TEMPDIFFS dbms=csv replace
|
||||||
outfile="&mpelocapprovals/&LOAD_REF/&tempDIFFS_CSV" ;
|
outfile="&mpelocapprovals/&TABLE/&tempDIFFS_CSV" ;
|
||||||
run;
|
run;
|
||||||
proc sql noprint;
|
proc sql noprint;
|
||||||
select filesize format=sizekmg10.1, filesize as filesize_raw
|
select filesize format=sizekmg10.1, filesize as filesize_raw
|
||||||
@ -545,7 +545,7 @@ run;
|
|||||||
proc sort data=&mpelib..mpe_submit(where=(
|
proc sort data=&mpelib..mpe_submit(where=(
|
||||||
submit_status_cd='SUBMITTED'
|
submit_status_cd='SUBMITTED'
|
||||||
and cats(base_lib,'.',base_ds)="&orig_libds"
|
and cats(base_lib,'.',base_ds)="&orig_libds"
|
||||||
and table_id ne "&LOAD_REF"
|
and table_id ne "&TABLE"
|
||||||
)) out=submits;
|
)) out=submits;
|
||||||
by descending submitted_on_dttm;
|
by descending submitted_on_dttm;
|
||||||
run;
|
run;
|
||||||
@ -599,7 +599,7 @@ run;
|
|||||||
data work.outds_mod; run;
|
data work.outds_mod; run;
|
||||||
data work.outds_del; run;
|
data work.outds_del; run;
|
||||||
%end;
|
%end;
|
||||||
libname approve "&mpelocapprovals/&LOAD_REF";
|
libname approve "&mpelocapprovals/&TABLE";
|
||||||
data; set &libds;stop;run;
|
data; set &libds;stop;run;
|
||||||
%let emptybasetable=&syslast;
|
%let emptybasetable=&syslast;
|
||||||
data approve.ActualDiffs;
|
data approve.ActualDiffs;
|
||||||
@ -621,7 +621,7 @@ run;
|
|||||||
run;
|
run;
|
||||||
|
|
||||||
proc export data=approve.ActualDiffs
|
proc export data=approve.ActualDiffs
|
||||||
outfile="&mpelocapprovals/&LOAD_REF/ActualDiffs.csv"
|
outfile="&mpelocapprovals/&TABLE/ActualDiffs.csv"
|
||||||
dbms=csv
|
dbms=csv
|
||||||
replace;
|
replace;
|
||||||
run;
|
run;
|
||||||
@ -631,7 +631,7 @@ run;
|
|||||||
%let apprno=%eval(&num_of_approvals_required-&num_of_approvals_remaining+1);
|
%let apprno=%eval(&num_of_approvals_required-&num_of_approvals_remaining+1);
|
||||||
data work.append_review;
|
data work.append_review;
|
||||||
if 0 then set &mpelib..mpe_review;
|
if 0 then set &mpelib..mpe_review;
|
||||||
TABLE_ID="&LOAD_REF";
|
TABLE_ID="&TABLE";
|
||||||
BASE_TABLE="&orig_libds";
|
BASE_TABLE="&orig_libds";
|
||||||
REVIEW_STATUS_ID="APPROVED";
|
REVIEW_STATUS_ID="APPROVED";
|
||||||
REVIEWED_BY_NM="&user";
|
REVIEWED_BY_NM="&user";
|
||||||
@ -641,7 +641,7 @@ run;
|
|||||||
stop;
|
stop;
|
||||||
run;
|
run;
|
||||||
%mp_lockanytable(LOCK,
|
%mp_lockanytable(LOCK,
|
||||||
lib=&mpelib,ds=mpe_review,ref=%str(&LOAD_REF Approval),
|
lib=&mpelib,ds=mpe_review,ref=%str(&table Approval),
|
||||||
ctl_ds=&mpelib..mpe_lockanytable
|
ctl_ds=&mpelib..mpe_lockanytable
|
||||||
)
|
)
|
||||||
proc append base=&mpelib..mpe_review data=work.append_review;
|
proc append base=&mpelib..mpe_review data=work.append_review;
|
||||||
@ -653,7 +653,7 @@ run;
|
|||||||
|
|
||||||
/* update mpe_submit table */
|
/* update mpe_submit table */
|
||||||
%mp_lockanytable(LOCK,
|
%mp_lockanytable(LOCK,
|
||||||
lib=&mpelib,ds=mpe_submit,ref=%str(&LOAD_REF Approval in auditors/postdata),
|
lib=&mpelib,ds=mpe_submit,ref=%str(&table Approval in auditors/postdata),
|
||||||
ctl_ds=&mpelib..mpe_lockanytable
|
ctl_ds=&mpelib..mpe_lockanytable
|
||||||
)
|
)
|
||||||
proc sql;
|
proc sql;
|
||||||
@ -662,7 +662,7 @@ run;
|
|||||||
num_of_approvals_remaining=&num_of_approvals_remaining-1,
|
num_of_approvals_remaining=&num_of_approvals_remaining-1,
|
||||||
reviewed_by_nm="&user",
|
reviewed_by_nm="&user",
|
||||||
reviewed_on_dttm=&sastime
|
reviewed_on_dttm=&sastime
|
||||||
where table_id="&LOAD_REF";
|
where table_id="&table";
|
||||||
%mp_lockanytable(UNLOCK,
|
%mp_lockanytable(UNLOCK,
|
||||||
lib=&mpelib,ds=mpe_submit,
|
lib=&mpelib,ds=mpe_submit,
|
||||||
ctl_ds=&mpelib..mpe_lockanytable
|
ctl_ds=&mpelib..mpe_lockanytable
|
||||||
@ -688,7 +688,7 @@ run;
|
|||||||
%mpe_alerts(alert_event=APPROVED
|
%mpe_alerts(alert_event=APPROVED
|
||||||
, alert_lib=&libref
|
, alert_lib=&libref
|
||||||
, alert_ds=&ds
|
, alert_ds=&ds
|
||||||
, dsid=&LOAD_REF
|
, dsid=&TABLE
|
||||||
)
|
)
|
||||||
|
|
||||||
%removecolsfromwork(___TMP___MD5)
|
%removecolsfromwork(___TMP___MD5)
|
||||||
|
@ -34,8 +34,7 @@ run;
|
|||||||
%mp_testservice(&_program,
|
%mp_testservice(&_program,
|
||||||
viyacontext=&defaultcontext,
|
viyacontext=&defaultcontext,
|
||||||
inputdatasets=work.sascontroltable work.jsdata,
|
inputdatasets=work.sascontroltable work.jsdata,
|
||||||
outlib=web1,
|
outlib=web1
|
||||||
mdebug=&sasjs_mdebug
|
|
||||||
)
|
)
|
||||||
|
|
||||||
%let status=0;
|
%let status=0;
|
||||||
|
@ -14,9 +14,8 @@
|
|||||||
<h5> sasdata </h5>
|
<h5> sasdata </h5>
|
||||||
<h5> sasparams </h5>
|
<h5> sasparams </h5>
|
||||||
Contains info on the request. One row is returned.
|
Contains info on the request. One row is returned.
|
||||||
@li CLS_FLG - set to 0 if there are no CLS rules (everything should be editable)
|
* CLS_FLG - set to 0 if there are no CLS rules (everything should be editable)
|
||||||
else set to 1 (CLS rules exist)
|
else set to 1 (CLS rules exist)
|
||||||
@li ISMAP - set to 1 if the target DS is an excel map target, else 0
|
|
||||||
|
|
||||||
<h5> approvers </h5>
|
<h5> approvers </h5>
|
||||||
<h5> dqrules </h5>
|
<h5> dqrules </h5>
|
||||||
@ -535,11 +534,6 @@ data _null_;
|
|||||||
run;
|
run;
|
||||||
|
|
||||||
%put params;
|
%put params;
|
||||||
%let ismap=0;
|
|
||||||
proc sql noprint;
|
|
||||||
select count(*) into: ismap from &mpelib..mpe_xlmap_info
|
|
||||||
where XLMAP_TARGETLIBDS="&orig_libds" and &dc_dttmtfmt. le TX_TO;
|
|
||||||
|
|
||||||
data sasparams;
|
data sasparams;
|
||||||
length colHeaders $20000 filter_text $32767;
|
length colHeaders $20000 filter_text $32767;
|
||||||
colHeaders=cats(upcase("%mf_getvarlist(sasdata1,dlm=%str(,))"));
|
colHeaders=cats(upcase("%mf_getvarlist(sasdata1,dlm=%str(,))"));
|
||||||
@ -557,11 +551,8 @@ data sasparams;
|
|||||||
if %mf_nobs(work.cls_rules)=0 then cls_flag=0;
|
if %mf_nobs(work.cls_rules)=0 then cls_flag=0;
|
||||||
else cls_flag=1;
|
else cls_flag=1;
|
||||||
put (_all_)(=);
|
put (_all_)(=);
|
||||||
if "&orig_libds"="&mpelib..MPE_XLMAP_DATA" or &ismap ne 0 then ismap=1;
|
|
||||||
else ismap=0;
|
|
||||||
run;
|
run;
|
||||||
|
|
||||||
|
|
||||||
/* Extract validation DQ Rules */
|
/* Extract validation DQ Rules */
|
||||||
proc sort data=&mpelib..mpe_validations
|
proc sort data=&mpelib..mpe_validations
|
||||||
(where=(&dc_dttmtfmt. le TX_TO
|
(where=(&dc_dttmtfmt. le TX_TO
|
||||||
@ -648,6 +639,8 @@ proc sort data=dqdata;
|
|||||||
by base_col selectbox_order;
|
by base_col selectbox_order;
|
||||||
run;
|
run;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
%mp_getmaxvarlengths(work.sasdata1,outds=maxvarlengths)
|
%mp_getmaxvarlengths(work.sasdata1,outds=maxvarlengths)
|
||||||
|
|
||||||
data maxvarlengths;
|
data maxvarlengths;
|
||||||
|
@ -141,15 +141,13 @@ run;
|
|||||||
data work.fmts;
|
data work.fmts;
|
||||||
length fmtname $32;
|
length fmtname $32;
|
||||||
fmtname="&fmtname";
|
fmtname="&fmtname";
|
||||||
type='N';
|
|
||||||
do start=1 to 10;
|
do start=1 to 10;
|
||||||
label= cats("&fmtname",start);
|
label= cats("&fmtname",start);
|
||||||
end=start;
|
|
||||||
output;
|
output;
|
||||||
end;
|
end;
|
||||||
run;
|
run;
|
||||||
proc sort data=work.fmts nodupkey;
|
proc sort data=work.fmts nodupkey;
|
||||||
by fmtname type start;
|
by fmtname;
|
||||||
run;
|
run;
|
||||||
proc format cntlin=work.fmts library=dctest.dcfmts;
|
proc format cntlin=work.fmts library=dctest.dcfmts;
|
||||||
run;
|
run;
|
||||||
@ -159,9 +157,8 @@ data work.inquery3;
|
|||||||
infile datalines4 dsd;
|
infile datalines4 dsd;
|
||||||
input GROUP_LOGIC:$3. SUBGROUP_LOGIC:$3. SUBGROUP_ID:8. VARIABLE_NM:$32.
|
input GROUP_LOGIC:$3. SUBGROUP_LOGIC:$3. SUBGROUP_ID:8. VARIABLE_NM:$32.
|
||||||
OPERATOR_NM:$10. RAW_VALUE:$4000.;
|
OPERATOR_NM:$10. RAW_VALUE:$4000.;
|
||||||
RAW_VALUE="'&fmtname'";
|
|
||||||
datalines4;
|
datalines4;
|
||||||
AND,AND,1,FMTNAME,CONTAINS,placeholder (see line above)
|
AND,AND,1,FMTNAME,CONTAINS,"'&fmtname'"
|
||||||
;;;;
|
;;;;
|
||||||
run;
|
run;
|
||||||
%mp_filterstore(
|
%mp_filterstore(
|
||||||
|
@ -1,82 +0,0 @@
|
|||||||
/**
|
|
||||||
@file getxlmaps.sas
|
|
||||||
@brief Returns a list of rules and other info for a specific xlmap_id
|
|
||||||
|
|
||||||
<h4> Service Inputs </h4>
|
|
||||||
|
|
||||||
<h5> getxlmaps_in </h5>
|
|
||||||
|
|
||||||
|XLMAP_ID|
|
|
||||||
|---|
|
|
||||||
|Sample|
|
|
||||||
|
|
||||||
<h4> Service Outputs </h4>
|
|
||||||
|
|
||||||
<h5> xlmaprules </h5>
|
|
||||||
|
|
||||||
Filtered output of the dc.MPE_XLMAP_RULES table
|
|
||||||
|
|
||||||
|XLMAP_ID|XLMAP_RANGE_ID|XLMAP_SHEET|XLMAP_START|XLMAP_FINISH|
|
|
||||||
|---|---|---|---|---|
|
|
||||||
|Sample|Range1|Sheet1|ABSOLUTE A1| |
|
|
||||||
|Sample|Range2|Sheet1|RELATIVE R[2]C[2]|ABSOLUTE H11|
|
|
||||||
|
|
||||||
<h5> xlmapinfo </h5>
|
|
||||||
Extra info for a map id
|
|
||||||
|
|
||||||
|TARGET_DS|
|
|
||||||
|---|
|
|
||||||
|DCXXX.MPE_XLMAP_DATA|
|
|
||||||
|
|
||||||
<h4> SAS Macros </h4>
|
|
||||||
@li mp_abort.sas
|
|
||||||
@li mpeinit.sas
|
|
||||||
|
|
||||||
@version 9.3
|
|
||||||
@author 4GL Apps Ltd
|
|
||||||
@copyright 4GL Apps Ltd. This code may only be used within Data Controller
|
|
||||||
and may not be re-distributed or re-sold without the express permission of
|
|
||||||
4GL Apps Ltd.
|
|
||||||
|
|
||||||
**/
|
|
||||||
|
|
||||||
%mpeinit()
|
|
||||||
|
|
||||||
data _null_;
|
|
||||||
set work.getxlmaps_in;
|
|
||||||
putlog (_all_)(=);
|
|
||||||
call symputx('xlmap_id',xlmap_id);
|
|
||||||
run;
|
|
||||||
|
|
||||||
proc sql noprint;
|
|
||||||
create table work.xlmaprules as
|
|
||||||
select xlmap_id
|
|
||||||
,XLMAP_RANGE_ID
|
|
||||||
,XLMAP_SHEET
|
|
||||||
,XLMAP_START
|
|
||||||
,XLMAP_FINISH
|
|
||||||
from &mpelib..MPE_XLMAP_RULES
|
|
||||||
where &dc_dttmtfmt. lt tx_to and xlmap_id="&xlmap_id"
|
|
||||||
order by xlmap_sheet, xlmap_range_id;
|
|
||||||
|
|
||||||
%global target_ds;
|
|
||||||
select XLMAP_TARGETLIBDS into: target_ds
|
|
||||||
from &mpelib..MPE_XLMAP_INFO
|
|
||||||
where &dc_dttmtfmt. lt tx_to and xlmap_id="&xlmap_id";
|
|
||||||
|
|
||||||
%mp_abort(iftrue= (&syscc ne 0)
|
|
||||||
,mac=&_program..sas
|
|
||||||
,msg=%str(syscc=&syscc)
|
|
||||||
)
|
|
||||||
|
|
||||||
data work.xlmapinfo;
|
|
||||||
target_ds=coalescec("&target_ds","&mpelib..MPE_XLMAP_DATA");
|
|
||||||
output;
|
|
||||||
stop;
|
|
||||||
run;
|
|
||||||
|
|
||||||
%webout(OPEN)
|
|
||||||
%webout(OBJ,xlmaprules)
|
|
||||||
%webout(OBJ,xlmapinfo)
|
|
||||||
%webout(CLOSE)
|
|
||||||
|
|
@ -1,58 +0,0 @@
|
|||||||
/**
|
|
||||||
@file
|
|
||||||
@brief testing getxlmaps service
|
|
||||||
|
|
||||||
<h4> SAS Macros </h4>
|
|
||||||
@li mf_getuniquefileref.sas
|
|
||||||
@li mx_testservice.sas
|
|
||||||
@li mp_assert.sas
|
|
||||||
@li mp_assertdsobs.sas
|
|
||||||
|
|
||||||
|
|
||||||
**/
|
|
||||||
|
|
||||||
%let _program=&appLoc/services/editors/getxlmaps;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Test 1 - basic send
|
|
||||||
*/
|
|
||||||
|
|
||||||
%let f1=%mf_getuniquefileref();
|
|
||||||
data _null_;
|
|
||||||
file &f1 termstr=crlf;
|
|
||||||
put 'XLMAP_ID:$char12.';
|
|
||||||
put "Sample";
|
|
||||||
run;
|
|
||||||
|
|
||||||
%mx_testservice(&_program,
|
|
||||||
viyacontext=&defaultcontext,
|
|
||||||
inputfiles=&f1:getxlmaps_in,
|
|
||||||
outlib=web1,
|
|
||||||
mdebug=&sasjs_mdebug
|
|
||||||
)
|
|
||||||
|
|
||||||
data work.xlmaprules;
|
|
||||||
set web1.xlmaprules;
|
|
||||||
putlog (_all_)(=);
|
|
||||||
run;
|
|
||||||
|
|
||||||
%mp_assertdsobs(work.xlmaprules,
|
|
||||||
test=ATLEAST 2,
|
|
||||||
desc=Checking successful return of at least 2 rules for the Sample map,
|
|
||||||
outds=work.test_results
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Test 2 - info returned
|
|
||||||
*/
|
|
||||||
data work.xlmapinfo;
|
|
||||||
set web1.xlmapinfo;
|
|
||||||
putlog (_all_)(=);
|
|
||||||
call symputx('tgtds',target_ds);
|
|
||||||
run;
|
|
||||||
%mp_assert(
|
|
||||||
iftrue=(&tgtds=&dclib..MPE_XLMAP_DATA),
|
|
||||||
desc=Checking correct target table is returned,
|
|
||||||
outds=work.test_results
|
|
||||||
)
|
|
@ -146,7 +146,7 @@ select count(*) into: nobs from &syslast;
|
|||||||
,msg=%str(Issue assigning library &orig_lib)
|
,msg=%str(Issue assigning library &orig_lib)
|
||||||
)
|
)
|
||||||
|
|
||||||
%global txfrom txto processed rk;
|
%global txfrom txto processed;
|
||||||
|
|
||||||
data _null_;
|
data _null_;
|
||||||
set &mpelib..MPE_TABLES;
|
set &mpelib..MPE_TABLES;
|
||||||
@ -154,13 +154,12 @@ data _null_;
|
|||||||
call symputx('txfrom',var_txfrom);
|
call symputx('txfrom',var_txfrom);
|
||||||
call symputx('txto',var_txto);
|
call symputx('txto',var_txto);
|
||||||
call symputx('processed',var_processed);
|
call symputx('processed',var_processed);
|
||||||
if not missing(RK_UNDERLYING) then call symputx('rk',buskey);
|
|
||||||
run;
|
run;
|
||||||
|
|
||||||
%mp_lockfilecheck(libds=&orig_libds)
|
%mp_lockfilecheck(libds=&orig_libds)
|
||||||
|
|
||||||
data compare;
|
data compare;
|
||||||
set &libds(drop=&txfrom &txto &processed &rk);
|
set &libds(drop=&txfrom &txto &processed);
|
||||||
stop;
|
stop;
|
||||||
run;
|
run;
|
||||||
|
|
||||||
|
@ -24,7 +24,6 @@ proc format lib=DCTEST.DCFMTS cntlout=work.fmtextract;
|
|||||||
run;
|
run;
|
||||||
data work.jsdata;
|
data work.jsdata;
|
||||||
set work.fmtextract;
|
set work.fmtextract;
|
||||||
fmtrow=_n_;
|
|
||||||
if _n_<5 then _____DELETE__THIS__RECORD_____='Yes';
|
if _n_<5 then _____DELETE__THIS__RECORD_____='Yes';
|
||||||
else _____DELETE__THIS__RECORD_____='No';
|
else _____DELETE__THIS__RECORD_____='No';
|
||||||
if _n_>20 then stop;
|
if _n_>20 then stop;
|
||||||
|
@ -34,7 +34,6 @@ data work.staging_ds;
|
|||||||
var_processed=upcase(var_processed);
|
var_processed=upcase(var_processed);
|
||||||
close_vars=upcase(close_vars);
|
close_vars=upcase(close_vars);
|
||||||
audit_libds=upcase(audit_libds);
|
audit_libds=upcase(audit_libds);
|
||||||
rk_underlying=upcase(rk_underlying);
|
|
||||||
|
|
||||||
/* check for valid loadtype */
|
/* check for valid loadtype */
|
||||||
if LOADTYPE not in ('UPDATE','TXTEMPORAL','FORMAT_CAT','BITEMPORAL','REPLACE')
|
if LOADTYPE not in ('UPDATE','TXTEMPORAL','FORMAT_CAT','BITEMPORAL','REPLACE')
|
||||||
@ -46,12 +45,8 @@ data work.staging_ds;
|
|||||||
/* force correct BUSKEY and DSN when loading format catalogs */
|
/* force correct BUSKEY and DSN when loading format catalogs */
|
||||||
if LOADTYPE='FORMAT_CAT' then do;
|
if LOADTYPE='FORMAT_CAT' then do;
|
||||||
BUSKEY='TYPE FMTNAME FMTROW';
|
BUSKEY='TYPE FMTNAME FMTROW';
|
||||||
DSN=scan(dsn,1,'-')!!'-FC';
|
if subpad(dsn,length(dsn)-3,3) ne '-FC' then dsn=cats(dsn,'-FC');
|
||||||
end;
|
end;
|
||||||
|
|
||||||
/* convert tabs into spaces */
|
|
||||||
buskey=translate(buskey," ","09"x);
|
|
||||||
rk_underlying=translate(rk_underlying," ","09"x);
|
|
||||||
run;
|
run;
|
||||||
|
|
||||||
%mp_abort(iftrue=(&errflag=1)
|
%mp_abort(iftrue=(&errflag=1)
|
||||||
|
@ -1,69 +0,0 @@
|
|||||||
/**
|
|
||||||
@file
|
|
||||||
@brief Post Edit Hook script for the MPE_XLMAP_INFO table
|
|
||||||
@details Post edit hooks provide additional backend validation for user
|
|
||||||
provided data. The incoming dataset is named `work.staging_ds` and is
|
|
||||||
provided in mpe_loader.sas.
|
|
||||||
|
|
||||||
Available macro variables:
|
|
||||||
@li DC_LIBREF - The DC control library
|
|
||||||
@li LIBREF - The library of the dataset being edited (is assigned)
|
|
||||||
@li DS - The dataset being edited
|
|
||||||
|
|
||||||
<h4> SAS Macros </h4>
|
|
||||||
@li mf_existds.sas
|
|
||||||
@li mf_getvarlist.sas
|
|
||||||
@li mf_wordsinstr1butnotstr2.sas
|
|
||||||
@li dc_assignlib.sas
|
|
||||||
@li mp_validatecol.sas
|
|
||||||
|
|
||||||
**/
|
|
||||||
|
|
||||||
data work.staging_ds;
|
|
||||||
set work.staging_ds;
|
|
||||||
|
|
||||||
/* apply the first excel map to all cells */
|
|
||||||
length tgtds $41;
|
|
||||||
retain tgtds;
|
|
||||||
drop tgtds is_libds;
|
|
||||||
if _n_=1 then do;
|
|
||||||
if missing(XLMAP_TARGETLIBDS) then tgtds="&dc_libref..MPE_XLMAP_DATA";
|
|
||||||
else tgtds=upcase(XLMAP_TARGETLIBDS);
|
|
||||||
%mp_validatecol(XLMAP_TARGETLIBDS,LIBDS,is_libds)
|
|
||||||
call symputx('tgtds',tgtds);
|
|
||||||
call symputx('is_libds',is_libds);
|
|
||||||
end;
|
|
||||||
XLMAP_TARGETLIBDS=tgtds;
|
|
||||||
|
|
||||||
run;
|
|
||||||
|
|
||||||
%mp_abort(iftrue=(&is_libds ne 1)
|
|
||||||
,mac=mpe_xlmap_info_postedit
|
|
||||||
,msg=Invalid target dataset (&tgtds)
|
|
||||||
)
|
|
||||||
|
|
||||||
/**
|
|
||||||
* make sure that the supplied target dataset exists and
|
|
||||||
* has the necessary columns
|
|
||||||
*/
|
|
||||||
%dc_assignlib(READ,%scan(&tgtds,1,.))
|
|
||||||
|
|
||||||
%mp_abort(iftrue=(%mf_existds(libds=&tgtds) ne 1)
|
|
||||||
,mac=mpe_xlmap_info_postedit
|
|
||||||
,msg=Target dataset (&tgtds) could not be opened
|
|
||||||
)
|
|
||||||
|
|
||||||
%let tgtvars=%upcase(%mf_getvarlist(&tgtds));
|
|
||||||
%let srcvars=%upcase(%mf_getvarlist(&dc_libref..MPE_XLMAP_DATA));
|
|
||||||
%let badvars1=%mf_wordsInStr1ButNotStr2(Str1=&srcvars,Str2=&tgtvars);
|
|
||||||
%let badvars2=%mf_wordsInStr1ButNotStr2(Str1=&tgtvars,Str2=&srcvars);
|
|
||||||
|
|
||||||
%mp_abort(iftrue=(%length(&badvars1.X)>1)
|
|
||||||
,mac=mpe_xlmap_info_postedit
|
|
||||||
,msg=%str(Target dataset (&tgtds) has missing vars: &badvars1)
|
|
||||||
)
|
|
||||||
|
|
||||||
%mp_abort(iftrue=(%length(&badvars2.X)>1)
|
|
||||||
,mac=mpe_xlmap_info_postedit
|
|
||||||
,msg=%str(Target dataset (&tgtds) has unrecognised vars: &badvars2)
|
|
||||||
)
|
|
@ -1,23 +0,0 @@
|
|||||||
/**
|
|
||||||
@file
|
|
||||||
@brief Post Edit Hook script for the MPE_XLMAP_RULES table
|
|
||||||
@details Post edit hooks provide additional backend validation for user
|
|
||||||
provided data. The incoming dataset is named `work.staging_ds` and is
|
|
||||||
provided in mpe_loader.sas.
|
|
||||||
|
|
||||||
Available macro variables:
|
|
||||||
@li DC_LIBREF - The DC control library
|
|
||||||
@li LIBREF - The library of the dataset being edited (is assigned)
|
|
||||||
@li DS - The dataset being edited
|
|
||||||
|
|
||||||
|
|
||||||
**/
|
|
||||||
|
|
||||||
data work.staging_ds;
|
|
||||||
set work.staging_ds;
|
|
||||||
|
|
||||||
/* ensure uppercasing */
|
|
||||||
XLMAP_ID=upcase(XLMAP_ID);
|
|
||||||
|
|
||||||
run;
|
|
||||||
|
|
@ -1,22 +0,0 @@
|
|||||||
/**
|
|
||||||
@file
|
|
||||||
@brief Sample XLMAP Data hook program (sample_xlmap_data_postapprove)
|
|
||||||
@details This hook script should NOT be modified in place, as the changes
|
|
||||||
would be lost in your next Data Controller deployment.
|
|
||||||
Instead, create a copy of this hook script and place it OUTSIDE the
|
|
||||||
Data Controller metadata folder.
|
|
||||||
|
|
||||||
Available macro variables:
|
|
||||||
@li LOAD_REF - The Load Reference (unique upload id)
|
|
||||||
@li ORIG_LIBDS - The target library.dataset that was just loaded
|
|
||||||
|
|
||||||
**/
|
|
||||||
|
|
||||||
|
|
||||||
data _null_;
|
|
||||||
set work.staging_ds;
|
|
||||||
putlog 'load ref is in the staged data: ' load_ref;
|
|
||||||
stop;
|
|
||||||
run;
|
|
||||||
|
|
||||||
%put the unique identifier (LOAD_REF) is also a macro variable: &LOAD_REF;
|
|
@ -1,49 +0,0 @@
|
|||||||
/**
|
|
||||||
@file
|
|
||||||
@brief Sample XLMAP Data hook program
|
|
||||||
@details This hook script should NOT be modified in place, as the changes
|
|
||||||
would be lost in your next Data Controller deployment.
|
|
||||||
Instead, create a copy of this hook script and place it OUTSIDE the
|
|
||||||
Data Controller metadata folder.
|
|
||||||
|
|
||||||
Available macro variables:
|
|
||||||
@li DC_LIBREF - The DC control library
|
|
||||||
@li LIBREF - The library of the dataset being edited (is assigned)
|
|
||||||
@li DS - The target dataset being loaded
|
|
||||||
|
|
||||||
**/
|
|
||||||
|
|
||||||
%let abort=0;
|
|
||||||
%let errmsg=;
|
|
||||||
|
|
||||||
data work.staging_ds;
|
|
||||||
set work.staging_ds;
|
|
||||||
length errmsg $1000;
|
|
||||||
drop err:;
|
|
||||||
/* KM1 validations */
|
|
||||||
if XLMAP_ID='BASEL-KM1' then do;
|
|
||||||
if XLMAP_RANGE_ID='KM1:a' & input(value_txt,8.)<100 then do;
|
|
||||||
errmsg='Should be greater than 100';
|
|
||||||
err=1;
|
|
||||||
end;
|
|
||||||
end;
|
|
||||||
/* CR2 Validations */
|
|
||||||
if XLMAP_ID='BASEL-CR2' then do;
|
|
||||||
if XLMAP_RANGE_ID='CR2-sec1' & row_no=3 & input(value_txt,8.)>0 then do;
|
|
||||||
errmsg='Should be negative';
|
|
||||||
err=1;
|
|
||||||
end;
|
|
||||||
end;
|
|
||||||
|
|
||||||
/* publish error message */
|
|
||||||
if err=1 then do;
|
|
||||||
errmsg=catx(' ',xlmap_range_id,':',value_txt,'->',errmsg);
|
|
||||||
call symputx('errmsg',errmsg);
|
|
||||||
call symputx('abort',1);
|
|
||||||
end;
|
|
||||||
run;
|
|
||||||
|
|
||||||
%mp_abort(iftrue=(&abort ne 0)
|
|
||||||
,mac=xlmap_data_postedit
|
|
||||||
,msg=%superq(errmsg)
|
|
||||||
)
|
|
@ -3,8 +3,10 @@
|
|||||||
@brief List the libraries and tables the mp-editor user can access
|
@brief List the libraries and tables the mp-editor user can access
|
||||||
@details If user is in a control group (&mpeadmins, configured in mpeinit.sas)
|
@details If user is in a control group (&mpeadmins, configured in mpeinit.sas)
|
||||||
then they have access to all libraries / tables. Otherwise a join is made
|
then they have access to all libraries / tables. Otherwise a join is made
|
||||||
to the &mpelib..mpe_security table.
|
to the &mpelib..mp_editor_access table.
|
||||||
|
|
||||||
|
This service is also callable from EUCs - just add EUCDLM= parameter.
|
||||||
|
EUCDLM values: TAB or CSV
|
||||||
|
|
||||||
<h4> SAS Macros </h4>
|
<h4> SAS Macros </h4>
|
||||||
@li mf_getuser.sas
|
@li mf_getuser.sas
|
||||||
@ -127,26 +129,10 @@ create table saslibs as
|
|||||||
,msg=%str(issue with security validation)
|
,msg=%str(issue with security validation)
|
||||||
)
|
)
|
||||||
|
|
||||||
proc sql;
|
|
||||||
create table work.xlmaps as
|
|
||||||
select distinct a.XLMAP_ID
|
|
||||||
,b.XLMAP_DESCRIPTION
|
|
||||||
,coalescec(b.XLMAP_TARGETLIBDS,"&mpelib..MPE_XLMAP_DATA")
|
|
||||||
as XLMAP_TARGETLIBDS
|
|
||||||
from &mpelib..MPE_XLMAP_RULES a
|
|
||||||
left join &mpelib..MPE_XLMAP_INFO(where=(&dc_dttmtfmt. lt tx_to)) b
|
|
||||||
on a.XLMAP_ID=b.XLMAP_ID
|
|
||||||
where &dc_dttmtfmt. lt a.tx_to;
|
|
||||||
|
|
||||||
/* we don't want the XLMAP target datasets to be directly editable */
|
|
||||||
delete from sasdatasets
|
|
||||||
where cats(libref,'.',dsn) in (select XLMAP_TARGETLIBDS from xlmaps);
|
|
||||||
|
|
||||||
%webout(OPEN)
|
%webout(OPEN)
|
||||||
%webout(OBJ,sasDatasets)
|
%webout(OBJ,sasDatasets)
|
||||||
%webout(OBJ,saslibs)
|
%webout(OBJ,saslibs)
|
||||||
%webout(OBJ,globvars)
|
%webout(OBJ,globvars)
|
||||||
%webout(ARR,xlmaps)
|
|
||||||
%webout(CLOSE)
|
%webout(CLOSE)
|
||||||
|
|
||||||
%mpeterm()
|
%mpeterm()
|
||||||
|
@ -16,24 +16,13 @@
|
|||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
data work.globvars;
|
data globvars;
|
||||||
set webout.globvars;
|
set webout.globvars;
|
||||||
putlog (_all_)(=);
|
putlog (_all_)(=);
|
||||||
run;
|
run;
|
||||||
|
|
||||||
data work.xlmaps;
|
|
||||||
set webout.xlmaps;
|
|
||||||
putlog (_all_)(=);
|
|
||||||
run;
|
|
||||||
|
|
||||||
%mp_assertdsobs(work.globvars,
|
%mp_assertdsobs(work.globvars,
|
||||||
desc=Fromsas table returned,
|
desc=Fromsas table returned,
|
||||||
test=HASOBS,
|
test=HASOBS,
|
||||||
outds=work.test_results
|
outds=work.test_results
|
||||||
)
|
)
|
||||||
|
|
||||||
%mp_assertdsobs(work.xlmaps,
|
|
||||||
desc=xlmaps table returned,
|
|
||||||
test=HASOBS,
|
|
||||||
outds=work.test_results
|
|
||||||
)
|
|
@ -290,7 +290,6 @@ run;
|
|||||||
run;
|
run;
|
||||||
data work.groups;
|
data work.groups;
|
||||||
length groupuri groupname $32 groupdesc $128 ;
|
length groupuri groupname $32 groupdesc $128 ;
|
||||||
call missing (of _all_);
|
|
||||||
output;
|
output;
|
||||||
stop;
|
stop;
|
||||||
run;
|
run;
|
||||||
|
Reference in New Issue
Block a user