Compare commits
	
		
			190 Commits
		
	
	
		
			5474fad9cc
			...
			css-refact
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					2e0c60cc0d | ||
| 
						 | 
					f9decbd366 | ||
| 
						 | 
					1dc69341ca | ||
| 
						 | 
					75ae19fa8e | ||
| 
						 | 
					9de04e9a0c | ||
| 
						 | 
					983f59cd51 | ||
| 
						 | 
					7b5e7ae184 | ||
| 
						 | 
					6e96b1daec | ||
| 
						 | 
					9604661f3b | ||
| 
						 | 
					4bd215491f | ||
| 
						 | 
					6a7dd451b5 | ||
| 
						 | 
					841201adab | ||
| 
						 | 
					e013e62776 | ||
| 
						 | 
					6c171a6394 | ||
| 
						 | 
					a377f6e8d6 | ||
| 0337318e0b | |||
| 
						 | 
					2844c70f95 | ||
| 
						 | 
					7e11c8f375 | ||
| 23cbbce964 | |||
| 
						 | 
					c63fcdd465 | ||
| 
						 | 
					82412b2659 | ||
| eef3832e40 | |||
| 63b75a1c61 | |||
| d1f0879f0a | |||
| 
						 | 
					36416aab2e | ||
| 
						 | 
					7f3577c3ef | ||
| 
						 | 
					69f883034f | ||
| 
						 | 
					1c56af01d0 | ||
| 
						 | 
					43c0f73c21 | ||
| 
						 | 
					e8cd3d63da | ||
| 
						 | 
					b7f564cb21 | ||
| 
						 | 
					3f5cb1e2de | ||
| 
						 | 
					7d8c0472f0 | ||
| b8b516ba77 | |||
| 
						 | 
					83b3d775b6 | ||
| 
						 | 
					aea252ccc6 | ||
| d8908f9c7f | |||
| 
						 | 
					95289aa952 | ||
| 
						 | 
					149e318a87 | ||
| 
						 | 
					8657826e60 | ||
| 
						 | 
					a45f5bb3b2 | ||
| 
						 | 
					ae9a91a7a1 | ||
| 
						 | 
					3638bde633 | ||
| 3a2361f42c | |||
| 
						 | 
					1bd0eef913 | ||
| 85aa3b38b7 | |||
| 678859a68d | |||
| 
						 | 
					e531acee3f | ||
| 
						 | 
					4a45ebfe3b | ||
| 8fb9a344f2 | |||
| 
						 | 
					8829b60220 | ||
| 
						 | 
					4e64f28732 | ||
| a0749de700 | |||
| f9623e046e | |||
| 
						 | 
					bce1fd57ef | ||
| 
						 | 
					c6a1c53b46 | ||
| e36229e4c4 | |||
| 
						 | 
					853c1bc23e | ||
| 
						 | 
					3b6f6853bc | ||
| 
						 | 
					b415437662 | ||
| 
						 | 
					24df878abe | ||
| 
						 | 
					f474673a14 | ||
| 
						 | 
					c53ab85107 | ||
| 
						 | 
					5c4dd7c9e3 | ||
| 
						 | 
					217220ffaa | ||
| 
						 | 
					c89a3049fa | ||
| bb12bff9db | |||
| 
						 | 
					30d5e51d0b | ||
| 
						 | 
					cd3bcc0ee3 | ||
| ef1ba824c3 | |||
| bc45e92138 | |||
| 
						 | 
					c2f36f5419 | ||
| 31a31612fc | |||
| 
						 | 
					f9c2491ab6 | ||
| 
						 | 
					3de095fe77 | ||
| 
						 | 
					e4e04a193f | ||
| 
						 | 
					a1a90519c5 | ||
| 
						 | 
					522979835c | ||
| fe049256b5 | |||
| 
						 | 
					31d1870198 | ||
| 
						 | 
					6edf0dfb31 | ||
| 3bf3dceaa2 | |||
| 
						 | 
					ce503653cd | ||
| 
						 | 
					caa9854ff0 | ||
| 
						 | 
					20c3a338c5 | ||
| 
						 | 
					6547461637 | ||
| 
						 | 
					bbb725c64c | ||
| 403d08c86a | |||
| 
						 | 
					f66d9f511a | ||
| 
						 | 
					dd2138ac5e | ||
| 74e9979c67 | |||
| 
						 | 
					ddc22e5200 | ||
| 
						 | 
					4218da91cd | ||
| 
						 | 
					857b94f44f | ||
| 
						 | 
					a3ce367950 | ||
| 
						 | 
					75dac54591 | ||
| a156c0111b | |||
| e5d93fd7d6 | |||
| 
						 | 
					67436f4ff9 | ||
| 
						 | 
					b712f851a2 | ||
| e2c0b8da86 | |||
| 
						 | 
					1b4560061d | ||
| 
						 | 
					d94df7f0eb | ||
| 
						 | 
					fa04d7bf4e | ||
| 
						 | 
					4d4cabb465 | ||
| 
						 | 
					d4fee791a7 | ||
| 
						 | 
					82c285e348 | ||
| 
						 | 
					4d276657b3 | ||
| 
						 | 
					cffeab813d | ||
| 
						 | 
					18363bbbeb | ||
| 
						 | 
					6df7d8d2ba | ||
| 
						 | 
					5deba44d2b | ||
| 
						 | 
					0a8b1e764c | ||
| 
						 | 
					fc52e8f41a | ||
| 
						 | 
					efcdc694dd | ||
| 
						 | 
					eb7c44333c | ||
| 
						 | 
					3c62ff6913 | ||
| f4f589af94 | |||
| 3521579dea | |||
| e97a6f52da | |||
| 
						 | 
					3584aa35c7 | ||
| 
						 | 
					b553520abe | ||
| 
						 | 
					e32d44b1bc | ||
| 
						 | 
					105da1503f | ||
| 29298072e5 | |||
| 
						 | 
					154c10fee5 | ||
| 7853f7cb6a | |||
| 
						 | 
					e54ecc8a35 | ||
| d8b95c5739 | |||
| 
						 | 
					2f8d0b764a | ||
| 
						 | 
					d7732ed206 | ||
| 
						 | 
					6e631cd9a5 | ||
| 
						 | 
					0a9e5dd834 | ||
| 
						 | 
					d14a4eaadd | ||
| 
						 | 
					5f7c7fcc7b | ||
| 
						 | 
					978f152ab6 | ||
| 
						 | 
					68a2a606f3 | ||
| 
						 | 
					bad43135d7 | ||
| 
						 | 
					110ad9a6e9 | ||
| 
						 | 
					e98f288302 | ||
| 
						 | 
					9a79f37bf1 | ||
| 
						 | 
					85909cfc1e | ||
| 
						 | 
					4330da520f | ||
| 
						 | 
					27907ed00f | ||
| 
						 | 
					31c90f3190 | ||
| 
						 | 
					35844e0cf1 | ||
| 
						 | 
					afa7e380aa | ||
| cb9a5f0eb4 | |||
| 
						 | 
					da522c557d | ||
| 
						 | 
					2c0afd0268 | ||
| 
						 | 
					20255c69c2 | ||
| 74f1c5416b | |||
| feed7f1ded | |||
| 967698e4ce | |||
| 
						 | 
					71bd81ae47 | ||
| 89a5153bb3 | |||
| c11bd9a2c5 | |||
| 59d46a9926 | |||
| 
						 | 
					39c3e5411f | ||
| aad419c55d | |||
| 
						 | 
					ee58fd5b4b | ||
| 
						 | 
					5564aea9c2 | ||
| aedd2c451b | |||
| 
						 | 
					6d597611b6 | ||
| 
						 | 
					9ffe5efe5d | ||
| 
						 | 
					2715950d86 | ||
| 
						 | 
					77a7190f4d | ||
| cc65890fea | |||
| 
						 | 
					93758efb27 | ||
| 
						 | 
					c0dc9191e3 | ||
| 
						 | 
					485783a782 | ||
| ee07bef2b8 | |||
| 
						 | 
					0de8481314 | ||
| 
						 | 
					ec0f539a33 | ||
| 
						 | 
					173ee2daff | ||
| 
						 | 
					5c0091b5e8 | ||
| 84dce7f6e8 | |||
| 
						 | 
					e8a943a35a | ||
| 
						 | 
					b3171a8125 | ||
| 
						 | 
					d77f2eb674 | ||
| 56cf271e77 | |||
| 
						 | 
					57aa6fa0fc | ||
| 
						 | 
					4a01f3d490 | ||
| 
						 | 
					80a0db951d | ||
| 
						 | 
					3202cb8e08 | ||
| 
						 | 
					ddf36230bf | ||
| 
						 | 
					b67c2be968 | ||
| 
						 | 
					dc2c8da92b | ||
| 
						 | 
					5d6c3701d0 | ||
| 
						 | 
					b024e263b4 | 
@@ -8,9 +8,16 @@ jobs:
 | 
			
		||||
 | 
			
		||||
    steps:
 | 
			
		||||
      - uses: actions/checkout@v3
 | 
			
		||||
      - uses: actions/setup-node@v3
 | 
			
		||||
      - uses: actions/setup-node@v4
 | 
			
		||||
        with:
 | 
			
		||||
          node-version: 18
 | 
			
		||||
          node-version: 20.14.0
 | 
			
		||||
 | 
			
		||||
      - name: Install Google Chrome
 | 
			
		||||
        run: |
 | 
			
		||||
          wget -q -O - https://dl-ssl.google.com/linux/linux_signing_key.pub | apt-key add -
 | 
			
		||||
          echo "deb http://dl.google.com/linux/chrome/deb/ stable main" > /etc/apt/sources.list.d/google.list
 | 
			
		||||
          apt-get update
 | 
			
		||||
          apt-get install -y google-chrome-stable xvfb
 | 
			
		||||
 | 
			
		||||
      - name: Write .npmrc file
 | 
			
		||||
        run: echo "$NPMRC" > client/.npmrc
 | 
			
		||||
@@ -24,13 +31,111 @@ jobs:
 | 
			
		||||
      - name: Install dependencies
 | 
			
		||||
        run: |
 | 
			
		||||
          cd client
 | 
			
		||||
          npm ci
 | 
			
		||||
          # Decrypt and Install sheet
 | 
			
		||||
          echo ${{ secrets.SHEET_PWD }} | gpg --batch --yes --passphrase-fd 0 ./libraries/sheet-crypto.tgz.gpg
 | 
			
		||||
          npm i ./libraries/sheet-crypto.tgz
 | 
			
		||||
          # End
 | 
			
		||||
          npm ci
 | 
			
		||||
 | 
			
		||||
      - name: Licence checker
 | 
			
		||||
        run: |
 | 
			
		||||
          cd client
 | 
			
		||||
          npm run license-checker
 | 
			
		||||
          npm run license-checker
 | 
			
		||||
 | 
			
		||||
      - name: Angular Tests
 | 
			
		||||
        run: |
 | 
			
		||||
          cd client
 | 
			
		||||
          npm run test:headless
 | 
			
		||||
 | 
			
		||||
      - name: Production Build
 | 
			
		||||
        run: |
 | 
			
		||||
          cd client
 | 
			
		||||
          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@v4
 | 
			
		||||
        with:
 | 
			
		||||
          node-version: 20.14.0
 | 
			
		||||
 | 
			
		||||
      - 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: |
 | 
			
		||||
          cd client
 | 
			
		||||
          # Decrypt and Install sheet
 | 
			
		||||
          echo ${{ secrets.SHEET_PWD }} | gpg --batch --yes --passphrase-fd 0 ./libraries/sheet-crypto.tgz.gpg
 | 
			
		||||
          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-multi-load.cy.ts,cypress/e2e/excel.cy.ts,cypress/e2e/csv.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
 | 
			
		||||
 
 | 
			
		||||
@@ -11,9 +11,9 @@ jobs:
 | 
			
		||||
 | 
			
		||||
    steps:
 | 
			
		||||
      - uses: actions/checkout@v3
 | 
			
		||||
      - uses: actions/setup-node@v3
 | 
			
		||||
      - uses: actions/setup-node@v4
 | 
			
		||||
        with:
 | 
			
		||||
          node-version: 18
 | 
			
		||||
          node-version: 20.14.0
 | 
			
		||||
 | 
			
		||||
      - name: Write .npmrc file
 | 
			
		||||
        run: |
 | 
			
		||||
@@ -36,11 +36,9 @@ jobs:
 | 
			
		||||
      - name: Install dependencies
 | 
			
		||||
        run: |
 | 
			
		||||
          cd client
 | 
			
		||||
          npm ci
 | 
			
		||||
          # Decrypt and Install sheet
 | 
			
		||||
          echo ${{ secrets.SHEET_PWD }} | gpg --batch --yes --passphrase-fd 0 ./libraries/sheet-crypto.tgz.gpg
 | 
			
		||||
          npm i ./libraries/sheet-crypto.tgz
 | 
			
		||||
          # End
 | 
			
		||||
          npm ci
 | 
			
		||||
 | 
			
		||||
      - name: Check audit
 | 
			
		||||
      # Audit should fail and stop the CI if critical vulnerability found
 | 
			
		||||
@@ -54,7 +52,7 @@ jobs:
 | 
			
		||||
      - name: Angular Tests
 | 
			
		||||
        run: |
 | 
			
		||||
          cd client
 | 
			
		||||
          npm test -- --no-watch --no-progress --browsers=ChromeHeadlessCI
 | 
			
		||||
          npm run test:headless
 | 
			
		||||
 | 
			
		||||
      - name: Angular Production Build
 | 
			
		||||
        run: |
 | 
			
		||||
@@ -68,9 +66,9 @@ jobs:
 | 
			
		||||
 | 
			
		||||
    steps:
 | 
			
		||||
      - uses: actions/checkout@v3
 | 
			
		||||
      - uses: actions/setup-node@v3
 | 
			
		||||
      - uses: actions/setup-node@v4
 | 
			
		||||
        with:
 | 
			
		||||
          node-version: 18
 | 
			
		||||
          node-version: 20.14.0
 | 
			
		||||
 | 
			
		||||
      - name: Write .npmrc file
 | 
			
		||||
        run: |
 | 
			
		||||
@@ -94,11 +92,9 @@ jobs:
 | 
			
		||||
      - name: Install dependencies
 | 
			
		||||
        run: |
 | 
			
		||||
          cd client
 | 
			
		||||
          npm ci
 | 
			
		||||
          # Decrypt and Install sheet
 | 
			
		||||
          echo ${{ secrets.SHEET_PWD }} | gpg --batch --yes --passphrase-fd 0 ./libraries/sheet-crypto.tgz.gpg
 | 
			
		||||
          npm i ./libraries/sheet-crypto.tgz
 | 
			
		||||
          # End
 | 
			
		||||
          npm ci
 | 
			
		||||
 | 
			
		||||
      # Install pm2 and prepare SASJS server
 | 
			
		||||
      - run: npm i -g pm2
 | 
			
		||||
@@ -140,7 +136,7 @@ jobs:
 | 
			
		||||
          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"
 | 
			
		||||
          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-multi-load.cy.ts,cypress/e2e/excel.cy.ts,cypress/e2e/csv.cy.ts,cypress/e2e/filtering.cy.ts,cypress/e2e/licensing.cy.ts"
 | 
			
		||||
 | 
			
		||||
      - name: Zip Cypress videos
 | 
			
		||||
        if: always()
 | 
			
		||||
@@ -160,9 +156,9 @@ jobs:
 | 
			
		||||
 | 
			
		||||
    steps:
 | 
			
		||||
      - uses: actions/checkout@v3
 | 
			
		||||
      - uses: actions/setup-node@v3
 | 
			
		||||
      - uses: actions/setup-node@v4
 | 
			
		||||
        with:
 | 
			
		||||
          node-version: 20
 | 
			
		||||
          node-version: 20.14.0
 | 
			
		||||
 | 
			
		||||
      - name: Write .npmrc file
 | 
			
		||||
        run: |
 | 
			
		||||
@@ -184,6 +180,16 @@ jobs:
 | 
			
		||||
          apt-get update
 | 
			
		||||
          apt-get install doxygen -y
 | 
			
		||||
 | 
			
		||||
      - name: Frontend Preliminary Build
 | 
			
		||||
        description: We want to prevent creating empty release if frontend fails
 | 
			
		||||
        run: |
 | 
			
		||||
          cd client
 | 
			
		||||
          # Decrypt and Install sheet
 | 
			
		||||
          echo ${{ secrets.SHEET_PWD }} | gpg --batch --yes --passphrase-fd 0 ./libraries/sheet-crypto.tgz.gpg
 | 
			
		||||
          npm ci
 | 
			
		||||
          npm i webpack
 | 
			
		||||
          npm run build
 | 
			
		||||
 | 
			
		||||
      - name: Create Empty Release (assets are posted later)
 | 
			
		||||
        run: |
 | 
			
		||||
          npm i
 | 
			
		||||
@@ -196,11 +202,6 @@ jobs:
 | 
			
		||||
        description: Must be created AFTER the release as the version (git tag) is used in the interface
 | 
			
		||||
        run: |
 | 
			
		||||
          cd client
 | 
			
		||||
          npm ci
 | 
			
		||||
          # Decrypt and Install sheet
 | 
			
		||||
          echo ${{ secrets.SHEET_PWD }} | gpg --batch --yes --passphrase-fd 0 ./libraries/sheet-crypto.tgz.gpg
 | 
			
		||||
          npm i ./libraries/sheet-crypto.tgz
 | 
			
		||||
          # End
 | 
			
		||||
          npm run build
 | 
			
		||||
 | 
			
		||||
      - name: Build SAS9 EBI Release
 | 
			
		||||
@@ -244,6 +245,7 @@ jobs:
 | 
			
		||||
          rm sasjsbuild/services/clickme.html
 | 
			
		||||
          sasjs b -t viya
 | 
			
		||||
          cp sasjsbuild/viya.sas ./viya.sas
 | 
			
		||||
          cp sasjsbuild/viya.json ./viya.json
 | 
			
		||||
 | 
			
		||||
      - name: Zip Frontend (including viya.json for full viya deploy)
 | 
			
		||||
        run: |
 | 
			
		||||
@@ -279,3 +281,4 @@ jobs:
 | 
			
		||||
          curl -k $URL -F attachment=@sas/sasjs_server.json.zip
 | 
			
		||||
          curl -k $URL -F attachment=@sas/sas9.sas
 | 
			
		||||
          curl -k $URL -F attachment=@sas/viya.sas
 | 
			
		||||
          curl -k $URL -F attachment=@sas/viya.json
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										4
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							@@ -11,7 +11,9 @@ client/cypress/screenshots
 | 
			
		||||
client/cypress/results
 | 
			
		||||
client/cypress/videos
 | 
			
		||||
client/documentation
 | 
			
		||||
client/sheet-crypto*
 | 
			
		||||
client/**/sheet-crypto.tgz
 | 
			
		||||
client/.nx
 | 
			
		||||
client/libraries/sheet-crypto.tgz
 | 
			
		||||
cypress.env.json
 | 
			
		||||
sasjsbuild
 | 
			
		||||
sasjsresults
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										253
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										253
									
								
								CHANGELOG.md
									
									
									
									
									
								
							@@ -1,3 +1,256 @@
 | 
			
		||||
## [6.14.7](https://git.datacontroller.io/dc/dc/compare/v6.14.6...v6.14.7) (2025-05-08)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
### Bug Fixes
 | 
			
		||||
 | 
			
		||||
* updated hot, clarity and improved accessibility score. ([2844c70](https://git.datacontroller.io/dc/dc/commit/2844c70f9507036216b8b621900c2bb9010c1d34))
 | 
			
		||||
 | 
			
		||||
## [6.14.6](https://git.datacontroller.io/dc/dc/compare/v6.14.5...v6.14.6) (2025-04-03)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
### Bug Fixes
 | 
			
		||||
 | 
			
		||||
* history table modal links styling ([c63fcdd](https://git.datacontroller.io/dc/dc/commit/c63fcdd465950ada439d7d69622a3886e8f3a783))
 | 
			
		||||
 | 
			
		||||
## [6.14.5](https://git.datacontroller.io/dc/dc/compare/v6.14.4...v6.14.5) (2025-03-24)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
### Bug Fixes
 | 
			
		||||
 | 
			
		||||
* improving accessibility lighthouse score ([7f3577c](https://git.datacontroller.io/dc/dc/commit/7f3577c3ef9f44e55a58bc64fbf89a3a64006dd4))
 | 
			
		||||
* prevent errors when using sqlrc in a DI job in a HOOK ([d1f0879](https://git.datacontroller.io/dc/dc/commit/d1f0879f0acf7e816c80f7635fd02f4f284214ed))
 | 
			
		||||
* user profile style fix, new select library and table icons ([69f8830](https://git.datacontroller.io/dc/dc/commit/69f883034fabbed31aa5d832e20561c4ae3042db))
 | 
			
		||||
 | 
			
		||||
## [6.14.4](https://git.datacontroller.io/dc/dc/compare/v6.14.3...v6.14.4) (2025-03-18)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
### Bug Fixes
 | 
			
		||||
 | 
			
		||||
* removing cli dependency warnings2 ([43c0f73](https://git.datacontroller.io/dc/dc/commit/43c0f73c2189ff762986a964caae6b0b108164fc))
 | 
			
		||||
 | 
			
		||||
## [6.14.3](https://git.datacontroller.io/dc/dc/compare/v6.14.2...v6.14.3) (2025-03-15)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
### Bug Fixes
 | 
			
		||||
 | 
			
		||||
* NLDAT & NLDATM formats are now being staged ([3f5cb1e](https://git.datacontroller.io/dc/dc/commit/3f5cb1e2defe390220e904e4bf04a165cb31fec4))
 | 
			
		||||
 | 
			
		||||
## [6.14.2](https://git.datacontroller.io/dc/dc/compare/v6.14.1...v6.14.2) (2025-03-10)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
### Bug Fixes
 | 
			
		||||
 | 
			
		||||
* improving instructions for setup ([83b3d77](https://git.datacontroller.io/dc/dc/commit/83b3d775b6e33653b087ca9f4eb3ad5b0dbbd479))
 | 
			
		||||
 | 
			
		||||
## [6.14.1](https://git.datacontroller.io/dc/dc/compare/v6.14.0...v6.14.1) (2025-03-05)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
### Bug Fixes
 | 
			
		||||
 | 
			
		||||
* handle national language datetime formats ([149e318](https://git.datacontroller.io/dc/dc/commit/149e318a8787be0109f25aeec3a1270ea75a97b2))
 | 
			
		||||
* updating logic to use NLDAT formats ([95289aa](https://git.datacontroller.io/dc/dc/commit/95289aa9524d3cb2b1c248cfb84f6b0d0a490c32))
 | 
			
		||||
 | 
			
		||||
# [6.14.0](https://git.datacontroller.io/dc/dc/compare/v6.13.2...v6.14.0) (2025-02-26)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
### Features
 | 
			
		||||
 | 
			
		||||
* uses SORTSEQ=LINGUISTIC for the services/metanav/metadetails service ([a45f5bb](https://git.datacontroller.io/dc/dc/commit/a45f5bb3b27a86da5f55ff28c9c7669956484ddf))
 | 
			
		||||
 | 
			
		||||
## [6.13.2](https://git.datacontroller.io/dc/dc/compare/v6.13.1...v6.13.2) (2025-02-26)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
### Bug Fixes
 | 
			
		||||
 | 
			
		||||
* get metadata email if exists ([1bd0eef](https://git.datacontroller.io/dc/dc/commit/1bd0eef913593579771c647d80010ce628c00819))
 | 
			
		||||
 | 
			
		||||
## [6.13.1](https://git.datacontroller.io/dc/dc/compare/v6.13.0...v6.13.1) (2025-02-18)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
### Bug Fixes
 | 
			
		||||
 | 
			
		||||
* Avoiding LATIN1 unprintables in various UI locations ([bce1fd5](https://git.datacontroller.io/dc/dc/commit/bce1fd57ef397cfdd030552c3f424a8407174729))
 | 
			
		||||
* updated @sasjs/adapter, crypto-browserify ([4e64f28](https://git.datacontroller.io/dc/dc/commit/4e64f28732868b07ec2ab403912bd384c62c7e1d))
 | 
			
		||||
 | 
			
		||||
# [6.13.0](https://git.datacontroller.io/dc/dc/compare/v6.12.3...v6.13.0) (2025-01-31)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
### Bug Fixes
 | 
			
		||||
 | 
			
		||||
* editor page csv upload ([217220f](https://git.datacontroller.io/dc/dc/commit/217220ffaaf688133321cc68d770aaf1e50590a9))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
### Features
 | 
			
		||||
 | 
			
		||||
* csv test ([c53ab85](https://git.datacontroller.io/dc/dc/commit/c53ab85107f10c023117a099cc06321afc3e1f03))
 | 
			
		||||
 | 
			
		||||
## [6.12.3](https://git.datacontroller.io/dc/dc/compare/v6.12.2...v6.12.3) (2025-01-27)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
### Bug Fixes
 | 
			
		||||
 | 
			
		||||
* adding missing=STRING to three services ([30d5e51](https://git.datacontroller.io/dc/dc/commit/30d5e51d0b9cf27774038476bd90559600952304))
 | 
			
		||||
 | 
			
		||||
## [6.12.2](https://git.datacontroller.io/dc/dc/compare/v6.12.1...v6.12.2) (2025-01-27)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
### Bug Fixes
 | 
			
		||||
 | 
			
		||||
* unnecessary zeros when adding new row (data schema default values) ([a1a9051](https://git.datacontroller.io/dc/dc/commit/a1a90519c535ca25e00822b4d3358c991ac9662e))
 | 
			
		||||
 | 
			
		||||
## [6.12.1](https://git.datacontroller.io/dc/dc/compare/v6.12.0...v6.12.1) (2024-12-31)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
### Bug Fixes
 | 
			
		||||
 | 
			
		||||
* no upcase of pk fields in MPE_TABLES in delete scenario ([3de095f](https://git.datacontroller.io/dc/dc/commit/3de095fe7797cde60f0e232c188305fe423c27eb)), closes [#134](https://git.datacontroller.io/dc/dc/issues/134)
 | 
			
		||||
* reduce length of tmp table names.  Closes [#130](https://git.datacontroller.io/dc/dc/issues/130) ([f9c2491](https://git.datacontroller.io/dc/dc/commit/f9c2491ab6e7b528b7ffc011fd9e45c963c5f6bf))
 | 
			
		||||
 | 
			
		||||
# [6.12.0](https://git.datacontroller.io/dc/dc/compare/v6.11.1...v6.12.0) (2024-09-02)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
### Bug Fixes
 | 
			
		||||
 | 
			
		||||
* added appLoc to the system page ([dd2138a](https://git.datacontroller.io/dc/dc/commit/dd2138ac5e6067de310e83d16fccc9b9764ba3ff))
 | 
			
		||||
* bumping core for passthrough fix, [#124](https://git.datacontroller.io/dc/dc/issues/124) ([caa9854](https://git.datacontroller.io/dc/dc/commit/caa9854ff0431ccbb6ff1d6d3509dc877362cceb))
 | 
			
		||||
* excel with password flow, introducing web worker for XLSX.read ([a3ce367](https://git.datacontroller.io/dc/dc/commit/a3ce36795007a4e3b6ac3499ffd119dc3758f387))
 | 
			
		||||
* implemented the new request wrapper usage, added XLSX read with a Web Worker, multi load preview data, full height ([4218da9](https://git.datacontroller.io/dc/dc/commit/4218da91cd193aa45346ad7e34ccc00ca89df4fb))
 | 
			
		||||
* **multi load:** xlsx read file ahead of time, while user choose datasets ([6547461](https://git.datacontroller.io/dc/dc/commit/65474616379e1dacc1329b3bdc5eb14f34428bb1))
 | 
			
		||||
* refactored adapter request wrapper function to return job log as well ([67436f4](https://git.datacontroller.io/dc/dc/commit/67436f4ff9bb4d77d5f897f47a3e3d472981f275))
 | 
			
		||||
* using temporary names for temporary tables ([ce50365](https://git.datacontroller.io/dc/dc/commit/ce503653cd9fc36f72fb172bd14816e07c792e14)), closes [#124](https://git.datacontroller.io/dc/dc/issues/124)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
### Features
 | 
			
		||||
 | 
			
		||||
* searching data in excel files using new algorithm (massive performance improvement) ([bbb725c](https://git.datacontroller.io/dc/dc/commit/bbb725c64cc23ed701b189623992408c42fdde8f))
 | 
			
		||||
 | 
			
		||||
## [6.11.1](https://git.datacontroller.io/dc/dc/compare/v6.11.0...v6.11.1) (2024-07-02)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
### Bug Fixes
 | 
			
		||||
 | 
			
		||||
* adding SYSSITE, part of [#116](https://git.datacontroller.io/dc/dc/issues/116) ([a156c01](https://git.datacontroller.io/dc/dc/commit/a156c0111b3de5e3744e38d377d6e9aa09915803))
 | 
			
		||||
* ensuring review_reason_txt in output.  Closes [#117](https://git.datacontroller.io/dc/dc/issues/117) ([e5d93fd](https://git.datacontroller.io/dc/dc/commit/e5d93fd7d6d86bc47ff56664bd812b4d9d0749a5))
 | 
			
		||||
 | 
			
		||||
# [6.11.0](https://git.datacontroller.io/dc/dc/compare/v6.10.1...v6.11.0) (2024-06-27)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
### Bug Fixes
 | 
			
		||||
 | 
			
		||||
* addressing PR comments ([d94df7f](https://git.datacontroller.io/dc/dc/commit/d94df7f0ebae8feab5e1d5cf8011af8c8be2ca18))
 | 
			
		||||
* **multi load:** fixed parsing algorithm reused for the multi load, the fix affects the normal upload as well. ([d4fee79](https://git.datacontroller.io/dc/dc/commit/d4fee791a72021e449cf9680c3e3a525dce41ac1))
 | 
			
		||||
* **multi load:** label rename ([fa04d7b](https://git.datacontroller.io/dc/dc/commit/fa04d7bf4e5ba337146bdaa926c60488f8851449))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
### Features
 | 
			
		||||
 | 
			
		||||
* **multi load:** added HOT for user datasets input ([18363bb](https://git.datacontroller.io/dc/dc/commit/18363bbbeb9cf96183ba4841da8134b2f66f735c))
 | 
			
		||||
* **multi load:** implemented matching libds and parsing of the multiple sheets ([efcdc69](https://git.datacontroller.io/dc/dc/commit/efcdc694dd275cdb9a4e19f26e5522b8dadc5fd9))
 | 
			
		||||
* **multi load:** licence submit limits ([cffeab8](https://git.datacontroller.io/dc/dc/commit/cffeab813d8d4b324f82710dfd73953d4cbf8ffe))
 | 
			
		||||
* **multi load:** multiple csv files ([4d27665](https://git.datacontroller.io/dc/dc/commit/4d276657b35a147a2233a03afcb1716348555f52))
 | 
			
		||||
* **multi load:** refactored range find function, unlocking excel with password is reusable ([eb7c443](https://git.datacontroller.io/dc/dc/commit/eb7c44333c865e7f7bbfb54dd7f73bfc110f86a7))
 | 
			
		||||
* **multi load:** submitting multiple found tables at once ([5deba44](https://git.datacontroller.io/dc/dc/commit/5deba44d2b7352866d821b70dbbfbbf54955dc47))
 | 
			
		||||
 | 
			
		||||
## [6.10.1](https://git.datacontroller.io/dc/dc/compare/v6.10.0...v6.10.1) (2024-06-07)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
### Bug Fixes
 | 
			
		||||
 | 
			
		||||
* adding 60 more colours to crayons table. Closes [#112](https://git.datacontroller.io/dc/dc/issues/112) ([3521579](https://git.datacontroller.io/dc/dc/commit/3521579dead089eebf62455686be3aee88bde687))
 | 
			
		||||
* terms and conditions colours, editor on smaller screens show only icons ([e32d44b](https://git.datacontroller.io/dc/dc/commit/e32d44b1bcdfeea43d19b21ec0ddf4af1ce3992a))
 | 
			
		||||
 | 
			
		||||
# [6.10.0](https://git.datacontroller.io/dc/dc/compare/v6.9.0...v6.10.0) (2024-06-07)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
### Features
 | 
			
		||||
 | 
			
		||||
* updated handsontable to v14 ([2f8d0b7](https://git.datacontroller.io/dc/dc/commit/2f8d0b764a957ad8c11cd1088fad5e0670aa1731))
 | 
			
		||||
 | 
			
		||||
# [6.9.0](https://git.datacontroller.io/dc/dc/compare/v6.8.5...v6.9.0) (2024-05-31)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
### Bug Fixes
 | 
			
		||||
 | 
			
		||||
* added colors.scss file, start of a refactor ([110ad9a](https://git.datacontroller.io/dc/dc/commit/110ad9a6e9ed39bd5591ae65c2d0005ba47ca758))
 | 
			
		||||
* added stealFocus directive ([9a79f37](https://git.datacontroller.io/dc/dc/commit/9a79f37bf143a1e05df7407358e2687c678e3e68))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
### Features
 | 
			
		||||
 | 
			
		||||
* added app settings service to handle theme persistance, fix: optimised dark mode contrast ([35844e0](https://git.datacontroller.io/dc/dc/commit/35844e0cf1a639553269f2ab0f8666a56ab5cc47))
 | 
			
		||||
* **dark mode:** clarity optimizations ([afa7e38](https://git.datacontroller.io/dc/dc/commit/afa7e380aa3bdabd380c038522b9d73d9a8a3b91))
 | 
			
		||||
* **dark mode:** lineage and metadata ([27907ed](https://git.datacontroller.io/dc/dc/commit/27907ed00fe81f4c752ffe99d2fb029d5c884f0a))
 | 
			
		||||
* **dark mode:** refactoring clarity to enable dark mode, added toggle button ([5564aea](https://git.datacontroller.io/dc/dc/commit/5564aea9c25f8e81ff85afa8352325b9992e4043))
 | 
			
		||||
* **dark mode:** removing custom css rules so clarity can handle dark/light modes. Handsontable css for dark mode ([2c0afd0](https://git.datacontroller.io/dc/dc/commit/2c0afd02684cdf3bda374731b0359665e00ed95d))
 | 
			
		||||
 | 
			
		||||
## [6.8.5](https://git.datacontroller.io/dc/dc/compare/v6.8.4...v6.8.5) (2024-05-23)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
### Bug Fixes
 | 
			
		||||
 | 
			
		||||
* bitemporal load issue [#105](https://git.datacontroller.io/dc/dc/issues/105) ([967698e](https://git.datacontroller.io/dc/dc/commit/967698e4ce1e0abcbc6f0aff8a4be6c512dee93c))
 | 
			
		||||
 | 
			
		||||
## [6.8.4](https://git.datacontroller.io/dc/dc/compare/v6.8.3...v6.8.4) (2024-05-22)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
### Bug Fixes
 | 
			
		||||
 | 
			
		||||
* new approach to fixing [#105](https://git.datacontroller.io/dc/dc/issues/105) ([c11bd9a](https://git.datacontroller.io/dc/dc/commit/c11bd9a2c55e49f10451962cb2e222c21206bce5))
 | 
			
		||||
 | 
			
		||||
## [6.8.3](https://git.datacontroller.io/dc/dc/compare/v6.8.2...v6.8.3) (2024-05-09)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
### Bug Fixes
 | 
			
		||||
 | 
			
		||||
* updating core to increase filename length, closes [#103](https://git.datacontroller.io/dc/dc/issues/103) ([ee58fd5](https://git.datacontroller.io/dc/dc/commit/ee58fd5b4bc0dd3e3f232c4f26bb85b2e7fe2b54))
 | 
			
		||||
 | 
			
		||||
## [6.8.2](https://git.datacontroller.io/dc/dc/compare/v6.8.1...v6.8.2) (2024-05-03)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
### Bug Fixes
 | 
			
		||||
 | 
			
		||||
* dc_request_logs option feature ([93758ef](https://git.datacontroller.io/dc/dc/commit/93758efb275966c181f1ee8b6c752010909a0282))
 | 
			
		||||
* release process ([c0dc919](https://git.datacontroller.io/dc/dc/commit/c0dc9191e3b95ea6f7e5021fc0bdbcab0af4cc64))
 | 
			
		||||
 | 
			
		||||
## [6.8.1](https://git.datacontroller.io/dc/dc/compare/v6.8.0...v6.8.1) (2024-05-02)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
### Bug Fixes
 | 
			
		||||
 | 
			
		||||
* hide approve button when table revertable ([ec0f539](https://git.datacontroller.io/dc/dc/commit/ec0f539a337b176c83a661ff520a6892d47efa02))
 | 
			
		||||
 | 
			
		||||
# [6.8.0](https://git.datacontroller.io/dc/dc/compare/v6.7.0...v6.8.0) (2024-05-02)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
### Bug Fixes
 | 
			
		||||
 | 
			
		||||
* ci sheet lib, submit message auto focus ([c5e4650](https://git.datacontroller.io/dc/dc/commit/c5e46503272f3f3d9cd83ac04225babf79d4de44))
 | 
			
		||||
* **clarity:** new version style issues ([8c7de5a](https://git.datacontroller.io/dc/dc/commit/8c7de5aad7e7e32a64769696af9b93eb9a6225d3))
 | 
			
		||||
* cypress tests ([3dd85cc](https://git.datacontroller.io/dc/dc/commit/3dd85cc60bd5ac99bc930b6b9c89a8e707e4d51d))
 | 
			
		||||
* ensuring that only restorable versions are restorable ([a402856](https://git.datacontroller.io/dc/dc/commit/a4028562ce91b32ff971ab9821328b97cd23f381))
 | 
			
		||||
* ensuring version history only includes loaded versions ([51ebd25](https://git.datacontroller.io/dc/dc/commit/51ebd25aa362aa8e66c83b29b2c64aa0f206f5bd))
 | 
			
		||||
* final testing on restore feature ([297a84d](https://git.datacontroller.io/dc/dc/commit/297a84d3a4ebb47bef7f3ca9758978d727afed8d))
 | 
			
		||||
* issue with multiple adds/deletes, [#84](https://git.datacontroller.io/dc/dc/issues/84) ([904ca30](https://git.datacontroller.io/dc/dc/commit/904ca30f918da085fa05dae066367b512933d1a9))
 | 
			
		||||
* load_ref var ([aaad9f7](https://git.datacontroller.io/dc/dc/commit/aaad9f7207115599a006980fff099d59738dd2cd))
 | 
			
		||||
* removing alerts dummy data, closes [#93](https://git.datacontroller.io/dc/dc/issues/93) ([eba21e9](https://git.datacontroller.io/dc/dc/commit/eba21e96b4fa34e63b4477281f47d9a01d621f2e))
 | 
			
		||||
* restore table version improvement ([549f357](https://git.datacontroller.io/dc/dc/commit/549f35766ba7b5bbe55694845e85bfefc4193375))
 | 
			
		||||
* **sas:** viewer versions fix ([c6595c1](https://git.datacontroller.io/dc/dc/commit/c6595c1f618803d9202cba1a1fe76986449cf2e2))
 | 
			
		||||
* stage and approve buttons renaming ([ef81e33](https://git.datacontroller.io/dc/dc/commit/ef81e33f704d0b4f99405d96498e1d29ac817982))
 | 
			
		||||
* supporting SCD2 data reversions ([fa8396f](https://git.datacontroller.io/dc/dc/commit/fa8396f0394cbddb6dbacb4b355de078fad49980))
 | 
			
		||||
* table info modal, versions - column names ([801c8c6](https://git.datacontroller.io/dc/dc/commit/801c8c6a9fb95388a06a6c6284fec4dc25bb77c5))
 | 
			
		||||
* **updates:** angular, clarity, resolved legacy-peer-deps ([c60dd65](https://git.datacontroller.io/dc/dc/commit/c60dd65a1637333f11a0c39ef697c7292a6ede07))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
### Features
 | 
			
		||||
 | 
			
		||||
* backend to show in getchangeinfo whether a user is allowed to restore ([8769841](https://git.datacontroller.io/dc/dc/commit/8769841f08694f672ef7ae1a17beacd0dbedda52))
 | 
			
		||||
* list versions of target tables (backend) ([f8a14d4](https://git.datacontroller.io/dc/dc/commit/f8a14d4bdef055b99930491d1f6fabe55a50a497))
 | 
			
		||||
* restore ([604c2e7](https://git.datacontroller.io/dc/dc/commit/604c2e70bdeeeb1aa5bb18b94f525ebd049397fa))
 | 
			
		||||
* SAS services & tests for RESTORE, [#84](https://git.datacontroller.io/dc/dc/issues/84) ([9ad7ae4](https://git.datacontroller.io/dc/dc/commit/9ad7ae47b5e793ce68ab21c9eeb8dee6cb85e496))
 | 
			
		||||
* staging page, restore buttons ([02a8a1c](https://git.datacontroller.io/dc/dc/commit/02a8a1c5654350cafc53b749cceb686fc6848c33))
 | 
			
		||||
* table metadata modal, versions tab (and link) ([b27fea5](https://git.datacontroller.io/dc/dc/commit/b27fea5b91e33b4673a3a991aedae558e366ca29))
 | 
			
		||||
* **versions:** getting list of versions (plus test) ([8003da9](https://git.datacontroller.io/dc/dc/commit/8003da94e615463ed3ddfd60b0cbf2e58615eab1))
 | 
			
		||||
 | 
			
		||||
# [6.7.0](https://git.datacontroller.io/dc/dc/compare/v6.6.4...v6.7.0) (2024-04-01)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -69,7 +69,8 @@
 | 
			
		||||
            ],
 | 
			
		||||
            "scripts": [
 | 
			
		||||
              "node_modules/marked/marked.min.js"
 | 
			
		||||
            ]
 | 
			
		||||
            ],
 | 
			
		||||
            "webWorkerTsConfig": "tsconfig.worker.json"
 | 
			
		||||
          },
 | 
			
		||||
          "configurations": {
 | 
			
		||||
            "production": {
 | 
			
		||||
@@ -133,25 +134,23 @@
 | 
			
		||||
        "test": {
 | 
			
		||||
          "builder": "@angular-devkit/build-angular:karma",
 | 
			
		||||
          "options": {
 | 
			
		||||
            "tsConfig": "tsconfig.spec.json",
 | 
			
		||||
            "inlineStyleLanguage": "scss",
 | 
			
		||||
            "codeCoverage": true,
 | 
			
		||||
            "polyfills": [
 | 
			
		||||
              "src/polyfills.ts",
 | 
			
		||||
              "zone.js",
 | 
			
		||||
              "zone.js/testing"
 | 
			
		||||
            ],
 | 
			
		||||
            "styles": [
 | 
			
		||||
              "src/styles.scss"
 | 
			
		||||
            ],
 | 
			
		||||
            "scripts": [
 | 
			
		||||
 | 
			
		||||
            ],
 | 
			
		||||
            "tsConfig": "tsconfig.spec.json",
 | 
			
		||||
            "inlineStyleLanguage": "scss",
 | 
			
		||||
            "assets": [
 | 
			
		||||
              "src/favicon.ico",
 | 
			
		||||
              "src/assets"
 | 
			
		||||
            ],
 | 
			
		||||
            "karmaConfig": "karma.conf.js"
 | 
			
		||||
            "styles": [
 | 
			
		||||
              "src/styles.scss"
 | 
			
		||||
            ],
 | 
			
		||||
            "scripts": [],
 | 
			
		||||
            "karmaConfig": "karma.conf.js",
 | 
			
		||||
            "webWorkerTsConfig": "tsconfig.worker.json"
 | 
			
		||||
          }
 | 
			
		||||
        },
 | 
			
		||||
        "lint": {
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										115
									
								
								client/cypress/e2e/csv.cy.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										115
									
								
								client/cypress/e2e/csv.cy.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,115 @@
 | 
			
		||||
import { Callbacks } from 'cypress/types/jquery/index'
 | 
			
		||||
 | 
			
		||||
const username = Cypress.env('username')
 | 
			
		||||
const password = Cypress.env('password')
 | 
			
		||||
const hostUrl = Cypress.env('hosturl')
 | 
			
		||||
const appLocation = Cypress.env('appLocation')
 | 
			
		||||
const longerCommandTimeout = Cypress.env('longerCommandTimeout')
 | 
			
		||||
const serverType = Cypress.env('serverType')
 | 
			
		||||
const libraryToOpenIncludes = Cypress.env(`libraryToOpenIncludes_${serverType}`)
 | 
			
		||||
const fixturePath = 'csvs/'
 | 
			
		||||
 | 
			
		||||
context('excel tests: ', function () {
 | 
			
		||||
  this.beforeAll(() => {
 | 
			
		||||
    cy.visit(`${hostUrl}/SASLogon/logout`)
 | 
			
		||||
    cy.loginAndUpdateValidKey(true)
 | 
			
		||||
  })
 | 
			
		||||
 | 
			
		||||
  this.beforeEach(() => {
 | 
			
		||||
    cy.visit(hostUrl + appLocation)
 | 
			
		||||
 | 
			
		||||
    visitPage('home')
 | 
			
		||||
 | 
			
		||||
    colorLog(
 | 
			
		||||
      `TEST START ---> ${
 | 
			
		||||
        Cypress.mocha.getRunner().suite.ctx.currentTest.title
 | 
			
		||||
      }`,
 | 
			
		||||
      '#3498DB'
 | 
			
		||||
    )
 | 
			
		||||
  })
 | 
			
		||||
 | 
			
		||||
  it('1 | Uploads regular csv file', () => {
 | 
			
		||||
    openTableFromTree(libraryToOpenIncludes, 'mpe_x_test')
 | 
			
		||||
 | 
			
		||||
    attachExcelFile('regular.csv', () => {
 | 
			
		||||
      cy.get('#approval-btn', { timeout: 60000 })
 | 
			
		||||
        .should('be.visible')
 | 
			
		||||
        // .then(() => {
 | 
			
		||||
        //   cy.get('#hotInstance', { timeout: 30000 })
 | 
			
		||||
        //     .find('div.ht_master.handsontable')
 | 
			
		||||
        //     .find('div.wtHolder')
 | 
			
		||||
        //     .find('div.wtHider')
 | 
			
		||||
        //     .find('div.wtSpreader')
 | 
			
		||||
        //     .find('table.htCore')
 | 
			
		||||
        //     .find('tbody')
 | 
			
		||||
        //     .then((data) => {
 | 
			
		||||
        //       let cell: any = data[0].children[0].children[1]
 | 
			
		||||
        //       expect(cell.innerText).to.equal('0')
 | 
			
		||||
        //       cell = data[0].children[0].children[2]
 | 
			
		||||
        //       expect(cell.innerText).to.equal('44')
 | 
			
		||||
        //       cell = data[0].children[0].children[3]
 | 
			
		||||
        //       expect(cell.innerText).to.equal('abc')
 | 
			
		||||
        //       cell = data[0].children[0].children[6]
 | 
			
		||||
        //       expect(cell.innerText).to.equal('Option abc')
 | 
			
		||||
        //     })
 | 
			
		||||
        // })
 | 
			
		||||
    })
 | 
			
		||||
  })
 | 
			
		||||
 | 
			
		||||
  this.afterEach(() => {
 | 
			
		||||
    colorLog(`TEST END -------------`, '#3498DB')
 | 
			
		||||
  })
 | 
			
		||||
})
 | 
			
		||||
 | 
			
		||||
const openTableFromTree = (libNameIncludes: string, tablename: string) => {
 | 
			
		||||
  cy.get('.app-loading', { timeout: longerCommandTimeout })
 | 
			
		||||
    .should('not.exist')
 | 
			
		||||
    .then(() => {
 | 
			
		||||
      cy.get('.nav-tree clr-tree > clr-tree-node', {
 | 
			
		||||
        timeout: longerCommandTimeout
 | 
			
		||||
      }).then((treeNodes: any) => {
 | 
			
		||||
        let viyaLib
 | 
			
		||||
 | 
			
		||||
        for (let node of treeNodes) {
 | 
			
		||||
          if (node.innerText.toLowerCase().includes(libNameIncludes)) {
 | 
			
		||||
            viyaLib = node
 | 
			
		||||
            break
 | 
			
		||||
          }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        cy.get(viyaLib).within(() => {
 | 
			
		||||
          cy.get('.clr-tree-node-content-container > button').click()
 | 
			
		||||
 | 
			
		||||
          cy.get('.clr-treenode-link').then((innerNodes: any) => {
 | 
			
		||||
            for (let innerNode of innerNodes) {
 | 
			
		||||
              if (innerNode.innerText.toLowerCase().includes(tablename)) {
 | 
			
		||||
                innerNode.click()
 | 
			
		||||
                break
 | 
			
		||||
              }
 | 
			
		||||
            }
 | 
			
		||||
          })
 | 
			
		||||
        })
 | 
			
		||||
      })
 | 
			
		||||
    })
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const attachExcelFile = (excelFilename: string, callback?: any) => {
 | 
			
		||||
  cy.get('.buttonBar button:last-child')
 | 
			
		||||
    .should('exist')
 | 
			
		||||
    .click()
 | 
			
		||||
    .then(() => {
 | 
			
		||||
      cy.get('input[type="file"]#file-upload')
 | 
			
		||||
        .attachFile(`/${fixturePath}/${excelFilename}`)
 | 
			
		||||
        .then(() => {
 | 
			
		||||
          if (callback) callback()
 | 
			
		||||
        })
 | 
			
		||||
    })
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const visitPage = (url: string) => {
 | 
			
		||||
  cy.visit(`${hostUrl}${appLocation}/#/${url}`)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const colorLog = (msg: string, color: string) => {
 | 
			
		||||
  console.log('%c' + msg, 'color:' + color + ';font-weight:bold;')
 | 
			
		||||
}
 | 
			
		||||
@@ -15,9 +15,6 @@ context('editor tests: ', function () {
 | 
			
		||||
 | 
			
		||||
  this.beforeEach(() => {
 | 
			
		||||
    cy.visit(hostUrl + appLocation)
 | 
			
		||||
    // cy.get('input.username').type(username)
 | 
			
		||||
    // cy.get('input.password').type(password)
 | 
			
		||||
    // cy.get('.login-group button').click()
 | 
			
		||||
 | 
			
		||||
    visitPage('home')
 | 
			
		||||
  })
 | 
			
		||||
@@ -118,10 +115,6 @@ context('editor tests: ', function () {
 | 
			
		||||
      })
 | 
			
		||||
    })
 | 
			
		||||
  })
 | 
			
		||||
 | 
			
		||||
  this.afterEach(() => {
 | 
			
		||||
    // cy.visit(`${hostUrl}/SASLogon/logout`)
 | 
			
		||||
  })
 | 
			
		||||
})
 | 
			
		||||
 | 
			
		||||
const clickOnEdit = (callback?: any) => {
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										228
									
								
								client/cypress/e2e/excel-multi-load.cy.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										228
									
								
								client/cypress/e2e/excel-multi-load.cy.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,228 @@
 | 
			
		||||
import { Callbacks } from 'cypress/types/jquery/index'
 | 
			
		||||
 | 
			
		||||
const username = Cypress.env('username')
 | 
			
		||||
const password = Cypress.env('password')
 | 
			
		||||
const hostUrl = Cypress.env('hosturl')
 | 
			
		||||
const appLocation = Cypress.env('appLocation')
 | 
			
		||||
const longerCommandTimeout = Cypress.env('longerCommandTimeout')
 | 
			
		||||
const serverType = Cypress.env('serverType')
 | 
			
		||||
const libraryToOpenIncludes = Cypress.env(`libraryToOpenIncludes_${serverType}`)
 | 
			
		||||
const fixturePath = 'excels_multi_load/'
 | 
			
		||||
 | 
			
		||||
const library = 'DC996664'
 | 
			
		||||
const mpeXTestTable = 'MPE_X_TEST'
 | 
			
		||||
const mpeTablesTable = 'MPE_TABLES'
 | 
			
		||||
 | 
			
		||||
context('excel multi load tests: ', function () {
 | 
			
		||||
  this.beforeAll(() => {
 | 
			
		||||
    cy.visit(`${hostUrl}/SASLogon/logout`)
 | 
			
		||||
    cy.loginAndUpdateValidKey(true)
 | 
			
		||||
  })
 | 
			
		||||
 | 
			
		||||
  this.beforeEach(() => {
 | 
			
		||||
    cy.visit(hostUrl + appLocation)
 | 
			
		||||
 | 
			
		||||
    visitPage('home/multi-load')
 | 
			
		||||
 | 
			
		||||
    colorLog(
 | 
			
		||||
      `TEST START ---> ${
 | 
			
		||||
        Cypress.mocha.getRunner().suite.ctx.currentTest.title
 | 
			
		||||
      }`,
 | 
			
		||||
      '#3498DB'
 | 
			
		||||
    )
 | 
			
		||||
  })
 | 
			
		||||
 | 
			
		||||
  it('1 | Uploads Excel file with multiple sheets, 3 sheets including data, 2 sheets matched with dataset', (done) => {
 | 
			
		||||
    attachExcelFile('multi_load_test_2.xlsx', () => {
 | 
			
		||||
      checkHotUserDatasetTable('hotTableUserDataset', [
 | 
			
		||||
        [library, mpeXTestTable],
 | 
			
		||||
        [library, mpeTablesTable]
 | 
			
		||||
      ], () => {
 | 
			
		||||
        cy.get('#continue-btn').trigger('click').then(() => {
 | 
			
		||||
          checkIfTreeHasTables([`${library}.${mpeXTestTable}`, `${library}.${mpeTablesTable}`], undefined, (includes: boolean) => {
 | 
			
		||||
            if (includes) {
 | 
			
		||||
              // MPE_TABLES sheet does not have data so 1 error image must be shown
 | 
			
		||||
              hasErrorTables(1, (valid: boolean) => {
 | 
			
		||||
                if (valid) done()
 | 
			
		||||
              })
 | 
			
		||||
            }
 | 
			
		||||
          })
 | 
			
		||||
        })
 | 
			
		||||
      })
 | 
			
		||||
    })
 | 
			
		||||
  })
 | 
			
		||||
 | 
			
		||||
  it('2 | Uploads Excel file with multiple sheets, 2 sheets matched with dataset, 1 matched sheet does not have data', (done) => {
 | 
			
		||||
    attachExcelFile('multi_load_test_1.xlsx', () => {
 | 
			
		||||
      checkHotUserDatasetTable('hotTableUserDataset', [
 | 
			
		||||
        [library, mpeXTestTable],
 | 
			
		||||
        [library, mpeTablesTable]
 | 
			
		||||
      ], () => {
 | 
			
		||||
        cy.get('#continue-btn').trigger('click').then(() => {
 | 
			
		||||
          checkIfTreeHasTables([`${library}.${mpeXTestTable}`, `${library}.${mpeTablesTable}`], `${library}.${mpeXTestTable}`, (includes: boolean) => {
 | 
			
		||||
            if (includes) {
 | 
			
		||||
              cy.get('#hotTable').should('be.visible').then(() => {
 | 
			
		||||
                checkHotUserDatasetTable('hotTable', [
 | 
			
		||||
                  ['No', '1', 'more dummy data'],
 | 
			
		||||
                  ['No', '1', 'It was a dark and stormy night.  The wind was blowing a gale!  The captain said to his mate - mate, tell us a tale.  And this, is the tale he told: It was a dark and stormy night.  The wind was blowing a gale!  The captain said to his mate - mate, tell us a tale.  And this, is the tale he told: It was a dark and stormy night.  The wind was blowing a gale!  The captain said to his mate - mate, tell us a tale.  And this, is the tale he told: It was a dark and stormy night.  The wind was blowing a gale!  The captain said to his mate - mate, tell us a tale.  And this, is the tale he told:'],
 | 
			
		||||
                  ['No', '1', 'if you can fill the unforgiving minute']
 | 
			
		||||
                ], () => {
 | 
			
		||||
                  submitTables()
 | 
			
		||||
 | 
			
		||||
                  hasSuccessSubmits(2, (valid: boolean) => {
 | 
			
		||||
                    if (valid) done()
 | 
			
		||||
                  })
 | 
			
		||||
 | 
			
		||||
                })
 | 
			
		||||
              })
 | 
			
		||||
            }
 | 
			
		||||
          })
 | 
			
		||||
        })
 | 
			
		||||
      })
 | 
			
		||||
    })
 | 
			
		||||
  })
 | 
			
		||||
 | 
			
		||||
  it('3 | Uploads Excel file with multiple sheets, 1 sheets has 2 tables', (done) => {
 | 
			
		||||
    attachExcelFile('multi_load_test_1.xlsx', () => {
 | 
			
		||||
      checkHotUserDatasetTable('hotTableUserDataset', [
 | 
			
		||||
        [library, mpeXTestTable],
 | 
			
		||||
        [library, mpeTablesTable]
 | 
			
		||||
      ], () => {
 | 
			
		||||
        cy.get('#continue-btn').trigger('click').then(() => {
 | 
			
		||||
          checkIfTreeHasTables([`${library}.${mpeXTestTable}`, `${library}.${mpeTablesTable}`], `${library}.${mpeXTestTable}`, (includes: boolean) => {
 | 
			
		||||
            if (includes) {
 | 
			
		||||
              cy.get('#hotTable').should('be.visible').then(() => {
 | 
			
		||||
                checkHotUserDatasetTable('hotTable', [
 | 
			
		||||
                  ['No', '1', 'more dummy data'],
 | 
			
		||||
                  ['No', '1', 'It was a dark and stormy night.  The wind was blowing a gale!  The captain said to his mate - mate, tell us a tale.  And this, is the tale he told: It was a dark and stormy night.  The wind was blowing a gale!  The captain said to his mate - mate, tell us a tale.  And this, is the tale he told: It was a dark and stormy night.  The wind was blowing a gale!  The captain said to his mate - mate, tell us a tale.  And this, is the tale he told: It was a dark and stormy night.  The wind was blowing a gale!  The captain said to his mate - mate, tell us a tale.  And this, is the tale he told:'],
 | 
			
		||||
                  ['No', '1', 'if you can fill the unforgiving minute']
 | 
			
		||||
                ], () => {
 | 
			
		||||
                  clickOnTreeNode('DC996664.MPE_TABLES', () => {
 | 
			
		||||
                    cy.wait(1000).then(() => {
 | 
			
		||||
                      cy.get('#hotTable').should('be.visible').then(() => {
 | 
			
		||||
                        checkHotUserDatasetTable('hotTable', [
 | 
			
		||||
                          ['No', 'DC914286', 'MPE_COLUMN_LEVEL_SECURITY'],
 | 
			
		||||
                          ['No', 'DC914286', 'MPE_XLMAP_INFO'],
 | 
			
		||||
                          ['No', 'DC914286', 'MPE_XLMAP_RULES']
 | 
			
		||||
                        ], () => {
 | 
			
		||||
                          submitTables()
 | 
			
		||||
 | 
			
		||||
                          hasSuccessSubmits(2, (valid: boolean) => {
 | 
			
		||||
                            if (valid) done()
 | 
			
		||||
                          })
 | 
			
		||||
 | 
			
		||||
                        })
 | 
			
		||||
                      })
 | 
			
		||||
                    })
 | 
			
		||||
                  })
 | 
			
		||||
                })
 | 
			
		||||
              })
 | 
			
		||||
            }
 | 
			
		||||
          })
 | 
			
		||||
        })
 | 
			
		||||
      })
 | 
			
		||||
    })
 | 
			
		||||
  })
 | 
			
		||||
 | 
			
		||||
  this.afterEach(() => {
 | 
			
		||||
    colorLog(`TEST END -------------`, '#3498DB')
 | 
			
		||||
  })
 | 
			
		||||
})
 | 
			
		||||
 | 
			
		||||
const attachExcelFile = (excelFilename: string, callback?: any) => {
 | 
			
		||||
  cy.get('#browse-file')
 | 
			
		||||
    .should('exist')
 | 
			
		||||
    .click()
 | 
			
		||||
    .then(() => {
 | 
			
		||||
      cy.get('input[type="file"]#file-upload')
 | 
			
		||||
        .attachFile(`/${fixturePath}/${excelFilename}`)
 | 
			
		||||
        .then(() => {
 | 
			
		||||
          if (callback) callback()
 | 
			
		||||
        })
 | 
			
		||||
    })
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const checkHotUserDatasetTable = (hotId: string, dataToContain: any[][], callback?: () => void) => {
 | 
			
		||||
  cy.get(`#${hotId}`, { timeout: longerCommandTimeout })
 | 
			
		||||
  .find('div.ht_master.handsontable')
 | 
			
		||||
  .find('div.wtHolder')
 | 
			
		||||
  .find('div.wtHider')
 | 
			
		||||
  .find('div.wtSpreader')
 | 
			
		||||
  .find('table.htCore')
 | 
			
		||||
  .find('tbody')
 | 
			
		||||
  .then((data) => {
 | 
			
		||||
    cy.wait(2000).then(() => {
 | 
			
		||||
      for (let rowI = 0; rowI < dataToContain.length; rowI++) {
 | 
			
		||||
        for (let colI = 0; colI < dataToContain[rowI].length; colI++) {
 | 
			
		||||
          expect(data[0].children[rowI].children[colI]).to.contain(dataToContain[rowI][colI])
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      if (callback) callback()
 | 
			
		||||
    })
 | 
			
		||||
  })
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const clickOnTreeNode = (clickOnNode: string, callback?: () => void) => {
 | 
			
		||||
  cy.get('.nav-tree clr-tree > clr-tree-node').then((treeNodes: any) => {
 | 
			
		||||
    for (let node of treeNodes) {
 | 
			
		||||
      if (node.innerText.toUpperCase().trim().includes(clickOnNode)) {
 | 
			
		||||
        cy.get(node).trigger('click')
 | 
			
		||||
        if (callback) callback()
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  })
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const checkIfTreeHasTables = (tables: string[], clickOnNode?: string, callback?: (includes: boolean) => void) => {
 | 
			
		||||
  cy.get('.nav-tree clr-tree > clr-tree-node').then((treeNodes: any) => {
 | 
			
		||||
    let datasets = tables
 | 
			
		||||
    let nodesCorrect = true
 | 
			
		||||
    let nodeToClick
 | 
			
		||||
 | 
			
		||||
    for (let node of treeNodes) {
 | 
			
		||||
      if (!datasets.includes(node.innerText.toUpperCase().trim())) {
 | 
			
		||||
        nodesCorrect = false
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      if (clickOnNode) {
 | 
			
		||||
        if (node.innerText.toUpperCase().trim().includes(clickOnNode)) {
 | 
			
		||||
          nodeToClick = node
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (nodeToClick) {
 | 
			
		||||
      cy.wait(1000)
 | 
			
		||||
      cy.get(nodeToClick).trigger('click')
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (callback) callback(nodesCorrect)
 | 
			
		||||
  })
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const submitTables = () => {
 | 
			
		||||
  cy.get('#submit-all').trigger('click')
 | 
			
		||||
  cy.get('#submit-tables').trigger('click')
 | 
			
		||||
  cy.wait(1000)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const hasSuccessSubmits = (expectedNoOfSubmits: number, callback: (valid: boolean) => void) => {
 | 
			
		||||
  cy.get('.nav-tree clr-tree > clr-tree-node cds-icon[status="success"]').should('be.visible').then(($nodes) => {
 | 
			
		||||
    callback(expectedNoOfSubmits === $nodes.length)
 | 
			
		||||
  })
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const hasErrorTables = (expectedNoOfErrors: number, callback: (valid: boolean) => void) => {
 | 
			
		||||
  cy.get('.nav-tree clr-tree > clr-tree-node cds-icon[status="danger"]').should('be.visible').then(($nodes) => {
 | 
			
		||||
    callback(expectedNoOfErrors === $nodes.length)
 | 
			
		||||
  })
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const visitPage = (url: string) => {
 | 
			
		||||
  cy.visit(`${hostUrl}${appLocation}/#/${url}`)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const colorLog = (msg: string, color: string) => {
 | 
			
		||||
  console.log('%c' + msg, 'color:' + color + ';font-weight:bold;')
 | 
			
		||||
}
 | 
			
		||||
@@ -17,9 +17,6 @@ context('excel tests: ', function () {
 | 
			
		||||
 | 
			
		||||
  this.beforeEach(() => {
 | 
			
		||||
    cy.visit(hostUrl + appLocation)
 | 
			
		||||
    // cy.get('input.username').type(username)
 | 
			
		||||
    // cy.get('input.password').type(password)
 | 
			
		||||
    // cy.get('.login-group button').click()
 | 
			
		||||
 | 
			
		||||
    visitPage('home')
 | 
			
		||||
 | 
			
		||||
@@ -112,13 +109,8 @@ context('excel tests: ', function () {
 | 
			
		||||
    openTableFromTree(libraryToOpenIncludes, 'mpe_x_test')
 | 
			
		||||
 | 
			
		||||
    attachExcelFile('duplicate_column_excel.xlsx', () => {
 | 
			
		||||
      cy.get('.abortMsg', { timeout: longerCommandTimeout })
 | 
			
		||||
        .should('exist')
 | 
			
		||||
        .then((elements: any) => {
 | 
			
		||||
          if (elements[0]) {
 | 
			
		||||
            if (elements[0].innerText.toLowerCase().includes('missing')) done()
 | 
			
		||||
          }
 | 
			
		||||
        })
 | 
			
		||||
      submitExcel()
 | 
			
		||||
      rejectExcel(done)
 | 
			
		||||
    })
 | 
			
		||||
  })
 | 
			
		||||
 | 
			
		||||
@@ -337,7 +329,6 @@ context('excel tests: ', function () {
 | 
			
		||||
 | 
			
		||||
  this.afterEach(() => {
 | 
			
		||||
    colorLog(`TEST END -------------`, '#3498DB')
 | 
			
		||||
    // cy.visit(`${hostUrl}/SASLogon/logout`)
 | 
			
		||||
  })
 | 
			
		||||
})
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -15,9 +15,7 @@ context('filtering tests: ', function () {
 | 
			
		||||
 | 
			
		||||
  this.beforeEach(() => {
 | 
			
		||||
    cy.visit(hostUrl + appLocation, { timeout: longerCommandTimeout })
 | 
			
		||||
    // cy.get('input.username').type(username)
 | 
			
		||||
    // cy.get('input.password').type(password)
 | 
			
		||||
    // cy.get('.login-group button').click()
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    visitPage('home')
 | 
			
		||||
  })
 | 
			
		||||
@@ -159,24 +157,21 @@ context('filtering tests: ', function () {
 | 
			
		||||
    })
 | 
			
		||||
  })
 | 
			
		||||
 | 
			
		||||
  it('7 | filter bestnum field BETWEEN', (done) => {
 | 
			
		||||
    openTableFromTree(libraryToOpenIncludes, 'mpe_x_test')
 | 
			
		||||
  // TODO: fix
 | 
			
		||||
  // it('7 | filter bestnum field BETWEEN', (done) => {
 | 
			
		||||
  //   openTableFromTree(libraryToOpenIncludes, 'mpe_x_test')
 | 
			
		||||
 | 
			
		||||
    openFilterPopup(() => {
 | 
			
		||||
      setFilterWithValue('SOME_BESTNUM', '0-10', 'between', () => {
 | 
			
		||||
        checkInfoBarIncludes(
 | 
			
		||||
          `AND,AND,0,SOME_BESTNUM,BETWEEN,0 AND 10`,
 | 
			
		||||
          (includes: boolean) => {
 | 
			
		||||
            if (includes) done()
 | 
			
		||||
          }
 | 
			
		||||
        )
 | 
			
		||||
      })
 | 
			
		||||
    })
 | 
			
		||||
  })
 | 
			
		||||
 | 
			
		||||
  this.afterEach(() => {
 | 
			
		||||
    // cy.visit(`${hostUrl}/SASLogon/logout`)
 | 
			
		||||
  })
 | 
			
		||||
  //   openFilterPopup(() => {
 | 
			
		||||
  //     setFilterWithValue('SOME_BESTNUM', '0-10', 'between', () => {
 | 
			
		||||
  //       checkInfoBarIncludes(
 | 
			
		||||
  //         `AND,AND,0,SOME_BESTNUM,BETWEEN,0 AND 10`,
 | 
			
		||||
  //         (includes: boolean) => {
 | 
			
		||||
  //           if (includes) done()
 | 
			
		||||
  //         }
 | 
			
		||||
  //       )
 | 
			
		||||
  //     })
 | 
			
		||||
  //   })
 | 
			
		||||
  // })
 | 
			
		||||
})
 | 
			
		||||
 | 
			
		||||
const checkInfoBarIncludes = (text: string, callback: any) => {
 | 
			
		||||
 
 | 
			
		||||
@@ -23,15 +23,12 @@ interface EditConfigTableCells {
 | 
			
		||||
 | 
			
		||||
context('licensing tests: ', function () {
 | 
			
		||||
  this.beforeAll(() => {
 | 
			
		||||
    // cy.visit(`${hostUrl}/SASLogon/logout`)
 | 
			
		||||
 | 
			
		||||
    cy.loginAndUpdateValidKey()
 | 
			
		||||
  })
 | 
			
		||||
 | 
			
		||||
  this.beforeEach(() => {
 | 
			
		||||
    cy.visit(hostUrl + appLocation)
 | 
			
		||||
    // cy.get('input.username').type(username)
 | 
			
		||||
    // cy.get('input.password').type(password)
 | 
			
		||||
    // cy.get('.login-group button').click()
 | 
			
		||||
 | 
			
		||||
    visitPage('home')
 | 
			
		||||
  })
 | 
			
		||||
@@ -375,9 +372,7 @@ context('licensing tests: ', function () {
 | 
			
		||||
    })
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  this.afterEach(() => {
 | 
			
		||||
    // cy.visit(`${hostUrl}/SASLogon/logout`)
 | 
			
		||||
  })
 | 
			
		||||
 | 
			
		||||
})
 | 
			
		||||
 | 
			
		||||
const logout = (callback?: any) => {
 | 
			
		||||
 
 | 
			
		||||
@@ -18,9 +18,6 @@ context('liveness tests: ', function () {
 | 
			
		||||
  this.beforeEach(() => {
 | 
			
		||||
    cy.visit(hostUrl + appLocation)
 | 
			
		||||
 | 
			
		||||
    // cy.get('input.username').type(username)
 | 
			
		||||
    // cy.get('input.password').type(password)
 | 
			
		||||
    // cy.get('.login-group button').click()
 | 
			
		||||
 | 
			
		||||
    visitPage('home')
 | 
			
		||||
  })
 | 
			
		||||
 
 | 
			
		||||
@@ -16,7 +16,6 @@ context('editor tests: ', function () {
 | 
			
		||||
 | 
			
		||||
  this.beforeEach(() => {
 | 
			
		||||
    cy.visit(hostUrl + appLocation)
 | 
			
		||||
 | 
			
		||||
    cy.wait(2000)
 | 
			
		||||
 | 
			
		||||
    cy.get('body').then(($body) => {
 | 
			
		||||
@@ -393,10 +392,6 @@ context('editor tests: ', function () {
 | 
			
		||||
  //       }
 | 
			
		||||
  //     )
 | 
			
		||||
  // })
 | 
			
		||||
 | 
			
		||||
  this.afterEach(() => {
 | 
			
		||||
    // cy.visit(`${hostUrl}/SASLogon/logout`)
 | 
			
		||||
  })
 | 
			
		||||
})
 | 
			
		||||
 | 
			
		||||
const removeAllColumns = () => {
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										497
									
								
								client/cypress/fixtures/csvs/regular.csv
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										497
									
								
								client/cypress/fixtures/csvs/regular.csv
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,497 @@
 | 
			
		||||
PRIMARY_KEY_FIELD,SOME_CHAR,SOME_DROPDOWN,SOME_NUM,SOME_DATE,SOME_DATETIME,SOME_TIME,SOME_SHORTNUM,SOME_BESTNUM
 | 
			
		||||
0,abc,Option abc,42,12FEB1960,01JAN1960:00:00:42,0:00:42,3,44
 | 
			
		||||
1,more dummy data,Option 2,42,12FEB1960,01JAN1960:00:00:42,0:07:02,3,44
 | 
			
		||||
2,even more dummy data,Option 3,42,12FEB1960,01JAN1960:00:00:42,0:02:22,3,44
 | 
			
		||||
3,"It was a dark and stormy night.  The wind was blowing a gale!  The captain said to his mate - mate, tell us a tale.  And this, is the tale he told: It was a dark and stormy night.  The wind was blowing a gale!  The captain said to his mate - mate, tell us a tale.  And this, is the tale he told: It was a dark and stormy night.  The wind was blowing a gale!  The captain said to his mate - mate, tell us a tale.  And this, is the tale he told: It was a dark and stormy night.  The wind was blowing a gale!  The captain said to his mate - mate, tell us a tale.  And this, is the tale he told:",Option 2,1613.001,27FEB1961,01JAN1960:00:07:03,0:00:44,3,44
 | 
			
		||||
4,if you can fill the unforgiving minute,Option 1,1613.0011235,02AUG1971,29MAY1973:06:12:03,0:06:52,3,44
 | 
			
		||||
1010,10 bottles of beer on the wall,Option 1,0.9153696885,04MAR1962,01JAN1960:12:47:55,0:01:40,92,76
 | 
			
		||||
1011,11 bottles of beer on the wall,Option 1,0.3531217558,29MAR1960,01JAN1960:03:33:24,0:01:03,80,29
 | 
			
		||||
1012,12 bottles of beer on the wall,Option 1,0.6743748717,02AUG1962,01JAN1960:07:25:59,0:00:10,16,98
 | 
			
		||||
1013,13 bottles of beer on the wall,Option 1,0.1305445992,11SEP1960,01JAN1960:13:51:32,0:00:35,73,15
 | 
			
		||||
1014,14 bottles of beer on the wall,Option 1,0.7409067949,26JUL1960,01JAN1960:05:18:10,0:00:41,30,89
 | 
			
		||||
1015,15 bottles of beer on the wall,Option 1,0.0869016028,28FEB1961,01JAN1960:13:23:45,0:00:44,80,3
 | 
			
		||||
1016,16 bottles of beer on the wall,Option 1,0.0462121419,09AUG1962,01JAN1960:07:42:38,0:01:17,62,2
 | 
			
		||||
1017,17 bottles of beer on the wall,Option 1,0.7501918947,14MAY1962,01JAN1960:04:40:20,0:00:15,53,65
 | 
			
		||||
1018,18 bottles of beer on the wall,Option 1,0.7300173294,03AUG1962,01JAN1960:03:20:41,0:00:41,21,60
 | 
			
		||||
1019,19 bottles of beer on the wall,Option 1,0.6960950437,01JUN1960,01JAN1960:01:58:52,0:01:08,38,5
 | 
			
		||||
1020,20 bottles of beer on the wall,Option 1,0.6190566065,30MAY1961,01JAN1960:09:04:20,0:01:07,92,23
 | 
			
		||||
1021,21 bottles of beer on the wall,Option 1,0.5173368238,07JAN1961,01JAN1960:07:52:34,0:00:52,57,21
 | 
			
		||||
1022,22 bottles of beer on the wall,Option 1,0.4720626452,07NOV1960,01JAN1960:12:12:00,0:00:26,53,32
 | 
			
		||||
1023,23 bottles of beer on the wall,Option 1,0.2856596393,08AUG1960,01JAN1960:06:09:25,0:00:28,40,12
 | 
			
		||||
1024,24 bottles of beer on the wall,Option 1,0.5160869418,02JUN1960,01JAN1960:06:36:06,0:01:10,41,81
 | 
			
		||||
1025,25 bottles of beer on the wall,Option 1,0.1683158517,05JAN1961,01JAN1960:08:14:35,0:00:06,18,53
 | 
			
		||||
1026,26 bottles of beer on the wall,Option 1,0.8951142248,28NOV1961,01JAN1960:03:31:17,0:00:58,79,54
 | 
			
		||||
1027,27 bottles of beer on the wall,Option 1,0.7037817481,01SEP1961,01JAN1960:05:48:34,0:00:29,50,15
 | 
			
		||||
1028,28 bottles of beer on the wall,Option 1,0.6193826714,31MAR1962,01JAN1960:02:49:39,0:00:24,78,87
 | 
			
		||||
1029,29 bottles of beer on the wall,Option 1,0.9339028457,06DEC1961,01JAN1960:02:57:57,0:00:24,73,64
 | 
			
		||||
1030,30 bottles of beer on the wall,Option 1,0.5647351339,10AUG1960,01JAN1960:11:02:59,0:00:55,39,28
 | 
			
		||||
1031,31 bottles of beer on the wall,Option 1,0.1218988607,19JUN1961,01JAN1960:04:19:32,0:00:58,51,32
 | 
			
		||||
1032,32 bottles of beer on the wall,Option 1,0.3459929113,14MAY1962,01JAN1960:05:42:48,0:00:54,96,46
 | 
			
		||||
1033,33 bottles of beer on the wall,Option 1,0.092664999,31AUG1962,01JAN1960:00:08:34,0:00:51,69,90
 | 
			
		||||
1034,34 bottles of beer on the wall,Option 1,0.9793458097,08FEB1960,01JAN1960:01:55:23,0:00:42,45,28
 | 
			
		||||
1035,35 bottles of beer on the wall,Option 1,0.8964386624,18DEC1961,01JAN1960:04:42:45,0:00:07,49,97
 | 
			
		||||
1036,36 bottles of beer on the wall,Option 1,0.0961652911,13NOV1960,01JAN1960:03:44:53,0:01:25,62,59
 | 
			
		||||
1037,37 bottles of beer on the wall,Option 1,0.3475089201,16JAN1962,01JAN1960:01:35:19,0:00:15,23,50
 | 
			
		||||
1038,38 bottles of beer on the wall,Option 1,0.3096271312,21MAY1960,01JAN1960:09:51:33,0:00:15,2,71
 | 
			
		||||
1039,39 bottles of beer on the wall,Option 1,0.9445223114,28AUG1962,01JAN1960:07:09:31,0:00:12,30,31
 | 
			
		||||
1040,40 bottles of beer on the wall,Option 1,0.5626084667,06NOV1960,01JAN1960:01:42:16,0:01:14,18,97
 | 
			
		||||
1041,41 bottles of beer on the wall,Option 1,0.9432962513,01JUN1962,01JAN1960:03:30:04,0:00:11,20,34
 | 
			
		||||
1042,42 bottles of beer on the wall,Option 1,0.5802429382,08JUL1961,01JAN1960:08:12:43,0:01:26,18,5
 | 
			
		||||
1043,43 bottles of beer on the wall,Option 1,0.1970176255,27MAR1961,01JAN1960:00:19:45,0:01:29,13,76
 | 
			
		||||
1044,44 bottles of beer on the wall,Option 1,0.4980671608,05JAN1961,01JAN1960:13:36:08,0:00:56,4,36
 | 
			
		||||
1045,45 bottles of beer on the wall,Option 1,0.2486515531,05MAY1962,01JAN1960:08:47:09,0:00:42,2,23
 | 
			
		||||
1046,46 bottles of beer on the wall,Option 1,0.4097825794,20JUN1960,01JAN1960:03:33:26,0:00:31,98,71
 | 
			
		||||
1047,47 bottles of beer on the wall,Option 1,0.138754441,28JAN1960,01JAN1960:00:57:41,0:00:18,80,32
 | 
			
		||||
1048,48 bottles of beer on the wall,Option 1,0.0249874415,03MAR1960,01JAN1960:11:33:53,0:00:04,96,76
 | 
			
		||||
1049,49 bottles of beer on the wall,Option 1,0.8395310011,06NOV1961,01JAN1960:09:54:04,0:00:52,28,45
 | 
			
		||||
1050,50 bottles of beer on the wall,Option 1,0.0942291618,14APR1962,01JAN1960:08:09:30,0:01:36,37,86
 | 
			
		||||
1051,51 bottles of beer on the wall,Option 1,0.1670458001,13NOV1961,01JAN1960:01:05:55,0:00:25,42,83
 | 
			
		||||
1052,52 bottles of beer on the wall,Option 1,0.3122402715,04JUN1960,01JAN1960:03:47:47,0:01:01,18,78
 | 
			
		||||
1053,53 bottles of beer on the wall,Option 1,0.3854694261,14JUN1960,01JAN1960:02:43:08,0:00:06,22,67
 | 
			
		||||
1054,54 bottles of beer on the wall,Option 1,0.1950434345,14NOV1961,01JAN1960:02:46:34,0:00:55,42,0
 | 
			
		||||
1055,55 bottles of beer on the wall,Option 1,0.4948673586,29MAR1962,01JAN1960:00:48:06,0:01:04,28,4
 | 
			
		||||
1056,56 bottles of beer on the wall,Option 1,0.6464513832,06SEP1962,01JAN1960:10:08:36,0:01:02,43,82
 | 
			
		||||
1057,57 bottles of beer on the wall,Option 1,0.0724864798,20JUN1961,01JAN1960:12:22:51,0:01:27,82,53
 | 
			
		||||
1058,58 bottles of beer on the wall,Option 1,0.8114467793,20MAR1962,01JAN1960:06:11:33,0:01:29,40,89
 | 
			
		||||
1059,59 bottles of beer on the wall,Option 1,0.6348024321,28JUN1962,01JAN1960:05:21:21,0:01:37,55,41
 | 
			
		||||
1060,60 bottles of beer on the wall,Option 1,0.8019492933,08APR1961,01JAN1960:12:37:00,0:01:29,49,88
 | 
			
		||||
1061,61 bottles of beer on the wall,Option 1,0.4695742002,29JAN1962,01JAN1960:08:54:24,0:00:15,40,91
 | 
			
		||||
1062,62 bottles of beer on the wall,Option 1,0.902706475,15JUN1961,01JAN1960:09:46:49,0:00:23,74,70
 | 
			
		||||
1063,63 bottles of beer on the wall,Option 1,0.4557614594,16JUL1961,01JAN1960:02:06:05,0:01:09,7,3
 | 
			
		||||
1064,64 bottles of beer on the wall,Option 1,0.6632444466,20MAY1961,01JAN1960:02:44:44,0:00:20,42,100
 | 
			
		||||
1065,65 bottles of beer on the wall,Option 1,0.3901674,31AUG1961,01JAN1960:07:56:49,0:00:32,98,3
 | 
			
		||||
1066,66 bottles of beer on the wall,Option 1,0.8453234848,30JUN1962,01JAN1960:04:51:54,0:01:02,51,22
 | 
			
		||||
1067,67 bottles of beer on the wall,Option 1,0.9370150906,26APR1960,01JAN1960:04:05:08,0:01:39,28,86
 | 
			
		||||
1068,68 bottles of beer on the wall,Option 1,0.8854161277,22MAR1962,01JAN1960:10:38:49,0:01:07,60,85
 | 
			
		||||
1069,69 bottles of beer on the wall,Option 1,0.1327841906,24MAY1960,01JAN1960:01:18:46,0:01:15,88,85
 | 
			
		||||
1070,70 bottles of beer on the wall,Option 1,0.5846563226,27JUL1962,01JAN1960:03:52:31,0:00:09,20,8
 | 
			
		||||
1071,71 bottles of beer on the wall,Option 1,0.0257193684,18FEB1961,01JAN1960:03:25:01,0:00:29,1,62
 | 
			
		||||
1072,72 bottles of beer on the wall,Option 1,0.9471486034,01JUN1962,01JAN1960:04:05:25,0:01:22,65,20
 | 
			
		||||
1073,73 bottles of beer on the wall,Option 1,0.3037446282,16MAY1962,01JAN1960:05:10:19,0:00:01,14,34
 | 
			
		||||
1074,74 bottles of beer on the wall,Option 1,0.2508690675,01NOV1961,01JAN1960:11:26:03,0:00:15,8,74
 | 
			
		||||
1075,75 bottles of beer on the wall,Option 1,0.814380363,17SEP1960,01JAN1960:09:00:38,0:00:25,95,1
 | 
			
		||||
1076,76 bottles of beer on the wall,Option 1,0.3761493621,16AUG1961,01JAN1960:01:48:17,0:00:52,5,38
 | 
			
		||||
1077,77 bottles of beer on the wall,Option 1,0.3621215761,25JUL1961,01JAN1960:11:48:47,0:01:20,86,90
 | 
			
		||||
1078,78 bottles of beer on the wall,Option 1,0.0268799584,20MAY1961,01JAN1960:12:43:34,0:01:00,70,96
 | 
			
		||||
1079,79 bottles of beer on the wall,Option 1,0.4112483945,27JUL1962,01JAN1960:01:20:24,0:01:26,66,20
 | 
			
		||||
1080,80 bottles of beer on the wall,Option 1,0.9501868011,15APR1961,01JAN1960:09:58:20,0:00:51,93,79
 | 
			
		||||
1081,81 bottles of beer on the wall,Option 1,0.9866548018,13SEP1961,01JAN1960:05:20:04,0:00:14,28,97
 | 
			
		||||
1082,82 bottles of beer on the wall,Option 1,0.9907830073,22FEB1962,01JAN1960:03:29:03,0:00:17,16,91
 | 
			
		||||
1083,83 bottles of beer on the wall,Option 1,0.8927816567,11MAR1960,01JAN1960:05:52:48,0:01:26,54,14
 | 
			
		||||
1084,84 bottles of beer on the wall,Option 1,0.12871663,02FEB1961,01JAN1960:10:34:37,0:00:44,52,90
 | 
			
		||||
1085,85 bottles of beer on the wall,Option 1,0.5490252802,02JAN1960,01JAN1960:06:11:58,0:00:27,4,98
 | 
			
		||||
1086,86 bottles of beer on the wall,Option 1,0.5432773864,11FEB1960,01JAN1960:08:40:00,0:01:15,79,19
 | 
			
		||||
1087,87 bottles of beer on the wall,Option 1,0.8223943137,01OCT1960,01JAN1960:08:11:33,0:01:19,2,86
 | 
			
		||||
1088,88 bottles of beer on the wall,Option 1,0.8496777699,09FEB1962,01JAN1960:03:10:35,0:00:15,95,6
 | 
			
		||||
1089,89 bottles of beer on the wall,Option 1,0.9308730536,27MAY1962,01JAN1960:11:57:53,0:01:18,86,90
 | 
			
		||||
1090,90 bottles of beer on the wall,Option 1,0.3072653344,23FEB1962,01JAN1960:04:52:38,0:00:25,85,17
 | 
			
		||||
1091,91 bottles of beer on the wall,Option 1,0.7687679575,12FEB1960,01JAN1960:08:47:11,0:01:20,8,7
 | 
			
		||||
1092,92 bottles of beer on the wall,Option 1,0.1873595105,29SEP1961,01JAN1960:04:29:58,0:00:29,17,78
 | 
			
		||||
1093,93 bottles of beer on the wall,Option 1,0.0495966631,03OCT1961,01JAN1960:03:18:50,0:00:39,89,56
 | 
			
		||||
1094,94 bottles of beer on the wall,Option 1,0.2607690526,19SEP1960,01JAN1960:03:22:28,0:00:29,81,16
 | 
			
		||||
1095,95 bottles of beer on the wall,Option 1,0.549640266,07JUN1962,01JAN1960:06:15:32,0:00:04,57,70
 | 
			
		||||
1096,96 bottles of beer on the wall,Option 1,0.9993291092,08MAR1961,01JAN1960:13:49:08,0:00:33,37,28
 | 
			
		||||
1097,97 bottles of beer on the wall,Option 1,0.9517237963,02SEP1960,01JAN1960:05:16:03,0:00:40,77,61
 | 
			
		||||
1098,98 bottles of beer on the wall,Option 1,0.5952155588,14FEB1962,01JAN1960:05:05:11,0:01:29,63,83
 | 
			
		||||
1099,99 bottles of beer on the wall,Option 1,0.7526210732,05MAY1961,01JAN1960:06:58:36,0:00:02,95,1
 | 
			
		||||
10100,100 bottles of beer on the wall,Option 1,0.307558153,17MAY1961,01JAN1960:06:13:01,0:01:37,68,7
 | 
			
		||||
10101,101 bottles of beer on the wall,Option 1,0.6596710829,15APR1962,01JAN1960:08:34:02,0:00:43,66,43
 | 
			
		||||
10102,102 bottles of beer on the wall,Option 1,0.0202811998,31AUG1961,01JAN1960:07:22:35,0:01:31,57,35
 | 
			
		||||
10103,103 bottles of beer on the wall,Option 1,0.6699061034,02MAY1962,01JAN1960:05:13:17,0:00:36,30,23
 | 
			
		||||
10104,104 bottles of beer on the wall,Option 1,0.330972748,04JUN1961,01JAN1960:06:47:20,0:01:05,69,82
 | 
			
		||||
10105,105 bottles of beer on the wall,Option 1,0.2274839176,25JAN1961,01JAN1960:05:34:51,0:00:56,63,68
 | 
			
		||||
10106,106 bottles of beer on the wall,Option 1,0.5612243989,27JUN1962,01JAN1960:04:32:03,0:01:15,19,73
 | 
			
		||||
10107,107 bottles of beer on the wall,Option 1,0.7398902111,03SEP1962,01JAN1960:08:34:07,0:00:17,90,6
 | 
			
		||||
10108,108 bottles of beer on the wall,Option 1,0.6124899791,08AUG1960,01JAN1960:04:59:34,0:00:25,56,12
 | 
			
		||||
10109,109 bottles of beer on the wall,Option 1,0.882404773,26JAN1961,01JAN1960:01:29:15,0:01:26,36,4
 | 
			
		||||
10110,110 bottles of beer on the wall,Option 1,0.4427004733,27FEB1961,01JAN1960:06:16:49,0:01:40,97,84
 | 
			
		||||
10111,111 bottles of beer on the wall,Option 1,0.3609524622,10JAN1962,01JAN1960:09:48:37,0:01:11,87,62
 | 
			
		||||
10112,112 bottles of beer on the wall,Option 1,0.9408929562,03AUG1960,01JAN1960:06:54:26,0:00:08,19,33
 | 
			
		||||
10113,113 bottles of beer on the wall,Option 1,0.3149107319,10AUG1962,01JAN1960:13:01:00,0:00:04,75,60
 | 
			
		||||
10114,114 bottles of beer on the wall,Option 1,0.0525069181,17APR1962,01JAN1960:13:00:52,0:00:35,9,23
 | 
			
		||||
10115,115 bottles of beer on the wall,Option 1,0.145448105,14FEB1962,01JAN1960:04:06:08,0:00:26,45,91
 | 
			
		||||
10116,116 bottles of beer on the wall,Option 1,0.2444279959,10OCT1961,01JAN1960:08:03:12,0:01:37,12,41
 | 
			
		||||
10117,117 bottles of beer on the wall,Option 1,0.4619846043,30JUL1960,01JAN1960:09:40:16,0:00:50,2,88
 | 
			
		||||
10118,118 bottles of beer on the wall,Option 1,0.0316203502,13JUL1961,01JAN1960:08:31:39,0:01:05,60,94
 | 
			
		||||
10119,119 bottles of beer on the wall,Option 1,0.4738720574,13AUG1960,01JAN1960:01:31:05,0:01:17,43,79
 | 
			
		||||
10120,120 bottles of beer on the wall,Option 1,0.8058761856,11JUN1960,01JAN1960:12:56:35,0:00:08,92,36
 | 
			
		||||
10121,121 bottles of beer on the wall,Option 1,0.2955600979,08JUL1962,01JAN1960:06:09:22,0:00:03,94,80
 | 
			
		||||
10122,122 bottles of beer on the wall,Option 1,0.0064115427,18SEP1962,01JAN1960:00:06:24,0:00:13,72,75
 | 
			
		||||
10123,123 bottles of beer on the wall,Option 1,0.5678159327,21APR1960,01JAN1960:10:54:21,0:00:16,75,67
 | 
			
		||||
10124,124 bottles of beer on the wall,Option 1,0.1431510994,10JAN1962,01JAN1960:01:57:00,0:00:12,48,31
 | 
			
		||||
10125,125 bottles of beer on the wall,Option 1,0.3805634409,26JAN1962,01JAN1960:03:03:19,0:01:29,83,52
 | 
			
		||||
10126,126 bottles of beer on the wall,Option 1,0.3833517993,26APR1960,01JAN1960:11:27:41,0:00:44,99,36
 | 
			
		||||
10127,127 bottles of beer on the wall,Option 1,0.5669089111,04MAR1961,01JAN1960:05:36:22,0:01:18,43,27
 | 
			
		||||
10128,128 bottles of beer on the wall,Option 1,0.1514211843,01NOV1960,01JAN1960:07:45:50,0:01:02,22,12
 | 
			
		||||
10129,129 bottles of beer on the wall,Option 1,0.0446588583,05JAN1961,01JAN1960:02:13:55,0:00:42,27,46
 | 
			
		||||
10130,130 bottles of beer on the wall,Option 1,0.7892141611,22APR1962,01JAN1960:04:17:54,0:01:05,75,84
 | 
			
		||||
10131,131 bottles of beer on the wall,Option 1,0.5012088001,24DEC1960,01JAN1960:13:03:23,0:01:22,87,82
 | 
			
		||||
10132,132 bottles of beer on the wall,Option 1,0.2327582944,07APR1961,01JAN1960:01:33:15,0:01:14,18,46
 | 
			
		||||
10133,133 bottles of beer on the wall,Option 1,0.2234651173,20MAR1961,01JAN1960:13:52:02,0:01:06,42,58
 | 
			
		||||
10134,134 bottles of beer on the wall,Option 1,0.4954405918,10FEB1961,01JAN1960:13:51:14,0:01:36,35,11
 | 
			
		||||
10135,135 bottles of beer on the wall,Option 1,0.7874922891,15AUG1960,01JAN1960:00:21:57,0:00:52,45,36
 | 
			
		||||
10136,136 bottles of beer on the wall,Option 1,0.3992494891,06SEP1961,01JAN1960:09:51:46,0:01:25,26,9
 | 
			
		||||
10137,137 bottles of beer on the wall,Option 1,0.3964866136,25MAY1960,01JAN1960:03:19:48,0:00:28,44,3
 | 
			
		||||
10138,138 bottles of beer on the wall,Option 1,0.9466173323,06APR1962,01JAN1960:13:19:18,0:01:27,78,51
 | 
			
		||||
10139,139 bottles of beer on the wall,Option 1,0.6525219277,09APR1960,01JAN1960:05:43:49,0:00:21,63,6
 | 
			
		||||
10140,140 bottles of beer on the wall,Option 1,0.4684071925,29MAY1961,01JAN1960:02:53:36,0:00:46,68,4
 | 
			
		||||
10141,141 bottles of beer on the wall,Option 1,0.8581724013,16MAY1960,01JAN1960:01:45:44,0:01:32,31,85
 | 
			
		||||
10142,142 bottles of beer on the wall,Option 1,0.825792401,23APR1961,01JAN1960:12:03:13,0:00:49,36,45
 | 
			
		||||
10143,143 bottles of beer on the wall,Option 1,0.3172852538,20FEB1962,01JAN1960:12:38:31,0:01:34,51,78
 | 
			
		||||
10144,144 bottles of beer on the wall,Option 1,0.670397946,27JAN1962,01JAN1960:04:59:37,0:00:39,38,99
 | 
			
		||||
10145,145 bottles of beer on the wall,Option 1,0.3304372441,04JUN1960,01JAN1960:00:39:12,0:00:29,88,76
 | 
			
		||||
10146,146 bottles of beer on the wall,Option 1,0.845151971,31JUL1962,01JAN1960:05:03:34,0:00:13,2,80
 | 
			
		||||
10147,147 bottles of beer on the wall,Option 1,0.7957223709,02FEB1961,01JAN1960:00:03:07,0:01:11,29,99
 | 
			
		||||
10148,148 bottles of beer on the wall,Option 1,0.323337108,29FEB1960,01JAN1960:01:58:05,0:01:17,23,65
 | 
			
		||||
10149,149 bottles of beer on the wall,Option 1,0.1813316611,29JUN1960,01JAN1960:02:18:08,0:00:40,45,52
 | 
			
		||||
10150,150 bottles of beer on the wall,Option 1,0.7860426655,05MAR1962,01JAN1960:01:57:15,0:00:26,31,91
 | 
			
		||||
10151,151 bottles of beer on the wall,Option 1,0.3305453571,09APR1960,01JAN1960:07:08:32,0:01:30,72,15
 | 
			
		||||
10152,152 bottles of beer on the wall,Option 1,0.9367212513,18AUG1962,01JAN1960:10:36:03,0:01:26,85,81
 | 
			
		||||
10153,153 bottles of beer on the wall,Option 1,0.3385623458,19MAR1962,01JAN1960:04:21:46,0:01:29,11,54
 | 
			
		||||
10154,154 bottles of beer on the wall,Option 1,0.9756794413,17JUN1961,01JAN1960:08:35:40,0:00:34,3,72
 | 
			
		||||
10155,155 bottles of beer on the wall,Option 1,0.6385958868,21OCT1961,01JAN1960:08:51:00,0:00:50,1,6
 | 
			
		||||
10156,156 bottles of beer on the wall,Option 1,0.3569769959,14AUG1960,01JAN1960:10:57:16,0:00:05,5,94
 | 
			
		||||
10157,157 bottles of beer on the wall,Option 1,0.8559997239,23MAR1962,01JAN1960:12:03:38,0:00:08,96,68
 | 
			
		||||
10158,158 bottles of beer on the wall,Option 1,0.2293701918,13AUG1960,01JAN1960:06:36:47,0:00:07,87,87
 | 
			
		||||
10159,159 bottles of beer on the wall,Option 1,0.0007910165,20SEP1962,01JAN1960:11:49:02,0:00:55,18,69
 | 
			
		||||
10160,160 bottles of beer on the wall,Option 1,0.5876370373,08JAN1960,01JAN1960:00:59:15,0:01:26,27,36
 | 
			
		||||
10161,161 bottles of beer on the wall,Option 1,0.2354667514,11OCT1961,01JAN1960:01:27:14,0:01:04,90,28
 | 
			
		||||
10162,162 bottles of beer on the wall,Option 1,0.0144103263,11AUG1961,01JAN1960:02:37:41,0:01:39,92,8
 | 
			
		||||
10163,163 bottles of beer on the wall,Option 1,0.7087855668,03JUL1962,01JAN1960:02:07:23,0:01:35,51,33
 | 
			
		||||
10164,164 bottles of beer on the wall,Option 1,0.7251478106,20MAY1960,01JAN1960:12:15:23,0:01:28,58,7
 | 
			
		||||
10165,165 bottles of beer on the wall,Option 1,0.9629398403,06APR1962,01JAN1960:01:05:24,0:01:39,84,6
 | 
			
		||||
10166,166 bottles of beer on the wall,Option 1,0.5155049164,14OCT1960,01JAN1960:04:02:06,0:00:30,63,96
 | 
			
		||||
10167,167 bottles of beer on the wall,Option 1,0.1016342775,11MAY1960,01JAN1960:05:55:03,0:00:01,31,44
 | 
			
		||||
10168,168 bottles of beer on the wall,Option 1,0.3690353596,12NOV1961,01JAN1960:12:49:02,0:00:03,2,79
 | 
			
		||||
10169,169 bottles of beer on the wall,Option 1,0.5573803501,02SEP1962,01JAN1960:12:59:56,0:00:31,7,95
 | 
			
		||||
10170,170 bottles of beer on the wall,Option 1,0.2008119497,10JUN1961,01JAN1960:05:59:06,0:01:29,88,4
 | 
			
		||||
10171,171 bottles of beer on the wall,Option 1,0.6939068505,25MAY1962,01JAN1960:07:20:06,0:01:08,87,8
 | 
			
		||||
10172,172 bottles of beer on the wall,Option 1,0.7013406594,14JUL1960,01JAN1960:04:04:24,0:00:11,44,96
 | 
			
		||||
10173,173 bottles of beer on the wall,Option 1,0.83506724,30APR1961,01JAN1960:12:44:40,0:00:10,74,70
 | 
			
		||||
10174,174 bottles of beer on the wall,Option 1,0.9339991943,26JAN1962,01JAN1960:04:59:32,0:01:09,13,66
 | 
			
		||||
10175,175 bottles of beer on the wall,Option 1,0.8333402787,18FEB1961,01JAN1960:07:25:44,0:00:28,47,2
 | 
			
		||||
10176,176 bottles of beer on the wall,Option 1,0.5998844433,03MAR1962,01JAN1960:03:45:33,0:00:52,2,61
 | 
			
		||||
10177,177 bottles of beer on the wall,Option 1,0.6161394634,18DEC1960,01JAN1960:05:35:25,0:00:50,70,22
 | 
			
		||||
10178,178 bottles of beer on the wall,Option 1,0.0821002392,21APR1960,01JAN1960:10:08:56,0:01:40,28,9
 | 
			
		||||
10179,179 bottles of beer on the wall,Option 1,0.6845213462,23MAY1960,01JAN1960:13:15:46,0:00:03,89,13
 | 
			
		||||
10180,180 bottles of beer on the wall,Option 1,0.3839034477,14MAY1960,01JAN1960:03:22:17,0:01:11,15,38
 | 
			
		||||
10181,181 bottles of beer on the wall,Option 1,0.7949567609,21AUG1962,01JAN1960:02:41:01,0:00:57,90,93
 | 
			
		||||
10182,182 bottles of beer on the wall,Option 1,0.5079025419,23SEP1962,01JAN1960:02:22:31,0:01:07,83,32
 | 
			
		||||
10183,183 bottles of beer on the wall,Option 1,0.3215162574,26DEC1961,01JAN1960:09:03:00,0:01:38,46,94
 | 
			
		||||
10184,184 bottles of beer on the wall,Option 1,0.3322958058,12MAY1961,01JAN1960:02:48:05,0:00:46,80,54
 | 
			
		||||
10185,185 bottles of beer on the wall,Option 1,0.6510801453,07SEP1960,01JAN1960:11:49:02,0:00:59,51,47
 | 
			
		||||
10186,186 bottles of beer on the wall,Option 1,0.060995535,15AUG1960,01JAN1960:02:21:08,0:01:40,5,61
 | 
			
		||||
10187,187 bottles of beer on the wall,Option 1,0.8541180551,14SEP1960,01JAN1960:13:29:33,0:01:23,17,14
 | 
			
		||||
10188,188 bottles of beer on the wall,Option 1,0.9427926219,23JUL1960,01JAN1960:05:19:05,0:01:13,22,97
 | 
			
		||||
10189,189 bottles of beer on the wall,Option 1,0.2325015186,01FEB1960,01JAN1960:11:50:07,0:01:22,6,53
 | 
			
		||||
10190,190 bottles of beer on the wall,Option 1,0.3687101493,21FEB1962,01JAN1960:06:44:23,0:00:13,16,30
 | 
			
		||||
10191,191 bottles of beer on the wall,Option 1,0.7647511232,09JAN1960,01JAN1960:13:06:29,0:01:35,6,97
 | 
			
		||||
10192,192 bottles of beer on the wall,Option 1,0.4105463565,17AUG1961,01JAN1960:11:04:32,0:01:14,38,33
 | 
			
		||||
10193,193 bottles of beer on the wall,Option 1,0.8785403831,12JUL1962,01JAN1960:04:11:05,0:00:29,19,82
 | 
			
		||||
10194,194 bottles of beer on the wall,Option 1,0.9304303433,11JUL1961,01JAN1960:12:36:57,0:01:02,20,35
 | 
			
		||||
10195,195 bottles of beer on the wall,Option 1,0.7302505256,01MAR1961,01JAN1960:01:38:35,0:00:42,16,35
 | 
			
		||||
10196,196 bottles of beer on the wall,Option 1,0.2536906177,04SEP1962,01JAN1960:05:18:23,0:01:18,91,50
 | 
			
		||||
10197,197 bottles of beer on the wall,Option 1,0.1181504503,08AUG1961,01JAN1960:09:27:54,0:01:26,20,13
 | 
			
		||||
10198,198 bottles of beer on the wall,Option 1,0.9275614228,17JUL1961,01JAN1960:01:52:18,0:00:06,52,73
 | 
			
		||||
10199,199 bottles of beer on the wall,Option 1,0.7495222128,04APR1961,01JAN1960:09:28:04,0:00:42,30,41
 | 
			
		||||
10200,200 bottles of beer on the wall,Option 1,0.925741082,02FEB1962,01JAN1960:12:23:10,0:00:07,79,17
 | 
			
		||||
10201,201 bottles of beer on the wall,Option 1,0.2591843359,04DEC1960,01JAN1960:12:46:41,0:00:00,53,58
 | 
			
		||||
10202,202 bottles of beer on the wall,Option 1,0.4289995704,17NOV1961,01JAN1960:02:20:52,0:00:35,41,25
 | 
			
		||||
10203,203 bottles of beer on the wall,Option 1,0.4625803807,24JAN1960,01JAN1960:08:20:44,0:01:11,84,66
 | 
			
		||||
10204,204 bottles of beer on the wall,Option 1,0.858440102,31AUG1962,01JAN1960:08:51:40,0:00:12,18,51
 | 
			
		||||
10205,205 bottles of beer on the wall,Option 1,0.8964499016,01SEP1962,01JAN1960:05:33:47,0:00:23,34,77
 | 
			
		||||
10206,206 bottles of beer on the wall,Option 1,0.5742789063,24OCT1961,01JAN1960:02:31:04,0:01:08,27,66
 | 
			
		||||
10207,207 bottles of beer on the wall,Option 1,0.4864150954,29SEP1960,01JAN1960:09:27:46,0:01:28,31,26
 | 
			
		||||
10208,208 bottles of beer on the wall,Option 1,0.4511992249,04DEC1960,01JAN1960:09:39:26,0:00:42,49,98
 | 
			
		||||
10209,209 bottles of beer on the wall,Option 1,0.4218624157,13SEP1961,01JAN1960:01:40:55,0:01:39,35,50
 | 
			
		||||
10210,210 bottles of beer on the wall,Option 1,0.1572868331,15FEB1960,01JAN1960:07:01:15,0:00:51,43,1
 | 
			
		||||
10211,211 bottles of beer on the wall,Option 1,0.713915177,23MAR1960,01JAN1960:11:08:53,0:00:15,18,61
 | 
			
		||||
10212,212 bottles of beer on the wall,Option 1,0.5677882165,19MAY1960,01JAN1960:01:27:23,0:01:02,34,89
 | 
			
		||||
10213,213 bottles of beer on the wall,Option 1,0.7552938581,12SEP1961,01JAN1960:11:47:33,0:00:38,44,46
 | 
			
		||||
10214,214 bottles of beer on the wall,Option 1,0.6071256071,28DEC1961,01JAN1960:05:28:18,0:01:23,84,66
 | 
			
		||||
10215,215 bottles of beer on the wall,Option 1,0.7717189266,12MAR1960,01JAN1960:01:21:26,0:01:00,28,22
 | 
			
		||||
10216,216 bottles of beer on the wall,Option 1,0.8985594329,24MAR1961,01JAN1960:10:48:58,0:01:31,93,2
 | 
			
		||||
10217,217 bottles of beer on the wall,Option 1,0.3156879904,13AUG1960,01JAN1960:07:10:46,0:01:18,100,54
 | 
			
		||||
10218,218 bottles of beer on the wall,Option 1,0.3408455315,08JUN1961,01JAN1960:02:26:49,0:00:05,65,82
 | 
			
		||||
10219,219 bottles of beer on the wall,Option 1,0.6263580553,08JUN1962,01JAN1960:05:59:46,0:01:03,76,88
 | 
			
		||||
10220,220 bottles of beer on the wall,Option 1,0.2878925355,19DEC1961,01JAN1960:08:23:41,0:00:00,92,1
 | 
			
		||||
10221,221 bottles of beer on the wall,Option 1,0.0901017348,19JUL1962,01JAN1960:09:50:47,0:00:43,21,84
 | 
			
		||||
10222,222 bottles of beer on the wall,Option 1,0.8967759362,14SEP1960,01JAN1960:12:25:58,0:01:22,34,50
 | 
			
		||||
10223,223 bottles of beer on the wall,Option 1,0.9878171943,03DEC1961,01JAN1960:03:43:09,0:00:17,11,84
 | 
			
		||||
10224,224 bottles of beer on the wall,Option 1,0.5275036886,13DEC1961,01JAN1960:03:12:56,0:01:36,85,49
 | 
			
		||||
10225,225 bottles of beer on the wall,Option 1,0.442012436,12JUN1960,01JAN1960:11:40:23,0:01:40,76,87
 | 
			
		||||
10226,226 bottles of beer on the wall,Option 1,0.582103689,10FEB1961,01JAN1960:01:50:49,0:00:59,53,29
 | 
			
		||||
10227,227 bottles of beer on the wall,Option 1,0.5757669842,01NOV1960,01JAN1960:13:47:33,0:00:43,55,6
 | 
			
		||||
10228,228 bottles of beer on the wall,Option 1,0.4786617507,07JAN1960,01JAN1960:13:36:24,0:01:22,91,53
 | 
			
		||||
10229,229 bottles of beer on the wall,Option 1,0.1386274957,06APR1962,01JAN1960:03:48:29,0:01:27,36,48
 | 
			
		||||
10230,230 bottles of beer on the wall,Option 1,0.4188394893,31MAY1962,01JAN1960:10:30:51,0:00:54,5,87
 | 
			
		||||
10231,231 bottles of beer on the wall,Option 1,0.9250617777,18OCT1960,01JAN1960:04:29:52,0:00:38,34,94
 | 
			
		||||
10232,232 bottles of beer on the wall,Option 1,0.3077528124,05FEB1960,01JAN1960:09:37:42,0:01:13,58,75
 | 
			
		||||
10233,233 bottles of beer on the wall,Option 1,0.7316332277,29NOV1960,01JAN1960:08:56:57,0:01:13,34,53
 | 
			
		||||
10234,234 bottles of beer on the wall,Option 1,0.5666298352,21NOV1960,01JAN1960:07:51:09,0:01:08,97,71
 | 
			
		||||
10235,235 bottles of beer on the wall,Option 1,0.5736639409,03JUL1962,01JAN1960:11:57:25,0:00:51,15,49
 | 
			
		||||
10236,236 bottles of beer on the wall,Option 1,0.6785667616,11FEB1962,01JAN1960:09:47:20,0:00:50,65,21
 | 
			
		||||
10237,237 bottles of beer on the wall,Option 1,0.3721726869,05JUL1962,01JAN1960:11:58:22,0:01:32,82,21
 | 
			
		||||
10238,238 bottles of beer on the wall,Option 1,0.0332283876,17AUG1961,01JAN1960:13:11:34,0:00:54,83,30
 | 
			
		||||
10239,239 bottles of beer on the wall,Option 1,0.9734656848,02JAN1961,01JAN1960:00:36:43,0:00:19,31,54
 | 
			
		||||
10240,240 bottles of beer on the wall,Option 1,0.3022106021,16FEB1961,01JAN1960:13:50:38,0:00:40,22,66
 | 
			
		||||
10241,241 bottles of beer on the wall,Option 1,0.7546903294,06JUL1961,01JAN1960:12:36:17,0:01:29,16,85
 | 
			
		||||
10242,242 bottles of beer on the wall,Option 1,0.2509871834,07MAR1962,01JAN1960:10:38:28,0:00:39,7,8
 | 
			
		||||
10243,243 bottles of beer on the wall,Option 1,0.9526996668,15JAN1960,01JAN1960:04:24:42,0:01:01,69,80
 | 
			
		||||
10244,244 bottles of beer on the wall,Option 1,0.1816610122,06FEB1962,01JAN1960:08:46:51,0:00:54,89,91
 | 
			
		||||
10245,245 bottles of beer on the wall,Option 1,0.3928658876,21JUL1962,01JAN1960:12:59:42,0:00:38,24,27
 | 
			
		||||
10246,246 bottles of beer on the wall,Option 1,0.3774878524,18FEB1961,01JAN1960:07:40:49,0:01:31,88,93
 | 
			
		||||
10247,247 bottles of beer on the wall,Option 1,0.6063659362,01NOV1960,01JAN1960:01:19:07,0:00:05,82,73
 | 
			
		||||
10248,248 bottles of beer on the wall,Option 1,0.119603098,14JUN1960,01JAN1960:04:29:22,0:00:58,87,47
 | 
			
		||||
10249,249 bottles of beer on the wall,Option 1,0.4833748445,03JUL1960,01JAN1960:01:53:54,0:00:37,34,33
 | 
			
		||||
10250,250 bottles of beer on the wall,Option 1,0.2244539946,10AUG1961,01JAN1960:06:19:01,0:01:15,87,97
 | 
			
		||||
10251,251 bottles of beer on the wall,Option 1,0.9368193191,11JUN1962,01JAN1960:06:37:14,0:00:46,94,39
 | 
			
		||||
10252,252 bottles of beer on the wall,Option 1,0.1791427751,10NOV1961,01JAN1960:00:49:22,0:00:47,96,21
 | 
			
		||||
10253,253 bottles of beer on the wall,Option 1,0.5836302874,06JUN1961,01JAN1960:08:39:34,0:01:01,78,49
 | 
			
		||||
10254,254 bottles of beer on the wall,Option 1,0.1289398275,28DEC1960,01JAN1960:12:25:05,0:00:43,67,99
 | 
			
		||||
10255,255 bottles of beer on the wall,Option 1,0.7833669785,05SEP1962,01JAN1960:02:47:35,0:00:20,25,2
 | 
			
		||||
10256,256 bottles of beer on the wall,Option 1,0.4945342483,29JAN1960,01JAN1960:00:54:13,0:01:13,72,56
 | 
			
		||||
10257,257 bottles of beer on the wall,Option 1,0.0635836129,05JAN1961,01JAN1960:08:10:04,0:00:52,11,10
 | 
			
		||||
10258,258 bottles of beer on the wall,Option 1,0.8188241654,09FEB1962,01JAN1960:06:33:00,0:01:21,41,96
 | 
			
		||||
10259,259 bottles of beer on the wall,Option 1,0.3398916076,11FEB1960,01JAN1960:07:12:29,0:00:56,18,76
 | 
			
		||||
10260,260 bottles of beer on the wall,Option 1,0.0814064155,21MAY1961,01JAN1960:11:03:51,0:01:18,78,29
 | 
			
		||||
10261,261 bottles of beer on the wall,Option 1,0.6653245542,20JAN1962,01JAN1960:08:03:31,0:00:18,39,95
 | 
			
		||||
10262,262 bottles of beer on the wall,Option 1,0.4036777021,04AUG1962,01JAN1960:12:32:27,0:00:08,57,63
 | 
			
		||||
10263,263 bottles of beer on the wall,Option 1,0.8931138603,07JAN1962,01JAN1960:09:04:24,0:00:32,6,27
 | 
			
		||||
10264,264 bottles of beer on the wall,Option 1,0.528584433,06APR1962,01JAN1960:09:43:19,0:01:00,24,41
 | 
			
		||||
10265,265 bottles of beer on the wall,Option 1,0.8267822945,29JUL1960,01JAN1960:00:48:11,0:00:01,81,78
 | 
			
		||||
10266,266 bottles of beer on the wall,Option 1,0.7218411401,17FEB1960,01JAN1960:07:30:38,0:00:08,35,81
 | 
			
		||||
10267,267 bottles of beer on the wall,Option 1,0.1475262773,11NOV1960,01JAN1960:13:44:20,0:00:57,36,68
 | 
			
		||||
10268,268 bottles of beer on the wall,Option 1,0.9412727286,30DEC1960,01JAN1960:02:46:30,0:01:19,5,92
 | 
			
		||||
10269,269 bottles of beer on the wall,Option 1,0.3038877548,27NOV1960,01JAN1960:10:50:10,0:01:21,43,95
 | 
			
		||||
10270,270 bottles of beer on the wall,Option 1,0.2756435532,15APR1962,01JAN1960:09:05:28,0:01:34,11,14
 | 
			
		||||
10271,271 bottles of beer on the wall,Option 1,0.7056001121,31AUG1960,01JAN1960:08:48:52,0:00:02,9,51
 | 
			
		||||
10272,272 bottles of beer on the wall,Option 1,0.5273708508,21SEP1962,01JAN1960:12:58:13,0:00:28,97,69
 | 
			
		||||
10273,273 bottles of beer on the wall,Option 1,0.6002807215,03MAY1960,01JAN1960:10:14:48,0:00:40,52,32
 | 
			
		||||
10274,274 bottles of beer on the wall,Option 1,0.6100557971,20JUN1960,01JAN1960:08:11:55,0:00:27,90,14
 | 
			
		||||
10275,275 bottles of beer on the wall,Option 1,0.4197408638,07JUN1961,01JAN1960:12:07:18,0:00:26,64,100
 | 
			
		||||
10276,276 bottles of beer on the wall,Option 1,0.4903712498,19JAN1960,01JAN1960:01:06:26,0:00:03,35,24
 | 
			
		||||
10277,277 bottles of beer on the wall,Option 1,0.6658435406,04NOV1960,01JAN1960:00:04:17,0:00:37,7,84
 | 
			
		||||
10278,278 bottles of beer on the wall,Option 1,0.5491365942,14JAN1961,01JAN1960:04:12:49,0:00:27,99,47
 | 
			
		||||
10279,279 bottles of beer on the wall,Option 1,0.4473488622,13MAY1961,01JAN1960:12:06:34,0:01:16,19,20
 | 
			
		||||
10280,280 bottles of beer on the wall,Option 1,0.4511988663,06JUL1962,01JAN1960:10:05:51,0:00:56,76,34
 | 
			
		||||
10281,281 bottles of beer on the wall,Option 1,0.0783031066,11JUN1961,01JAN1960:09:58:43,0:01:05,9,63
 | 
			
		||||
10282,282 bottles of beer on the wall,Option 1,0.776985302,20JUL1962,01JAN1960:10:44:29,0:01:00,59,10
 | 
			
		||||
10283,283 bottles of beer on the wall,Option 1,0.468099362,31AUG1962,01JAN1960:05:26:33,0:00:20,35,52
 | 
			
		||||
10284,284 bottles of beer on the wall,Option 1,0.4040679696,20FEB1962,01JAN1960:06:27:25,0:00:04,76,30
 | 
			
		||||
10285,285 bottles of beer on the wall,Option 1,0.4549995947,20FEB1962,01JAN1960:10:36:57,0:00:34,2,43
 | 
			
		||||
10286,286 bottles of beer on the wall,Option 1,0.7455339361,16SEP1961,01JAN1960:08:39:35,0:01:00,42,44
 | 
			
		||||
10287,287 bottles of beer on the wall,Option 1,0.0209561712,04JAN1960,01JAN1960:05:52:58,0:00:24,32,7
 | 
			
		||||
10288,288 bottles of beer on the wall,Option 1,0.4955981842,04JAN1962,01JAN1960:02:56:03,0:00:30,85,31
 | 
			
		||||
10289,289 bottles of beer on the wall,Option 1,0.4131368219,10FEB1960,01JAN1960:11:57:31,0:00:16,37,88
 | 
			
		||||
10290,290 bottles of beer on the wall,Option 1,0.3282186721,17OCT1960,01JAN1960:10:54:04,0:00:56,72,28
 | 
			
		||||
10291,291 bottles of beer on the wall,Option 1,0.2116929005,18JAN1962,01JAN1960:06:56:27,0:00:11,87,82
 | 
			
		||||
10292,292 bottles of beer on the wall,Option 1,0.8483731937,12FEB1962,01JAN1960:05:05:41,0:01:36,12,83
 | 
			
		||||
10293,293 bottles of beer on the wall,Option 1,0.1560111345,13NOV1960,01JAN1960:10:04:22,0:00:03,94,4
 | 
			
		||||
10294,294 bottles of beer on the wall,Option 1,0.7046207808,12APR1962,01JAN1960:13:50:47,0:00:32,31,97
 | 
			
		||||
10295,295 bottles of beer on the wall,Option 1,0.2716620403,04AUG1961,01JAN1960:01:52:29,0:00:57,99,44
 | 
			
		||||
10296,296 bottles of beer on the wall,Option 1,0.5543203496,12SEP1960,01JAN1960:13:43:54,0:00:44,49,1
 | 
			
		||||
10297,297 bottles of beer on the wall,Option 1,0.983109036,31JUL1962,01JAN1960:01:07:33,0:00:36,4,10
 | 
			
		||||
10298,298 bottles of beer on the wall,Option 1,0.8123072115,14SEP1962,01JAN1960:06:16:12,0:01:25,88,96
 | 
			
		||||
10299,299 bottles of beer on the wall,Option 1,0.4276896559,05OCT1960,01JAN1960:02:55:07,0:00:58,83,76
 | 
			
		||||
10300,300 bottles of beer on the wall,Option 1,0.8921809042,19JAN1962,01JAN1960:02:05:38,0:00:12,80,13
 | 
			
		||||
10301,301 bottles of beer on the wall,Option 1,0.6041374279,10DEC1961,01JAN1960:01:06:29,0:01:27,62,9
 | 
			
		||||
10302,302 bottles of beer on the wall,Option 1,0.0460550185,31MAY1962,01JAN1960:03:03:56,0:00:05,33,88
 | 
			
		||||
10303,303 bottles of beer on the wall,Option 1,0.1868385622,12APR1962,01JAN1960:12:42:44,0:01:05,65,18
 | 
			
		||||
10304,304 bottles of beer on the wall,Option 1,0.3386632657,28SEP1961,01JAN1960:11:24:06,0:00:42,2,93
 | 
			
		||||
10305,305 bottles of beer on the wall,Option 1,0.6400271019,01JUN1960,01JAN1960:13:33:07,0:01:30,60,72
 | 
			
		||||
10306,306 bottles of beer on the wall,Option 1,0.9534907304,18NOV1961,01JAN1960:02:02:51,0:00:54,7,57
 | 
			
		||||
10307,307 bottles of beer on the wall,Option 1,0.6663103745,06SEP1961,01JAN1960:05:36:49,0:00:43,88,2
 | 
			
		||||
10308,308 bottles of beer on the wall,Option 1,0.5392553073,13FEB1962,01JAN1960:11:28:18,0:01:08,16,8
 | 
			
		||||
10309,309 bottles of beer on the wall,Option 1,0.0747909025,17OCT1961,01JAN1960:08:36:12,0:00:41,49,42
 | 
			
		||||
10310,310 bottles of beer on the wall,Option 1,0.3249381847,30SEP1960,01JAN1960:08:12:54,0:00:09,96,89
 | 
			
		||||
10311,311 bottles of beer on the wall,Option 1,0.9231011951,19MAY1962,01JAN1960:05:10:33,0:00:50,30,9
 | 
			
		||||
10312,312 bottles of beer on the wall,Option 1,0.4658221637,21MAY1961,01JAN1960:12:55:25,0:01:39,16,20
 | 
			
		||||
10313,313 bottles of beer on the wall,Option 1,0.7215524673,21FEB1960,01JAN1960:02:00:07,0:01:40,95,94
 | 
			
		||||
10314,314 bottles of beer on the wall,Option 1,0.7328679942,28OCT1961,01JAN1960:09:07:00,0:00:25,42,71
 | 
			
		||||
10315,315 bottles of beer on the wall,Option 1,0.1276036776,12JUN1960,01JAN1960:01:54:08,0:00:56,57,42
 | 
			
		||||
10316,316 bottles of beer on the wall,Option 1,0.1270824723,15SEP1960,01JAN1960:03:19:25,0:00:21,85,9
 | 
			
		||||
10317,317 bottles of beer on the wall,Option 1,0.3750520117,13JUN1961,01JAN1960:04:33:09,0:01:15,24,20
 | 
			
		||||
10318,318 bottles of beer on the wall,Option 1,0.5777822102,10DEC1960,01JAN1960:13:32:14,0:00:09,98,28
 | 
			
		||||
10319,319 bottles of beer on the wall,Option 1,0.140476402,27AUG1962,01JAN1960:08:52:46,0:01:08,64,83
 | 
			
		||||
10320,320 bottles of beer on the wall,Option 1,0.2589205551,31MAY1961,01JAN1960:08:33:06,0:00:53,28,98
 | 
			
		||||
10321,321 bottles of beer on the wall,Option 1,0.7350722825,16SEP1962,01JAN1960:05:47:44,0:01:17,79,95
 | 
			
		||||
10322,322 bottles of beer on the wall,Option 1,0.1476364542,15JAN1960,01JAN1960:12:21:20,0:00:20,86,62
 | 
			
		||||
10323,323 bottles of beer on the wall,Option 1,0.8700561099,15MAY1962,01JAN1960:00:47:05,0:00:20,90,15
 | 
			
		||||
10324,324 bottles of beer on the wall,Option 1,0.6408788802,12SEP1962,01JAN1960:11:50:31,0:00:53,41,72
 | 
			
		||||
10325,325 bottles of beer on the wall,Option 1,0.6961101623,27NOV1960,01JAN1960:00:10:49,0:01:17,28,72
 | 
			
		||||
10326,326 bottles of beer on the wall,Option 1,0.1467710059,24FEB1961,01JAN1960:01:13:38,0:00:33,14,5
 | 
			
		||||
10327,327 bottles of beer on the wall,Option 1,0.0215573572,09JUN1961,01JAN1960:11:47:17,0:00:21,57,10
 | 
			
		||||
10328,328 bottles of beer on the wall,Option 1,0.4173900054,25JUL1962,01JAN1960:12:28:20,0:00:23,73,90
 | 
			
		||||
10329,329 bottles of beer on the wall,Option 1,0.6395625713,02NOV1961,01JAN1960:08:49:34,0:00:37,77,79
 | 
			
		||||
10330,330 bottles of beer on the wall,Option 1,0.0091438908,18MAY1962,01JAN1960:05:10:05,0:00:41,15,31
 | 
			
		||||
10331,331 bottles of beer on the wall,Option 1,0.1024675197,11DEC1960,01JAN1960:13:12:57,0:00:23,50,13
 | 
			
		||||
10332,332 bottles of beer on the wall,Option 1,0.057470562,11MAY1961,01JAN1960:03:43:04,0:00:17,48,14
 | 
			
		||||
10333,333 bottles of beer on the wall,Option 1,0.8478633872,21JUL1961,01JAN1960:03:45:42,0:01:31,22,40
 | 
			
		||||
10334,334 bottles of beer on the wall,Option 1,0.3442252541,24JUN1960,01JAN1960:01:19:31,0:00:48,82,25
 | 
			
		||||
10335,335 bottles of beer on the wall,Option 1,0.7338460184,06JUN1962,01JAN1960:03:32:34,0:01:04,6,31
 | 
			
		||||
10336,336 bottles of beer on the wall,Option 1,0.6217917342,09MAR1961,01JAN1960:06:37:39,0:00:50,70,84
 | 
			
		||||
10337,337 bottles of beer on the wall,Option 1,0.6684890807,10OCT1961,01JAN1960:05:34:24,0:01:20,66,18
 | 
			
		||||
10338,338 bottles of beer on the wall,Option 1,0.3695247562,05SEP1962,01JAN1960:00:25:21,0:01:18,48,37
 | 
			
		||||
10339,339 bottles of beer on the wall,Option 1,0.9429265987,06DEC1960,01JAN1960:07:11:17,0:00:38,59,1
 | 
			
		||||
10340,340 bottles of beer on the wall,Option 1,0.9266307265,17JUL1960,01JAN1960:06:33:59,0:00:21,12,13
 | 
			
		||||
10341,341 bottles of beer on the wall,Option 1,0.7280535543,23FEB1961,01JAN1960:05:01:10,0:00:34,73,25
 | 
			
		||||
10342,342 bottles of beer on the wall,Option 1,0.488654495,15AUG1962,01JAN1960:01:24:33,0:00:56,59,25
 | 
			
		||||
10343,343 bottles of beer on the wall,Option 1,0.9526806548,28DEC1960,01JAN1960:07:26:17,0:00:58,97,61
 | 
			
		||||
10344,344 bottles of beer on the wall,Option 1,0.526025336,14JAN1960,01JAN1960:10:02:08,0:00:55,11,77
 | 
			
		||||
10345,345 bottles of beer on the wall,Option 1,0.807215352,03JUL1961,01JAN1960:12:49:47,0:00:01,40,7
 | 
			
		||||
10346,346 bottles of beer on the wall,Option 1,0.9305162979,28FEB1960,01JAN1960:09:46:40,0:00:59,56,28
 | 
			
		||||
10347,347 bottles of beer on the wall,Option 1,0.7591318552,18FEB1962,01JAN1960:13:25:32,0:01:10,41,9
 | 
			
		||||
10348,348 bottles of beer on the wall,Option 1,0.4177664911,11SEP1961,01JAN1960:09:55:17,0:01:39,76,82
 | 
			
		||||
10349,349 bottles of beer on the wall,Option 1,0.4690050443,05DEC1961,01JAN1960:11:05:15,0:01:09,63,40
 | 
			
		||||
10350,350 bottles of beer on the wall,Option 1,0.7541399979,31AUG1961,01JAN1960:12:30:45,0:00:33,57,12
 | 
			
		||||
10351,351 bottles of beer on the wall,Option 1,0.1392844325,17MAR1962,01JAN1960:08:20:38,0:00:41,85,45
 | 
			
		||||
10352,352 bottles of beer on the wall,Option 1,0.1020530235,23DEC1961,01JAN1960:09:46:20,0:00:01,55,56
 | 
			
		||||
10353,353 bottles of beer on the wall,Option 1,0.0257998794,04DEC1961,01JAN1960:09:47:10,0:00:31,100,2
 | 
			
		||||
10354,354 bottles of beer on the wall,Option 1,0.1238113316,20MAR1962,01JAN1960:09:15:30,0:00:01,74,11
 | 
			
		||||
10355,355 bottles of beer on the wall,Option 1,0.4214245292,24NOV1960,01JAN1960:04:24:09,0:01:01,79,83
 | 
			
		||||
10356,356 bottles of beer on the wall,Option 1,0.3243381057,12FEB1961,01JAN1960:00:55:59,0:00:50,30,52
 | 
			
		||||
10357,357 bottles of beer on the wall,Option 1,0.9735697345,24NOV1960,01JAN1960:07:10:56,0:01:33,64,2
 | 
			
		||||
10358,358 bottles of beer on the wall,Option 1,0.4796259461,28JAN1961,01JAN1960:11:51:29,0:01:03,19,29
 | 
			
		||||
10359,359 bottles of beer on the wall,Option 1,0.003359966,01SEP1960,01JAN1960:13:24:25,0:00:09,66,60
 | 
			
		||||
10360,360 bottles of beer on the wall,Option 1,0.5773700334,21JAN1960,01JAN1960:10:15:32,0:00:40,9,21
 | 
			
		||||
10361,361 bottles of beer on the wall,Option 1,0.0792848342,25JAN1962,01JAN1960:06:00:35,0:01:25,73,73
 | 
			
		||||
10362,362 bottles of beer on the wall,Option 1,0.9339359365,30MAY1961,01JAN1960:09:08:13,0:00:56,12,56
 | 
			
		||||
10363,363 bottles of beer on the wall,Option 1,0.3517632132,12FEB1961,01JAN1960:12:07:19,0:00:01,74,69
 | 
			
		||||
10364,364 bottles of beer on the wall,Option 1,0.4088954895,17MAR1961,01JAN1960:08:04:51,0:01:01,70,66
 | 
			
		||||
10365,365 bottles of beer on the wall,Option 1,0.1254259623,30DEC1961,01JAN1960:06:47:12,0:00:01,79,43
 | 
			
		||||
10366,366 bottles of beer on the wall,Option 1,0.9190132958,28MAY1961,01JAN1960:06:35:42,0:00:07,19,31
 | 
			
		||||
10367,367 bottles of beer on the wall,Option 1,0.3033860015,17MAY1962,01JAN1960:05:32:03,0:00:32,57,73
 | 
			
		||||
10368,368 bottles of beer on the wall,Option 1,0.1463442846,02SEP1962,01JAN1960:01:24:29,0:00:53,19,64
 | 
			
		||||
10369,369 bottles of beer on the wall,Option 1,0.5516236343,18JUN1962,01JAN1960:10:52:23,0:00:11,51,40
 | 
			
		||||
10370,370 bottles of beer on the wall,Option 1,0.5205378246,19JAN1960,01JAN1960:06:54:14,0:00:58,2,72
 | 
			
		||||
10371,371 bottles of beer on the wall,Option 1,0.9941610768,28MAR1962,01JAN1960:04:55:58,0:00:22,46,65
 | 
			
		||||
10372,372 bottles of beer on the wall,Option 1,0.065678,07MAY1961,01JAN1960:10:17:35,0:00:10,54,100
 | 
			
		||||
10373,373 bottles of beer on the wall,Option 1,0.7222646138,17JUL1961,01JAN1960:01:47:32,0:00:51,26,96
 | 
			
		||||
10374,374 bottles of beer on the wall,Option 1,0.8772228294,23JUL1960,01JAN1960:00:30:51,0:00:40,18,45
 | 
			
		||||
10375,375 bottles of beer on the wall,Option 1,0.3651081847,11DEC1961,01JAN1960:00:46:15,0:00:15,14,90
 | 
			
		||||
10376,376 bottles of beer on the wall,Option 1,0.3800431529,15AUG1960,01JAN1960:05:30:55,0:00:19,13,74
 | 
			
		||||
10377,377 bottles of beer on the wall,Option 1,0.1077003503,26FEB1960,01JAN1960:04:48:40,0:00:08,51,53
 | 
			
		||||
10378,378 bottles of beer on the wall,Option 1,0.7945664035,06MAR1961,01JAN1960:05:36:47,0:00:45,65,39
 | 
			
		||||
10379,379 bottles of beer on the wall,Option 1,0.8754883054,06JUN1960,01JAN1960:06:09:33,0:00:04,3,3
 | 
			
		||||
10380,380 bottles of beer on the wall,Option 1,0.9975561108,10AUG1960,01JAN1960:10:34:33,0:00:28,92,29
 | 
			
		||||
10381,381 bottles of beer on the wall,Option 1,0.9817031599,07JUL1960,01JAN1960:01:40:00,0:00:39,63,45
 | 
			
		||||
10382,382 bottles of beer on the wall,Option 1,0.4427802341,07JAN1962,01JAN1960:01:21:32,0:01:31,7,54
 | 
			
		||||
10383,383 bottles of beer on the wall,Option 1,0.03180474,17JUL1962,01JAN1960:07:15:54,0:01:08,72,60
 | 
			
		||||
10384,384 bottles of beer on the wall,Option 1,0.1031627707,10MAY1962,01JAN1960:02:52:58,0:01:31,6,64
 | 
			
		||||
10385,385 bottles of beer on the wall,Option 1,0.911744344,01MAY1960,01JAN1960:03:51:16,0:01:04,1,54
 | 
			
		||||
10386,386 bottles of beer on the wall,Option 1,0.4912374353,13FEB1961,01JAN1960:07:22:49,0:01:21,70,71
 | 
			
		||||
10387,387 bottles of beer on the wall,Option 1,0.8803869509,04JUL1960,01JAN1960:12:14:05,0:00:18,78,88
 | 
			
		||||
10388,388 bottles of beer on the wall,Option 1,0.0596609544,17DEC1960,01JAN1960:08:29:20,0:00:53,13,84
 | 
			
		||||
10389,389 bottles of beer on the wall,Option 1,0.6625022132,12JAN1961,01JAN1960:11:15:39,0:01:05,10,31
 | 
			
		||||
10390,390 bottles of beer on the wall,Option 1,0.2669677358,05OCT1961,01JAN1960:00:48:03,0:01:33,29,31
 | 
			
		||||
10391,391 bottles of beer on the wall,Option 1,0.8095563454,04DEC1961,01JAN1960:07:54:13,0:01:15,95,2
 | 
			
		||||
10392,392 bottles of beer on the wall,Option 1,0.6406704796,27JAN1961,01JAN1960:03:35:55,0:00:36,92,47
 | 
			
		||||
10393,393 bottles of beer on the wall,Option 1,0.2188341847,02MAR1960,01JAN1960:04:05:21,0:00:11,28,34
 | 
			
		||||
10394,394 bottles of beer on the wall,Option 1,0.7052043424,09JUN1962,01JAN1960:07:00:21,0:01:14,84,19
 | 
			
		||||
10395,395 bottles of beer on the wall,Option 1,0.9843204464,18APR1960,01JAN1960:04:54:31,0:01:29,46,46
 | 
			
		||||
10396,396 bottles of beer on the wall,Option 1,0.329249541,10SEP1961,01JAN1960:13:21:04,0:00:21,70,11
 | 
			
		||||
10397,397 bottles of beer on the wall,Option 1,0.2073628675,13JUL1962,01JAN1960:11:16:26,0:01:23,17,97
 | 
			
		||||
10398,398 bottles of beer on the wall,Option 1,0.7881676665,29JUN1962,01JAN1960:11:36:47,0:00:50,39,72
 | 
			
		||||
10399,399 bottles of beer on the wall,Option 1,0.4347905537,31AUG1962,01JAN1960:02:30:30,0:00:21,86,61
 | 
			
		||||
10400,400 bottles of beer on the wall,Option 1,0.2937454084,05APR1962,01JAN1960:02:44:06,0:00:32,27,81
 | 
			
		||||
10401,401 bottles of beer on the wall,Option 1,0.6659902565,10APR1961,01JAN1960:11:53:59,0:00:01,80,17
 | 
			
		||||
10402,402 bottles of beer on the wall,Option 1,0.7637229686,07APR1962,01JAN1960:05:45:25,0:01:28,11,17
 | 
			
		||||
10403,403 bottles of beer on the wall,Option 1,0.3325480941,19MAR1961,01JAN1960:09:57:05,0:00:27,9,99
 | 
			
		||||
10404,404 bottles of beer on the wall,Option 1,0.580015553,10AUG1960,01JAN1960:06:15:44,0:00:25,27,99
 | 
			
		||||
10405,405 bottles of beer on the wall,Option 1,0.2514696071,08APR1961,01JAN1960:10:37:45,0:00:17,29,59
 | 
			
		||||
10406,406 bottles of beer on the wall,Option 1,0.3792107284,19MAR1962,01JAN1960:04:49:30,0:00:07,44,38
 | 
			
		||||
10407,407 bottles of beer on the wall,Option 1,0.4702913125,13DEC1961,01JAN1960:09:03:31,0:00:38,95,90
 | 
			
		||||
10408,408 bottles of beer on the wall,Option 1,0.4207190659,03FEB1961,01JAN1960:00:37:47,0:00:38,37,65
 | 
			
		||||
10409,409 bottles of beer on the wall,Option 1,0.2485069117,08DEC1961,01JAN1960:07:18:28,0:01:39,91,64
 | 
			
		||||
10410,410 bottles of beer on the wall,Option 1,0.3257893502,27NOV1961,01JAN1960:11:33:02,0:00:54,41,7
 | 
			
		||||
10411,411 bottles of beer on the wall,Option 1,0.0967283533,11AUG1962,01JAN1960:07:23:15,0:00:21,90,1
 | 
			
		||||
10412,412 bottles of beer on the wall,Option 1,0.532676542,01SEP1960,01JAN1960:02:46:29,0:00:45,31,48
 | 
			
		||||
10413,413 bottles of beer on the wall,Option 1,0.2564602151,15JAN1961,01JAN1960:01:09:11,0:01:21,16,89
 | 
			
		||||
10414,414 bottles of beer on the wall,Option 1,0.0865634024,20MAR1962,01JAN1960:12:56:45,0:00:05,75,51
 | 
			
		||||
10415,415 bottles of beer on the wall,Option 1,0.2926342321,07FEB1960,01JAN1960:06:48:55,0:00:36,66,33
 | 
			
		||||
10416,416 bottles of beer on the wall,Option 1,0.0678594029,06OCT1961,01JAN1960:07:14:58,0:01:26,50,16
 | 
			
		||||
10417,417 bottles of beer on the wall,Option 1,0.6376664767,09JUN1960,01JAN1960:05:00:13,0:01:37,7,92
 | 
			
		||||
10418,418 bottles of beer on the wall,Option 1,0.4795483525,14JUN1962,01JAN1960:00:00:50,0:00:50,33,74
 | 
			
		||||
10419,419 bottles of beer on the wall,Option 1,0.3492934342,22MAR1960,01JAN1960:08:54:52,0:01:26,61,72
 | 
			
		||||
10420,420 bottles of beer on the wall,Option 1,0.5085071356,14MAR1960,01JAN1960:09:47:51,0:00:56,36,63
 | 
			
		||||
10421,421 bottles of beer on the wall,Option 1,0.414953564,25MAR1961,01JAN1960:03:35:19,0:00:30,47,30
 | 
			
		||||
10422,422 bottles of beer on the wall,Option 1,0.2431976652,25SEP1960,01JAN1960:08:47:25,0:00:17,6,42
 | 
			
		||||
10423,423 bottles of beer on the wall,Option 1,0.2370444998,04MAR1962,01JAN1960:00:16:44,0:01:36,54,100
 | 
			
		||||
10424,424 bottles of beer on the wall,Option 1,0.8687893324,12OCT1961,01JAN1960:03:16:33,0:00:56,97,25
 | 
			
		||||
10425,425 bottles of beer on the wall,Option 1,0.0470510335,10MAY1960,01JAN1960:02:44:07,0:01:05,83,59
 | 
			
		||||
10426,426 bottles of beer on the wall,Option 1,0.7114342887,12APR1960,01JAN1960:02:17:40,0:00:39,57,97
 | 
			
		||||
10427,427 bottles of beer on the wall,Option 1,0.6176668283,05JUL1962,01JAN1960:09:09:12,0:01:03,92,38
 | 
			
		||||
10428,428 bottles of beer on the wall,Option 1,0.0657907743,01JUL1962,01JAN1960:11:03:37,0:01:35,97,63
 | 
			
		||||
10429,429 bottles of beer on the wall,Option 1,0.280104912,13MAR1962,01JAN1960:10:35:08,0:01:07,42,35
 | 
			
		||||
10430,430 bottles of beer on the wall,Option 1,0.3639827088,02APR1960,01JAN1960:12:00:50,0:00:40,91,84
 | 
			
		||||
10431,431 bottles of beer on the wall,Option 1,0.2130561137,25AUG1961,01JAN1960:12:07:22,0:01:37,70,9
 | 
			
		||||
10432,432 bottles of beer on the wall,Option 1,0.9656705889,23DEC1960,01JAN1960:02:36:48,0:01:28,23,92
 | 
			
		||||
10433,433 bottles of beer on the wall,Option 1,0.5824601052,22JAN1961,01JAN1960:01:19:48,0:00:32,73,6
 | 
			
		||||
10434,434 bottles of beer on the wall,Option 1,0.5207124942,23APR1961,01JAN1960:10:45:16,0:01:08,45,85
 | 
			
		||||
10435,435 bottles of beer on the wall,Option 1,0.9280530661,01NOV1960,01JAN1960:12:46:59,0:01:36,57,4
 | 
			
		||||
10436,436 bottles of beer on the wall,Option 1,0.8982810149,29NOV1961,01JAN1960:09:23:13,0:01:15,100,68
 | 
			
		||||
10437,437 bottles of beer on the wall,Option 1,0.3268259467,04MAR1960,01JAN1960:01:22:42,0:00:04,99,61
 | 
			
		||||
10438,438 bottles of beer on the wall,Option 1,0.2038002704,12FEB1962,01JAN1960:09:03:24,0:00:40,97,78
 | 
			
		||||
10439,439 bottles of beer on the wall,Option 1,0.0798850833,11MAY1960,01JAN1960:00:32:50,0:00:37,7,49
 | 
			
		||||
10440,440 bottles of beer on the wall,Option 1,0.6697000566,01SEP1960,01JAN1960:08:04:03,0:00:30,7,68
 | 
			
		||||
10441,441 bottles of beer on the wall,Option 1,0.550699767,11JAN1960,01JAN1960:06:33:15,0:01:09,19,10
 | 
			
		||||
10442,442 bottles of beer on the wall,Option 1,0.9514032798,07FEB1962,01JAN1960:06:03:16,0:01:13,65,87
 | 
			
		||||
10443,443 bottles of beer on the wall,Option 1,0.1182718324,08FEB1960,01JAN1960:11:28:52,0:00:46,61,59
 | 
			
		||||
10444,444 bottles of beer on the wall,Option 1,0.5772118874,11FEB1962,01JAN1960:01:14:29,0:01:07,16,22
 | 
			
		||||
10445,445 bottles of beer on the wall,Option 1,0.9828714463,06DEC1960,01JAN1960:05:47:08,0:01:40,66,4
 | 
			
		||||
10446,446 bottles of beer on the wall,Option 1,0.1229167269,02JUN1960,01JAN1960:09:43:51,0:01:34,29,24
 | 
			
		||||
10447,447 bottles of beer on the wall,Option 1,0.2102257522,09MAR1961,01JAN1960:04:06:09,0:01:27,53,78
 | 
			
		||||
10448,448 bottles of beer on the wall,Option 1,0.9958884935,20FEB1962,01JAN1960:06:43:53,0:00:07,42,65
 | 
			
		||||
10449,449 bottles of beer on the wall,Option 1,0.3530408085,04MAY1961,01JAN1960:11:26:59,0:01:37,76,35
 | 
			
		||||
10450,450 bottles of beer on the wall,Option 1,0.2848354258,03SEP1960,01JAN1960:13:26:36,0:00:25,69,38
 | 
			
		||||
10451,451 bottles of beer on the wall,Option 1,0.1461860818,10JUN1960,01JAN1960:01:24:25,0:00:02,61,23
 | 
			
		||||
10452,452 bottles of beer on the wall,Option 1,0.2033735766,29AUG1960,01JAN1960:12:30:17,0:00:19,32,75
 | 
			
		||||
10453,453 bottles of beer on the wall,Option 1,0.7973149958,26JUN1961,01JAN1960:04:46:06,0:00:21,94,14
 | 
			
		||||
10454,454 bottles of beer on the wall,Option 1,0.6812086425,07NOV1961,01JAN1960:01:59:51,0:00:12,89,59
 | 
			
		||||
10455,455 bottles of beer on the wall,Option 1,0.6111464974,24FEB1962,01JAN1960:07:42:56,0:00:15,23,92
 | 
			
		||||
10456,456 bottles of beer on the wall,Option 1,0.9542484209,09APR1961,01JAN1960:09:09:17,0:01:26,87,93
 | 
			
		||||
10457,457 bottles of beer on the wall,Option 1,0.9818577077,18AUG1960,01JAN1960:12:24:47,0:01:08,18,20
 | 
			
		||||
10458,458 bottles of beer on the wall,Option 1,0.9331120881,06JUL1961,01JAN1960:10:10:02,0:01:40,78,36
 | 
			
		||||
10459,459 bottles of beer on the wall,Option 1,0.8953045587,22APR1960,01JAN1960:04:47:34,0:00:37,13,46
 | 
			
		||||
10460,460 bottles of beer on the wall,Option 1,0.1584420461,08JAN1961,01JAN1960:07:47:49,0:00:05,85,61
 | 
			
		||||
10461,461 bottles of beer on the wall,Option 1,0.9324573814,24DEC1960,01JAN1960:02:13:36,0:00:22,55,94
 | 
			
		||||
10462,462 bottles of beer on the wall,Option 1,0.0813189871,21AUG1961,01JAN1960:09:18:28,0:00:36,42,49
 | 
			
		||||
10463,463 bottles of beer on the wall,Option 1,0.8736312594,17MAY1962,01JAN1960:12:52:58,0:00:30,45,7
 | 
			
		||||
10464,464 bottles of beer on the wall,Option 1,0.1894013794,06NOV1960,01JAN1960:12:56:13,0:01:30,11,50
 | 
			
		||||
10465,465 bottles of beer on the wall,Option 1,0.9149773772,07JUN1960,01JAN1960:08:53:28,0:00:22,23,17
 | 
			
		||||
10466,466 bottles of beer on the wall,Option 1,0.6329479598,27JUN1962,01JAN1960:07:38:23,0:00:06,68,14
 | 
			
		||||
10467,467 bottles of beer on the wall,Option 1,0.1897066413,21AUG1961,01JAN1960:10:15:28,0:01:12,59,46
 | 
			
		||||
10468,468 bottles of beer on the wall,Option 1,0.8352205767,16MAR1961,01JAN1960:06:48:50,0:00:44,60,37
 | 
			
		||||
10469,469 bottles of beer on the wall,Option 1,0.3339443432,09OCT1960,01JAN1960:00:52:53,0:00:27,70,44
 | 
			
		||||
10470,470 bottles of beer on the wall,Option 1,0.1096137567,09JUL1962,01JAN1960:08:28:10,0:01:15,88,81
 | 
			
		||||
10471,471 bottles of beer on the wall,Option 1,0.1481936733,10JAN1962,01JAN1960:06:38:41,0:00:56,21,77
 | 
			
		||||
10472,472 bottles of beer on the wall,Option 1,0.5630390432,01MAR1960,01JAN1960:06:35:09,0:01:08,70,32
 | 
			
		||||
10473,473 bottles of beer on the wall,Option 1,0.0604085713,12APR1962,01JAN1960:02:41:26,0:00:46,36,17
 | 
			
		||||
10474,474 bottles of beer on the wall,Option 1,0.6011423606,25DEC1961,01JAN1960:02:57:45,0:01:11,36,18
 | 
			
		||||
10475,475 bottles of beer on the wall,Option 1,0.7698335782,31JUL1962,01JAN1960:07:46:29,0:01:34,72,17
 | 
			
		||||
10476,476 bottles of beer on the wall,Option 1,0.3823641224,17MAR1962,01JAN1960:03:34:37,0:00:09,27,86
 | 
			
		||||
10477,477 bottles of beer on the wall,Option 1,0.9225378446,12JAN1962,01JAN1960:08:41:25,0:01:06,45,20
 | 
			
		||||
10478,478 bottles of beer on the wall,Option 1,0.2979750453,23JUN1962,01JAN1960:10:06:36,0:01:28,88,7
 | 
			
		||||
10479,479 bottles of beer on the wall,Option 1,0.4988665942,15JUL1961,01JAN1960:02:33:06,0:01:33,85,1
 | 
			
		||||
10480,480 bottles of beer on the wall,Option 1,0.5243853585,29JUL1960,01JAN1960:00:26:18,0:00:31,48,77
 | 
			
		||||
10481,481 bottles of beer on the wall,Option 1,0.6049248826,12JUL1962,01JAN1960:10:01:41,0:00:16,66,45
 | 
			
		||||
10482,482 bottles of beer on the wall,Option 1,0.6312438024,01OCT1961,01JAN1960:05:18:12,0:00:04,43,27
 | 
			
		||||
10483,483 bottles of beer on the wall,Option 1,0.3366865974,17MAY1962,01JAN1960:01:44:17,0:00:01,31,25
 | 
			
		||||
10484,484 bottles of beer on the wall,Option 1,0.0813266188,08AUG1962,01JAN1960:06:18:27,0:00:08,71,63
 | 
			
		||||
10485,485 bottles of beer on the wall,Option 1,0.3601163455,23JAN1962,01JAN1960:11:26:01,0:00:37,41,88
 | 
			
		||||
10486,486 bottles of beer on the wall,Option 1,0.9033777345,13AUG1961,01JAN1960:00:16:14,0:00:56,82,33
 | 
			
		||||
10487,487 bottles of beer on the wall,Option 1,0.1716986667,23DEC1960,01JAN1960:12:06:34,0:01:13,2,32
 | 
			
		||||
10488,488 bottles of beer on the wall,Option 1,0.9734818912,11SEP1961,01JAN1960:00:34:41,0:00:28,17,7
 | 
			
		||||
10489,489 bottles of beer on the wall,Option 1,0.0618140223,17DEC1961,01JAN1960:06:26:30,0:00:38,94,40
 | 
			
		||||
10490,490 bottles of beer on the wall,Option 1,0.0204490144,22AUG1960,01JAN1960:01:50:18,0:00:19,40,56
 | 
			
		||||
10491,491 bottles of beer on the wall,Option 1,0.8719323929,23MAY1960,01JAN1960:12:36:06,0:00:08,49,44
 | 
			
		||||
10492,492 bottles of beer on the wall,Option 1,0.2656292181,28JUN1962,01JAN1960:05:32:50,0:01:35,15,39
 | 
			
		||||
10493,493 bottles of beer on the wall,Option 1,0.37794835,23JUL1962,01JAN1960:13:15:43,0:00:56,6,86
 | 
			
		||||
10494,494 bottles of beer on the wall,Option 1,0.6395865733,17JAN1961,01JAN1960:12:20:12,0:01:31,22,87
 | 
			
		||||
10495,495 bottles of beer on the wall,Option 1,0.4408595098,05JUL1960,01JAN1960:12:21:53,0:01:40,25,27
 | 
			
		||||
10496,496 bottles of beer on the wall,Option 1,0.4846261169,02DEC1961,01JAN1960:12:41:08,0:00:06,7,63
 | 
			
		||||
10497,497 bottles of beer on the wall,Option 1,0.7903671478,14MAY1962,01JAN1960:05:09:21,0:01:03,98,25
 | 
			
		||||
10498,498 bottles of beer on the wall,Option 1,0.60474184,13OCT1961,01JAN1960:03:52:59,0:01:22,98,98
 | 
			
		||||
10499,499 bottles of beer on the wall,Option 1,0.1213259218,23APR1962,01JAN1960:06:29:47,0:01:35,38,70
 | 
			
		||||
10500,500 bottles of beer on the wall,Option 1,0.7882370487,17FEB1962,01JAN1960:02:00:25,0:00:12,1,74
 | 
			
		||||
		
		
			
  | 
							
								
								
									
										
											BIN
										
									
								
								client/cypress/fixtures/excels_multi_load/multi_load_test_1.xlsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								client/cypress/fixtures/excels_multi_load/multi_load_test_1.xlsx
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								client/cypress/fixtures/excels_multi_load/multi_load_test_2.xlsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								client/cypress/fixtures/excels_multi_load/multi_load_test_2.xlsx
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								client/cypress/fixtures/excels_multi_load/multi_load_test_3.xlsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								client/cypress/fixtures/excels_multi_load/multi_load_test_3.xlsx
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							@@ -15,9 +15,6 @@ context('editor tests: ', function () {
 | 
			
		||||
 | 
			
		||||
  this.beforeEach(() => {
 | 
			
		||||
    cy.visit(hostUrl + appLocation)
 | 
			
		||||
    // cy.get('input.username').type(username)
 | 
			
		||||
    // cy.get('input.password').type(password)
 | 
			
		||||
    // cy.get('.login-group button').click()
 | 
			
		||||
 | 
			
		||||
    visitPage('home')
 | 
			
		||||
  })
 | 
			
		||||
@@ -118,10 +115,6 @@ context('editor tests: ', function () {
 | 
			
		||||
      })
 | 
			
		||||
    })
 | 
			
		||||
  })
 | 
			
		||||
 | 
			
		||||
  this.afterEach(() => {
 | 
			
		||||
    // cy.visit(`${hostUrl}/SASLogon/logout`)
 | 
			
		||||
  })
 | 
			
		||||
})
 | 
			
		||||
 | 
			
		||||
const clickOnEdit = (callback?: any) => {
 | 
			
		||||
 
 | 
			
		||||
@@ -19,9 +19,6 @@ context('excel tests: ', function () {
 | 
			
		||||
 | 
			
		||||
  this.beforeEach(() => {
 | 
			
		||||
    cy.visit(hostUrl + appLocation)
 | 
			
		||||
    // cy.get('input.username').type(username)
 | 
			
		||||
    // cy.get('input.password').type(password)
 | 
			
		||||
    // cy.get('.login-group button').click()
 | 
			
		||||
 | 
			
		||||
    visitPage('home')
 | 
			
		||||
 | 
			
		||||
@@ -339,7 +336,6 @@ context('excel tests: ', function () {
 | 
			
		||||
 | 
			
		||||
  this.afterEach(() => {
 | 
			
		||||
    colorLog(`TEST END -------------`, '#3498DB')
 | 
			
		||||
    // cy.visit(`${hostUrl}/SASLogon/logout`)
 | 
			
		||||
  })
 | 
			
		||||
})
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -15,9 +15,6 @@ context('filtering tests: ', function () {
 | 
			
		||||
 | 
			
		||||
  this.beforeEach(() => {
 | 
			
		||||
    cy.visit(hostUrl + appLocation, { timeout: longerCommandTimeout })
 | 
			
		||||
    // cy.get('input.username').type(username)
 | 
			
		||||
    // cy.get('input.password').type(password)
 | 
			
		||||
    // cy.get('.login-group button').click()
 | 
			
		||||
 | 
			
		||||
    visitPage('home')
 | 
			
		||||
  })
 | 
			
		||||
@@ -173,10 +170,6 @@ context('filtering tests: ', function () {
 | 
			
		||||
      })
 | 
			
		||||
    })
 | 
			
		||||
  })
 | 
			
		||||
 | 
			
		||||
  this.afterEach(() => {
 | 
			
		||||
    // cy.visit(`${hostUrl}/SASLogon/logout`)
 | 
			
		||||
  })
 | 
			
		||||
})
 | 
			
		||||
 | 
			
		||||
const checkInfoBarIncludes = (text: string, callback: any) => {
 | 
			
		||||
 
 | 
			
		||||
@@ -23,15 +23,11 @@ interface EditConfigTableCells {
 | 
			
		||||
 | 
			
		||||
context('licensing tests: ', function () {
 | 
			
		||||
  this.beforeAll(() => {
 | 
			
		||||
    // cy.visit(`${hostUrl}/SASLogon/logout`)
 | 
			
		||||
    cy.loginAndUpdateValidKey()
 | 
			
		||||
  })
 | 
			
		||||
 | 
			
		||||
  this.beforeEach(() => {
 | 
			
		||||
    cy.visit(hostUrl + appLocation)
 | 
			
		||||
    // cy.get('input.username').type(username)
 | 
			
		||||
    // cy.get('input.password').type(password)
 | 
			
		||||
    // cy.get('.login-group button').click()
 | 
			
		||||
 | 
			
		||||
    visitPage('home')
 | 
			
		||||
  })
 | 
			
		||||
@@ -374,10 +370,6 @@ context('licensing tests: ', function () {
 | 
			
		||||
      })
 | 
			
		||||
    })
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  this.afterEach(() => {
 | 
			
		||||
    // cy.visit(`${hostUrl}/SASLogon/logout`)
 | 
			
		||||
  })
 | 
			
		||||
})
 | 
			
		||||
 | 
			
		||||
const logout = (callback?: any) => {
 | 
			
		||||
 
 | 
			
		||||
@@ -18,10 +18,6 @@ context('liveness tests: ', function () {
 | 
			
		||||
  this.beforeEach(() => {
 | 
			
		||||
    cy.visit(hostUrl + appLocation)
 | 
			
		||||
 | 
			
		||||
    // cy.get('input.username').type(username)
 | 
			
		||||
    // cy.get('input.password').type(password)
 | 
			
		||||
    // cy.get('.login-group button').click()
 | 
			
		||||
 | 
			
		||||
    visitPage('home')
 | 
			
		||||
  })
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -17,7 +17,6 @@ context('editor tests: ', function () {
 | 
			
		||||
 | 
			
		||||
  this.beforeEach(() => {
 | 
			
		||||
    cy.visit(hostUrl + appLocation)
 | 
			
		||||
 | 
			
		||||
    cy.wait(2000)
 | 
			
		||||
 | 
			
		||||
    cy.get('body').then(($body) => {
 | 
			
		||||
@@ -386,10 +385,6 @@ context('editor tests: ', function () {
 | 
			
		||||
      }
 | 
			
		||||
    )
 | 
			
		||||
  })
 | 
			
		||||
 | 
			
		||||
  this.afterEach(() => {
 | 
			
		||||
    // cy.visit(`${hostUrl}/SASLogon/logout`)
 | 
			
		||||
  })
 | 
			
		||||
})
 | 
			
		||||
 | 
			
		||||
const checkColumns = (columns: string[], callback: () => void) => {
 | 
			
		||||
 
 | 
			
		||||
@@ -6,5 +6,8 @@
 | 
			
		||||
      "lib": ["es2019", "dom"],
 | 
			
		||||
      "types": ["cypress", "cypress-real-events"]
 | 
			
		||||
    },
 | 
			
		||||
    "include": ["**/*.ts"]
 | 
			
		||||
    "include": [
 | 
			
		||||
      "**/*.ts",
 | 
			
		||||
      "../cypress.config.ts"
 | 
			
		||||
    ]
 | 
			
		||||
  }
 | 
			
		||||
@@ -42,4 +42,4 @@ module.exports = function (config) {
 | 
			
		||||
      }
 | 
			
		||||
    },
 | 
			
		||||
  });
 | 
			
		||||
};
 | 
			
		||||
};
 | 
			
		||||
							
								
								
									
										
											BIN
										
									
								
								client/libraries/clr-angular-17.9.0.tgz
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								client/libraries/clr-angular-17.9.0.tgz
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								client/libraries/clr-ui-17.9.0.tgz
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								client/libraries/clr-ui-17.9.0.tgz
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							@@ -10,7 +10,7 @@ const check = (cwd) => {
 | 
			
		||||
        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;',
 | 
			
		||||
        excludePackages:
 | 
			
		||||
          '@cds/city@1.1.0;@handsontable/angular@13.1.0;handsontable@13.1.0;hyperformula@2.6.2;jackspeak@2.2.0;path-scurry@1.7.0'
 | 
			
		||||
          '@cds/city@1.1.0;@handsontable/angular@15.3.0;handsontable@15.3.0;hyperformula@2.7.1;hyperformula@3.0.0;jackspeak@3.4.3;path-scurry@1.11.1;package-json-from-dist@1.0.1'
 | 
			
		||||
      },
 | 
			
		||||
      (error, json) => {
 | 
			
		||||
        if (error) {
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										11042
									
								
								client/package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										11042
									
								
								client/package-lock.json
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							@@ -18,8 +18,8 @@
 | 
			
		||||
    "deploy_sasjs": "rsync -avhe ssh ./dist/* --delete root@${npm_config_account}.4gl.io:/var/www/html/dc/dev",
 | 
			
		||||
    "viyabuild": "cd build; ./viyabuild.sh",
 | 
			
		||||
    "lint": "cd .. && npm run lint",
 | 
			
		||||
    "test": "ng test",
 | 
			
		||||
    "test:headless": "ng test --browsers ChromeHeadless",
 | 
			
		||||
    "test": "npx ng test",
 | 
			
		||||
    "test:headless": "npx ng test --no-watch --no-progress --browsers ChromeHeadlessCI",
 | 
			
		||||
    "watch": "ng test watch=true",
 | 
			
		||||
    "pree2e": "webdriver-manager update",
 | 
			
		||||
    "e2e": "protractor protractor.config.js",
 | 
			
		||||
@@ -44,22 +44,23 @@
 | 
			
		||||
    "@angular/platform-browser": "^17.3.3",
 | 
			
		||||
    "@angular/platform-browser-dynamic": "^17.3.3",
 | 
			
		||||
    "@angular/router": "^17.3.3",
 | 
			
		||||
    "@cds/core": "^6.10.0",
 | 
			
		||||
    "@clr/angular": "^17.0.1",
 | 
			
		||||
    "@cds/core": "^6.15.1",
 | 
			
		||||
    "@clr/angular": "file:libraries/clr-angular-17.9.0.tgz",
 | 
			
		||||
    "@clr/icons": "^13.0.2",
 | 
			
		||||
    "@clr/ui": "^17.0.1",
 | 
			
		||||
    "@handsontable/angular": "^13.1.0",
 | 
			
		||||
    "@sasjs/adapter": "4.10.2",
 | 
			
		||||
    "@clr/ui": "file:libraries/clr-ui-17.9.0.tgz",
 | 
			
		||||
    "@handsontable/angular": "^15.3.0",
 | 
			
		||||
    "@sasjs/adapter": "^4.11.0",
 | 
			
		||||
    "@sasjs/utils": "^3.4.0",
 | 
			
		||||
    "@sheet/crypto": "file:libraries/sheet-crypto.tgz",
 | 
			
		||||
    "@types/d3-graphviz": "^2.6.7",
 | 
			
		||||
    "@types/text-encoding": "0.0.35",
 | 
			
		||||
    "base64-arraybuffer": "^0.2.0",
 | 
			
		||||
    "buffer": "^5.4.3",
 | 
			
		||||
    "crypto-browserify": "3.12.0",
 | 
			
		||||
    "crypto-browserify": "^3.12.1",
 | 
			
		||||
    "crypto-js": "^4.2.0",
 | 
			
		||||
    "d3-graphviz": "^5.0.2",
 | 
			
		||||
    "fs-extra": "^7.0.1",
 | 
			
		||||
    "handsontable": "^13.1.0",
 | 
			
		||||
    "handsontable": "^15.3.0",
 | 
			
		||||
    "https-browserify": "1.0.0",
 | 
			
		||||
    "hyperformula": "^2.5.0",
 | 
			
		||||
    "iconv-lite": "^0.5.0",
 | 
			
		||||
@@ -78,6 +79,8 @@
 | 
			
		||||
    "text-encoding": "^0.7.0",
 | 
			
		||||
    "tslib": "^2.3.0",
 | 
			
		||||
    "vm": "^0.1.0",
 | 
			
		||||
    "webpack": "^5.91.0",
 | 
			
		||||
    "xlsx": "^0.18.5",
 | 
			
		||||
    "zone.js": "~0.14.4"
 | 
			
		||||
  },
 | 
			
		||||
  "devDependencies": {
 | 
			
		||||
 
 | 
			
		||||
@@ -43,6 +43,10 @@ export interface XLMapListItem {
 | 
			
		||||
  targetDS: string
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export interface HandsontableStaticConfig {
 | 
			
		||||
  darkTableHeaderClass: string
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Cached filtering values across whole app (editor, viewer, viewboxes)
 | 
			
		||||
 * Cached lineage libraries, tables
 | 
			
		||||
@@ -62,6 +66,7 @@ export const globals: {
 | 
			
		||||
  viyaApi: any
 | 
			
		||||
  usernav: any
 | 
			
		||||
  operators: any
 | 
			
		||||
  handsontable: HandsontableStaticConfig
 | 
			
		||||
  [key: string]: any
 | 
			
		||||
} = {
 | 
			
		||||
  rootParam: <string>'',
 | 
			
		||||
@@ -140,5 +145,11 @@ export const globals: {
 | 
			
		||||
  operators: {
 | 
			
		||||
    numOperators: ['=', '<', '>', '<=', '>=', 'BETWEEN', 'IN', 'NOT IN', 'NE'],
 | 
			
		||||
    charOperators: ['=', '<', '>', '<=', '>=', 'CONTAINS', 'IN', 'NOT IN', 'NE']
 | 
			
		||||
  },
 | 
			
		||||
  handsontable: {
 | 
			
		||||
    darkTableHeaderClass: 'darkTH'
 | 
			
		||||
  },
 | 
			
		||||
  userDropdownConfig: {
 | 
			
		||||
    closeOnDebugClick: false
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -139,10 +139,15 @@
 | 
			
		||||
        [routerLink]="['/']"
 | 
			
		||||
        class="nav-link"
 | 
			
		||||
      >
 | 
			
		||||
        <img class="without-text d-block d-md-none" src="images/dc-logo.svg" />
 | 
			
		||||
        <img
 | 
			
		||||
          class="without-text d-block d-md-none"
 | 
			
		||||
          src="images/dc-logo.svg"
 | 
			
		||||
          alt="datacontroller logo without text"
 | 
			
		||||
        />
 | 
			
		||||
        <img
 | 
			
		||||
          class="with-text d-none d-md-block"
 | 
			
		||||
          src="images/datacontroller.svg"
 | 
			
		||||
          alt="datacontroller logo"
 | 
			
		||||
        />
 | 
			
		||||
      </a>
 | 
			
		||||
 | 
			
		||||
@@ -245,6 +250,7 @@
 | 
			
		||||
 | 
			
		||||
  <app-alerts *ngIf="!errTop"></app-alerts>
 | 
			
		||||
  <app-requests-modal [(opened)]="requestsModal"></app-requests-modal>
 | 
			
		||||
  <app-excel-password-modal></app-excel-password-modal>
 | 
			
		||||
 | 
			
		||||
  <!-- <app-terms *ngIf="showRegistration"></app-terms> -->
 | 
			
		||||
 | 
			
		||||
@@ -282,7 +288,11 @@
 | 
			
		||||
 | 
			
		||||
<!-- App Loading Page -->
 | 
			
		||||
<div *ngIf="!startupDataLoaded" class="app-loading">
 | 
			
		||||
  <img class="loading-logo" src="images/datacontroller.svg" />
 | 
			
		||||
  <img
 | 
			
		||||
    class="loading-logo"
 | 
			
		||||
    src="images/datacontroller.svg"
 | 
			
		||||
    alt="datacontroller logo"
 | 
			
		||||
  />
 | 
			
		||||
 | 
			
		||||
  <div *ngIf="appActive === null" class="slider">
 | 
			
		||||
    <div class="line"></div>
 | 
			
		||||
 
 | 
			
		||||
@@ -1,440 +0,0 @@
 | 
			
		||||
// Copyright (c) 2016 VMware, Inc. All Rights Reserved.
 | 
			
		||||
// This software is released under MIT license.
 | 
			
		||||
// The full license information can be found in LICENSE in the root directory of this project.
 | 
			
		||||
app-requests-modal {
 | 
			
		||||
  z-index: 10000;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
header.app-header {
 | 
			
		||||
  background: #314351 !important;
 | 
			
		||||
  color: #fff;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.logo img.without-text {
 | 
			
		||||
  width: 30px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.logo img.with-text {
 | 
			
		||||
  width: 210px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.header-hamburger-trigger {
 | 
			
		||||
  display: block;
 | 
			
		||||
  background: transparent;
 | 
			
		||||
  border: 0;
 | 
			
		||||
  margin-left: 10px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.demo-expired-notice {
 | 
			
		||||
  display: flex;
 | 
			
		||||
  justify-content: center;
 | 
			
		||||
  align-items: center;
 | 
			
		||||
  position: fixed;
 | 
			
		||||
  left: 0;
 | 
			
		||||
  top: 0;
 | 
			
		||||
  height: 100vh !important;
 | 
			
		||||
  width: 100vw !important;
 | 
			
		||||
  z-index: 105;
 | 
			
		||||
  background: rgba(33, 33, 33, .5);
 | 
			
		||||
 | 
			
		||||
  .expired-details {
 | 
			
		||||
    flex-direction: column;
 | 
			
		||||
    align-items: center;
 | 
			
		||||
    padding: 30px;
 | 
			
		||||
    z-index: 110;
 | 
			
		||||
    background: #314351;
 | 
			
		||||
 | 
			
		||||
    .expired-notice {
 | 
			
		||||
      color: #e0e0e0;
 | 
			
		||||
      font-size: 16px;
 | 
			
		||||
 | 
			
		||||
      .mailto {
 | 
			
		||||
        color: #8dc53e;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.main-container .update-key {
 | 
			
		||||
  display: flex;
 | 
			
		||||
  align-items: center;
 | 
			
		||||
  color: white;
 | 
			
		||||
  padding: 0px 10px;
 | 
			
		||||
  background: #00000026;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.alert-icon-wrapper {
 | 
			
		||||
  margin-top: 0 !important;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.nav-text {
 | 
			
		||||
  margin-right: 20px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.sidebar-toggle {
 | 
			
		||||
  display: flex;
 | 
			
		||||
  height: 100%;
 | 
			
		||||
  align-items: center;
 | 
			
		||||
  padding-left: 10px;
 | 
			
		||||
 | 
			
		||||
  clr-icon {
 | 
			
		||||
    cursor: pointer;
 | 
			
		||||
    width: 30px;
 | 
			
		||||
    height: 30px;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
header {
 | 
			
		||||
  .header-actions {
 | 
			
		||||
    .dropdown {
 | 
			
		||||
      position: unset; //without it, when opening user dropdown scrollbar was displaying without reason
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .nav-link:hover {
 | 
			
		||||
      color: #fafafa;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .nav-link.active {
 | 
			
		||||
    background: #61717D;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.notf {
 | 
			
		||||
  background: #16a57a;
 | 
			
		||||
  color: #fffcfc;
 | 
			
		||||
  font-size: 12px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.btn.btn-success {
 | 
			
		||||
  border-color: #62a420;
 | 
			
		||||
  background-color: #16a57a!important;
 | 
			
		||||
  color: #fff;
 | 
			
		||||
}
 | 
			
		||||
.btn.btn-success:hover {
 | 
			
		||||
  background-color: #2add39;
 | 
			
		||||
  color: #fff;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.toggle-switch input[type=checkbox]:checked+label:before {
 | 
			
		||||
  border-color: #61717D;
 | 
			
		||||
  background-color: #61717D;
 | 
			
		||||
  transition: .15s ease-in;
 | 
			
		||||
  transition-property: border-color,background-color;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.main-container {
 | 
			
		||||
  min-height: 100vh !important;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.main-container .content-container .content-area {
 | 
			
		||||
  padding: 0rem 1rem 1rem 1rem;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.content-container {
 | 
			
		||||
  z-index: 0!important;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.navBarResp {
 | 
			
		||||
  display: flex;
 | 
			
		||||
  justify-content: center;
 | 
			
		||||
  background: #495A67;
 | 
			
		||||
  color: #fff;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@media screen and (max-width: 768px) {
 | 
			
		||||
  .navBarResp {
 | 
			
		||||
    display: flex;
 | 
			
		||||
    justify-content: flex-start;
 | 
			
		||||
    background: #495A67;
 | 
			
		||||
    color: #fff;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .main-container .sub-nav.clr-nav-level-1 .nav .nav-link, .main-container .sub-nav.clr-nav-level-2 .nav .nav-link, .main-container .subnav.clr-nav-level-1 .nav .nav-link, .main-container .subnav.clr-nav-level-2 .nav .nav-link {
 | 
			
		||||
      padding: 0 .5rem 0 1rem;
 | 
			
		||||
      width: 100%;
 | 
			
		||||
      max-width: 100%;
 | 
			
		||||
      overflow: hidden;
 | 
			
		||||
      text-overflow: ellipsis;
 | 
			
		||||
      border-radius: .125rem 0 0 .125rem;
 | 
			
		||||
      color: #95c84b;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
  .card-block, .card-footer {
 | 
			
		||||
    padding: 10px 0px 0px 0px;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .main-container[_ngcontent-c0] .content-container[_ngcontent-c0] .content-area[_ngcontent-c0] {
 | 
			
		||||
    padding: 0rem 0rem 0rem 0rem;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
::ng-deep {
 | 
			
		||||
  .htInvalid {
 | 
			
		||||
  background: black!important;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
  @media screen and (max-width:480px) {
 | 
			
		||||
  h2 {
 | 
			
		||||
    font-size: .7rem!important;
 | 
			
		||||
 | 
			
		||||
  }
 | 
			
		||||
  h3 {
 | 
			
		||||
    font-size: .7rem;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
  .nav-link {
 | 
			
		||||
    padding: 0rem 1rem 0rem 1rem;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .btn-primary .btn, .btn.btn-primary {
 | 
			
		||||
    border-color: #314351;
 | 
			
		||||
    background-color: #314351;
 | 
			
		||||
    color: #fff;
 | 
			
		||||
  }
 | 
			
		||||
  .btn {
 | 
			
		||||
    cursor: pointer;
 | 
			
		||||
    display: inline-block;
 | 
			
		||||
    -webkit-appearance: none!important;
 | 
			
		||||
    border-radius: .125rem;
 | 
			
		||||
    border: 1px solid;
 | 
			
		||||
    min-width: 3rem;
 | 
			
		||||
    max-width: 15rem;
 | 
			
		||||
    white-space: nowrap;
 | 
			
		||||
    text-overflow: ellipsis;
 | 
			
		||||
    overflow: hidden;
 | 
			
		||||
    text-align: center;
 | 
			
		||||
    text-transform: uppercase;
 | 
			
		||||
    vertical-align: middle;
 | 
			
		||||
    line-height: 1.5rem;
 | 
			
		||||
    letter-spacing: .12em;
 | 
			
		||||
    font-size: .5rem;
 | 
			
		||||
    font-weight: 500;
 | 
			
		||||
    height: 1.5rem;
 | 
			
		||||
    padding: 0 .5rem;
 | 
			
		||||
    border-color: #314351;
 | 
			
		||||
    background-color: transparent;
 | 
			
		||||
    color: #314351;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .btn.btn-outline {
 | 
			
		||||
    border-color: #314351;
 | 
			
		||||
    background-color: transparent;
 | 
			
		||||
    color: #314351;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .btn.btn-outline:hover {
 | 
			
		||||
    border-color: #314351;
 | 
			
		||||
    background-color: #495A67;
 | 
			
		||||
    color: #fff;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .btn.btn-success-outline:hover {
 | 
			
		||||
    background-color: #5ea71f;
 | 
			
		||||
    color: #fff7f7;
 | 
			
		||||
    border-color: #9a9696;
 | 
			
		||||
}
 | 
			
		||||
// .btn.btn-success-outline {
 | 
			
		||||
//     border-color: #266900;
 | 
			
		||||
//     background-color: transparent;
 | 
			
		||||
//     color: #318700;
 | 
			
		||||
// }
 | 
			
		||||
  // .wtSpreader {
 | 
			
		||||
 | 
			
		||||
  // }
 | 
			
		||||
 | 
			
		||||
  .htMobileEditorContainer .inputs textarea {
 | 
			
		||||
    font-size: 13pt;
 | 
			
		||||
    border: 2px solid #485967;
 | 
			
		||||
    border-radius: 4px;
 | 
			
		||||
    -webkit-appearance: none;
 | 
			
		||||
    box-shadow: none;
 | 
			
		||||
    position: absolute;
 | 
			
		||||
    left: 14px;
 | 
			
		||||
    right: 0px;
 | 
			
		||||
    top: 0;
 | 
			
		||||
    bottom: 0;
 | 
			
		||||
    padding: 7pt;
 | 
			
		||||
    width: 290px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.htMobileEditorContainer .positionControls {
 | 
			
		||||
    width: 333px;
 | 
			
		||||
    position: absolute;
 | 
			
		||||
    right: 5pt;
 | 
			
		||||
    top: 50px;
 | 
			
		||||
    bottom: 0;
 | 
			
		||||
    display: flex;
 | 
			
		||||
    justify-content: center;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.htMobileEditorContainer.active {
 | 
			
		||||
    display: block;
 | 
			
		||||
    height: 120px;
 | 
			
		||||
    width: 350px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
  .handsontable  {
 | 
			
		||||
  background-color: #ffffff;
 | 
			
		||||
  // border: 1px solid #ccc;
 | 
			
		||||
  border-radius: 3px;
 | 
			
		||||
  }
 | 
			
		||||
  .handsontable th {
 | 
			
		||||
  background-color: #fafafa;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /* Left and right */
 | 
			
		||||
  .ht_clone_left th {
 | 
			
		||||
    border-right: 1px solid #ccc;
 | 
			
		||||
    border-left: 1px solid #ccc;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /* Column headers */
 | 
			
		||||
  .ht_clone_top th {
 | 
			
		||||
    border-top: 1px solid #ccc;
 | 
			
		||||
    border-right: 1px solid #ccc;
 | 
			
		||||
    border-bottom: 1px solid #ccc;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .ht_clone_top_left_corner th {
 | 
			
		||||
    border-right: 1px solid #ccc;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .ht_master tr:nth-of-type(odd) > td {
 | 
			
		||||
    background-color: #f3f3f3;
 | 
			
		||||
    border: 1px solid rgb(197, 197, 197);
 | 
			
		||||
    border-bottom: 1px solid rgb(236, 235, 235);
 | 
			
		||||
    // padding: 1px 1px;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .ht_master tr:nth-of-type(even) > td {
 | 
			
		||||
    background-color: white;
 | 
			
		||||
    border: 1px solid rgb(197, 197, 197);
 | 
			
		||||
    border-bottom: 1px solid rgb(236, 235, 235);
 | 
			
		||||
    // padding: 1px 1px;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .wtBorder {
 | 
			
		||||
    background-color: #495A67!important;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .handsontable .handsontable.ht_clone_top .wtHider {
 | 
			
		||||
    padding: 0 0 0px 0!important;
 | 
			
		||||
    margin: 0px;
 | 
			
		||||
    border-bottom: 3px solid #d6d3d3;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
  .content-container {
 | 
			
		||||
    background: #F5F6FF;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .card {
 | 
			
		||||
    box-shadow: 0 0.125rem 0 0 #d7d7d7;
 | 
			
		||||
    border-radius: .0rem;
 | 
			
		||||
    border: 1px solid transparent;
 | 
			
		||||
    // min-height: calc(100vh - 150px);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .datagrid-compact, .datagrid-history{
 | 
			
		||||
    .datagrid {
 | 
			
		||||
      border-collapse: separate;
 | 
			
		||||
      border: 1px solid transparent;
 | 
			
		||||
      border-radius: .125rem;
 | 
			
		||||
      background-color: #fff;
 | 
			
		||||
      color: #565656;
 | 
			
		||||
      margin: 0;
 | 
			
		||||
      margin-top: 1rem;
 | 
			
		||||
      max-width: 100%;
 | 
			
		||||
      width: 100%;
 | 
			
		||||
      padding: 15px 15px 50px 15px;
 | 
			
		||||
    }
 | 
			
		||||
    .datagrid-foot {
 | 
			
		||||
      -webkit-box-pack: end;
 | 
			
		||||
      -ms-flex-pack: end;
 | 
			
		||||
      justify-content: flex-end;
 | 
			
		||||
      height: 1.5rem;
 | 
			
		||||
      padding: 0 .5rem;
 | 
			
		||||
      line-height: calc(1.5rem - 3px);
 | 
			
		||||
      font-size: .45833rem;
 | 
			
		||||
      background-color: #fff;
 | 
			
		||||
      border-top: 1px solid #ccc;
 | 
			
		||||
      border-radius: 0px;
 | 
			
		||||
      // border-radius: 0 0 .125rem .125rem;
 | 
			
		||||
    }
 | 
			
		||||
    .datagrid-footer {
 | 
			
		||||
      position: absolute;
 | 
			
		||||
      right: 15px;
 | 
			
		||||
      top: 2px;
 | 
			
		||||
    }
 | 
			
		||||
    .datagrid .datagrid-head {
 | 
			
		||||
      background-color: #fff;
 | 
			
		||||
      border-bottom: 1px solid #ccc;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .dropdown-menu {
 | 
			
		||||
    position: absolute;
 | 
			
		||||
    top: 100%;
 | 
			
		||||
    left: 0;
 | 
			
		||||
    margin-top: .083333rem;
 | 
			
		||||
    display: -webkit-box;
 | 
			
		||||
    display: -ms-flexbox;
 | 
			
		||||
    display: flex;
 | 
			
		||||
    -webkit-box-orient: vertical;
 | 
			
		||||
    -webkit-box-direction: normal;
 | 
			
		||||
    -ms-flex-direction: column;
 | 
			
		||||
    flex-direction: column;
 | 
			
		||||
    background: #f5f6ff;
 | 
			
		||||
    padding: .5rem 0;
 | 
			
		||||
    border: 1px solid #ccc;
 | 
			
		||||
    box-shadow: 0 1px 0.125rem hsla(0,0%,45%,.25);
 | 
			
		||||
    min-width: 5rem;
 | 
			
		||||
    max-width: 15rem;
 | 
			
		||||
    border-radius: .125rem;
 | 
			
		||||
    visibility: hidden;
 | 
			
		||||
    z-index: 1000;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .table {
 | 
			
		||||
    border-collapse: separate;
 | 
			
		||||
    border: 1px solid transparent;
 | 
			
		||||
    border-radius: 0px;
 | 
			
		||||
    background-color: #fff;
 | 
			
		||||
    color: #565656;
 | 
			
		||||
    margin: 0;
 | 
			
		||||
    margin-top: 1rem;
 | 
			
		||||
    max-width: 100%;
 | 
			
		||||
    width: 100%;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .table th {
 | 
			
		||||
    font-size: .45833rem;
 | 
			
		||||
    font-weight: 600;
 | 
			
		||||
    letter-spacing: .03em;
 | 
			
		||||
    background-color: #fff;
 | 
			
		||||
    vertical-align: bottom;
 | 
			
		||||
    border-bottom: 1px solid #ccc;
 | 
			
		||||
    text-transform: uppercase;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .modal-header {
 | 
			
		||||
    border-bottom: 2px solid #e4e4e4;
 | 
			
		||||
    padding: 0 0 .5rem 0;
 | 
			
		||||
    margin-bottom: 1rem;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .main-container .content-container {
 | 
			
		||||
    min-height: 0px;
 | 
			
		||||
    position: relative;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.app-loading {
 | 
			
		||||
  .loading-logo {
 | 
			
		||||
    max-width: 400px;
 | 
			
		||||
    width: 100%;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,4 +1,9 @@
 | 
			
		||||
import { ChangeDetectorRef, Component, ElementRef } from '@angular/core'
 | 
			
		||||
import {
 | 
			
		||||
  ChangeDetectorRef,
 | 
			
		||||
  Component,
 | 
			
		||||
  ElementRef,
 | 
			
		||||
  ViewEncapsulation
 | 
			
		||||
} from '@angular/core'
 | 
			
		||||
import { Router } from '@angular/router'
 | 
			
		||||
import { VERSION } from '../environments/version'
 | 
			
		||||
import { ActivatedRoute } from '@angular/router'
 | 
			
		||||
@@ -13,11 +18,31 @@ import { InfoModal } from './models/InfoModal'
 | 
			
		||||
import { DcAdapterSettings } from './models/DcAdapterSettings'
 | 
			
		||||
import { AppStoreService } from './services/app-store.service'
 | 
			
		||||
import { LicenceService } from './services/licence.service'
 | 
			
		||||
import '@cds/core/icon/register.js'
 | 
			
		||||
import {
 | 
			
		||||
  ClarityIcons,
 | 
			
		||||
  exclamationTriangleIcon,
 | 
			
		||||
  moonIcon,
 | 
			
		||||
  processOnVmIcon,
 | 
			
		||||
  sunIcon,
 | 
			
		||||
  tableIcon,
 | 
			
		||||
  trashIcon
 | 
			
		||||
} from '@cds/core/icon'
 | 
			
		||||
 | 
			
		||||
ClarityIcons.addIcons(
 | 
			
		||||
  moonIcon,
 | 
			
		||||
  sunIcon,
 | 
			
		||||
  exclamationTriangleIcon,
 | 
			
		||||
  tableIcon,
 | 
			
		||||
  trashIcon,
 | 
			
		||||
  processOnVmIcon
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
@Component({
 | 
			
		||||
  selector: 'my-app',
 | 
			
		||||
  templateUrl: './app.component.html',
 | 
			
		||||
  styleUrls: ['./app.component.scss']
 | 
			
		||||
  styleUrls: ['./app.component.scss'],
 | 
			
		||||
  encapsulation: ViewEncapsulation.None
 | 
			
		||||
})
 | 
			
		||||
export class AppComponent {
 | 
			
		||||
  private dcAdapterSettings: DcAdapterSettings | undefined
 | 
			
		||||
 
 | 
			
		||||
@@ -20,10 +20,10 @@ import { UsernavRouteComponent } from './routes/usernav-route/usernav-route.comp
 | 
			
		||||
import { AppService } from './services/app.service'
 | 
			
		||||
import { InfoModalComponent } from './shared/abort-modal/info-modal.component'
 | 
			
		||||
import { RequestsModalComponent } from './shared/requests-modal/requests-modal.component'
 | 
			
		||||
import { HomeModule } from './home/home.module'
 | 
			
		||||
import { DirectivesModule } from './directives/directives.module'
 | 
			
		||||
import { ViyaApiExplorerComponent } from './viya-api-explorer/viya-api-explorer.component'
 | 
			
		||||
import { NgxJsonViewerModule } from 'ngx-json-viewer'
 | 
			
		||||
import { AppSettingsService } from './services/app-settings.service'
 | 
			
		||||
 | 
			
		||||
@NgModule({
 | 
			
		||||
  declarations: [
 | 
			
		||||
@@ -50,7 +50,7 @@ import { NgxJsonViewerModule } from 'ngx-json-viewer'
 | 
			
		||||
    DirectivesModule,
 | 
			
		||||
    NgxJsonViewerModule
 | 
			
		||||
  ],
 | 
			
		||||
  providers: [AppService, SasStoreService, LicensingGuard],
 | 
			
		||||
  providers: [AppService, SasStoreService, LicensingGuard, AppSettingsService],
 | 
			
		||||
  bootstrap: [AppComponent]
 | 
			
		||||
})
 | 
			
		||||
export class AppModule {}
 | 
			
		||||
 
 | 
			
		||||
@@ -5,7 +5,7 @@
 | 
			
		||||
      <div class="card-header">Terms and Conditions</div>
 | 
			
		||||
      <div class="card-block">
 | 
			
		||||
        <div class="card-text">
 | 
			
		||||
          <p>
 | 
			
		||||
          <p class="mt-0">
 | 
			
		||||
            The Demo version of Data Controller is free for EVALUATION purposes
 | 
			
		||||
            only. Before proceeding with configuration, please confirm that you
 | 
			
		||||
            have read, understood, and agreed to the
 | 
			
		||||
 
 | 
			
		||||
@@ -1,50 +0,0 @@
 | 
			
		||||
.card {
 | 
			
		||||
    margin-top: 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.btn {
 | 
			
		||||
    margin-top: 10px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.log-wrapper {
 | 
			
		||||
    width: 100%;
 | 
			
		||||
    background: #f0f0f0;
 | 
			
		||||
    border: 1px solid #c9c9c9;
 | 
			
		||||
    padding: 10px;
 | 
			
		||||
    overflow: auto;
 | 
			
		||||
    white-space: pre-wrap;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#contexts-btn {
 | 
			
		||||
    padding: 0;
 | 
			
		||||
    min-width: 30px;
 | 
			
		||||
    margin-left: 10px;
 | 
			
		||||
    height: 30px;
 | 
			
		||||
    display: inline-flex;
 | 
			
		||||
    justify-content: center;
 | 
			
		||||
    align-items: center;
 | 
			
		||||
    padding-top: 3px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.validation-bar {
 | 
			
		||||
    display: flex;
 | 
			
		||||
    margin-top: 20px;
 | 
			
		||||
    align-items: center;
 | 
			
		||||
 | 
			
		||||
    clr-icon {
 | 
			
		||||
        margin-right: 5px;
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.autodeploy-section {
 | 
			
		||||
    padding: 0px 15px;
 | 
			
		||||
 | 
			
		||||
    .clr-checkbox-wrapper {
 | 
			
		||||
        margin: 20px 0 20px 0;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    .btn-autodeploy {
 | 
			
		||||
        display: block;
 | 
			
		||||
        margin: 15px 0 15px 0;
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,4 +1,4 @@
 | 
			
		||||
import { Component, OnInit } from '@angular/core'
 | 
			
		||||
import { Component, OnInit, ViewEncapsulation } from '@angular/core'
 | 
			
		||||
import { SasService } from '../services/sas.service'
 | 
			
		||||
import { SASjsConfig } from '@sasjs/adapter'
 | 
			
		||||
import { Router } from '@angular/router'
 | 
			
		||||
@@ -13,7 +13,8 @@ import { DcAdapterSettings } from '../models/DcAdapterSettings'
 | 
			
		||||
  styleUrls: ['./deploy.component.scss'],
 | 
			
		||||
  host: {
 | 
			
		||||
    class: 'content-container'
 | 
			
		||||
  }
 | 
			
		||||
  },
 | 
			
		||||
  encapsulation: ViewEncapsulation.None
 | 
			
		||||
})
 | 
			
		||||
export class DeployComponent implements OnInit {
 | 
			
		||||
  public step: number = 0
 | 
			
		||||
@@ -56,25 +57,6 @@ export class DeployComponent implements OnInit {
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  ngOnInit() {
 | 
			
		||||
    if (this.sasJsConfig.serverType === ServerType.SasViya) {
 | 
			
		||||
      fetch('sasbuild/viya.json')
 | 
			
		||||
        .then((res) => res.text())
 | 
			
		||||
        .then((res) => {
 | 
			
		||||
          let initJsonFile: any = null
 | 
			
		||||
 | 
			
		||||
          try {
 | 
			
		||||
            initJsonFile = JSON.parse(res)
 | 
			
		||||
          } catch (err) {
 | 
			
		||||
            console.error(err)
 | 
			
		||||
          }
 | 
			
		||||
 | 
			
		||||
          if (initJsonFile) {
 | 
			
		||||
            this.jsonFile = initJsonFile
 | 
			
		||||
            this.loggerService.log(this.jsonFile)
 | 
			
		||||
          }
 | 
			
		||||
        })
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    this.setDeployDefaults()
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -9,14 +9,17 @@
 | 
			
		||||
      <p class="m-0 align-self-start">Done</p>
 | 
			
		||||
      <hr class="w-100" />
 | 
			
		||||
 | 
			
		||||
      <div class="deploy-status-row">
 | 
			
		||||
      <div
 | 
			
		||||
        *ngIf="autoDeployStatus.deployServicePack !== null"
 | 
			
		||||
        class="deploy-status-row"
 | 
			
		||||
      >
 | 
			
		||||
        <clr-icon
 | 
			
		||||
          *ngIf="autoDeployStatus.deployServicePack"
 | 
			
		||||
          *ngIf="autoDeployStatus.deployServicePack === true"
 | 
			
		||||
          class="deploy-success"
 | 
			
		||||
          shape="success-standard"
 | 
			
		||||
        ></clr-icon>
 | 
			
		||||
        <clr-icon
 | 
			
		||||
          *ngIf="!autoDeployStatus.deployServicePack"
 | 
			
		||||
          *ngIf="!autoDeployStatus.deployServicePack === false"
 | 
			
		||||
          class="deploy-error"
 | 
			
		||||
          shape="times-circle"
 | 
			
		||||
        ></clr-icon>
 | 
			
		||||
@@ -52,7 +55,7 @@
 | 
			
		||||
            class="deploy-error"
 | 
			
		||||
            shape="times-circle"
 | 
			
		||||
          ></clr-icon>
 | 
			
		||||
          LAUNCH / CONFIGURE
 | 
			
		||||
          LAUNCH
 | 
			
		||||
        </button>
 | 
			
		||||
 | 
			
		||||
        <button
 | 
			
		||||
@@ -94,20 +97,36 @@
 | 
			
		||||
</div>
 | 
			
		||||
 | 
			
		||||
<label for="dcloc" class="mt-20 clr-control-label">DC Loc</label>
 | 
			
		||||
<div class="mb-10 clr-control-container">
 | 
			
		||||
<div class="mb-10 clr-control-container dc-loc-input-wrapper">
 | 
			
		||||
  <div class="clr-input-wrapper">
 | 
			
		||||
    <p class="mt-0">{{ dcPath }}</p>
 | 
			
		||||
    <input clrInput name="dcloc" [(ngModel)]="dcPath" />
 | 
			
		||||
  </div>
 | 
			
		||||
</div>
 | 
			
		||||
 | 
			
		||||
<label for="dcloc" class="mt-20 clr-control-label">SAS Admin group</label>
 | 
			
		||||
<div class="mb-10 clr-control-container">
 | 
			
		||||
  <div class="clr-input-wrapper">
 | 
			
		||||
    <p class="mt-0">{{ selectedAdminGroup }}</p>
 | 
			
		||||
    <select
 | 
			
		||||
      *ngIf="!adminGroupsLoading"
 | 
			
		||||
      clrSelect
 | 
			
		||||
      name="options"
 | 
			
		||||
      [(ngModel)]="selectedAdminGroup"
 | 
			
		||||
    >
 | 
			
		||||
      <option *ngFor="let adminGroup of adminGroups" [value]="adminGroup.id">
 | 
			
		||||
        {{ adminGroup.name }}
 | 
			
		||||
      </option>
 | 
			
		||||
    </select>
 | 
			
		||||
    <clr-spinner
 | 
			
		||||
      clrInline
 | 
			
		||||
      class="spinner-sm"
 | 
			
		||||
      *ngIf="adminGroupsLoading"
 | 
			
		||||
    ></clr-spinner>
 | 
			
		||||
  </div>
 | 
			
		||||
</div>
 | 
			
		||||
 | 
			
		||||
<clr-checkbox-wrapper>
 | 
			
		||||
<!-- Keeping this for a reference in case future VIYA changes and starts allowing separate backend and frontend) -->
 | 
			
		||||
 | 
			
		||||
<!-- <clr-checkbox-wrapper>
 | 
			
		||||
  <input
 | 
			
		||||
    clrCheckbox
 | 
			
		||||
    [(ngModel)]="recreateDatabase"
 | 
			
		||||
@@ -116,19 +135,28 @@
 | 
			
		||||
    checked
 | 
			
		||||
  />
 | 
			
		||||
  <label>Recreate database</label>
 | 
			
		||||
</clr-checkbox-wrapper>
 | 
			
		||||
</clr-checkbox-wrapper> -->
 | 
			
		||||
 | 
			
		||||
<hr />
 | 
			
		||||
 | 
			
		||||
<button
 | 
			
		||||
  (click)="runAutoDeploy()"
 | 
			
		||||
  class="btn-autodeploy btn btn-primary d-inline-block mr-10"
 | 
			
		||||
>
 | 
			
		||||
  Deploy
 | 
			
		||||
</button>
 | 
			
		||||
 | 
			
		||||
<!-- Keeping this for a reference in case future VIYA changes and starts allowing separate backend and frontend) -->
 | 
			
		||||
 | 
			
		||||
<!-- <button
 | 
			
		||||
  (click)="executeJson()"
 | 
			
		||||
  class="btn-autodeploy btn btn-primary d-inline-block mr-10"
 | 
			
		||||
  [disabled]="!jsonFile"
 | 
			
		||||
>
 | 
			
		||||
  Deploy {{ !jsonFile ? '(json file is not available)' : '' }}
 | 
			
		||||
</button>
 | 
			
		||||
</button> -->
 | 
			
		||||
 | 
			
		||||
<button
 | 
			
		||||
<!-- <button
 | 
			
		||||
  (click)="uploadJsonAuto.click()"
 | 
			
		||||
  class="btn-autodeploy btn btn-primary d-inline-block mr-10"
 | 
			
		||||
>
 | 
			
		||||
@@ -140,7 +168,7 @@
 | 
			
		||||
  hidden
 | 
			
		||||
  (click)="clearUploadInput($event)"
 | 
			
		||||
  (change)="onJsonFileChange($event)"
 | 
			
		||||
/>
 | 
			
		||||
/> -->
 | 
			
		||||
 | 
			
		||||
<clr-modal [(clrModalOpen)]="recreateDatabaseModal" [clrModalClosable]="false">
 | 
			
		||||
  <h3 class="modal-title">Warning</h3>
 | 
			
		||||
 
 | 
			
		||||
@@ -1,61 +0,0 @@
 | 
			
		||||
.auto-deploy {
 | 
			
		||||
    display: flex;
 | 
			
		||||
    justify-content: center;
 | 
			
		||||
    align-items: center;
 | 
			
		||||
 | 
			
		||||
    position: fixed;
 | 
			
		||||
    left: 0;
 | 
			
		||||
    right: 0;
 | 
			
		||||
    top: 0;
 | 
			
		||||
    bottom: 0;
 | 
			
		||||
    background: rgba(0, 0, 0, 0.4);
 | 
			
		||||
 | 
			
		||||
    z-index: 100;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.spinner-box {
 | 
			
		||||
    width: 400px;
 | 
			
		||||
    padding: 20px;
 | 
			
		||||
    border-radius: 3px;
 | 
			
		||||
    background: #fff;
 | 
			
		||||
    display: flex;
 | 
			
		||||
    justify-content: center;
 | 
			
		||||
    align-items: center;
 | 
			
		||||
    flex-direction: column;
 | 
			
		||||
    box-shadow: 1px 1px 8px 0px #00000082;
 | 
			
		||||
 | 
			
		||||
    .buttons {
 | 
			
		||||
        display: flex;
 | 
			
		||||
        justify-content: space-between;
 | 
			
		||||
        width: 100%;
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.deploy-status-row {
 | 
			
		||||
    display: flex;
 | 
			
		||||
    align-items: center;
 | 
			
		||||
    align-self: flex-start;
 | 
			
		||||
 | 
			
		||||
    p {
 | 
			
		||||
        margin: 0 0 0 10px;
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.deploy-success {
 | 
			
		||||
    color: #6ECF44;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.deploy-error {
 | 
			
		||||
    color: #E74C3C;
 | 
			
		||||
    // width: 20px;
 | 
			
		||||
    // height: 20px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.deploy-undeterminated {
 | 
			
		||||
    color: #cacaca;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
hr {
 | 
			
		||||
    border: 0;
 | 
			
		||||
    border-bottom: 1px solid #00000045;
 | 
			
		||||
}
 | 
			
		||||
@@ -1,15 +1,28 @@
 | 
			
		||||
import { Component, EventEmitter, Input, OnInit, Output } from '@angular/core'
 | 
			
		||||
import {
 | 
			
		||||
  Component,
 | 
			
		||||
  EventEmitter,
 | 
			
		||||
  Input,
 | 
			
		||||
  OnInit,
 | 
			
		||||
  Output,
 | 
			
		||||
  ViewEncapsulation
 | 
			
		||||
} from '@angular/core'
 | 
			
		||||
import SASjs, { SASjsConfig } from '@sasjs/adapter'
 | 
			
		||||
import { DcAdapterSettings } from 'src/app/models/DcAdapterSettings'
 | 
			
		||||
import { DeployService } from 'src/app/services/deploy.service'
 | 
			
		||||
import { EventService } from 'src/app/services/event.service'
 | 
			
		||||
import { LoggerService } from 'src/app/services/logger.service'
 | 
			
		||||
import { SasViyaService } from 'src/app/services/sas-viya.service'
 | 
			
		||||
import { SasService } from 'src/app/services/sas.service'
 | 
			
		||||
import {
 | 
			
		||||
  Item,
 | 
			
		||||
  ViyaApiIdentities
 | 
			
		||||
} from 'src/app/viya-api-explorer/models/viya-api-identities.model'
 | 
			
		||||
 | 
			
		||||
@Component({
 | 
			
		||||
  selector: 'app-automatic-deploy',
 | 
			
		||||
  templateUrl: './automatic.component.html',
 | 
			
		||||
  styleUrls: ['./automatic.component.scss']
 | 
			
		||||
  styleUrls: ['./automatic.component.scss'],
 | 
			
		||||
  encapsulation: ViewEncapsulation.None
 | 
			
		||||
})
 | 
			
		||||
export class AutomaticComponent implements OnInit {
 | 
			
		||||
  @Input() sasJs!: SASjs
 | 
			
		||||
@@ -28,8 +41,14 @@ export class AutomaticComponent implements OnInit {
 | 
			
		||||
  public recreateDatabaseModal: boolean = false
 | 
			
		||||
  public isSubmittingJson: boolean = false
 | 
			
		||||
  public isJsonSubmitted: boolean = false
 | 
			
		||||
  public recreateDatabase: boolean = false
 | 
			
		||||
  /**
 | 
			
		||||
   * Default was `false` when deploy was done with frontend and backend separately.
 | 
			
		||||
   * Now we are using only streaming app, so we always want to recreate database (makedata)
 | 
			
		||||
   */
 | 
			
		||||
  public recreateDatabase: boolean = true
 | 
			
		||||
  public createDatabaseLoading: boolean = false
 | 
			
		||||
  public adminGroupsLoading: boolean = false
 | 
			
		||||
  public adminGroups: { id: string; name: string }[] = []
 | 
			
		||||
 | 
			
		||||
  /** autoDeployStatus
 | 
			
		||||
   * This object presents the status for two steps that we have for deploy.
 | 
			
		||||
@@ -50,10 +69,33 @@ export class AutomaticComponent implements OnInit {
 | 
			
		||||
    private eventService: EventService,
 | 
			
		||||
    private deployService: DeployService,
 | 
			
		||||
    private sasService: SasService,
 | 
			
		||||
    private sasViyaService: SasViyaService,
 | 
			
		||||
    private loggerService: LoggerService
 | 
			
		||||
  ) {}
 | 
			
		||||
 | 
			
		||||
  ngOnInit(): void {}
 | 
			
		||||
  ngOnInit(): void {
 | 
			
		||||
    this.getAdminGroups()
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  public async getAdminGroups() {
 | 
			
		||||
    this.adminGroupsLoading = true
 | 
			
		||||
 | 
			
		||||
    this.sasViyaService.getAdminGroups().subscribe((res: ViyaApiIdentities) => {
 | 
			
		||||
      this.adminGroupsLoading = false
 | 
			
		||||
      // Map admin groups with only needed fields
 | 
			
		||||
      this.adminGroups = res.items.map((item: Item) => {
 | 
			
		||||
        return {
 | 
			
		||||
          id: item.id,
 | 
			
		||||
          name: item.name
 | 
			
		||||
        }
 | 
			
		||||
      })
 | 
			
		||||
    }),
 | 
			
		||||
      (err: any) => {
 | 
			
		||||
        this.adminGroupsLoading = false
 | 
			
		||||
        this.loggerService.error('Error while getting admin groups', err)
 | 
			
		||||
        this.eventService.showAbortModal('admin groups', err)
 | 
			
		||||
      }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * Executes sas.json file to deploy the backend
 | 
			
		||||
@@ -63,7 +105,6 @@ export class AutomaticComponent implements OnInit {
 | 
			
		||||
   * to create database if checkbox is toggled on
 | 
			
		||||
   */
 | 
			
		||||
  public async executeJson() {
 | 
			
		||||
    this.autodeploying = true
 | 
			
		||||
    this.isSubmittingJson = true
 | 
			
		||||
 | 
			
		||||
    try {
 | 
			
		||||
@@ -98,6 +139,14 @@ export class AutomaticComponent implements OnInit {
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    this.isSubmittingJson = false
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  public async runAutoDeploy(executeJson: boolean = false) {
 | 
			
		||||
    this.autodeploying = true
 | 
			
		||||
 | 
			
		||||
    if (executeJson) {
 | 
			
		||||
      this.executeJson()
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (this.recreateDatabase) {
 | 
			
		||||
      this.createDatabase()
 | 
			
		||||
@@ -125,7 +174,7 @@ export class AutomaticComponent implements OnInit {
 | 
			
		||||
     * contextName: null is the MUST field for it.
 | 
			
		||||
     */
 | 
			
		||||
    let overrideConfig = {
 | 
			
		||||
      useComputeApi: false,
 | 
			
		||||
      useComputeApi: null,
 | 
			
		||||
      contextName: this.sasJsConfig.contextName,
 | 
			
		||||
      debug: true
 | 
			
		||||
    }
 | 
			
		||||
@@ -148,8 +197,21 @@ export class AutomaticComponent implements OnInit {
 | 
			
		||||
        } else {
 | 
			
		||||
          this.autoDeployStatus.runMakeData = false
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        if (typeof res.sasjsAbort !== 'undefined') {
 | 
			
		||||
          const abortRes = res
 | 
			
		||||
          const abortMsg = abortRes.sasjsAbort[0].MSG
 | 
			
		||||
          const macMsg = abortRes.sasjsAbort[0].MAC
 | 
			
		||||
 | 
			
		||||
          this.eventService.showAbortModal('makedata', abortMsg, {
 | 
			
		||||
            SYSWARNINGTEXT: abortRes.SYSWARNINGTEXT,
 | 
			
		||||
            SYSERRORTEXT: abortRes.SYSERRORTEXT,
 | 
			
		||||
            MAC: macMsg
 | 
			
		||||
          })
 | 
			
		||||
        }
 | 
			
		||||
      })
 | 
			
		||||
      .catch((err: any) => {
 | 
			
		||||
        this.eventService.showAbortModal('makedata', JSON.stringify(err))
 | 
			
		||||
        this.autoDeployStatus.runMakeData = false
 | 
			
		||||
        this.autodeployDone = true
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -1,4 +0,0 @@
 | 
			
		||||
.clear-memory-button {
 | 
			
		||||
  right: 10px;
 | 
			
		||||
  top: 2px;
 | 
			
		||||
}
 | 
			
		||||
@@ -1,6 +1,14 @@
 | 
			
		||||
import { Component, Input, OnInit, Output, EventEmitter } from '@angular/core'
 | 
			
		||||
import {
 | 
			
		||||
  Component,
 | 
			
		||||
  Input,
 | 
			
		||||
  OnInit,
 | 
			
		||||
  Output,
 | 
			
		||||
  EventEmitter,
 | 
			
		||||
  ViewEncapsulation
 | 
			
		||||
} from '@angular/core'
 | 
			
		||||
import SASjs, { SASjsConfig } from '@sasjs/adapter'
 | 
			
		||||
import { DcAdapterSettings } from 'src/app/models/DcAdapterSettings'
 | 
			
		||||
import { RequestWrapperResponse } from 'src/app/models/request-wrapper/RequestWrapperResponse'
 | 
			
		||||
import { DeployService } from 'src/app/services/deploy.service'
 | 
			
		||||
import { EventService } from 'src/app/services/event.service'
 | 
			
		||||
import { LoggerService } from 'src/app/services/logger.service'
 | 
			
		||||
@@ -9,7 +17,8 @@ import { SasService } from 'src/app/services/sas.service'
 | 
			
		||||
@Component({
 | 
			
		||||
  selector: 'app-manual-deploy',
 | 
			
		||||
  templateUrl: './manual.component.html',
 | 
			
		||||
  styleUrls: ['./manual.component.scss']
 | 
			
		||||
  styleUrls: ['./manual.component.scss'],
 | 
			
		||||
  encapsulation: ViewEncapsulation.None
 | 
			
		||||
})
 | 
			
		||||
export class ManualComponent implements OnInit {
 | 
			
		||||
  @Input() sasJs!: SASjs
 | 
			
		||||
@@ -265,7 +274,7 @@ export class ManualComponent implements OnInit {
 | 
			
		||||
     * contextName: null is the MUST field for it.
 | 
			
		||||
     */
 | 
			
		||||
    let overrideConfig = {
 | 
			
		||||
      useComputeApi: false,
 | 
			
		||||
      useComputeApi: null,
 | 
			
		||||
      contextName: this.sasJsConfig.contextName,
 | 
			
		||||
      debug: true
 | 
			
		||||
    }
 | 
			
		||||
@@ -303,10 +312,10 @@ export class ManualComponent implements OnInit {
 | 
			
		||||
 | 
			
		||||
    this.sasService
 | 
			
		||||
      .request('public/startupservice', null)
 | 
			
		||||
      .then((res: any) => {
 | 
			
		||||
        this.loggerService.log(res)
 | 
			
		||||
      .then((res: RequestWrapperResponse) => {
 | 
			
		||||
        this.loggerService.log(res.adapterResponse)
 | 
			
		||||
 | 
			
		||||
        if (res.saslibs) {
 | 
			
		||||
        if (res.adapterResponse.saslibs) {
 | 
			
		||||
          this.validationState = 'success'
 | 
			
		||||
        } else {
 | 
			
		||||
          this.validationState = 'error'
 | 
			
		||||
 
 | 
			
		||||
@@ -10,11 +10,13 @@
 | 
			
		||||
</p>
 | 
			
		||||
 | 
			
		||||
<p class="m-0 mt-10">
 | 
			
		||||
  Please specify a physical directory below, to which user
 | 
			
		||||
  <strong>{{ SYSUSERID }}</strong> can write, on behalf of Data Controller:
 | 
			
		||||
  Please specify a physical directory (on the
 | 
			
		||||
  <strong> {{ SYSHOSTNAME }}</strong>
 | 
			
		||||
  compute server) below, to which user
 | 
			
		||||
  <strong>{{ SYSUSERID }}</strong> can write, on behalf of Data Controller.
 | 
			
		||||
</p>
 | 
			
		||||
 | 
			
		||||
<label class="mt-20 clr-control-label">DC Directory</label>
 | 
			
		||||
<label class="mt-20 clr-control-label">DC Staging Directory</label>
 | 
			
		||||
<div class="mb-10 clr-control-container">
 | 
			
		||||
  <div class="clr-input-wrapper">
 | 
			
		||||
    <input
 | 
			
		||||
 
 | 
			
		||||
@@ -1,23 +0,0 @@
 | 
			
		||||
.clr-control-container {
 | 
			
		||||
  width: 50vw;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.clr-input-wrapper {
 | 
			
		||||
  width: 100%;
 | 
			
		||||
 | 
			
		||||
  input {
 | 
			
		||||
    width: 100%;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.thinProgress {
 | 
			
		||||
  left: 0px;
 | 
			
		||||
  right: 0;
 | 
			
		||||
  width: unset;
 | 
			
		||||
  height: 1px;
 | 
			
		||||
  margin-top: 0 !important;
 | 
			
		||||
 | 
			
		||||
  &::after {
 | 
			
		||||
    top: 0;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,8 +1,16 @@
 | 
			
		||||
import { Location } from '@angular/common'
 | 
			
		||||
import { Component, EventEmitter, Input, OnInit, Output } from '@angular/core'
 | 
			
		||||
import {
 | 
			
		||||
  Component,
 | 
			
		||||
  EventEmitter,
 | 
			
		||||
  Input,
 | 
			
		||||
  OnInit,
 | 
			
		||||
  Output,
 | 
			
		||||
  ViewEncapsulation
 | 
			
		||||
} from '@angular/core'
 | 
			
		||||
import SASjs, { SASjsConfig } from '@sasjs/adapter'
 | 
			
		||||
import { ServerType } from '@sasjs/utils/types/serverType'
 | 
			
		||||
import { DcAdapterSettings } from 'src/app/models/DcAdapterSettings'
 | 
			
		||||
import { RequestWrapperResponse } from 'src/app/models/request-wrapper/RequestWrapperResponse'
 | 
			
		||||
import { SASGroup } from 'src/app/models/sas/public-getgroups.model'
 | 
			
		||||
import { SASjsApiServerInfo } from 'src/app/models/sasjs-api/SASjsApiServerInfo.model'
 | 
			
		||||
import { SasService } from 'src/app/services/sas.service'
 | 
			
		||||
@@ -11,7 +19,8 @@ import { SasjsService } from 'src/app/services/sasjs.service'
 | 
			
		||||
@Component({
 | 
			
		||||
  selector: 'app-sasjs-configurator',
 | 
			
		||||
  templateUrl: './sasjs-configurator.component.html',
 | 
			
		||||
  styleUrls: ['./sasjs-configurator.component.scss']
 | 
			
		||||
  styleUrls: ['./sasjs-configurator.component.scss'],
 | 
			
		||||
  encapsulation: ViewEncapsulation.None
 | 
			
		||||
})
 | 
			
		||||
export class SasjsConfiguratorComponent implements OnInit {
 | 
			
		||||
  @Input() sasJs!: SASjs
 | 
			
		||||
@@ -68,11 +77,11 @@ export class SasjsConfiguratorComponent implements OnInit {
 | 
			
		||||
    this.loading = true
 | 
			
		||||
 | 
			
		||||
    this.sasService.request('usernav/usergroupsbymember', null).then(
 | 
			
		||||
      (res: any) => {
 | 
			
		||||
        this.METAPERSON = res.MF_GETUSER
 | 
			
		||||
        this.SYSUSERID = res.SYSUSERID
 | 
			
		||||
        this.SYSHOSTNAME = res.SYSHOSTNAME
 | 
			
		||||
        this.SYSVLONG = res.SYSVLONG
 | 
			
		||||
      (res: RequestWrapperResponse) => {
 | 
			
		||||
        this.METAPERSON = res.adapterResponse.MF_GETUSER
 | 
			
		||||
        this.SYSUSERID = res.adapterResponse.SYSUSERID
 | 
			
		||||
        this.SYSHOSTNAME = res.adapterResponse.SYSHOSTNAME
 | 
			
		||||
        this.SYSVLONG = res.adapterResponse.SYSVLONG
 | 
			
		||||
 | 
			
		||||
        /*
 | 
			
		||||
          We would like to present a default DCPATH (deployment path) to the
 | 
			
		||||
@@ -88,12 +97,14 @@ export class SasjsConfiguratorComponent implements OnInit {
 | 
			
		||||
        */
 | 
			
		||||
        this.dcDirectory =
 | 
			
		||||
          this.tmpDirectories[
 | 
			
		||||
            ['L', 'H', 'A', 'S'].includes(res.SYSSCPL.substring(0, 1))
 | 
			
		||||
            ['L', 'H', 'A', 'S'].includes(
 | 
			
		||||
              res.adapterResponse.SYSSCPL.substring(0, 1)
 | 
			
		||||
            )
 | 
			
		||||
              ? 'linux'
 | 
			
		||||
              : 'windows'
 | 
			
		||||
          ]
 | 
			
		||||
 | 
			
		||||
        this.dcAdminGroupList = res.groups
 | 
			
		||||
        this.dcAdminGroupList = res.adapterResponse.groups
 | 
			
		||||
        this.dcAdminGroup = this.dcAdminGroupList[0].GROUPNAME
 | 
			
		||||
 | 
			
		||||
        this.loading = false
 | 
			
		||||
 
 | 
			
		||||
@@ -4,20 +4,23 @@ import { NgVarDirective } from './ng-var.directive'
 | 
			
		||||
import { DragNdropDirective } from './drag-ndrop.directive'
 | 
			
		||||
import { FileDropDirective } from './file-drop.directive'
 | 
			
		||||
import { FileSelectDirective } from './file-select.directive'
 | 
			
		||||
import { StealFocusDirective } from './steal-focus.directive'
 | 
			
		||||
 | 
			
		||||
@NgModule({
 | 
			
		||||
  declarations: [
 | 
			
		||||
    NgVarDirective,
 | 
			
		||||
    DragNdropDirective,
 | 
			
		||||
    FileDropDirective,
 | 
			
		||||
    FileSelectDirective
 | 
			
		||||
    FileSelectDirective,
 | 
			
		||||
    StealFocusDirective
 | 
			
		||||
  ],
 | 
			
		||||
  imports: [CommonModule],
 | 
			
		||||
  exports: [
 | 
			
		||||
    NgVarDirective,
 | 
			
		||||
    DragNdropDirective,
 | 
			
		||||
    FileDropDirective,
 | 
			
		||||
    FileSelectDirective
 | 
			
		||||
    FileSelectDirective,
 | 
			
		||||
    StealFocusDirective
 | 
			
		||||
  ]
 | 
			
		||||
})
 | 
			
		||||
export class DirectivesModule {}
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										17
									
								
								client/src/app/directives/steal-focus.directive.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								client/src/app/directives/steal-focus.directive.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,17 @@
 | 
			
		||||
import { Directive, HostListener } from '@angular/core'
 | 
			
		||||
 | 
			
		||||
@Directive({
 | 
			
		||||
  selector: '[appStealFocus]'
 | 
			
		||||
})
 | 
			
		||||
export class StealFocusDirective {
 | 
			
		||||
  constructor() {}
 | 
			
		||||
 | 
			
		||||
  /**
 | 
			
		||||
   * For some reason newest version of Clarity v17.0.1 is stealing focus when
 | 
			
		||||
   * clicking on the input inside of the clr-tree-view
 | 
			
		||||
   * This is workaround
 | 
			
		||||
   */
 | 
			
		||||
  @HostListener('click', ['$event']) onClick(event: any) {
 | 
			
		||||
    event.target.focus()
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
@@ -277,7 +277,7 @@
 | 
			
		||||
    <div>
 | 
			
		||||
      <button
 | 
			
		||||
        type="button"
 | 
			
		||||
        class="btn btn-outline focusable"
 | 
			
		||||
        class="btn btn-outline focusable mr-5i"
 | 
			
		||||
        (click)="currentRecord!.noLinkOption = false; closeRecordEdit()"
 | 
			
		||||
      >
 | 
			
		||||
        Cancel
 | 
			
		||||
 
 | 
			
		||||
@@ -1,241 +0,0 @@
 | 
			
		||||
.record-edit-modal {
 | 
			
		||||
  .column-entry {
 | 
			
		||||
    display: flex;
 | 
			
		||||
    justify-content: space-between;
 | 
			
		||||
    .name-input-row {
 | 
			
		||||
      width: 100%;
 | 
			
		||||
      max-width: 260px;
 | 
			
		||||
 | 
			
		||||
      .cell-desc {
 | 
			
		||||
        margin-right: 30px;
 | 
			
		||||
        margin-top: 10px;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    .inputs-wrapper {
 | 
			
		||||
      flex: 1;
 | 
			
		||||
      display: flex;
 | 
			
		||||
      align-items: center;
 | 
			
		||||
 | 
			
		||||
      ::ng-deep >*:not(.date-field):not(clr-select-container) {
 | 
			
		||||
        flex: 1;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    p {
 | 
			
		||||
      margin-top: 0px;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    ::ng-deep {
 | 
			
		||||
      .clr-textarea-wrapper {
 | 
			
		||||
        margin-top: 0 !important;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      .clr-form-control {
 | 
			
		||||
        margin-top: 0px !important;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      app-soft-select {
 | 
			
		||||
        display: block;
 | 
			
		||||
        width: 224px;
 | 
			
		||||
        background: #fff;
 | 
			
		||||
        border: 1px solid #999;
 | 
			
		||||
        color: #000;
 | 
			
		||||
        padding: calc(.25rem + 2px) .5rem;
 | 
			
		||||
        border-radius: .125rem;
 | 
			
		||||
        font-size: .541667rem;
 | 
			
		||||
        margin-right: 6px;
 | 
			
		||||
 | 
			
		||||
        input {
 | 
			
		||||
          width: 100%;
 | 
			
		||||
          border: 0;
 | 
			
		||||
          background-color: #fff;
 | 
			
		||||
 | 
			
		||||
          &:focus {
 | 
			
		||||
            background: none;
 | 
			
		||||
            border: 0 !important;
 | 
			
		||||
          }
 | 
			
		||||
 | 
			
		||||
          &::-webkit-outer-spin-button,
 | 
			
		||||
          &::-webkit-inner-spin-button {
 | 
			
		||||
            -webkit-appearance: none;
 | 
			
		||||
            margin: 0;
 | 
			
		||||
          }
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    &:first-child p:first-child {
 | 
			
		||||
      margin-top: 0;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .date-field {
 | 
			
		||||
    position: relative;
 | 
			
		||||
    display: inline-block;
 | 
			
		||||
 | 
			
		||||
    textarea {
 | 
			
		||||
      width: 230px;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    .date-picker {
 | 
			
		||||
      position: absolute;
 | 
			
		||||
      right: 0;
 | 
			
		||||
      top: 4px;
 | 
			
		||||
 | 
			
		||||
      ::ng-deep {
 | 
			
		||||
        // clr-datepicker-view-manager {
 | 
			
		||||
        //   transform: unset !important;
 | 
			
		||||
        //   left: unset !important;
 | 
			
		||||
        //   right: 70px !important;
 | 
			
		||||
        // }
 | 
			
		||||
        .clr-input-group {
 | 
			
		||||
          border: 0 !important;
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .modal-body {
 | 
			
		||||
    padding-bottom: 10px;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  ::ng-deep {
 | 
			
		||||
    clr-select-container {
 | 
			
		||||
      border: 1px solid #999;
 | 
			
		||||
      color: #000;
 | 
			
		||||
      border-radius: .125rem;
 | 
			
		||||
      margin-right: 5px;
 | 
			
		||||
 | 
			
		||||
      .clr-select-wrapper {
 | 
			
		||||
        max-height: unset;
 | 
			
		||||
 | 
			
		||||
        &::after {
 | 
			
		||||
          top: 15px;
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      select {
 | 
			
		||||
        height: auto;
 | 
			
		||||
        padding: 10px;
 | 
			
		||||
        padding-right: 20px;
 | 
			
		||||
        border: 0 !important;
 | 
			
		||||
 | 
			
		||||
        &:focus {
 | 
			
		||||
          background: 0 0 !important;
 | 
			
		||||
        }
 | 
			
		||||
        &:hover {
 | 
			
		||||
          background: transparent;
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    clr-input-container {
 | 
			
		||||
      width: 224px;
 | 
			
		||||
      background: #fff;
 | 
			
		||||
      border: 1px solid #999;
 | 
			
		||||
      color: #000;
 | 
			
		||||
      padding: calc(.25rem + 2px) .5rem;
 | 
			
		||||
      border-radius: .125rem;
 | 
			
		||||
      font-size: .541667rem;
 | 
			
		||||
      margin-right: 6px;
 | 
			
		||||
 | 
			
		||||
      input {
 | 
			
		||||
        width: 100%;
 | 
			
		||||
        border: 0;
 | 
			
		||||
 | 
			
		||||
        &:focus {
 | 
			
		||||
          background: none;
 | 
			
		||||
          border: 0 !important;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        &::-webkit-outer-spin-button,
 | 
			
		||||
        &::-webkit-inner-spin-button {
 | 
			
		||||
          -webkit-appearance: none;
 | 
			
		||||
          margin: 0;
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      &.invalid-data {
 | 
			
		||||
        border-color: red;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    .modal-dialog {
 | 
			
		||||
      width: 80vw;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    .clr-control-container {
 | 
			
		||||
      width: 100%;
 | 
			
		||||
 | 
			
		||||
      textarea {
 | 
			
		||||
        width: 100%;
 | 
			
		||||
        resize: none;
 | 
			
		||||
        border-color: #999;
 | 
			
		||||
 | 
			
		||||
        &.invalid-data {
 | 
			
		||||
          border-color: red;
 | 
			
		||||
          outline: 0;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        &.not-char {
 | 
			
		||||
          font-family: "Lucida Console", Monaco, monospace;
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    .generate-record-url {
 | 
			
		||||
      right: 40px;
 | 
			
		||||
      top: 40px;
 | 
			
		||||
      font-size: 12px;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    .generate-record-url-button {
 | 
			
		||||
      right: 25px;
 | 
			
		||||
      top: 5px;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    .modal-header {
 | 
			
		||||
      padding: 0 0 1rem 0;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    .modal-footer {
 | 
			
		||||
      display: flex;
 | 
			
		||||
      align-items: center;
 | 
			
		||||
      justify-content: space-between;
 | 
			
		||||
 | 
			
		||||
      // height: 65px;
 | 
			
		||||
 | 
			
		||||
      .alert {
 | 
			
		||||
        margin: 0;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.prev-next {
 | 
			
		||||
  display: flex;
 | 
			
		||||
  align-items: center;
 | 
			
		||||
 | 
			
		||||
  p {
 | 
			
		||||
    margin: 0;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  button {
 | 
			
		||||
    margin: 0px 10px;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.focusable {
 | 
			
		||||
  &:focus {
 | 
			
		||||
    box-shadow: 0 0 3px 0px #5aa220;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.entry-input-left-offset {
 | 
			
		||||
  left: -30px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.validation-info-alert {
 | 
			
		||||
  width: 310px
 | 
			
		||||
}
 | 
			
		||||
@@ -1,5 +1,12 @@
 | 
			
		||||
import { KeyValue } from '@angular/common'
 | 
			
		||||
import { Component, EventEmitter, Input, OnInit, Output } from '@angular/core'
 | 
			
		||||
import {
 | 
			
		||||
  Component,
 | 
			
		||||
  EventEmitter,
 | 
			
		||||
  Input,
 | 
			
		||||
  OnInit,
 | 
			
		||||
  Output,
 | 
			
		||||
  ViewEncapsulation
 | 
			
		||||
} from '@angular/core'
 | 
			
		||||
import moment from 'moment'
 | 
			
		||||
import { ValidateFilterSASResponse } from 'src/app/models/sas/validate-filter.model'
 | 
			
		||||
import { QueryClause } from 'src/app/models/TableData'
 | 
			
		||||
@@ -16,7 +23,8 @@ import { EditRecordModal } from '../../models/EditRecordModal'
 | 
			
		||||
@Component({
 | 
			
		||||
  selector: 'app-edit-record',
 | 
			
		||||
  templateUrl: './edit-record.component.html',
 | 
			
		||||
  styleUrls: ['./edit-record.component.scss']
 | 
			
		||||
  styleUrls: ['./edit-record.component.scss'],
 | 
			
		||||
  encapsulation: ViewEncapsulation.None
 | 
			
		||||
})
 | 
			
		||||
export class EditRecordComponent implements OnInit {
 | 
			
		||||
  @Input() currentRecord!: EditRecordModal
 | 
			
		||||
 
 | 
			
		||||
@@ -1,8 +0,0 @@
 | 
			
		||||
:host {
 | 
			
		||||
    display: block;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
p {
 | 
			
		||||
    margin: 0;
 | 
			
		||||
    text-align: center;
 | 
			
		||||
}
 | 
			
		||||
@@ -1,4 +1,4 @@
 | 
			
		||||
import { Component, OnInit } from '@angular/core'
 | 
			
		||||
import { Component, OnInit, ViewEncapsulation } from '@angular/core'
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Goal of this component is to recieve array of strings where every element is one state
 | 
			
		||||
@@ -10,7 +10,8 @@ import { Component, OnInit } from '@angular/core'
 | 
			
		||||
@Component({
 | 
			
		||||
  selector: 'app-upload-stater',
 | 
			
		||||
  templateUrl: './upload-stater.component.html',
 | 
			
		||||
  styleUrls: ['./upload-stater.component.scss']
 | 
			
		||||
  styleUrls: ['./upload-stater.component.scss'],
 | 
			
		||||
  encapsulation: ViewEncapsulation.None
 | 
			
		||||
})
 | 
			
		||||
export class UploadStaterComponent implements OnInit {
 | 
			
		||||
  public statesList: string[] = [] //States appended to be displayed
 | 
			
		||||
 
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
    appFileDrop
 | 
			
		||||
    (fileOver)="fileOverBase($event)"
 | 
			
		||||
    [uploader]="uploader"
 | 
			
		||||
    (fileDrop)="getFileDesc($event, true)"
 | 
			
		||||
    (fileDrop)="attachFile($event, true)"
 | 
			
		||||
    [clrModalSize]="'xl'"
 | 
			
		||||
    [clrModalStaticBackdrop]="false"
 | 
			
		||||
    [clrModalClosable]="excelUploadState === 'Validating-DQ'"
 | 
			
		||||
@@ -36,7 +36,7 @@
 | 
			
		||||
        <div class="clr-row card-block mt-15 d-flex justify-content-between">
 | 
			
		||||
          <div class="clr-col-md-auto">
 | 
			
		||||
            <div class="encoding-block">
 | 
			
		||||
              <clr-radio-container class="mt-0-i" clrInline>
 | 
			
		||||
              <clr-radio-container class="mt-0" clrInline>
 | 
			
		||||
                <clr-radio-wrapper>
 | 
			
		||||
                  <input
 | 
			
		||||
                    type="radio"
 | 
			
		||||
@@ -81,7 +81,7 @@
 | 
			
		||||
              type="file"
 | 
			
		||||
              appFileSelect
 | 
			
		||||
              [uploader]="uploader"
 | 
			
		||||
              (change)="getFileDesc($event)"
 | 
			
		||||
              (change)="attachFile($event)"
 | 
			
		||||
            />
 | 
			
		||||
          </div>
 | 
			
		||||
 | 
			
		||||
@@ -92,7 +92,7 @@
 | 
			
		||||
            <button
 | 
			
		||||
              [disabled]="true"
 | 
			
		||||
              class="btnView btn btn-sm btn-success profile-buttons w-100"
 | 
			
		||||
              (click)="getFile()"
 | 
			
		||||
              (click)="uploadParsedFiles()"
 | 
			
		||||
            >
 | 
			
		||||
              Upload
 | 
			
		||||
            </button>
 | 
			
		||||
@@ -164,19 +164,28 @@
 | 
			
		||||
      <div
 | 
			
		||||
        class="card-header clr-row buttonBar headerBar clr-flex-md-row clr-justify-content-center clr-justify-content-lg-end"
 | 
			
		||||
      >
 | 
			
		||||
        <div *ngIf="tableTrue" class="clr-col-12 clr-col-lg-4 backBtn">
 | 
			
		||||
          <span class="btn btn-sm" [routerLink]="['/home']">
 | 
			
		||||
            <clr-icon shape="caret" dir="left" size="20"></clr-icon>Back to
 | 
			
		||||
            table selection
 | 
			
		||||
        <div
 | 
			
		||||
          *ngIf="tableTrue"
 | 
			
		||||
          class="clr-col-12 clr-col-md-3 clr-col-lg-4 backBtn"
 | 
			
		||||
        >
 | 
			
		||||
          <span
 | 
			
		||||
            class="btn icon-collapse btn-sm btn-icon btn-dimmed"
 | 
			
		||||
            [routerLink]="['/home']"
 | 
			
		||||
          >
 | 
			
		||||
            <clr-icon shape="caret" dir="left" size="20"></clr-icon>
 | 
			
		||||
            <span class="text">Back to table selection</span>
 | 
			
		||||
          </span>
 | 
			
		||||
          <span (click)="viewboxManager()" class="btn btn-sm viewbox-open">
 | 
			
		||||
          <span
 | 
			
		||||
            (click)="viewboxManager()"
 | 
			
		||||
            class="btn icon-collapse btn-sm btn-icon btn-dimmed viewbox-open"
 | 
			
		||||
          >
 | 
			
		||||
            <clr-icon shape="view-cards" size="20"></clr-icon>
 | 
			
		||||
            Viewboxes
 | 
			
		||||
            <span class="text">Viewboxes</span>
 | 
			
		||||
          </span>
 | 
			
		||||
        </div>
 | 
			
		||||
 | 
			
		||||
        <div
 | 
			
		||||
          class="clr-col-12 clr-col-lg-4 d-flex flex-column align-items-center"
 | 
			
		||||
          class="clr-col-12 clr-col-md-5 clr-col-lg-4 d-flex flex-column align-items-center"
 | 
			
		||||
          [class.clr-col-lg-12]="!tableTrue"
 | 
			
		||||
        >
 | 
			
		||||
          <h4
 | 
			
		||||
@@ -184,13 +193,14 @@
 | 
			
		||||
              libName: (libds?.split('.'))![0],
 | 
			
		||||
              tableName: (libds?.split('.'))![1]
 | 
			
		||||
            } as libdsParsed"
 | 
			
		||||
            class="editor-title text-center mt-0-i"
 | 
			
		||||
            class="editor-title text-center mt-0"
 | 
			
		||||
          >
 | 
			
		||||
            <clr-tooltip>
 | 
			
		||||
              <clr-icon
 | 
			
		||||
                clrTooltipTrigger
 | 
			
		||||
                (click)="datasetInfo = true"
 | 
			
		||||
                shape="info-circle"
 | 
			
		||||
                aria-label="View dataset meta info"
 | 
			
		||||
                class="is-highlight cursor-pointer"
 | 
			
		||||
                size="24"
 | 
			
		||||
              ></clr-icon>
 | 
			
		||||
@@ -231,34 +241,37 @@
 | 
			
		||||
            </ng-container>
 | 
			
		||||
          </h4>
 | 
			
		||||
        </div>
 | 
			
		||||
        <div *ngIf="tableTrue" class="clr-col-12 clr-col-lg-4 btnCtrl">
 | 
			
		||||
        <div
 | 
			
		||||
          *ngIf="tableTrue"
 | 
			
		||||
          class="clr-col-12 clr-col-md-4 clr-col-lg-4 btnCtrl"
 | 
			
		||||
        >
 | 
			
		||||
          <ng-container *ngIf="hotTable.readOnly && !uploadPreview">
 | 
			
		||||
            <button
 | 
			
		||||
              type="button"
 | 
			
		||||
              class="btnView btn btn-sm btn-icon btn-block"
 | 
			
		||||
              class="btnView btn icon-collapse btn-sm btn-icon btn-block btn-dimmed"
 | 
			
		||||
              (click)="openQb()"
 | 
			
		||||
            >
 | 
			
		||||
              <clr-icon shape="filter"></clr-icon>
 | 
			
		||||
              <span>Filter</span>
 | 
			
		||||
              <span class="text">Filter</span>
 | 
			
		||||
            </button>
 | 
			
		||||
 | 
			
		||||
            <button
 | 
			
		||||
              type="button"
 | 
			
		||||
              class="btn btn-sm btn-primary btn-block"
 | 
			
		||||
              class="btn icon-collapse btn-sm btn-primary btn-block"
 | 
			
		||||
              (click)="editTable()"
 | 
			
		||||
            >
 | 
			
		||||
              <clr-icon shape="note"></clr-icon>
 | 
			
		||||
              <span>Edit</span>
 | 
			
		||||
              <span class="text">Edit</span>
 | 
			
		||||
            </button>
 | 
			
		||||
 | 
			
		||||
            <button
 | 
			
		||||
              *ngIf="!columnLevelSecurityFlag"
 | 
			
		||||
              (click)="onShowUploadModal()"
 | 
			
		||||
              type="button"
 | 
			
		||||
              class="btn btn-sm btn-success btn-block mr-0"
 | 
			
		||||
              class="btn icon-collapse btn-sm btn-success btn-block mr-0"
 | 
			
		||||
            >
 | 
			
		||||
              <clr-icon shape="upload"></clr-icon>
 | 
			
		||||
              <span>Upload</span>
 | 
			
		||||
              <span class="text">Upload</span>
 | 
			
		||||
            </button>
 | 
			
		||||
          </ng-container>
 | 
			
		||||
 | 
			
		||||
@@ -362,8 +375,8 @@
 | 
			
		||||
            <ng-container *ngIf="!getdataError">
 | 
			
		||||
              <span class="spinner"> Loading... </span>
 | 
			
		||||
 | 
			
		||||
              <div>
 | 
			
		||||
                <h3>Loading table</h3>
 | 
			
		||||
              <div class="mt-10">
 | 
			
		||||
                <p cds-text="section">Loading table</p>
 | 
			
		||||
              </div>
 | 
			
		||||
            </ng-container>
 | 
			
		||||
 | 
			
		||||
@@ -372,8 +385,8 @@
 | 
			
		||||
                <clr-icon shape="error-standard" class="error-icon"></clr-icon>
 | 
			
		||||
              </span>
 | 
			
		||||
 | 
			
		||||
              <div>
 | 
			
		||||
                <h3>Loading table error</h3>
 | 
			
		||||
              <div class="mt-10">
 | 
			
		||||
                <p cds-text="section">Loading table error</p>
 | 
			
		||||
              </div>
 | 
			
		||||
            </ng-container>
 | 
			
		||||
          </div>
 | 
			
		||||
@@ -398,6 +411,7 @@
 | 
			
		||||
              hotId="hotInstance"
 | 
			
		||||
              id="hotTable"
 | 
			
		||||
              class="edit-hot"
 | 
			
		||||
              className="htDark"
 | 
			
		||||
              [class.hidden]="hotTable.hidden"
 | 
			
		||||
              [licenseKey]="hotTable.licenseKey"
 | 
			
		||||
            >
 | 
			
		||||
@@ -486,11 +500,15 @@
 | 
			
		||||
                    support@datacontroller.io</span
 | 
			
		||||
                  >
 | 
			
		||||
                  <div *ngIf="tableTrue" class="clr-offset-md-2 clr-col-md-8">
 | 
			
		||||
                    <div class="form-group">
 | 
			
		||||
                      <label for="formFields_8">Message</label>
 | 
			
		||||
                    <div class="text-area-full-width">
 | 
			
		||||
                      <label for="formFields_8" class="mb-5 d-block"
 | 
			
		||||
                        >Message</label
 | 
			
		||||
                      >
 | 
			
		||||
                      <textarea
 | 
			
		||||
                        clrTextarea
 | 
			
		||||
                        [(ngModel)]="message"
 | 
			
		||||
                        [disabled]="!validationDone"
 | 
			
		||||
                        tabindex="0"
 | 
			
		||||
                        [value]="
 | 
			
		||||
                          !validationDone
 | 
			
		||||
                            ? 'Please wait while we validate ' +
 | 
			
		||||
@@ -498,10 +516,9 @@
 | 
			
		||||
                              ' cells.'
 | 
			
		||||
                            : ''
 | 
			
		||||
                        "
 | 
			
		||||
                        class="w-100"
 | 
			
		||||
                        class="submit-reason"
 | 
			
		||||
                        type="text"
 | 
			
		||||
                        id="formFields_8"
 | 
			
		||||
                        rows="5"
 | 
			
		||||
                      ></textarea>
 | 
			
		||||
                    </div>
 | 
			
		||||
                    <!-- TODO:approvers list -->
 | 
			
		||||
@@ -520,6 +537,7 @@
 | 
			
		||||
                    [disabled]="!validationDone"
 | 
			
		||||
                    type="submit"
 | 
			
		||||
                    class="btn btn-sm btn-success-outline m-0"
 | 
			
		||||
                    tabindex="0"
 | 
			
		||||
                    (click)="saveTable(hotTable.data)"
 | 
			
		||||
                  >
 | 
			
		||||
                    Submit
 | 
			
		||||
@@ -528,6 +546,7 @@
 | 
			
		||||
                    id="cancelSubmitBtn"
 | 
			
		||||
                    type="button"
 | 
			
		||||
                    class="btn btn-sm btn-outline"
 | 
			
		||||
                    tabindex="0"
 | 
			
		||||
                    (click)="cancelSubmit(); submit = false; validationDone = 0"
 | 
			
		||||
                  >
 | 
			
		||||
                    Cancel
 | 
			
		||||
@@ -558,7 +577,7 @@
 | 
			
		||||
              <button
 | 
			
		||||
                type="button"
 | 
			
		||||
                class="btn btn-sm btn-primary"
 | 
			
		||||
                (click)="getFile(); submitLimitNotice = false"
 | 
			
		||||
                (click)="uploadParsedFiles(); submitLimitNotice = false"
 | 
			
		||||
              >
 | 
			
		||||
                Submit
 | 
			
		||||
              </button>
 | 
			
		||||
 
 | 
			
		||||
@@ -1,227 +0,0 @@
 | 
			
		||||
.card {
 | 
			
		||||
  margin-top: 0;
 | 
			
		||||
  border: 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.buttonBar {
 | 
			
		||||
  padding: 2px 10px 2px 10px;
 | 
			
		||||
  align-items: center;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.testRed {
 | 
			
		||||
  color: white;
 | 
			
		||||
  background: rgba(255,0,0, 0.8) !important;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
hot-table {
 | 
			
		||||
  ::ng-deep {
 | 
			
		||||
    .firstColumnHeaderStyle button.changeType {
 | 
			
		||||
      display: none;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    .handsontable tbody th.ht__highlight, .handsontable thead th.ht__highlight {
 | 
			
		||||
      &.primaryKeyHeaderStyle {
 | 
			
		||||
        background: #306b00b0;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    .primaryKeyHeaderStyle {
 | 
			
		||||
      background: #306b006e;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    th.readonlyCell {
 | 
			
		||||
      div {
 | 
			
		||||
        opacity: 0.4;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    td.readonlyCell {
 | 
			
		||||
      opacity: 0.5
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.infoBar {
 | 
			
		||||
  margin-top:14px;
 | 
			
		||||
  background: #495967;
 | 
			
		||||
  color: white;
 | 
			
		||||
  text-align:center;
 | 
			
		||||
  padding: 3px;
 | 
			
		||||
  font-size: 16px;
 | 
			
		||||
 | 
			
		||||
  height: 30px;
 | 
			
		||||
 | 
			
		||||
  text-overflow: ellipsis;
 | 
			
		||||
  overflow: hidden;
 | 
			
		||||
  white-space: nowrap;
 | 
			
		||||
 | 
			
		||||
  span {
 | 
			
		||||
    width: 80%;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  &:hover {
 | 
			
		||||
    height: unset;
 | 
			
		||||
    white-space: normal;
 | 
			
		||||
 | 
			
		||||
    span {
 | 
			
		||||
      width: unset;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.pkHeader {
 | 
			
		||||
  background: #687682;
 | 
			
		||||
  color: #fff;
 | 
			
		||||
  margin: -1px -1px -1px -1px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.headerBar {
 | 
			
		||||
  //  padding: 13px 0px 14px 0px;
 | 
			
		||||
  -webkit-box-align: center;
 | 
			
		||||
  -ms-flex-align: center;
 | 
			
		||||
  align-items: center;
 | 
			
		||||
  background: #ffffff;
 | 
			
		||||
  background: #f5f6fe;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.error-icon {
 | 
			
		||||
  width: 30px;
 | 
			
		||||
  height: 30px;
 | 
			
		||||
  color: red;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.btnCtrl {
 | 
			
		||||
  display:flex;
 | 
			
		||||
  justify-content:flex-end;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.card-header {
 | 
			
		||||
  border-bottom: 1px solid transparent;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.hidden {
 | 
			
		||||
  visibility: hidden;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.my-drop-zone {
 | 
			
		||||
  border: solid 1px lightgray;
 | 
			
		||||
  border-radius: 10px;
 | 
			
		||||
  background: whitesmoke;
 | 
			
		||||
  box-shadow: inset 0px 0px 4px 2px #a7a5a52b;
 | 
			
		||||
  height: 50vh;
 | 
			
		||||
}
 | 
			
		||||
.nv-file-over {
 | 
			
		||||
  border: solid 2px green;
 | 
			
		||||
} /* Default class applied to drop zones on over */
 | 
			
		||||
 | 
			
		||||
.file-drop-text{
 | 
			
		||||
  text-align: center;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.nv-file-over {
 | 
			
		||||
  border: solid 2px green;
 | 
			
		||||
} /* Default class applied to drop zones on over */
 | 
			
		||||
 | 
			
		||||
.file-drop-text{
 | 
			
		||||
  text-align: center;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@media screen and (max-width: 768px) {
 | 
			
		||||
 | 
			
		||||
  .progresStatic {
 | 
			
		||||
    margin-top:9px!important;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .progress, .progress-static {
 | 
			
		||||
    width: calc(100% - 14px);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.hotEditor {
 | 
			
		||||
  position: relative;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.excel-parsing {
 | 
			
		||||
  display: flex;
 | 
			
		||||
  flex-direction: column;
 | 
			
		||||
  align-items: center;
 | 
			
		||||
 | 
			
		||||
  position: relative;
 | 
			
		||||
 | 
			
		||||
  .details {
 | 
			
		||||
    margin: 0;
 | 
			
		||||
    position: absolute;
 | 
			
		||||
    top: -45px;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.edit-record-spinner {
 | 
			
		||||
  display: flex;
 | 
			
		||||
  justify-content: center;
 | 
			
		||||
  align-items: center;
 | 
			
		||||
  background: rgba(255, 255, 255, 0.6);
 | 
			
		||||
  position: absolute;
 | 
			
		||||
  top: 0px;
 | 
			
		||||
  bottom: 0px;
 | 
			
		||||
  width: 100%;
 | 
			
		||||
  z-index: 500;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@media screen and (max-width: 480px) {
 | 
			
		||||
 | 
			
		||||
   .progresStatic {
 | 
			
		||||
      margin-top:32px!important;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
   .card-block, .card-footer {
 | 
			
		||||
    padding: 10px 0px 0px 0px;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.content-area {
 | 
			
		||||
  padding: 0 0.8rem 0.8rem 0.8rem !important;
 | 
			
		||||
  padding-top: 0;
 | 
			
		||||
 | 
			
		||||
  // .card {
 | 
			
		||||
  //   min-height: calc(100vh - 160px);
 | 
			
		||||
  // }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.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;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#submitBtn, #cancelSubmitBtn {
 | 
			
		||||
  width: 150px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.view-table {
 | 
			
		||||
  font-size: inherit !important;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// FIXME
 | 
			
		||||
// Let's leave it here for a reference if there
 | 
			
		||||
// is an issue with viewboxes/filter modal overlaying
 | 
			
		||||
// we will remove it if no issues found
 | 
			
		||||
// .filter-modal {
 | 
			
		||||
//   z-index: 1210;
 | 
			
		||||
// }
 | 
			
		||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							@@ -11,9 +11,11 @@ export const errorRenderer = (
 | 
			
		||||
  value: any,
 | 
			
		||||
  cellProperties: any
 | 
			
		||||
) => {
 | 
			
		||||
  addDarkClass(td)
 | 
			
		||||
 | 
			
		||||
  td.innerHTML = `${
 | 
			
		||||
    value ? value.toString() : ''
 | 
			
		||||
  } <clr-icon shape="exclamation-circle" status="warning"></clr-icon>`
 | 
			
		||||
  } <cds-icon shape="exclamation-triangle" status="warning"></cds-icon>`
 | 
			
		||||
 | 
			
		||||
  return td
 | 
			
		||||
}
 | 
			
		||||
@@ -31,6 +33,8 @@ export const noSpinnerRenderer = (
 | 
			
		||||
  value: any,
 | 
			
		||||
  cellProperties: any
 | 
			
		||||
) => {
 | 
			
		||||
  addDarkClass(td)
 | 
			
		||||
 | 
			
		||||
  td.innerHTML = value ? value : ''
 | 
			
		||||
 | 
			
		||||
  return td
 | 
			
		||||
@@ -50,9 +54,20 @@ export const spinnerRenderer = (
 | 
			
		||||
  value: any,
 | 
			
		||||
  cellProperties: any
 | 
			
		||||
) => {
 | 
			
		||||
  addDarkClass(td)
 | 
			
		||||
 | 
			
		||||
  td.innerHTML = `${
 | 
			
		||||
    value ? value.toString() : ''
 | 
			
		||||
  } <span class="spinner spinner-sm vertical-align-middle"></span>`
 | 
			
		||||
 | 
			
		||||
  return td
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Adds a htDark class to a TD element if not existing
 | 
			
		||||
 */
 | 
			
		||||
const addDarkClass = (td: any) => {
 | 
			
		||||
  if (!td.classList.contains('htDark')) {
 | 
			
		||||
    td.classList.add('htDark')
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -3,6 +3,7 @@
 | 
			
		||||
    <clr-tree-node *ngIf="groups" class="search-node">
 | 
			
		||||
      <div class="tree-search-wrapper">
 | 
			
		||||
        <input
 | 
			
		||||
          appStealFocus
 | 
			
		||||
          clrInput
 | 
			
		||||
          #searchLibTreeInput
 | 
			
		||||
          placeholder="Filter by Groups"
 | 
			
		||||
@@ -27,7 +28,7 @@
 | 
			
		||||
      <clr-tree-node
 | 
			
		||||
        (click)="groupOnClick(group)"
 | 
			
		||||
        *ngIf="!group['hidden']"
 | 
			
		||||
        [class.table-active]="group.GROUPURI === groupUri"
 | 
			
		||||
        [class.active]="group.GROUPURI === groupUri"
 | 
			
		||||
      >
 | 
			
		||||
        <p class="m-0 cursor-pointer list-padding">
 | 
			
		||||
          <clr-icon shape="users"></clr-icon>
 | 
			
		||||
 
 | 
			
		||||
@@ -1,52 +0,0 @@
 | 
			
		||||
.sidebar-height{
 | 
			
		||||
    height: 100%;
 | 
			
		||||
}
 | 
			
		||||
.group-info-text{
 | 
			
		||||
    display: inline;    
 | 
			
		||||
    font-size: 20px;
 | 
			
		||||
}
 | 
			
		||||
.group-info{
 | 
			
		||||
    background-color: #f9f9f9;
 | 
			
		||||
    border: 1px solid #a7a7a7;
 | 
			
		||||
    border-radius: 3px;
 | 
			
		||||
    box-shadow: 0px 2px 5px #dad7d7;
 | 
			
		||||
}
 | 
			
		||||
.group-info td{
 | 
			
		||||
    text-align: center;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.group-data{
 | 
			
		||||
  background-color: #f9f9f9;
 | 
			
		||||
  border: 1px solid #a7a7a7;
 | 
			
		||||
  border-radius: 3px;
 | 
			
		||||
  box-shadow: 0px 2px 5px #dad7d7;
 | 
			
		||||
}
 | 
			
		||||
.group-data{
 | 
			
		||||
  min-height: auto;
 | 
			
		||||
  h3, h5{
 | 
			
		||||
      text-align: center;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.member-table{
 | 
			
		||||
  background-color: #f9f9f9;
 | 
			
		||||
  width: 100%;
 | 
			
		||||
}
 | 
			
		||||
.member-table thead{
 | 
			
		||||
  background-color: #dadada;
 | 
			
		||||
}
 | 
			
		||||
.member-table tbody{
 | 
			
		||||
  tr:hover{
 | 
			
		||||
      background-color: #e6e6e6;
 | 
			
		||||
      cursor: pointer;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
}
 | 
			
		||||
.table-container{
 | 
			
		||||
  overflow: auto;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@media screen and (max-width: 768px){
 | 
			
		||||
  .group-data{
 | 
			
		||||
    min-height: unset !important;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,11 +1,12 @@
 | 
			
		||||
import { Location } from '@angular/common'
 | 
			
		||||
import { Component, OnInit } from '@angular/core'
 | 
			
		||||
import { Component, OnInit, ViewEncapsulation } from '@angular/core'
 | 
			
		||||
import { ActivatedRoute, Router } from '@angular/router'
 | 
			
		||||
import { SASjsConfig } from '@sasjs/adapter'
 | 
			
		||||
import { ServerType } from '@sasjs/utils/types/serverType'
 | 
			
		||||
import { HelperService } from '../services/helper.service'
 | 
			
		||||
import { SasService } from '../services/sas.service'
 | 
			
		||||
import { globals } from '../_globals'
 | 
			
		||||
import { RequestWrapperResponse } from '../models/request-wrapper/RequestWrapperResponse'
 | 
			
		||||
 | 
			
		||||
@Component({
 | 
			
		||||
  selector: 'app-group',
 | 
			
		||||
@@ -13,7 +14,8 @@ import { globals } from '../_globals'
 | 
			
		||||
  styleUrls: ['./group.component.scss'],
 | 
			
		||||
  host: {
 | 
			
		||||
    class: 'content-container'
 | 
			
		||||
  }
 | 
			
		||||
  },
 | 
			
		||||
  encapsulation: ViewEncapsulation.None
 | 
			
		||||
})
 | 
			
		||||
export class GroupComponent implements OnInit {
 | 
			
		||||
  public groups: Array<any> | undefined
 | 
			
		||||
@@ -82,11 +84,13 @@ export class GroupComponent implements OnInit {
 | 
			
		||||
              globals.usernav.groupList = groups
 | 
			
		||||
            })
 | 
			
		||||
        } else {
 | 
			
		||||
          this.sasService.request('public/getgroups', null).then((res: any) => {
 | 
			
		||||
            this.loading = false
 | 
			
		||||
            this.groups = res.groups
 | 
			
		||||
            globals.usernav.groupList = res.groups
 | 
			
		||||
          })
 | 
			
		||||
          this.sasService
 | 
			
		||||
            .request('public/getgroups', null)
 | 
			
		||||
            .then((res: RequestWrapperResponse) => {
 | 
			
		||||
              this.loading = false
 | 
			
		||||
              this.groups = res.adapterResponse.groups
 | 
			
		||||
              globals.usernav.groupList = res.adapterResponse.groups
 | 
			
		||||
            })
 | 
			
		||||
        }
 | 
			
		||||
      } else {
 | 
			
		||||
        this.groups = globals.usernav.groupList
 | 
			
		||||
@@ -128,14 +132,15 @@ export class GroupComponent implements OnInit {
 | 
			
		||||
          let data = { iwant: [{ groupid: this.paramURI }] }
 | 
			
		||||
          this.sasService
 | 
			
		||||
            .request('usernav/usermembersbygroup', data)
 | 
			
		||||
            .then((res: any) => {
 | 
			
		||||
              this.groupMembers = res.sasmembers
 | 
			
		||||
              this.groupMemberCount = res.sasmembers.length
 | 
			
		||||
              if (res.sasmembers[0] !== undefined) {
 | 
			
		||||
            .then((res: RequestWrapperResponse) => {
 | 
			
		||||
              this.groupMembers = res.adapterResponse.sasmembers
 | 
			
		||||
              this.groupMemberCount = res.adapterResponse.sasmembers.length
 | 
			
		||||
              if (res.adapterResponse.sasmembers[0] !== undefined) {
 | 
			
		||||
                this.loading = false
 | 
			
		||||
                this.groupUri = res.sasmembers[0].URIMEM || this.paramURI
 | 
			
		||||
                this.groupName = res.sasmembers[0].GROUPNAME
 | 
			
		||||
                this.groupDesc = res.sasmembers[0].GROUPDESC
 | 
			
		||||
                this.groupUri =
 | 
			
		||||
                  res.adapterResponse.sasmembers[0].URIMEM || this.paramURI
 | 
			
		||||
                this.groupName = res.adapterResponse.sasmembers[0].GROUPNAME
 | 
			
		||||
                this.groupDesc = res.adapterResponse.sasmembers[0].GROUPDESC
 | 
			
		||||
 | 
			
		||||
                if (!this.groupName) {
 | 
			
		||||
                  this.groupName = this.paramURI
 | 
			
		||||
@@ -202,13 +207,13 @@ export class GroupComponent implements OnInit {
 | 
			
		||||
 | 
			
		||||
      this.sasService
 | 
			
		||||
        .request('usernav/usermembersbygroup', data)
 | 
			
		||||
        .then((res: any) => {
 | 
			
		||||
        .then((res: RequestWrapperResponse) => {
 | 
			
		||||
          this.loading = false
 | 
			
		||||
          this.groupUri = group.GROUPURI
 | 
			
		||||
          this.groupName = group.GROUPNAME
 | 
			
		||||
          this.groupDesc = group.GROUPDESC
 | 
			
		||||
          this.groupMembers = res.sasmembers
 | 
			
		||||
          this.groupMemberCount = res.sasmembers.length
 | 
			
		||||
          this.groupMembers = res.adapterResponse.sasmembers
 | 
			
		||||
          this.groupMemberCount = res.adapterResponse.sasmembers.length
 | 
			
		||||
        })
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 
 | 
			
		||||
@@ -3,6 +3,7 @@ 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'
 | 
			
		||||
import { MultiDatasetModule } from '../multi-dataset/multi-dataset.module'
 | 
			
		||||
 | 
			
		||||
const routes: Routes = [
 | 
			
		||||
  {
 | 
			
		||||
@@ -11,7 +12,8 @@ const routes: Routes = [
 | 
			
		||||
    children: [
 | 
			
		||||
      { path: '', pathMatch: 'full', redirectTo: 'tables' },
 | 
			
		||||
      { path: 'tables', component: HomeComponent },
 | 
			
		||||
      { path: 'files', loadChildren: () => XLMapModule }
 | 
			
		||||
      { path: 'excel-maps', loadChildren: () => XLMapModule },
 | 
			
		||||
      { path: 'multi-load', loadChildren: () => MultiDatasetModule }
 | 
			
		||||
    ]
 | 
			
		||||
  }
 | 
			
		||||
]
 | 
			
		||||
 
 | 
			
		||||
@@ -4,6 +4,7 @@
 | 
			
		||||
      <div class="tree-search-wrapper">
 | 
			
		||||
        <input
 | 
			
		||||
          clrInput
 | 
			
		||||
          appStealFocus
 | 
			
		||||
          #searchLibTreeInput
 | 
			
		||||
          placeholder="Libraries"
 | 
			
		||||
          name="input"
 | 
			
		||||
@@ -46,6 +47,7 @@
 | 
			
		||||
        <clr-tree-node *ngIf="library['tables']" class="search-node">
 | 
			
		||||
          <div class="tree-search-wrapper">
 | 
			
		||||
            <input
 | 
			
		||||
              appStealFocus
 | 
			
		||||
              clrInput
 | 
			
		||||
              #searchTreeInput
 | 
			
		||||
              placeholder="Tables"
 | 
			
		||||
@@ -85,7 +87,7 @@
 | 
			
		||||
              (click)="!tableLocked ? onTableClick(libTable, library) : ''"
 | 
			
		||||
              class="clr-treenode-link"
 | 
			
		||||
              [class.dc-locked-control]="tableLocked"
 | 
			
		||||
              [class.table-active]="libTabActive(library.LIBRARYREF, libTable)"
 | 
			
		||||
              [class.active]="libTabActive(library.LIBRARYREF, libTable)"
 | 
			
		||||
            >
 | 
			
		||||
              <ng-container [ngSwitch]="libTable.includes('-FC')">
 | 
			
		||||
                <clr-icon *ngSwitchCase="true" shape="bolt"></clr-icon>
 | 
			
		||||
@@ -121,17 +123,25 @@
 | 
			
		||||
    </div>
 | 
			
		||||
 | 
			
		||||
    <div *ngIf="!loading" class="no-table-selected">
 | 
			
		||||
      <clr-icon
 | 
			
		||||
        shape="warning-standard"
 | 
			
		||||
        size="60"
 | 
			
		||||
        class="is-info icon-dc-fill"
 | 
			
		||||
      ></clr-icon>
 | 
			
		||||
      <h3 *ngIf="treeNodeLibraries?.length! > 0" class="text-center color-gray">
 | 
			
		||||
      <img
 | 
			
		||||
        src="images/select-table.png"
 | 
			
		||||
        class="select-table-icon"
 | 
			
		||||
        alt="select table icon"
 | 
			
		||||
      />
 | 
			
		||||
      <p
 | 
			
		||||
        *ngIf="treeNodeLibraries?.length! > 0"
 | 
			
		||||
        class="text-center color-gray mt-10"
 | 
			
		||||
        cds-text="section"
 | 
			
		||||
      >
 | 
			
		||||
        Please select a table
 | 
			
		||||
      </h3>
 | 
			
		||||
      <h3 *ngIf="treeNodeLibraries?.length! < 1" class="text-center color-gray">
 | 
			
		||||
      </p>
 | 
			
		||||
      <p
 | 
			
		||||
        *ngIf="treeNodeLibraries?.length! < 1"
 | 
			
		||||
        class="text-center color-gray mt-10"
 | 
			
		||||
        cds-text="section"
 | 
			
		||||
      >
 | 
			
		||||
        No Editable Tables Configured
 | 
			
		||||
      </h3>
 | 
			
		||||
      </p>
 | 
			
		||||
    </div>
 | 
			
		||||
  </div>
 | 
			
		||||
</div>
 | 
			
		||||
 
 | 
			
		||||
@@ -1,32 +0,0 @@
 | 
			
		||||
clr-tree-node button {
 | 
			
		||||
  white-space: nowrap;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.card-block {
 | 
			
		||||
  height: 100%;
 | 
			
		||||
  padding: 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.no-table-selected {
 | 
			
		||||
  position: relative;
 | 
			
		||||
  height: 100%;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
::ng-deep {
 | 
			
		||||
  // .badge.badge-info {
 | 
			
		||||
  //     background: #6a9235!important;
 | 
			
		||||
  //     color: #fff;
 | 
			
		||||
  // }
 | 
			
		||||
  clr-icon.is-blue, clr-icon.is-info {
 | 
			
		||||
    fill: #6a9235;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.spinner-wrapper-fullpage {
 | 
			
		||||
  display: flex;
 | 
			
		||||
  justify-content: center;
 | 
			
		||||
  align-items: center;
 | 
			
		||||
 | 
			
		||||
  width: 100%;
 | 
			
		||||
  height: 100%;
 | 
			
		||||
}
 | 
			
		||||
@@ -3,7 +3,7 @@
 | 
			
		||||
 * This software is released under MIT license.
 | 
			
		||||
 * The full license information can be found in LICENSE in the root directory of this project.
 | 
			
		||||
 */
 | 
			
		||||
import { Component, AfterContentInit } from '@angular/core'
 | 
			
		||||
import { Component, AfterContentInit, ViewEncapsulation } from '@angular/core'
 | 
			
		||||
import { Router } from '@angular/router'
 | 
			
		||||
import { ActivatedRoute } from '@angular/router'
 | 
			
		||||
import { globals } from '../_globals'
 | 
			
		||||
@@ -18,7 +18,8 @@ import { LicenceService } from '../services/licence.service'
 | 
			
		||||
  templateUrl: './home.component.html',
 | 
			
		||||
  host: {
 | 
			
		||||
    class: 'content-container'
 | 
			
		||||
  }
 | 
			
		||||
  },
 | 
			
		||||
  encapsulation: ViewEncapsulation.None
 | 
			
		||||
})
 | 
			
		||||
export class HomeComponent implements AfterContentInit {
 | 
			
		||||
  public treeNodeLibraries: Array<any> | null = null
 | 
			
		||||
 
 | 
			
		||||
@@ -1,51 +0,0 @@
 | 
			
		||||
:host {
 | 
			
		||||
  height: calc(100% - 96px);
 | 
			
		||||
  padding: 20px 20px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.card {
 | 
			
		||||
  margin-top: 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.key-error {
 | 
			
		||||
  font-size: 16px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.misskey {
 | 
			
		||||
  color: #E74C3C;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.license-key-form, .activation-key-form {
 | 
			
		||||
  padding: 0;
 | 
			
		||||
 | 
			
		||||
  .clr-control-container {
 | 
			
		||||
    width: 100%;
 | 
			
		||||
 | 
			
		||||
    textarea {
 | 
			
		||||
      width: 100%;
 | 
			
		||||
      height: 170px;
 | 
			
		||||
      max-height: 170px;
 | 
			
		||||
      min-height: 170px;
 | 
			
		||||
      resize: none;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.apply-keys {
 | 
			
		||||
  height: 40px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.drop-area {
 | 
			
		||||
  display: flex;
 | 
			
		||||
  justify-content: center;
 | 
			
		||||
  align-items: center;
 | 
			
		||||
  padding: 15px;
 | 
			
		||||
  border: 2px dashed #b2b2b2;
 | 
			
		||||
  border-radius: 4px;
 | 
			
		||||
  cursor: pointer;
 | 
			
		||||
  margin: 10px 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
clr-tabs button {
 | 
			
		||||
  box-shadow: none !important
 | 
			
		||||
}
 | 
			
		||||
@@ -1,7 +1,8 @@
 | 
			
		||||
import { Component, OnInit } from '@angular/core'
 | 
			
		||||
import { Component, OnInit, ViewEncapsulation } from '@angular/core'
 | 
			
		||||
import { ActivatedRoute, Router } from '@angular/router'
 | 
			
		||||
import { AppService, LicenceService, SasService } from '../services'
 | 
			
		||||
import { LicenseKeyData } from '../models/LicenseKeyData'
 | 
			
		||||
import { RequestWrapperResponse } from '../models/request-wrapper/RequestWrapperResponse'
 | 
			
		||||
 | 
			
		||||
enum LicenseActions {
 | 
			
		||||
  key = 'key',
 | 
			
		||||
@@ -13,7 +14,8 @@ enum LicenseActions {
 | 
			
		||||
@Component({
 | 
			
		||||
  selector: 'app-licensing',
 | 
			
		||||
  templateUrl: './licensing.component.html',
 | 
			
		||||
  styleUrls: ['./licensing.component.scss']
 | 
			
		||||
  styleUrls: ['./licensing.component.scss'],
 | 
			
		||||
  encapsulation: ViewEncapsulation.None
 | 
			
		||||
})
 | 
			
		||||
export class LicensingComponent implements OnInit {
 | 
			
		||||
  public action: LicenseActions | null = null
 | 
			
		||||
@@ -116,8 +118,12 @@ export class LicensingComponent implements OnInit {
 | 
			
		||||
 | 
			
		||||
    this.sasService
 | 
			
		||||
      .request('admin/registerkey', table)
 | 
			
		||||
      .then((res: any) => {
 | 
			
		||||
        if (res.return && res.return[0] && res.return[0].MSG === 'SUCCESS') {
 | 
			
		||||
      .then((res: RequestWrapperResponse) => {
 | 
			
		||||
        if (
 | 
			
		||||
          res.adapterResponse.return &&
 | 
			
		||||
          res.adapterResponse.return[0] &&
 | 
			
		||||
          res.adapterResponse.return[0].MSG === 'SUCCESS'
 | 
			
		||||
        ) {
 | 
			
		||||
          location.replace(location.href.split('#')[0])
 | 
			
		||||
        }
 | 
			
		||||
      })
 | 
			
		||||
 
 | 
			
		||||
@@ -3,6 +3,7 @@
 | 
			
		||||
    <clr-tree-node *ngIf="libraryList" class="search-node">
 | 
			
		||||
      <div class="tree-search-wrapper">
 | 
			
		||||
        <input
 | 
			
		||||
          appStealFocus
 | 
			
		||||
          clrInput
 | 
			
		||||
          #searchLibTreeInput
 | 
			
		||||
          placeholder="Libraries"
 | 
			
		||||
@@ -42,6 +43,7 @@
 | 
			
		||||
        <clr-tree-node *ngIf="library['tables']" class="search-node">
 | 
			
		||||
          <div class="tree-search-wrapper">
 | 
			
		||||
            <input
 | 
			
		||||
              appStealFocus
 | 
			
		||||
              clrInput
 | 
			
		||||
              #searchTreeInput
 | 
			
		||||
              placeholder="Tables"
 | 
			
		||||
@@ -85,6 +87,7 @@
 | 
			
		||||
          <clr-tree-node *ngIf="libTable['columns']" class="search-node">
 | 
			
		||||
            <div class="tree-search-wrapper">
 | 
			
		||||
              <input
 | 
			
		||||
                appStealFocus
 | 
			
		||||
                clrInput
 | 
			
		||||
                #searchTreeInput
 | 
			
		||||
                placeholder="Columns"
 | 
			
		||||
@@ -138,7 +141,9 @@
 | 
			
		||||
        size="60"
 | 
			
		||||
        class="is-info icon-dc-fill"
 | 
			
		||||
      ></clr-icon>
 | 
			
		||||
      <h3 class="text-center color-gray">Please select a column or table</h3>
 | 
			
		||||
      <p class="text-center color-gray mt-10" cds-text="section">
 | 
			
		||||
        Please select a column or table
 | 
			
		||||
      </p>
 | 
			
		||||
    </div>
 | 
			
		||||
 | 
			
		||||
    <ng-container *ngIf="column || table">
 | 
			
		||||
@@ -180,13 +185,13 @@
 | 
			
		||||
            <button
 | 
			
		||||
              (click)="limitDotDepth = true"
 | 
			
		||||
              type="button"
 | 
			
		||||
              class="btn btn-outline"
 | 
			
		||||
              class="btn btn-outline mr-5"
 | 
			
		||||
            >
 | 
			
		||||
              Limit depth
 | 
			
		||||
            </button>
 | 
			
		||||
 | 
			
		||||
            <!-- <button class="btn btn-outline" (click)='showSvg()'> Open in New Tab </button> -->
 | 
			
		||||
            <div class="btn-group d-block">
 | 
			
		||||
            <div class="btn-group direction d-block">
 | 
			
		||||
              <div
 | 
			
		||||
                class="radio btn"
 | 
			
		||||
                (click)="
 | 
			
		||||
 
 | 
			
		||||
@@ -1,79 +0,0 @@
 | 
			
		||||
.toggle-switch input[type=checkbox]:checked+label:before {
 | 
			
		||||
    border-color: #314351;
 | 
			
		||||
    background-color: #314351!important;
 | 
			
		||||
    transition: .15s ease-in;
 | 
			
		||||
    transition-property: border-color,background-color;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#graph{
 | 
			
		||||
    height: calc(100vh - 195px);
 | 
			
		||||
    overflow: hidden;
 | 
			
		||||
    text-align: center;
 | 
			
		||||
    display: block;
 | 
			
		||||
    width: 100%;
 | 
			
		||||
    border: 1px solid #e4e4e4;
 | 
			
		||||
    margin-top: 10px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.selection-wrapper {
 | 
			
		||||
    width: 100%;
 | 
			
		||||
    max-width: 670px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.column-active {
 | 
			
		||||
    background: #d8e3e9;
 | 
			
		||||
    color: black;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.content-area {
 | 
			
		||||
    padding: 0.5rem !important;
 | 
			
		||||
 | 
			
		||||
    .card {
 | 
			
		||||
        min-height: calc(100vh - 120px);
 | 
			
		||||
 | 
			
		||||
        .card-block {
 | 
			
		||||
            padding: 0.5rem 0.35rem !important;
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
clr-tree-node button {
 | 
			
		||||
    white-space: nowrap;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.graph-render-spinner {
 | 
			
		||||
    position: absolute;
 | 
			
		||||
    top: 0;
 | 
			
		||||
    width: 100%;
 | 
			
		||||
    display: flex;
 | 
			
		||||
    justify-content: center;
 | 
			
		||||
    margin-top: 10px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.biglineage-row {
 | 
			
		||||
    display: flex;
 | 
			
		||||
    justify-content: space-between;
 | 
			
		||||
    align-items: center;
 | 
			
		||||
 | 
			
		||||
    margin-bottom: 20px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.modal-footer {
 | 
			
		||||
    p {
 | 
			
		||||
        margin: 0;
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.lineage-title-wrapper {
 | 
			
		||||
    left: 12px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.max-depth-input {
 | 
			
		||||
    width: 100%;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@media (max-width: 768px) {
 | 
			
		||||
    .toggle-switch-container {
 | 
			
		||||
        margin-bottom: 20px;
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,4 +1,4 @@
 | 
			
		||||
import { Component } from '@angular/core'
 | 
			
		||||
import { Component, ViewEncapsulation } from '@angular/core'
 | 
			
		||||
import { Location } from '@angular/common'
 | 
			
		||||
import { globals } from '../_globals'
 | 
			
		||||
import * as d3Viz from 'd3-graphviz'
 | 
			
		||||
@@ -9,6 +9,7 @@ import { SasService } from '../services/sas.service'
 | 
			
		||||
import * as saveSvg from 'save-svg-as-png'
 | 
			
		||||
import { LoggerService } from '../services/logger.service'
 | 
			
		||||
import { LicenceService } from '../services/licence.service'
 | 
			
		||||
import { RequestWrapperResponse } from '../models/request-wrapper/RequestWrapperResponse'
 | 
			
		||||
const moment = require('moment')
 | 
			
		||||
 | 
			
		||||
@Component({
 | 
			
		||||
@@ -17,7 +18,8 @@ const moment = require('moment')
 | 
			
		||||
  templateUrl: './lineage.component.html',
 | 
			
		||||
  host: {
 | 
			
		||||
    class: 'content-container'
 | 
			
		||||
  }
 | 
			
		||||
  },
 | 
			
		||||
  encapsulation: ViewEncapsulation.None
 | 
			
		||||
})
 | 
			
		||||
export class LineageComponent {
 | 
			
		||||
  public switchFlag: boolean = false
 | 
			
		||||
@@ -115,8 +117,8 @@ export class LineageComponent {
 | 
			
		||||
 | 
			
		||||
    await this.sasService
 | 
			
		||||
      .request('lineage/getmetacols', libTable)
 | 
			
		||||
      .then((res: any) => {
 | 
			
		||||
        this.columnsList = res.metacols
 | 
			
		||||
      .then((res: RequestWrapperResponse) => {
 | 
			
		||||
        this.columnsList = res.adapterResponse.metacols
 | 
			
		||||
        if (this.columnsList && this.columnsList.length > 0) {
 | 
			
		||||
          // this.column = this.columnsList[0]['COLURI']
 | 
			
		||||
 | 
			
		||||
@@ -174,8 +176,8 @@ export class LineageComponent {
 | 
			
		||||
    let libTable = { SASControlTable: [{ liburi: $event }] }
 | 
			
		||||
    await this.sasService
 | 
			
		||||
      .request('lineage/getmetatables', libTable)
 | 
			
		||||
      .then((res: any) => {
 | 
			
		||||
        this.tablesList = res.metatables
 | 
			
		||||
      .then((res: RequestWrapperResponse) => {
 | 
			
		||||
        this.tablesList = res.adapterResponse.metatables
 | 
			
		||||
 | 
			
		||||
        if (this.tablesList && this.tablesList.length > 0) {
 | 
			
		||||
          library['tables'] = this.tablesList
 | 
			
		||||
@@ -295,8 +297,8 @@ export class LineageComponent {
 | 
			
		||||
      } else {
 | 
			
		||||
        await this.sasService
 | 
			
		||||
          .request('public/viewlibs', null)
 | 
			
		||||
          .then((res: any) => {
 | 
			
		||||
            this.libraryList = res.saslibs
 | 
			
		||||
          .then((res: RequestWrapperResponse) => {
 | 
			
		||||
            this.libraryList = res.adapterResponse.saslibs
 | 
			
		||||
            this.helperService.displayLibraries(this.libraryList)
 | 
			
		||||
 | 
			
		||||
            if (this.libraryList) {
 | 
			
		||||
@@ -402,8 +404,8 @@ export class LineageComponent {
 | 
			
		||||
    return new Promise<void>((resolve, reject) => {
 | 
			
		||||
      this.sasService
 | 
			
		||||
        .request('lineage/fetchtablelineage', libTable)
 | 
			
		||||
        .then(async (res: any) => {
 | 
			
		||||
          if (res.flatdata.length > 0) {
 | 
			
		||||
        .then(async (res: RequestWrapperResponse) => {
 | 
			
		||||
          if (res.adapterResponse.flatdata.length > 0) {
 | 
			
		||||
            if (this.licenceService.checkLineageLimit()) {
 | 
			
		||||
              this.eventService.showInfoModal(
 | 
			
		||||
                'Notice',
 | 
			
		||||
@@ -421,20 +423,22 @@ export class LineageComponent {
 | 
			
		||||
          }
 | 
			
		||||
 | 
			
		||||
          this.lineageTableName =
 | 
			
		||||
            res.info[0].LIBREF + '.' + res.info[0].TABLENAME
 | 
			
		||||
            res.adapterResponse.info[0].LIBREF +
 | 
			
		||||
            '.' +
 | 
			
		||||
            res.adapterResponse.info[0].TABLENAME
 | 
			
		||||
 | 
			
		||||
          let dotArray = res.finalfinal
 | 
			
		||||
          let dotArray = res.adapterResponse.finalfinal
 | 
			
		||||
          let vizTmp: string = ''
 | 
			
		||||
 | 
			
		||||
          for (let i = 0; i < dotArray.length; i++) {
 | 
			
		||||
            vizTmp += unescape(dotArray[i].LINE) + '\n'
 | 
			
		||||
          }
 | 
			
		||||
 | 
			
		||||
          this.flatdata = res.flatdata
 | 
			
		||||
          this.flatdata = res.adapterResponse.flatdata
 | 
			
		||||
 | 
			
		||||
          if (this.libraryList) {
 | 
			
		||||
            let libraryToSelect = this.libraryList.find((library: any) =>
 | 
			
		||||
              res.info[0].LIBURI.toUpperCase().includes(
 | 
			
		||||
              res.adapterResponse.info[0].LIBURI.toUpperCase().includes(
 | 
			
		||||
                library.LIBRARYID.toUpperCase()
 | 
			
		||||
              )
 | 
			
		||||
            )
 | 
			
		||||
@@ -450,7 +454,7 @@ export class LineageComponent {
 | 
			
		||||
              if (libraryToSelect['tables']) {
 | 
			
		||||
                tableToSelect = libraryToSelect['tables'].find((table: any) =>
 | 
			
		||||
                  table.TABLEURI.toUpperCase().includes(
 | 
			
		||||
                    res.info[0].TABLEID.toUpperCase()
 | 
			
		||||
                    res.adapterResponse.info[0].TABLEID.toUpperCase()
 | 
			
		||||
                  )
 | 
			
		||||
                )
 | 
			
		||||
 | 
			
		||||
@@ -495,10 +499,10 @@ export class LineageComponent {
 | 
			
		||||
            .replace(/\sds:/g, '\nds:')
 | 
			
		||||
            .replace(/\s\n/g, '\n')
 | 
			
		||||
 | 
			
		||||
          this.idlookup = res.idlookup
 | 
			
		||||
          this.idlookup = res.adapterResponse.idlookup
 | 
			
		||||
 | 
			
		||||
          if (res.finalfinal.length > this.largeDotFileLimit) {
 | 
			
		||||
            this.largeDotFileLines = res.finalfinal.length
 | 
			
		||||
          if (res.adapterResponse.finalfinal.length > this.largeDotFileLimit) {
 | 
			
		||||
            this.largeDotFileLines = res.adapterResponse.finalfinal.length
 | 
			
		||||
          } else {
 | 
			
		||||
            this.buildGraph()
 | 
			
		||||
          }
 | 
			
		||||
@@ -619,8 +623,8 @@ export class LineageComponent {
 | 
			
		||||
    return new Promise<void>((resolve, reject) => {
 | 
			
		||||
      this.sasService
 | 
			
		||||
        .request('lineage/fetchcollineage', libTable)
 | 
			
		||||
        .then(async (res: any) => {
 | 
			
		||||
          if (res.flatdata.length > 0) {
 | 
			
		||||
        .then(async (res: RequestWrapperResponse) => {
 | 
			
		||||
          if (res.adapterResponse.flatdata.length > 0) {
 | 
			
		||||
            if (this.licenceService.checkLineageLimit()) {
 | 
			
		||||
              this.eventService.showInfoModal(
 | 
			
		||||
                'Notice',
 | 
			
		||||
@@ -631,18 +635,21 @@ export class LineageComponent {
 | 
			
		||||
            }
 | 
			
		||||
          }
 | 
			
		||||
 | 
			
		||||
          if (typeof res === 'string') {
 | 
			
		||||
          if (typeof res.adapterResponse === 'string') {
 | 
			
		||||
            this.vizInput = 'digraph G {SAS Error}'
 | 
			
		||||
            this.buildGraph()
 | 
			
		||||
            return
 | 
			
		||||
          }
 | 
			
		||||
 | 
			
		||||
          this.lineageTableName = res.info[0].LIBREF + '.' + res.info[0].TABNAME
 | 
			
		||||
          this.lineageColumnName = res.info[0].COLNAME
 | 
			
		||||
          this.lineageTableName =
 | 
			
		||||
            res.adapterResponse.info[0].LIBREF +
 | 
			
		||||
            '.' +
 | 
			
		||||
            res.adapterResponse.info[0].TABNAME
 | 
			
		||||
          this.lineageColumnName = res.adapterResponse.info[0].COLNAME
 | 
			
		||||
 | 
			
		||||
          this.idlookup = res.idlookup
 | 
			
		||||
          this.idlookup = res.adapterResponse.idlookup
 | 
			
		||||
 | 
			
		||||
          let dotArray = res.fromsas
 | 
			
		||||
          let dotArray = res.adapterResponse.fromsas
 | 
			
		||||
          let vizTmp: string = ''
 | 
			
		||||
          for (let i = 0; i < dotArray.length; i++) {
 | 
			
		||||
            vizTmp += unescape(dotArray[i].STRING) + '\n'
 | 
			
		||||
@@ -653,11 +660,11 @@ export class LineageComponent {
 | 
			
		||||
            .replace(/\sds:/g, '\nds:')
 | 
			
		||||
            .replace(/\s\n/g, '\n')
 | 
			
		||||
 | 
			
		||||
          this.flatdata = res.flatdata
 | 
			
		||||
          this.flatdata = res.adapterResponse.flatdata
 | 
			
		||||
 | 
			
		||||
          if (this.libraryList) {
 | 
			
		||||
            let libraryToSelect = this.libraryList.find((library: any) =>
 | 
			
		||||
              res.info[0]?.LIBURI?.toUpperCase()?.includes(
 | 
			
		||||
              res.adapterResponse.info[0]?.LIBURI?.toUpperCase()?.includes(
 | 
			
		||||
                library?.LIBRARYID?.toUpperCase()
 | 
			
		||||
              )
 | 
			
		||||
            )
 | 
			
		||||
@@ -672,7 +679,8 @@ export class LineageComponent {
 | 
			
		||||
 | 
			
		||||
              if (libraryToSelect['tables']) {
 | 
			
		||||
                tableToSelect = libraryToSelect['tables'].find(
 | 
			
		||||
                  (table: any) => table.TABLEURI === res.info[0].TABURI
 | 
			
		||||
                  (table: any) =>
 | 
			
		||||
                    table.TABLEURI === res.adapterResponse.info[0].TABURI
 | 
			
		||||
                )
 | 
			
		||||
 | 
			
		||||
                if (tableToSelect) {
 | 
			
		||||
@@ -714,8 +722,8 @@ export class LineageComponent {
 | 
			
		||||
            }
 | 
			
		||||
          }
 | 
			
		||||
 | 
			
		||||
          if (res.fromsas.length > this.largeDotFileLimit) {
 | 
			
		||||
            this.largeDotFileLines = res.fromsas.length
 | 
			
		||||
          if (res.adapterResponse.fromsas.length > this.largeDotFileLimit) {
 | 
			
		||||
            this.largeDotFileLines = res.adapterResponse.fromsas.length
 | 
			
		||||
          } else {
 | 
			
		||||
            this.buildGraph()
 | 
			
		||||
          }
 | 
			
		||||
 
 | 
			
		||||
@@ -22,6 +22,7 @@
 | 
			
		||||
    <clr-tree-node *ngIf="metaDataList" class="search-node">
 | 
			
		||||
      <div class="tree-search-wrapper">
 | 
			
		||||
        <input
 | 
			
		||||
          appStealFocus
 | 
			
		||||
          clrInput
 | 
			
		||||
          #searchLibTreeInput
 | 
			
		||||
          placeholder="search SAS Types"
 | 
			
		||||
@@ -72,7 +73,9 @@
 | 
			
		||||
        size="60"
 | 
			
		||||
        class="is-info icon-dc-fill"
 | 
			
		||||
      ></clr-icon>
 | 
			
		||||
      <h3 class="text-center color-gray">Please select a type</h3>
 | 
			
		||||
      <p class="text-center color-gray mt-10" cds-text="section">
 | 
			
		||||
        Please select a type
 | 
			
		||||
      </p>
 | 
			
		||||
    </div>
 | 
			
		||||
 | 
			
		||||
    <div class="loadingSpinner" *ngIf="loading">
 | 
			
		||||
@@ -123,12 +126,18 @@
 | 
			
		||||
                        [class.object-header]="!entry.count"
 | 
			
		||||
                        class="full-width"
 | 
			
		||||
                      >
 | 
			
		||||
                        <clr-icon
 | 
			
		||||
                          *ngIf="!entry.count"
 | 
			
		||||
                          shape="rack-server"
 | 
			
		||||
                        ></clr-icon>
 | 
			
		||||
                        <clr-icon *ngIf="entry.count" shape="block"></clr-icon>
 | 
			
		||||
                        {{ entry.display }}
 | 
			
		||||
                        <div>
 | 
			
		||||
                          <clr-icon
 | 
			
		||||
                            *ngIf="!entry.count"
 | 
			
		||||
                            shape="rack-server"
 | 
			
		||||
                          ></clr-icon>
 | 
			
		||||
                          <clr-icon
 | 
			
		||||
                            *ngIf="entry.count"
 | 
			
		||||
                            shape="block"
 | 
			
		||||
                          ></clr-icon>
 | 
			
		||||
                          {{ entry.display }}
 | 
			
		||||
                        </div>
 | 
			
		||||
 | 
			
		||||
                        <p class="float-right object-uri" *ngIf="!entry.count">
 | 
			
		||||
                          {{ entry.URI }}
 | 
			
		||||
                        </p>
 | 
			
		||||
@@ -163,9 +172,15 @@
 | 
			
		||||
                [clrExpandable]="true"
 | 
			
		||||
              >
 | 
			
		||||
                <div [class.object-header]="!entry.count" class="full-width">
 | 
			
		||||
                  <clr-icon *ngIf="!entry.count" shape="rack-server"></clr-icon>
 | 
			
		||||
                  <clr-icon *ngIf="entry.count" shape="block"></clr-icon>
 | 
			
		||||
                  {{ entry.display }}
 | 
			
		||||
                  <div>
 | 
			
		||||
                    <clr-icon
 | 
			
		||||
                      *ngIf="!entry.count"
 | 
			
		||||
                      shape="rack-server"
 | 
			
		||||
                    ></clr-icon>
 | 
			
		||||
                    <clr-icon *ngIf="entry.count" shape="block"></clr-icon>
 | 
			
		||||
                    {{ entry.display }}
 | 
			
		||||
                  </div>
 | 
			
		||||
 | 
			
		||||
                  <p class="float-right object-uri" *ngIf="!entry.count">
 | 
			
		||||
                    {{ entry.URI }}
 | 
			
		||||
                  </p>
 | 
			
		||||
 
 | 
			
		||||
@@ -1,67 +0,0 @@
 | 
			
		||||
 | 
			
		||||
.objects-col{
 | 
			
		||||
  height: 75vh;
 | 
			
		||||
  overflow: scroll;
 | 
			
		||||
  border: 1px solid #cccccc;
 | 
			
		||||
  background: white;
 | 
			
		||||
  border-radius: 4px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.cols-head {
 | 
			
		||||
  background: #fafafa;
 | 
			
		||||
  border: 1px solid #cccccc;
 | 
			
		||||
  padding: 10px;
 | 
			
		||||
  display: flex;
 | 
			
		||||
}
 | 
			
		||||
.object-text {
 | 
			
		||||
  display: flex;
 | 
			
		||||
  align-items: flex-start;
 | 
			
		||||
  justify-content: space-between;
 | 
			
		||||
  margin-left: 10px;
 | 
			
		||||
  flex: 1;
 | 
			
		||||
}
 | 
			
		||||
.repo-dropdown{
 | 
			
		||||
  margin-right: 15px;
 | 
			
		||||
  margin-left: 15px;
 | 
			
		||||
  margin-bottom: 10px;
 | 
			
		||||
}
 | 
			
		||||
.clr-accordion-title{
 | 
			
		||||
  width: 100%;
 | 
			
		||||
}
 | 
			
		||||
.float-right{
 | 
			
		||||
  margin: 0px;
 | 
			
		||||
  float: right;
 | 
			
		||||
}
 | 
			
		||||
.full-width{
 | 
			
		||||
  width: 100%;
 | 
			
		||||
}
 | 
			
		||||
.object-uri{
 | 
			
		||||
  margin: 0px;
 | 
			
		||||
  margin-top: 5px;
 | 
			
		||||
}
 | 
			
		||||
.object-header{
 | 
			
		||||
  padding-left: 3px;
 | 
			
		||||
  padding-right: 3px;
 | 
			
		||||
}
 | 
			
		||||
.object-header:hover{
 | 
			
		||||
  background-color: #d8e3e9;
 | 
			
		||||
  border-radius: 3px;
 | 
			
		||||
}
 | 
			
		||||
.datagrid-host{
 | 
			
		||||
  display: unset !important;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.card {
 | 
			
		||||
  margin-top: 0;
 | 
			
		||||
 | 
			
		||||
  flex: 1;
 | 
			
		||||
  display: flex;
 | 
			
		||||
  flex-direction: column;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.content-area {
 | 
			
		||||
  padding: 0.5rem !important;
 | 
			
		||||
 | 
			
		||||
  display: flex;
 | 
			
		||||
  flex-direction: column;
 | 
			
		||||
}
 | 
			
		||||
@@ -1,5 +1,5 @@
 | 
			
		||||
import { Location } from '@angular/common'
 | 
			
		||||
import { Component, OnInit } from '@angular/core'
 | 
			
		||||
import { Component, OnInit, ViewEncapsulation } from '@angular/core'
 | 
			
		||||
import { ActivatedRoute, Router } from '@angular/router'
 | 
			
		||||
import { ClrDatagridStringFilterInterface } from '@clr/angular'
 | 
			
		||||
import { Observable, of } from 'rxjs'
 | 
			
		||||
@@ -9,6 +9,7 @@ import { SasService } from '../services/sas.service'
 | 
			
		||||
import { globals } from '../_globals'
 | 
			
		||||
 | 
			
		||||
import { Injectable } from '@angular/core'
 | 
			
		||||
import { RequestWrapperResponse } from '../models/request-wrapper/RequestWrapperResponse'
 | 
			
		||||
 | 
			
		||||
interface MetaData {
 | 
			
		||||
  NAME: any
 | 
			
		||||
@@ -49,7 +50,8 @@ class ValueFilter implements ClrDatagridStringFilterInterface<MetaData> {
 | 
			
		||||
  styleUrls: ['./metadata.component.scss'],
 | 
			
		||||
  host: {
 | 
			
		||||
    class: 'content-container'
 | 
			
		||||
  }
 | 
			
		||||
  },
 | 
			
		||||
  encapsulation: ViewEncapsulation.None
 | 
			
		||||
})
 | 
			
		||||
export class MetadataComponent implements OnInit {
 | 
			
		||||
  metaDataList: Array<any> | undefined
 | 
			
		||||
@@ -109,44 +111,52 @@ export class MetadataComponent implements OnInit {
 | 
			
		||||
      this.metatypesLoading = false
 | 
			
		||||
      this.metaDataSearch = globals.metadata.metaDataSearch
 | 
			
		||||
    } else {
 | 
			
		||||
      this.sasService.request('metanav/metatypes', null).then((res: any) => {
 | 
			
		||||
        this.metaDataList = res.types
 | 
			
		||||
        globals.metadata.metaDataList = this.metaDataList
 | 
			
		||||
        this.loading = false
 | 
			
		||||
        this.metatypesLoading = false
 | 
			
		||||
      })
 | 
			
		||||
      this.sasService
 | 
			
		||||
        .request('metanav/metatypes', null)
 | 
			
		||||
        .then((res: RequestWrapperResponse) => {
 | 
			
		||||
          this.metaDataList = res.adapterResponse.types
 | 
			
		||||
          globals.metadata.metaDataList = this.metaDataList
 | 
			
		||||
          this.loading = false
 | 
			
		||||
          this.metatypesLoading = false
 | 
			
		||||
        })
 | 
			
		||||
 | 
			
		||||
      this.sasService.request('metanav/metarepos', null).then((res: any) => {
 | 
			
		||||
        let foundation = false
 | 
			
		||||
        this.repositories = []
 | 
			
		||||
        for (let index = 0; index < res.outrepos.length; index++) {
 | 
			
		||||
          this.repositories.push(res.outrepos[index].NAME)
 | 
			
		||||
          if (res.outrepos[index].NAME === 'Foundation') {
 | 
			
		||||
            foundation = true
 | 
			
		||||
      this.sasService
 | 
			
		||||
        .request('metanav/metarepos', null)
 | 
			
		||||
        .then((res: RequestWrapperResponse) => {
 | 
			
		||||
          let foundation = false
 | 
			
		||||
          this.repositories = []
 | 
			
		||||
          for (
 | 
			
		||||
            let index = 0;
 | 
			
		||||
            index < res.adapterResponse.outrepos.length;
 | 
			
		||||
            index++
 | 
			
		||||
          ) {
 | 
			
		||||
            this.repositories.push(res.adapterResponse.outrepos[index].NAME)
 | 
			
		||||
            if (res.adapterResponse.outrepos[index].NAME === 'Foundation') {
 | 
			
		||||
              foundation = true
 | 
			
		||||
            }
 | 
			
		||||
          }
 | 
			
		||||
        }
 | 
			
		||||
        if (foundation) {
 | 
			
		||||
          this.repository = 'Foundation'
 | 
			
		||||
        } else {
 | 
			
		||||
          this.repository = res.outrepos[0].NAME
 | 
			
		||||
        }
 | 
			
		||||
        globals.metadata.metaRepositories = this.repositories
 | 
			
		||||
        globals.metadata.selectedRepository = this.repository
 | 
			
		||||
        if (this.objectRoute) {
 | 
			
		||||
          this.eventService.closeSidebar()
 | 
			
		||||
          this.showData = true
 | 
			
		||||
          let name = ''
 | 
			
		||||
          let id = this.route.snapshot.params['objectID']
 | 
			
		||||
          // let temp = this.router.url.split("%20").join(" ").split("/").reverse();
 | 
			
		||||
          this.metaObjectList = []
 | 
			
		||||
          this.metaObjectList.push({ ID: id, NAME: name })
 | 
			
		||||
          this.metaObjectShowList = this.metaObjectList
 | 
			
		||||
          this.metaObjectOnClick(
 | 
			
		||||
            this.metaObjectShowList[0].ID,
 | 
			
		||||
            this.metaObjectShowList[0]
 | 
			
		||||
          )
 | 
			
		||||
        }
 | 
			
		||||
      })
 | 
			
		||||
          if (foundation) {
 | 
			
		||||
            this.repository = 'Foundation'
 | 
			
		||||
          } else {
 | 
			
		||||
            this.repository = res.adapterResponse.outrepos[0].NAME
 | 
			
		||||
          }
 | 
			
		||||
          globals.metadata.metaRepositories = this.repositories
 | 
			
		||||
          globals.metadata.selectedRepository = this.repository
 | 
			
		||||
          if (this.objectRoute) {
 | 
			
		||||
            this.eventService.closeSidebar()
 | 
			
		||||
            this.showData = true
 | 
			
		||||
            let name = ''
 | 
			
		||||
            let id = this.route.snapshot.params['objectID']
 | 
			
		||||
            // let temp = this.router.url.split("%20").join(" ").split("/").reverse();
 | 
			
		||||
            this.metaObjectList = []
 | 
			
		||||
            this.metaObjectList.push({ ID: id, NAME: name })
 | 
			
		||||
            this.metaObjectShowList = this.metaObjectList
 | 
			
		||||
            this.metaObjectOnClick(
 | 
			
		||||
              this.metaObjectShowList[0].ID,
 | 
			
		||||
              this.metaObjectShowList[0]
 | 
			
		||||
            )
 | 
			
		||||
          }
 | 
			
		||||
        })
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
@@ -183,56 +193,64 @@ export class MetadataComponent implements OnInit {
 | 
			
		||||
    const data: any = {
 | 
			
		||||
      SASControlTable: [{ metatype: $event, repo: this.repository }]
 | 
			
		||||
    }
 | 
			
		||||
    this.sasService.request('metanav/metaobjects', data).then((res: any) => {
 | 
			
		||||
      this.metaObjectList = res.objects
 | 
			
		||||
      this.getMetaObjectAttributes(this.metaObjectSize)
 | 
			
		||||
      this.loading = false
 | 
			
		||||
      this.assoTypeSelected = $event
 | 
			
		||||
      this.eventService.closeSidebar()
 | 
			
		||||
      this.showData = true
 | 
			
		||||
    })
 | 
			
		||||
    this.sasService
 | 
			
		||||
      .request('metanav/metaobjects', data)
 | 
			
		||||
      .then((res: RequestWrapperResponse) => {
 | 
			
		||||
        this.metaObjectList = res.adapterResponse.objects
 | 
			
		||||
        this.getMetaObjectAttributes(this.metaObjectSize)
 | 
			
		||||
        this.loading = false
 | 
			
		||||
        this.assoTypeSelected = $event
 | 
			
		||||
        this.eventService.closeSidebar()
 | 
			
		||||
        this.showData = true
 | 
			
		||||
      })
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  public async selectmetaObject($event: any, metaData?: any) {
 | 
			
		||||
    let data: any = {
 | 
			
		||||
      SASControlTable: [{ objecturi: $event }]
 | 
			
		||||
    }
 | 
			
		||||
    this.sasService.request('metanav/metadetails', data).then((res: any) => {
 | 
			
		||||
      this.metaObjectAssociations = res.associations
 | 
			
		||||
      this.root$ = of(this.getAssosiationsCount(res.associations))
 | 
			
		||||
      this.showAcc = true
 | 
			
		||||
      this.showTable = true
 | 
			
		||||
      let metaObjectName = res.attributes.find(
 | 
			
		||||
        (x: any) => x.NAME === 'Name'
 | 
			
		||||
      ).VALUE
 | 
			
		||||
      this.assoObjectSelected = metaObjectName
 | 
			
		||||
      metaData.NAME = metaObjectName
 | 
			
		||||
      let url = this.router.url
 | 
			
		||||
      if (this.objectRoute) {
 | 
			
		||||
        // this.location.replaceState(url.slice(0, url.lastIndexOf("object")) + "object/" + $event.slice(1 + $event.indexOf("\\")) + "/" + escape(metaData.NAME));
 | 
			
		||||
        this.location.replaceState(
 | 
			
		||||
          url.slice(0, url.lastIndexOf('object')) +
 | 
			
		||||
            'object/' +
 | 
			
		||||
            $event.slice(1 + $event.indexOf('\\'))
 | 
			
		||||
    this.sasService
 | 
			
		||||
      .request('metanav/metadetails', data)
 | 
			
		||||
      .then((res: RequestWrapperResponse) => {
 | 
			
		||||
        this.metaObjectAssociations = res.adapterResponse.associations
 | 
			
		||||
        this.root$ = of(
 | 
			
		||||
          this.getAssosiationsCount(res.adapterResponse.associations)
 | 
			
		||||
        )
 | 
			
		||||
      } else {
 | 
			
		||||
        // this.location.replaceState(url + "/object/" + $event.slice(1 + $event.indexOf("\\")) + "/" + escape(metaData.NAME));
 | 
			
		||||
        this.location.replaceState(
 | 
			
		||||
          url + '/object/' + $event.slice(1 + $event.indexOf('\\'))
 | 
			
		||||
        )
 | 
			
		||||
      }
 | 
			
		||||
      this.metaObjectAttributes = res.attributes
 | 
			
		||||
    })
 | 
			
		||||
        this.showAcc = true
 | 
			
		||||
        this.showTable = true
 | 
			
		||||
        let metaObjectName = res.adapterResponse.attributes.find(
 | 
			
		||||
          (x: any) => x.NAME === 'Name'
 | 
			
		||||
        ).VALUE
 | 
			
		||||
        this.assoObjectSelected = metaObjectName
 | 
			
		||||
        metaData.NAME = metaObjectName
 | 
			
		||||
        let url = this.router.url
 | 
			
		||||
        if (this.objectRoute) {
 | 
			
		||||
          // this.location.replaceState(url.slice(0, url.lastIndexOf("object")) + "object/" + $event.slice(1 + $event.indexOf("\\")) + "/" + escape(metaData.NAME));
 | 
			
		||||
          this.location.replaceState(
 | 
			
		||||
            url.slice(0, url.lastIndexOf('object')) +
 | 
			
		||||
              'object/' +
 | 
			
		||||
              $event.slice(1 + $event.indexOf('\\'))
 | 
			
		||||
          )
 | 
			
		||||
        } else {
 | 
			
		||||
          // this.location.replaceState(url + "/object/" + $event.slice(1 + $event.indexOf("\\")) + "/" + escape(metaData.NAME));
 | 
			
		||||
          this.location.replaceState(
 | 
			
		||||
            url + '/object/' + $event.slice(1 + $event.indexOf('\\'))
 | 
			
		||||
          )
 | 
			
		||||
        }
 | 
			
		||||
        this.metaObjectAttributes = res.adapterResponse.attributes
 | 
			
		||||
      })
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  public async selectAssosiationsDetails($event: any, metaData?: any) {
 | 
			
		||||
    let data: any = {
 | 
			
		||||
      SASControlTable: [{ objecturi: $event }]
 | 
			
		||||
    }
 | 
			
		||||
    this.sasService.request('metanav/metadetails', data).then((res: any) => {
 | 
			
		||||
      this.metaObjectAttributes = res.attributes
 | 
			
		||||
      this.showTable = true
 | 
			
		||||
    })
 | 
			
		||||
    this.sasService
 | 
			
		||||
      .request('metanav/metadetails', data)
 | 
			
		||||
      .then((res: RequestWrapperResponse) => {
 | 
			
		||||
        this.metaObjectAttributes = res.adapterResponse.attributes
 | 
			
		||||
        this.showTable = true
 | 
			
		||||
      })
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  public getAssosiationsCount(assosiationList: Array<any>) {
 | 
			
		||||
@@ -244,7 +262,7 @@ export class MetadataComponent implements OnInit {
 | 
			
		||||
          details: []
 | 
			
		||||
        })
 | 
			
		||||
      }
 | 
			
		||||
      let assocObj = assosiationsHash.get(assosiation.ASSOC)
 | 
			
		||||
      let assocObj: any = assosiationsHash.get(assosiation.ASSOC)
 | 
			
		||||
      assocObj.count++
 | 
			
		||||
      assocObj.details.push({
 | 
			
		||||
        ASSOCURI: assosiation.ASSOCURI,
 | 
			
		||||
@@ -254,7 +272,7 @@ export class MetadataComponent implements OnInit {
 | 
			
		||||
      })
 | 
			
		||||
    }
 | 
			
		||||
    let assocGrouped: Array<any> = []
 | 
			
		||||
    assosiationsHash.forEach(function (val, key) {
 | 
			
		||||
    assosiationsHash.forEach(function (val: any, key) {
 | 
			
		||||
      assocGrouped.push({
 | 
			
		||||
        ASSOC: key,
 | 
			
		||||
        count: val.count,
 | 
			
		||||
@@ -294,9 +312,9 @@ export class MetadataComponent implements OnInit {
 | 
			
		||||
    }
 | 
			
		||||
    return this.sasService
 | 
			
		||||
      .request('metanav/metadetails', data)
 | 
			
		||||
      .then((res: any) => {
 | 
			
		||||
      .then((res: RequestWrapperResponse) => {
 | 
			
		||||
        this.showTable = true
 | 
			
		||||
        this.metaObjectAttributes = res.attributes
 | 
			
		||||
        this.metaObjectAttributes = res.adapterResponse.attributes
 | 
			
		||||
        this.assoObjectSelected = asso.NAME
 | 
			
		||||
        let url = this.router.url
 | 
			
		||||
        if (this.objectRoute) {
 | 
			
		||||
@@ -314,7 +332,7 @@ export class MetadataComponent implements OnInit {
 | 
			
		||||
              asso.ASSOCURI.slice(1 + asso.ASSOCURI.indexOf('\\'))
 | 
			
		||||
          )
 | 
			
		||||
        }
 | 
			
		||||
        return this.getAssosiationsCount(res.associations)
 | 
			
		||||
        return this.getAssosiationsCount(res.adapterResponse.associations)
 | 
			
		||||
      })
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										9
									
								
								client/src/app/models/AppSettings.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								client/src/app/models/AppSettings.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,9 @@
 | 
			
		||||
export interface AppSettings {
 | 
			
		||||
  persistSelectedTheme: boolean
 | 
			
		||||
  selectedTheme: AppThemes
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export enum AppThemes {
 | 
			
		||||
  light = 'light',
 | 
			
		||||
  dark = 'dark'
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										1
									
								
								client/src/app/models/FileUploadEncoding.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								client/src/app/models/FileUploadEncoding.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1 @@
 | 
			
		||||
export type FileUploadEncoding = 'UTF-8' | 'WLATIN1'
 | 
			
		||||
							
								
								
									
										29
									
								
								client/src/app/models/ParseParams.interface.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								client/src/app/models/ParseParams.interface.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,29 @@
 | 
			
		||||
import { DcValidator } from '../shared/dc-validator/dc-validator'
 | 
			
		||||
import { FileUploadEncoding } from './FileUploadEncoding'
 | 
			
		||||
import { FileUploader } from './FileUploader.class'
 | 
			
		||||
import { ExcelRule } from './TableData'
 | 
			
		||||
import XLSX from 'xlsx'
 | 
			
		||||
 | 
			
		||||
export interface ParseParams {
 | 
			
		||||
  file: File
 | 
			
		||||
  password?: string
 | 
			
		||||
  dcValidator: DcValidator
 | 
			
		||||
  /**
 | 
			
		||||
   * If workbook is provided, parse function will not run a XLSX.read()
 | 
			
		||||
   * it will use this property instead. So the client must do a file read beforehand
 | 
			
		||||
   */
 | 
			
		||||
  workbook?: XLSX.WorkBook
 | 
			
		||||
  /**
 | 
			
		||||
   * Parse function will manipulate and return the uploader array which can be provided with files already in the queue
 | 
			
		||||
   * Otherwise new empty instance will be created.
 | 
			
		||||
   */
 | 
			
		||||
  uploader?: FileUploader
 | 
			
		||||
  headerPks: string[]
 | 
			
		||||
  headerArray: string[]
 | 
			
		||||
  headerShow: string[]
 | 
			
		||||
  timeHeaders: string[]
 | 
			
		||||
  dateHeaders: string[]
 | 
			
		||||
  dateTimeHeaders: string[]
 | 
			
		||||
  xlRules: ExcelRule[]
 | 
			
		||||
  encoding?: FileUploadEncoding
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										15
									
								
								client/src/app/models/ParseResult.interface.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								client/src/app/models/ParseResult.interface.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,15 @@
 | 
			
		||||
import { FileUploader } from './FileUploader.class'
 | 
			
		||||
import FoundRangeInfo from './RangeInfo'
 | 
			
		||||
 | 
			
		||||
export interface ParseResult {
 | 
			
		||||
  /**
 | 
			
		||||
   * In case of CSV file, won't be returned
 | 
			
		||||
   */
 | 
			
		||||
  data?: any[]
 | 
			
		||||
  /**
 | 
			
		||||
   * In case of CSV file, won't be returned
 | 
			
		||||
   */
 | 
			
		||||
  headerShow?: string[]
 | 
			
		||||
  rangeSheetRes?: FoundRangeInfo
 | 
			
		||||
  uploader: FileUploader
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										13
									
								
								client/src/app/models/RangeInfo.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								client/src/app/models/RangeInfo.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,13 @@
 | 
			
		||||
export default interface FoundRangeInfo {
 | 
			
		||||
  found: boolean
 | 
			
		||||
  sheetName: string
 | 
			
		||||
  rangeStartAddress: string
 | 
			
		||||
  rangeEndAddress: string
 | 
			
		||||
  rangeAddress: string
 | 
			
		||||
  missingHeaders: MissingHeaders[]
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export interface MissingHeaders {
 | 
			
		||||
  sheetName: string
 | 
			
		||||
  missingHeaders: string[]
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										13
									
								
								client/src/app/models/SearchDataExcelResult.interface.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								client/src/app/models/SearchDataExcelResult.interface.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,13 @@
 | 
			
		||||
import { MissingHeaders } from './RangeInfo'
 | 
			
		||||
 | 
			
		||||
export interface SearchDataExcelResult {
 | 
			
		||||
  missing?: MissingHeaders[]
 | 
			
		||||
  found?: {
 | 
			
		||||
    data: any
 | 
			
		||||
    arrayData: any[]
 | 
			
		||||
    sheetName: string
 | 
			
		||||
    headers: string[]
 | 
			
		||||
    startAddress?: string
 | 
			
		||||
    endAddress?: string
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
@@ -7,4 +7,5 @@ export default interface SheetInfo {
 | 
			
		||||
  missingHeaders: string[]
 | 
			
		||||
  rangeStartRow: number
 | 
			
		||||
  rangeStartCol: number
 | 
			
		||||
  rangeAddress?: string
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										4
									
								
								client/src/app/models/UploadFile.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								client/src/app/models/UploadFile.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,4 @@
 | 
			
		||||
export interface UploadFileResponse {
 | 
			
		||||
  adapterResponse: any
 | 
			
		||||
  log?: string
 | 
			
		||||
}
 | 
			
		||||
@@ -0,0 +1,4 @@
 | 
			
		||||
export interface RequestWrapperResponse<responseType = any> {
 | 
			
		||||
  adapterResponse: responseType
 | 
			
		||||
  log?: string
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										34
									
								
								client/src/app/models/sas/editors-stagedata.model.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								client/src/app/models/sas/editors-stagedata.model.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,34 @@
 | 
			
		||||
import { BaseSASResponse } from './common/BaseSASResponse'
 | 
			
		||||
 | 
			
		||||
export interface EditorsStageDataSASResponse extends BaseSASResponse {
 | 
			
		||||
  SYSDATE: string
 | 
			
		||||
  SYSTIME: string
 | 
			
		||||
  sasparams: Sasparam[]
 | 
			
		||||
  _DEBUG: string
 | 
			
		||||
  _PROGRAM: string
 | 
			
		||||
  AUTOEXEC: string
 | 
			
		||||
  MF_GETUSER: string
 | 
			
		||||
  SYSCC: string
 | 
			
		||||
  SYSENCODING: string
 | 
			
		||||
  SYSERRORTEXT: string
 | 
			
		||||
  SYSHOSTINFOLONG: string
 | 
			
		||||
  SYSHOSTNAME: string
 | 
			
		||||
  SYSPROCESSID: string
 | 
			
		||||
  SYSPROCESSMODE: string
 | 
			
		||||
  SYSPROCESSNAME: string
 | 
			
		||||
  SYSJOBID: string
 | 
			
		||||
  SYSSCPL: string
 | 
			
		||||
  SYSSITE: string
 | 
			
		||||
  SYSTCPIPHOSTNAME: string
 | 
			
		||||
  SYSUSERID: string
 | 
			
		||||
  SYSVLONG: string
 | 
			
		||||
  SYSWARNINGTEXT: string
 | 
			
		||||
  END_DTTM: string
 | 
			
		||||
  MEMSIZE: string
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export interface Sasparam {
 | 
			
		||||
  STATUS: string | 'SUCCESS'
 | 
			
		||||
  DSID: string
 | 
			
		||||
  URL: string
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										19
									
								
								client/src/app/multi-dataset/multi-dataset-routing.module.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								client/src/app/multi-dataset/multi-dataset-routing.module.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,19 @@
 | 
			
		||||
import { NgModule } from '@angular/core'
 | 
			
		||||
import { RouterModule, Routes } from '@angular/router'
 | 
			
		||||
 | 
			
		||||
import { MultiDatasetRouteComponent } from '../routes/multi-dataset-route/multi-dataset-route.component'
 | 
			
		||||
import { MultiDatasetComponent } from './multi-dataset.component'
 | 
			
		||||
 | 
			
		||||
const routes: Routes = [
 | 
			
		||||
  {
 | 
			
		||||
    path: '',
 | 
			
		||||
    component: MultiDatasetRouteComponent,
 | 
			
		||||
    children: [{ path: '', component: MultiDatasetComponent }]
 | 
			
		||||
  }
 | 
			
		||||
]
 | 
			
		||||
 | 
			
		||||
@NgModule({
 | 
			
		||||
  imports: [RouterModule.forChild(routes)],
 | 
			
		||||
  exports: [RouterModule]
 | 
			
		||||
})
 | 
			
		||||
export class MultiDatasetRoutingModule {}
 | 
			
		||||
							
								
								
									
										543
									
								
								client/src/app/multi-dataset/multi-dataset.component.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										543
									
								
								client/src/app/multi-dataset/multi-dataset.component.html
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,543 @@
 | 
			
		||||
<app-sidebar>
 | 
			
		||||
  <div *ngIf="datasetsLoading" class="my-10-mx-auto text-center">
 | 
			
		||||
    <clr-spinner clrMedium></clr-spinner>
 | 
			
		||||
  </div>
 | 
			
		||||
 | 
			
		||||
  <div *ngIf="!parsedDatasets.length" class="text-center mb-10">
 | 
			
		||||
    <button
 | 
			
		||||
      (click)="fileUploadInput.click()"
 | 
			
		||||
      id="browse-file"
 | 
			
		||||
      class="btn btn-primary btn-sm"
 | 
			
		||||
      [disabled]="selectedFile !== null || submittingCsv"
 | 
			
		||||
    >
 | 
			
		||||
      Browse file
 | 
			
		||||
    </button>
 | 
			
		||||
    <input
 | 
			
		||||
      hidden
 | 
			
		||||
      #fileUploadInput
 | 
			
		||||
      id="file-upload"
 | 
			
		||||
      type="file"
 | 
			
		||||
      (change)="onFileChange($event)"
 | 
			
		||||
      multiple
 | 
			
		||||
    />
 | 
			
		||||
  </div>
 | 
			
		||||
 | 
			
		||||
  <ng-container *ngIf="parsedDatasets.length && !submittedCsvDatasets.length">
 | 
			
		||||
    <div *ngIf="!excelsSubmitted" class="text-center mb-10">
 | 
			
		||||
      <button (click)="onDiscard()" class="btn btn-danger btn-sm mr-10">
 | 
			
		||||
        Discard
 | 
			
		||||
      </button>
 | 
			
		||||
      <button
 | 
			
		||||
        (click)="onSubmitAll()"
 | 
			
		||||
        id="submit-all"
 | 
			
		||||
        class="btn btn-primary btn-sm"
 | 
			
		||||
      >
 | 
			
		||||
        Submit All
 | 
			
		||||
      </button>
 | 
			
		||||
    </div>
 | 
			
		||||
 | 
			
		||||
    <p cds-text="caption" class="ml-10 mb-10">Found tables:</p>
 | 
			
		||||
    <clr-tree>
 | 
			
		||||
      <clr-tree-node *ngFor="let dataset of parsedDatasets">
 | 
			
		||||
        <button
 | 
			
		||||
          (click)="onParsedDatasetClick(dataset)"
 | 
			
		||||
          class="clr-treenode-link whitespace-nowrap d-flex clr-align-items-center"
 | 
			
		||||
          [class.active]="dataset.active"
 | 
			
		||||
        >
 | 
			
		||||
          <ng-container *ngIf="dataset.submitResult">
 | 
			
		||||
            <cds-icon
 | 
			
		||||
              *ngIf="dataset.submitResult.error"
 | 
			
		||||
              status="danger"
 | 
			
		||||
              shape="exclamation-circle"
 | 
			
		||||
            ></cds-icon>
 | 
			
		||||
            <cds-icon
 | 
			
		||||
              *ngIf="dataset.submitResult.success"
 | 
			
		||||
              status="success"
 | 
			
		||||
              shape="check-circle"
 | 
			
		||||
            ></cds-icon>
 | 
			
		||||
          </ng-container>
 | 
			
		||||
 | 
			
		||||
          <ng-container *ngIf="!dataset.submitResult">
 | 
			
		||||
            <ng-container *ngIf="dataset.datasource">
 | 
			
		||||
              <cds-icon
 | 
			
		||||
                *ngIf="!(dataset.datasource.length && dataset.parseResult)"
 | 
			
		||||
                status="danger"
 | 
			
		||||
                shape="exclamation-circle"
 | 
			
		||||
              ></cds-icon>
 | 
			
		||||
              <cds-icon
 | 
			
		||||
                *ngIf="dataset.datasource.length && dataset.parseResult"
 | 
			
		||||
                shape="table"
 | 
			
		||||
              ></cds-icon>
 | 
			
		||||
            </ng-container>
 | 
			
		||||
 | 
			
		||||
            <ng-container *ngIf="!dataset.datasource">
 | 
			
		||||
              <cds-icon *ngIf="!dataset.parsingTable" shape="table"></cds-icon>
 | 
			
		||||
 | 
			
		||||
              <clr-spinner *ngIf="dataset.parsingTable" clrSmall></clr-spinner>
 | 
			
		||||
            </ng-container>
 | 
			
		||||
          </ng-container>
 | 
			
		||||
 | 
			
		||||
          <span class="ml-5"> {{ dataset.libds }} </span>
 | 
			
		||||
        </button>
 | 
			
		||||
      </clr-tree-node>
 | 
			
		||||
    </clr-tree>
 | 
			
		||||
  </ng-container>
 | 
			
		||||
 | 
			
		||||
  <ng-container *ngIf="submittedCsvDatasets.length">
 | 
			
		||||
    <p cds-text="caption" class="ml-10 mb-10 mt-10">Submitted tables:</p>
 | 
			
		||||
    <clr-tree>
 | 
			
		||||
      <clr-tree-node *ngFor="let dataset of submittedCsvDatasets">
 | 
			
		||||
        <button
 | 
			
		||||
          (click)="onSubmittedCsvDatasetClick(dataset)"
 | 
			
		||||
          class="clr-treenode-link whitespace-nowrap"
 | 
			
		||||
          [class.active]="dataset.active"
 | 
			
		||||
        >
 | 
			
		||||
          <cds-icon
 | 
			
		||||
            *ngIf="dataset.error"
 | 
			
		||||
            status="danger"
 | 
			
		||||
            shape="exclamation-circle"
 | 
			
		||||
          ></cds-icon>
 | 
			
		||||
          <cds-icon
 | 
			
		||||
            *ngIf="dataset.success"
 | 
			
		||||
            status="success"
 | 
			
		||||
            shape="check-circle"
 | 
			
		||||
          ></cds-icon>
 | 
			
		||||
          <cds-icon shape="table"></cds-icon>
 | 
			
		||||
          {{ dataset.libds }}
 | 
			
		||||
        </button>
 | 
			
		||||
      </clr-tree-node>
 | 
			
		||||
    </clr-tree>
 | 
			
		||||
  </ng-container>
 | 
			
		||||
 | 
			
		||||
  <!-- <div *ngIf="librariesPaging" class="w-100 text-center">
 | 
			
		||||
    <span class="spinner spinner-sm"> Loading... </span>
 | 
			
		||||
  </div> -->
 | 
			
		||||
</app-sidebar>
 | 
			
		||||
 | 
			
		||||
<div #contentArea class="content-area">
 | 
			
		||||
  <div class="card no-borders h-100 d-flex clr-flex-column">
 | 
			
		||||
    <div
 | 
			
		||||
      class="header-row clr-row justify-content-between clr-justify-content-center w-100 m-0"
 | 
			
		||||
    >
 | 
			
		||||
      <p cds-text="section">Multi Dataset Load</p>
 | 
			
		||||
    </div>
 | 
			
		||||
 | 
			
		||||
    <div
 | 
			
		||||
      *ngIf="selectedFile === null && !submittingCsv"
 | 
			
		||||
      class="no-table-selected pointer-events-none"
 | 
			
		||||
    >
 | 
			
		||||
      <clr-icon
 | 
			
		||||
        shape="upload-cloud"
 | 
			
		||||
        size="40"
 | 
			
		||||
        class="is-info icon-dc-fill"
 | 
			
		||||
      ></clr-icon>
 | 
			
		||||
      <p class="text-center color-gray mt-10" cds-text="section">
 | 
			
		||||
        Please upload a file
 | 
			
		||||
      </p>
 | 
			
		||||
    </div>
 | 
			
		||||
 | 
			
		||||
    <ng-container *ngIf="selectedFile !== null || submittingCsv">
 | 
			
		||||
      <ng-container *ngIf="!parsedDatasets.length && selectedFile !== null">
 | 
			
		||||
        <div class="d-flex clr-justify-content-center mt-15">
 | 
			
		||||
          <div class="dataset-input-wrapper">
 | 
			
		||||
            <p cds-text="secondary regular" class="mb-5">
 | 
			
		||||
              Selected file: <strong>{{ selectedFile.name }}</strong>
 | 
			
		||||
              <clr-tooltip>
 | 
			
		||||
                <cds-icon
 | 
			
		||||
                  clrTooltipTrigger
 | 
			
		||||
                  (click)="onDiscardFile()"
 | 
			
		||||
                  shape="trash"
 | 
			
		||||
                  status="danger"
 | 
			
		||||
                  class="ml-5 cursor-pointer"
 | 
			
		||||
                ></cds-icon>
 | 
			
		||||
                <clr-tooltip-content> Discard the file </clr-tooltip-content>
 | 
			
		||||
              </clr-tooltip>
 | 
			
		||||
            </p>
 | 
			
		||||
            <p cds-text="secondary regular" class="mb-20">
 | 
			
		||||
              File size: <strong>{{ selectedFile.sizeMB }} MB</strong>
 | 
			
		||||
            </p>
 | 
			
		||||
            <p cds-text="secondary regular" class="mb-15">
 | 
			
		||||
              Paste or type the list of datasets to upload:
 | 
			
		||||
            </p>
 | 
			
		||||
 | 
			
		||||
            <clr-control-helper class="mb-5"
 | 
			
		||||
              >Each row is one dataset. We will automatically detect tables by
 | 
			
		||||
              the sheetname and populate if any.</clr-control-helper
 | 
			
		||||
            >
 | 
			
		||||
 | 
			
		||||
            <hot-table
 | 
			
		||||
              hotId="hotInstanceUserDataset"
 | 
			
		||||
              id="hotTableUserDataset"
 | 
			
		||||
              class="mt-15"
 | 
			
		||||
              [afterGetColHeader]="afterGetColHeader"
 | 
			
		||||
              [settings]="hotUserDatasets"
 | 
			
		||||
              [licenseKey]="hotTableLicenseKey"
 | 
			
		||||
              stretchH="all"
 | 
			
		||||
            >
 | 
			
		||||
            </hot-table>
 | 
			
		||||
 | 
			
		||||
            <div class="dataset-selection-actions text-right mt-10">
 | 
			
		||||
              <button
 | 
			
		||||
                (click)="onStartParsingFile()"
 | 
			
		||||
                id="continue-btn"
 | 
			
		||||
                class="btn btn-primary btn-sm"
 | 
			
		||||
                [disabled]="!matchedDatasets.length"
 | 
			
		||||
                [clrLoading]="uploadLoading"
 | 
			
		||||
              >
 | 
			
		||||
                Continue
 | 
			
		||||
              </button>
 | 
			
		||||
            </div>
 | 
			
		||||
          </div>
 | 
			
		||||
        </div>
 | 
			
		||||
      </ng-container>
 | 
			
		||||
 | 
			
		||||
      <ng-container
 | 
			
		||||
        *ngIf="parsedDatasets.length && !submittedCsvDatasets.length"
 | 
			
		||||
      >
 | 
			
		||||
        <div
 | 
			
		||||
          *ngIf="!activeParsedDataset"
 | 
			
		||||
          class="no-table-selected pointer-events-none"
 | 
			
		||||
        >
 | 
			
		||||
          <ng-container *ngIf="fileLoadingState !== FileLoadingState.parsed">
 | 
			
		||||
            <clr-icon
 | 
			
		||||
              shape="process-on-vm"
 | 
			
		||||
              size="40"
 | 
			
		||||
              class="is-info icon-dc-fill"
 | 
			
		||||
            ></clr-icon>
 | 
			
		||||
 | 
			
		||||
            <p class="text-center color-gray mt-10" cds-text="section">
 | 
			
		||||
              {{ fileLoadingState }}...
 | 
			
		||||
            </p>
 | 
			
		||||
          </ng-container>
 | 
			
		||||
 | 
			
		||||
          <ng-container *ngIf="fileLoadingState === FileLoadingState.parsed">
 | 
			
		||||
            <clr-icon
 | 
			
		||||
              shape="warning-standard"
 | 
			
		||||
              size="40"
 | 
			
		||||
              class="is-info icon-dc-fill"
 | 
			
		||||
            ></clr-icon>
 | 
			
		||||
            <p class="text-center color-gray mt-10" cds-text="section">
 | 
			
		||||
              Please select a dataset on the left to review the data
 | 
			
		||||
            </p>
 | 
			
		||||
          </ng-container>
 | 
			
		||||
        </div>
 | 
			
		||||
 | 
			
		||||
        <ng-container *ngIf="activeParsedDataset">
 | 
			
		||||
          <div
 | 
			
		||||
            *ngIf="activeParsedDataset.submitResult"
 | 
			
		||||
            class="d-flex clr-justify-content-between p-10 mt-15 submission-results"
 | 
			
		||||
          >
 | 
			
		||||
            <div>
 | 
			
		||||
              <p cds-text="secondary regular" class="mb-10">
 | 
			
		||||
                Submit Status:
 | 
			
		||||
                <span
 | 
			
		||||
                  *ngIf="activeParsedDataset.submitResult?.success"
 | 
			
		||||
                  class="color-green"
 | 
			
		||||
                  ><strong>SUCCESS</strong></span
 | 
			
		||||
                >
 | 
			
		||||
                <span
 | 
			
		||||
                  *ngIf="activeParsedDataset.submitResult?.error"
 | 
			
		||||
                  class="color-red"
 | 
			
		||||
                  ><strong>ERROR</strong></span
 | 
			
		||||
                >
 | 
			
		||||
              </p>
 | 
			
		||||
              <p
 | 
			
		||||
                *ngIf="activeParsedDataset.submitResult?.error"
 | 
			
		||||
                cds-text="secondary regular"
 | 
			
		||||
              >
 | 
			
		||||
                Error details:
 | 
			
		||||
              </p>
 | 
			
		||||
            </div>
 | 
			
		||||
 | 
			
		||||
            <div>
 | 
			
		||||
              <button
 | 
			
		||||
                *ngIf="
 | 
			
		||||
                  !submittingCsv && activeParsedDataset.submitResult?.error
 | 
			
		||||
                "
 | 
			
		||||
                (click)="reSubmitTable(activeParsedDataset)"
 | 
			
		||||
                class="btn btn-primary mt-10"
 | 
			
		||||
                [clrLoading]="submitLoading"
 | 
			
		||||
              >
 | 
			
		||||
                Resubmit
 | 
			
		||||
              </button>
 | 
			
		||||
              <button
 | 
			
		||||
                (click)="
 | 
			
		||||
                  downloadFile(
 | 
			
		||||
                    activeParsedDataset.submitResult.log ||
 | 
			
		||||
                      activeParsedDataset.submitResult.success ||
 | 
			
		||||
                      activeParsedDataset.submitResult.error
 | 
			
		||||
                  )
 | 
			
		||||
                "
 | 
			
		||||
                class="btn btn-primary-outline mt-10"
 | 
			
		||||
              >
 | 
			
		||||
                Download log
 | 
			
		||||
              </button>
 | 
			
		||||
            </div>
 | 
			
		||||
          </div>
 | 
			
		||||
 | 
			
		||||
          <div
 | 
			
		||||
            *ngIf="activeParsedDataset.submitResult?.error"
 | 
			
		||||
            class="error-field mt-15"
 | 
			
		||||
          >
 | 
			
		||||
            <div class="log-wrapper">
 | 
			
		||||
              {{ activeParsedDataset.submitResult?.error | json }}
 | 
			
		||||
            </div>
 | 
			
		||||
          </div>
 | 
			
		||||
 | 
			
		||||
          <div class="d-flex clr-justify-content-between p-10 mt-15">
 | 
			
		||||
            <div>
 | 
			
		||||
              <p cds-text="secondary regular" class="mb-10">
 | 
			
		||||
                Found in range:
 | 
			
		||||
 | 
			
		||||
                <ng-container *ngIf="activeParsedDataset.parseResult">
 | 
			
		||||
                  <strong
 | 
			
		||||
                    >"{{
 | 
			
		||||
                      activeParsedDataset.parseResult.rangeSheetRes?.sheetName
 | 
			
		||||
                    }}"!{{
 | 
			
		||||
                      activeParsedDataset.parseResult.rangeSheetRes
 | 
			
		||||
                        ?.rangeAddress
 | 
			
		||||
                    }}</strong
 | 
			
		||||
                  >
 | 
			
		||||
                </ng-container>
 | 
			
		||||
 | 
			
		||||
                <ng-container *ngIf="!activeParsedDataset.parseResult">
 | 
			
		||||
                  <strong *ngIf="!activeParsedDataset.parsingTable"
 | 
			
		||||
                    >No data found</strong
 | 
			
		||||
                  >
 | 
			
		||||
 | 
			
		||||
                  <span
 | 
			
		||||
                    *ngIf="activeParsedDataset.parsingTable"
 | 
			
		||||
                    class="d-flex clr-align-items-center"
 | 
			
		||||
                  >
 | 
			
		||||
                    <strong>Searching for the data...</strong>
 | 
			
		||||
                    <clr-spinner class="ml-5" clrSmall></clr-spinner>
 | 
			
		||||
                  </span>
 | 
			
		||||
                </ng-container>
 | 
			
		||||
              </p>
 | 
			
		||||
              <p cds-text="secondary regular">
 | 
			
		||||
                Dataset:
 | 
			
		||||
                <strong>
 | 
			
		||||
                  <clr-tooltip>
 | 
			
		||||
                    <a
 | 
			
		||||
                      clrTooltipTrigger
 | 
			
		||||
                      [routerLink]="'/editor/' + activeParsedDataset.libds"
 | 
			
		||||
                      >{{ activeParsedDataset.libds }}</a
 | 
			
		||||
                    >
 | 
			
		||||
                    <clr-tooltip-content
 | 
			
		||||
                      [clrPosition]="'top-right'"
 | 
			
		||||
                      [clrSize]="'sm'"
 | 
			
		||||
                    >
 | 
			
		||||
                      Click to edit the table
 | 
			
		||||
                    </clr-tooltip-content>
 | 
			
		||||
                  </clr-tooltip>
 | 
			
		||||
                </strong>
 | 
			
		||||
              </p>
 | 
			
		||||
            </div>
 | 
			
		||||
 | 
			
		||||
            <div>
 | 
			
		||||
              <clr-toggle-wrapper>
 | 
			
		||||
                <input
 | 
			
		||||
                  type="checkbox"
 | 
			
		||||
                  clrToggle
 | 
			
		||||
                  [(ngModel)]="activeParsedDataset.includeInSubmission"
 | 
			
		||||
                  name="options"
 | 
			
		||||
                  [disabled]="
 | 
			
		||||
                    !(
 | 
			
		||||
                      activeParsedDataset.datasource &&
 | 
			
		||||
                      activeParsedDataset.parseResult
 | 
			
		||||
                    )
 | 
			
		||||
                  "
 | 
			
		||||
                  required
 | 
			
		||||
                  value="option1"
 | 
			
		||||
                />
 | 
			
		||||
                <label>Include in submission</label>
 | 
			
		||||
              </clr-toggle-wrapper>
 | 
			
		||||
            </div>
 | 
			
		||||
          </div>
 | 
			
		||||
 | 
			
		||||
          <div *ngIf="isHotHidden" class="text-center w-100">
 | 
			
		||||
            <clr-spinner class="spinner-md"></clr-spinner>
 | 
			
		||||
          </div>
 | 
			
		||||
 | 
			
		||||
          <hot-table
 | 
			
		||||
            hotId="hotInstance"
 | 
			
		||||
            id="hotTable"
 | 
			
		||||
            class="mt-15"
 | 
			
		||||
            [afterGetColHeader]="afterGetColHeader"
 | 
			
		||||
            [className]="['htDark', 'htCustomHidden']"
 | 
			
		||||
            [licenseKey]="hotTableLicenseKey"
 | 
			
		||||
            [multiColumnSorting]="true"
 | 
			
		||||
            [viewportRowRenderingOffset]="50"
 | 
			
		||||
            [manualColumnResize]="true"
 | 
			
		||||
            [filters]="true"
 | 
			
		||||
            stretchH="all"
 | 
			
		||||
          >
 | 
			
		||||
          </hot-table>
 | 
			
		||||
        </ng-container>
 | 
			
		||||
      </ng-container>
 | 
			
		||||
 | 
			
		||||
      <ng-container *ngIf="submittedCsvDatasets.length">
 | 
			
		||||
        <div
 | 
			
		||||
          *ngIf="!activeSubmittedCsvDataset"
 | 
			
		||||
          class="no-table-selected pointer-events-none"
 | 
			
		||||
        >
 | 
			
		||||
          <clr-icon
 | 
			
		||||
            shape="warning-standard"
 | 
			
		||||
            size="40"
 | 
			
		||||
            class="is-info icon-dc-fill"
 | 
			
		||||
          ></clr-icon>
 | 
			
		||||
          <p class="text-center color-gray mt-10" cds-text="section">
 | 
			
		||||
            Please select a dataset on the left to review the submit results
 | 
			
		||||
          </p>
 | 
			
		||||
        </div>
 | 
			
		||||
      </ng-container>
 | 
			
		||||
 | 
			
		||||
      <ng-container *ngIf="activeSubmittedCsvDataset">
 | 
			
		||||
        <div class="d-flex clr-justify-content-between p-10">
 | 
			
		||||
          <div>
 | 
			
		||||
            <p cds-text="secondary regular" class="mb-10">
 | 
			
		||||
              Matched with dataset:
 | 
			
		||||
              <strong>
 | 
			
		||||
                <clr-tooltip>
 | 
			
		||||
                  <a
 | 
			
		||||
                    clrTooltipTrigger
 | 
			
		||||
                    [routerLink]="'/editor/' + activeSubmittedCsvDataset.libds"
 | 
			
		||||
                    >{{ activeSubmittedCsvDataset.libds }}</a
 | 
			
		||||
                  >
 | 
			
		||||
                  <clr-tooltip-content
 | 
			
		||||
                    [clrPosition]="'top-right'"
 | 
			
		||||
                    [clrSize]="'sm'"
 | 
			
		||||
                  >
 | 
			
		||||
                    Click to edit the table
 | 
			
		||||
                  </clr-tooltip-content>
 | 
			
		||||
                </clr-tooltip>
 | 
			
		||||
              </strong>
 | 
			
		||||
            </p>
 | 
			
		||||
            <p cds-text="secondary regular" class="mb-10">
 | 
			
		||||
              Status:
 | 
			
		||||
              <span
 | 
			
		||||
                *ngIf="activeSubmittedCsvDataset.success"
 | 
			
		||||
                class="color-green"
 | 
			
		||||
                ><strong>SUCCESS</strong></span
 | 
			
		||||
              >
 | 
			
		||||
              <span *ngIf="activeSubmittedCsvDataset.error" class="color-red"
 | 
			
		||||
                ><strong>ERROR</strong></span
 | 
			
		||||
              >
 | 
			
		||||
            </p>
 | 
			
		||||
            <p
 | 
			
		||||
              *ngIf="activeSubmittedCsvDataset.error"
 | 
			
		||||
              cds-text="secondary regular"
 | 
			
		||||
            >
 | 
			
		||||
              Error details:
 | 
			
		||||
            </p>
 | 
			
		||||
          </div>
 | 
			
		||||
 | 
			
		||||
          <div>
 | 
			
		||||
            <button
 | 
			
		||||
              (click)="
 | 
			
		||||
                downloadFile(
 | 
			
		||||
                  activeSubmittedCsvDataset.success ||
 | 
			
		||||
                    activeSubmittedCsvDataset.error
 | 
			
		||||
                )
 | 
			
		||||
              "
 | 
			
		||||
              class="btn btn-primary-outline mt-10"
 | 
			
		||||
            >
 | 
			
		||||
              Download log
 | 
			
		||||
            </button>
 | 
			
		||||
          </div>
 | 
			
		||||
        </div>
 | 
			
		||||
 | 
			
		||||
        <div *ngIf="activeSubmittedCsvDataset.error" class="error-field mt-15">
 | 
			
		||||
          <div class="log-wrapper">
 | 
			
		||||
            {{ activeSubmittedCsvDataset.error | json }}
 | 
			
		||||
          </div>
 | 
			
		||||
        </div>
 | 
			
		||||
      </ng-container>
 | 
			
		||||
    </ng-container>
 | 
			
		||||
 | 
			
		||||
    <!-- <div>
 | 
			
		||||
      <p
 | 
			
		||||
        *ngIf="
 | 
			
		||||
          licenceState.value.viewer_rows_allowed !== Infinity &&
 | 
			
		||||
          hotTable.data &&
 | 
			
		||||
          hotTable.data.length > licenceState.value.viewer_rows_allowed
 | 
			
		||||
        "
 | 
			
		||||
        class="mt-2-i w-100 text-center"
 | 
			
		||||
      >
 | 
			
		||||
        To display more than {{ licenceState.value.viewer_rows_allowed }} rows,
 | 
			
		||||
        contact <contact-link />
 | 
			
		||||
      </p>
 | 
			
		||||
    </div> -->
 | 
			
		||||
  </div>
 | 
			
		||||
</div>
 | 
			
		||||
 | 
			
		||||
<clr-modal [(clrModalOpen)]="showSubmitReasonModal" [clrModalClosable]="false">
 | 
			
		||||
  <h3 class="modal-title">
 | 
			
		||||
    Submit {{ tablesToSubmit.length }}
 | 
			
		||||
    {{ tablesToSubmit.length === 1 ? 'table' : 'tables' }} for approval
 | 
			
		||||
  </h3>
 | 
			
		||||
  <div class="modal-body">
 | 
			
		||||
    <p
 | 
			
		||||
      *ngIf="licenceState.value.submit_rows_limit !== Infinity"
 | 
			
		||||
      cds-text="body"
 | 
			
		||||
      class="licence-limit-notice mt-0 mb-15"
 | 
			
		||||
    >
 | 
			
		||||
      Due to current licence, only
 | 
			
		||||
      {{ licenceState.value.submit_rows_limit }} rows in each file will be
 | 
			
		||||
      submitted. To remove the restriction, contact
 | 
			
		||||
      support@datacontroller.io.
 | 
			
		||||
    </p>
 | 
			
		||||
 | 
			
		||||
    <div class="text-area-full-width">
 | 
			
		||||
      <label for="formFields_8" class="mb-5 d-block">Message</label>
 | 
			
		||||
      <textarea
 | 
			
		||||
        clrTextarea
 | 
			
		||||
        [(ngModel)]="submitReasonMessage"
 | 
			
		||||
        tabindex="0"
 | 
			
		||||
        class="submit-reason"
 | 
			
		||||
        type="text"
 | 
			
		||||
        id="formFields_8"
 | 
			
		||||
      ></textarea>
 | 
			
		||||
    </div>
 | 
			
		||||
 | 
			
		||||
    <p cds-text="caption_clean" class="mt-10">
 | 
			
		||||
      Tables will be sent sequentially, logs will be available after all tables
 | 
			
		||||
      are submitted.
 | 
			
		||||
    </p>
 | 
			
		||||
  </div>
 | 
			
		||||
  <div class="modal-footer">
 | 
			
		||||
    <button
 | 
			
		||||
      type="button"
 | 
			
		||||
      class="btn btn-outline"
 | 
			
		||||
      [disabled]="submitLoading"
 | 
			
		||||
      (click)="showSubmitReasonModal = false"
 | 
			
		||||
    >
 | 
			
		||||
      Cancel
 | 
			
		||||
    </button>
 | 
			
		||||
    <button
 | 
			
		||||
      type="button"
 | 
			
		||||
      id="submit-tables"
 | 
			
		||||
      class="btn btn-primary"
 | 
			
		||||
      [clrLoading]="submitLoading"
 | 
			
		||||
      (click)="submitTables()"
 | 
			
		||||
    >
 | 
			
		||||
      Submit
 | 
			
		||||
    </button>
 | 
			
		||||
  </div>
 | 
			
		||||
</clr-modal>
 | 
			
		||||
 | 
			
		||||
<clr-modal [(clrModalOpen)]="csvSubmitting" [clrModalClosable]="false">
 | 
			
		||||
  <h3 class="modal-title">
 | 
			
		||||
    Submitting {{ csvFiles.length }} CSV
 | 
			
		||||
    {{ csvFiles.length === 1 ? 'file' : 'files' }}
 | 
			
		||||
  </h3>
 | 
			
		||||
  <div class="modal-body">
 | 
			
		||||
    <div class="text-center">
 | 
			
		||||
      <clr-spinner clrMedium></clr-spinner>
 | 
			
		||||
    </div>
 | 
			
		||||
 | 
			
		||||
    <p cds-text="caption_clean" class="mt-10 text-center">
 | 
			
		||||
      This will take few moments
 | 
			
		||||
    </p>
 | 
			
		||||
  </div>
 | 
			
		||||
</clr-modal>
 | 
			
		||||
							
								
								
									
										1095
									
								
								client/src/app/multi-dataset/multi-dataset.component.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1095
									
								
								client/src/app/multi-dataset/multi-dataset.component.ts
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										31
									
								
								client/src/app/multi-dataset/multi-dataset.module.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								client/src/app/multi-dataset/multi-dataset.module.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,31 @@
 | 
			
		||||
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 { DcTreeModule } from '../shared/dc-tree/dc-tree.module'
 | 
			
		||||
import { MultiDatasetComponent } from './multi-dataset.component'
 | 
			
		||||
import { MultiDatasetRoutingModule } from './multi-dataset-routing.module'
 | 
			
		||||
import { MultiDatasetRouteComponent } from '../routes/multi-dataset-route/multi-dataset-route.component'
 | 
			
		||||
 | 
			
		||||
// register Handsontable's modules
 | 
			
		||||
registerAllModules()
 | 
			
		||||
 | 
			
		||||
@NgModule({
 | 
			
		||||
  declarations: [MultiDatasetRouteComponent, MultiDatasetComponent],
 | 
			
		||||
  imports: [
 | 
			
		||||
    HotTableModule,
 | 
			
		||||
    MultiDatasetRoutingModule,
 | 
			
		||||
    FormsModule,
 | 
			
		||||
    ClarityModule,
 | 
			
		||||
    AppSharedModule,
 | 
			
		||||
    CommonModule,
 | 
			
		||||
    DcTreeModule,
 | 
			
		||||
    DirectivesModule
 | 
			
		||||
  ],
 | 
			
		||||
  exports: [MultiDatasetComponent]
 | 
			
		||||
})
 | 
			
		||||
export class MultiDatasetModule {}
 | 
			
		||||
@@ -1,4 +1,4 @@
 | 
			
		||||
import { Component, OnInit } from '@angular/core'
 | 
			
		||||
import { Component, OnInit, ViewEncapsulation } from '@angular/core'
 | 
			
		||||
 | 
			
		||||
@Component({
 | 
			
		||||
  selector: 'app-not-found',
 | 
			
		||||
@@ -6,7 +6,8 @@ import { Component, OnInit } from '@angular/core'
 | 
			
		||||
  styleUrls: ['./not-found.component.scss'],
 | 
			
		||||
  host: {
 | 
			
		||||
    class: 'content-container'
 | 
			
		||||
  }
 | 
			
		||||
  },
 | 
			
		||||
  encapsulation: ViewEncapsulation.None
 | 
			
		||||
})
 | 
			
		||||
export class NotFoundComponent implements OnInit {
 | 
			
		||||
  constructor() {}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,283 +0,0 @@
 | 
			
		||||
.content {
 | 
			
		||||
    display: flex;
 | 
			
		||||
 | 
			
		||||
    .clauses-container {
 | 
			
		||||
        display: flex;
 | 
			
		||||
        flex-direction: column;
 | 
			
		||||
 | 
			
		||||
        .clause-logic {
 | 
			
		||||
            display: flex;
 | 
			
		||||
            justify-content: center;
 | 
			
		||||
            flex-direction: column;
 | 
			
		||||
            background: #e9e9e9;
 | 
			
		||||
            padding: 15px;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        .clause-query {
 | 
			
		||||
            padding: 30px 0px 20px 20px;
 | 
			
		||||
            background: #fbf8f8;
 | 
			
		||||
            display: flex;
 | 
			
		||||
            justify-content: center;
 | 
			
		||||
            flex-direction: column;
 | 
			
		||||
            position: relative;
 | 
			
		||||
 | 
			
		||||
            & > .clr-row {
 | 
			
		||||
                justify-content: space-between;
 | 
			
		||||
 | 
			
		||||
                &:not(:last-child) {
 | 
			
		||||
                    padding-bottom: 15px;
 | 
			
		||||
                    margin-bottom: 15px;
 | 
			
		||||
                    border-bottom: 1px solid rgba(0, 0, 0, 0.16)
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            .remove-group-clause-button {
 | 
			
		||||
                position: absolute;
 | 
			
		||||
                top: 0px;
 | 
			
		||||
                right: 10px;
 | 
			
		||||
                color: gray;
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            .variable-col {
 | 
			
		||||
                display: flex;
 | 
			
		||||
                align-items: flex-start;
 | 
			
		||||
                padding-bottom: 1px;
 | 
			
		||||
 | 
			
		||||
                .datalist-wrapper {
 | 
			
		||||
                    width: 100%;
 | 
			
		||||
 | 
			
		||||
                    input {
 | 
			
		||||
                        width: 100%;
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            .operator-col {
 | 
			
		||||
                display: flex;
 | 
			
		||||
                align-items: flex-start;
 | 
			
		||||
 | 
			
		||||
                clr-select-container {
 | 
			
		||||
                    height: 45px;
 | 
			
		||||
                    margin-top: 0;
 | 
			
		||||
                    width: 100%;
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            .value-col {
 | 
			
		||||
                display: flex;
 | 
			
		||||
                align-items: flex-start;
 | 
			
		||||
                padding-bottom: 1px;
 | 
			
		||||
 | 
			
		||||
                .checkbox-vals {
 | 
			
		||||
                    width: 100%;
 | 
			
		||||
                    padding: 0px 5px;
 | 
			
		||||
                    border-bottom: 1px solid rgba(0, 0, 0, 0.3);
 | 
			
		||||
 | 
			
		||||
                    clr-checkbox-container {
 | 
			
		||||
                        margin-top: 0;
 | 
			
		||||
                    }
 | 
			
		||||
 | 
			
		||||
                    section {
 | 
			
		||||
                        max-height: 120px;
 | 
			
		||||
                        overflow-y: scroll;
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
                .single-field-vals {
 | 
			
		||||
                    width: 100%;
 | 
			
		||||
 | 
			
		||||
                    ::ng-deep {
 | 
			
		||||
                        .clr-control-container {
 | 
			
		||||
                            width: 100%;
 | 
			
		||||
 | 
			
		||||
                            .clr-input-wrapper {
 | 
			
		||||
                                max-width: none;
 | 
			
		||||
 | 
			
		||||
                                .clr-input-group {
 | 
			
		||||
                                    width: 100%;
 | 
			
		||||
                                }
 | 
			
		||||
                            }
 | 
			
		||||
                        }
 | 
			
		||||
                    }
 | 
			
		||||
 | 
			
		||||
                    & > input {
 | 
			
		||||
                        width: 100%;
 | 
			
		||||
                    }
 | 
			
		||||
 | 
			
		||||
                    input[type=time] {
 | 
			
		||||
                        width: 100%;
 | 
			
		||||
                        padding-right: 17px;
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
                .range-vals {
 | 
			
		||||
                    width: 100%;
 | 
			
		||||
 | 
			
		||||
                    ::ng-deep {
 | 
			
		||||
                        .clr-control-container {
 | 
			
		||||
                            width: 100%;
 | 
			
		||||
 | 
			
		||||
                            .clr-input-wrapper {
 | 
			
		||||
                                max-width: none;
 | 
			
		||||
 | 
			
		||||
                                .clr-input-group {
 | 
			
		||||
                                    width: 100%;
 | 
			
		||||
                                }
 | 
			
		||||
                            }
 | 
			
		||||
                        }
 | 
			
		||||
                    }
 | 
			
		||||
 | 
			
		||||
                    .from {
 | 
			
		||||
                        margin-bottom: 10px;
 | 
			
		||||
 | 
			
		||||
                        & > input {
 | 
			
		||||
                            width: 100%;
 | 
			
		||||
                        }
 | 
			
		||||
 | 
			
		||||
                        input[type=time] {
 | 
			
		||||
                            width: 100%;
 | 
			
		||||
                            padding-right: 17px;
 | 
			
		||||
                        }
 | 
			
		||||
                    }
 | 
			
		||||
 | 
			
		||||
                    .from, .to {
 | 
			
		||||
                        min-width: 100px;
 | 
			
		||||
 | 
			
		||||
                        & > input {
 | 
			
		||||
                            width: 100%;
 | 
			
		||||
                        }
 | 
			
		||||
 | 
			
		||||
                        input[type=time] {
 | 
			
		||||
                            width: 100%;
 | 
			
		||||
                            padding-right: 17px;
 | 
			
		||||
                        }
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
                .contains-vals {
 | 
			
		||||
                    width: 100%;
 | 
			
		||||
 | 
			
		||||
                    ::ng-deep {
 | 
			
		||||
                        .clr-control-container {
 | 
			
		||||
                            width: 100%;
 | 
			
		||||
 | 
			
		||||
                            .clr-input-wrapper {
 | 
			
		||||
                                max-width: none;
 | 
			
		||||
 | 
			
		||||
                                .clr-input-group {
 | 
			
		||||
                                    width: 100%;
 | 
			
		||||
                                }
 | 
			
		||||
                            }
 | 
			
		||||
                        }
 | 
			
		||||
                    }
 | 
			
		||||
 | 
			
		||||
                    & > input {
 | 
			
		||||
                        width: 100%;
 | 
			
		||||
                    }
 | 
			
		||||
 | 
			
		||||
                    input[type=time] {
 | 
			
		||||
                        width: 100%;
 | 
			
		||||
                        padding-right: 17px;
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            .clause-buttons {
 | 
			
		||||
                display: flex;
 | 
			
		||||
                justify-content: space-around;
 | 
			
		||||
                align-items: center;
 | 
			
		||||
                flex-direction: row;
 | 
			
		||||
                align-items: center;
 | 
			
		||||
 | 
			
		||||
                button {
 | 
			
		||||
                    min-width: auto;
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.invalid-clause {
 | 
			
		||||
    border-left: 2px solid #d94b31;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.clause-row {
 | 
			
		||||
    clr-icon {
 | 
			
		||||
        margin: 0;
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.clause-row:after {
 | 
			
		||||
    position: relative;
 | 
			
		||||
    content: "";
 | 
			
		||||
    height: .41667rem;
 | 
			
		||||
    width: .41667rem;
 | 
			
		||||
    top: .29167rem;
 | 
			
		||||
    right: .25rem;
 | 
			
		||||
    background-image: url(data:image/svg+xml;charset=utf8,%3Csvg%20xmlns%3D%22http%3A%2F%2Fwww.w3.org…%2C9.84%2C3.24a0.68%2C0.68%2C0%2C1%2C1%2C1%2C1Z%22%2F%3E%0A%3C%2Fsvg%3E%0A);
 | 
			
		||||
    background-repeat: no-repeat;
 | 
			
		||||
    background-size: contain;
 | 
			
		||||
    vertical-align: middle;
 | 
			
		||||
    margin: 0;
 | 
			
		||||
}
 | 
			
		||||
:not(pre) > code[class*="language-"], pre[class*="language-"] {
 | 
			
		||||
    background-color: #fbf8f8;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
pre[class*="language-"] {
 | 
			
		||||
    padding: 8px;
 | 
			
		||||
    margin: 0;
 | 
			
		||||
    border-radius: 1px;
 | 
			
		||||
 | 
			
		||||
    display: flex;
 | 
			
		||||
    justify-content: center;
 | 
			
		||||
    align-items: center;
 | 
			
		||||
    min-height: 66px;
 | 
			
		||||
 | 
			
		||||
    position: relative;
 | 
			
		||||
 | 
			
		||||
    span.spinner {
 | 
			
		||||
        position: absolute;
 | 
			
		||||
        left: 10px;
 | 
			
		||||
        top: 10px;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    code {
 | 
			
		||||
        white-space: pre-wrap;
 | 
			
		||||
        word-break: break-word;
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.input-val {
 | 
			
		||||
    border: 0px;
 | 
			
		||||
    background: #fbf8f8;
 | 
			
		||||
    border-bottom: 1px solid #999999;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
clr-date-container {
 | 
			
		||||
    margin-top: 2px !important;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
input[type="time"] {
 | 
			
		||||
    border: 0;
 | 
			
		||||
    background: transparent;
 | 
			
		||||
    border-bottom: 1px solid #b3b3b3;
 | 
			
		||||
 | 
			
		||||
    &:focus {
 | 
			
		||||
        outline: none;
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.in-values-modal {
 | 
			
		||||
    .modal-footer {
 | 
			
		||||
        border-top: 1px solid #d8d8d8;
 | 
			
		||||
        margin-top: 10px;
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.progress, .progress-static {
 | 
			
		||||
    background-color: transparent;
 | 
			
		||||
    width: 100%;
 | 
			
		||||
    height: 4px;
 | 
			
		||||
    top: 3px;
 | 
			
		||||
}
 | 
			
		||||
@@ -6,7 +6,8 @@ import {
 | 
			
		||||
  OnDestroy,
 | 
			
		||||
  ChangeDetectorRef,
 | 
			
		||||
  LOCALE_ID,
 | 
			
		||||
  Input
 | 
			
		||||
  Input,
 | 
			
		||||
  ViewEncapsulation
 | 
			
		||||
} from '@angular/core'
 | 
			
		||||
import { SasStoreService } from '../services/sas-store.service'
 | 
			
		||||
import { globals } from '../_globals'
 | 
			
		||||
@@ -27,7 +28,8 @@ registerLocaleData(localeEnGB)
 | 
			
		||||
  selector: 'app-query',
 | 
			
		||||
  templateUrl: './query.component.html',
 | 
			
		||||
  styleUrls: ['./query.component.scss'],
 | 
			
		||||
  providers: [{ provide: LOCALE_ID, useValue: 'en-GB' }]
 | 
			
		||||
  providers: [{ provide: LOCALE_ID, useValue: 'en-GB' }],
 | 
			
		||||
  encapsulation: ViewEncapsulation.None
 | 
			
		||||
})
 | 
			
		||||
export class QueryComponent
 | 
			
		||||
  implements AfterViewInit, AfterContentInit, OnDestroy
 | 
			
		||||
 
 | 
			
		||||
@@ -284,19 +284,15 @@
 | 
			
		||||
                    >
 | 
			
		||||
                      <span class="label label-warning">
 | 
			
		||||
                        Changed Rows
 | 
			
		||||
                        <span class="badge badge-warning">{{
 | 
			
		||||
                          lens.updated
 | 
			
		||||
                        }}</span>
 | 
			
		||||
                        <span class="badge">{{ lens.updated }}</span>
 | 
			
		||||
                      </span>
 | 
			
		||||
                      <span class="label label-success">
 | 
			
		||||
                        Added Rows
 | 
			
		||||
                        <span class="badge badge-success">{{ lens.new }}</span>
 | 
			
		||||
                        <span class="badge">{{ lens.new }}</span>
 | 
			
		||||
                      </span>
 | 
			
		||||
                      <span class="label label-danger">
 | 
			
		||||
                        Deleted Rows
 | 
			
		||||
                        <span class="badge badge-danger">{{
 | 
			
		||||
                          lens.deleted
 | 
			
		||||
                        }}</span>
 | 
			
		||||
                        <span class="badge">{{ lens.deleted }}</span>
 | 
			
		||||
                      </span>
 | 
			
		||||
                    </div>
 | 
			
		||||
                  </div>
 | 
			
		||||
@@ -311,8 +307,8 @@
 | 
			
		||||
          class="h-24vh d-flex flex-column justify-content-center align-items-center"
 | 
			
		||||
        >
 | 
			
		||||
          <span class="spinner"> Loading... </span>
 | 
			
		||||
          <div *ngIf="!loadingTable">
 | 
			
		||||
            <h3>Loading table</h3>
 | 
			
		||||
          <div *ngIf="!loadingTable" class="mt-10">
 | 
			
		||||
            <p cds-text="section">Loading table</p>
 | 
			
		||||
          </div>
 | 
			
		||||
        </div>
 | 
			
		||||
 | 
			
		||||
@@ -477,15 +473,15 @@
 | 
			
		||||
                >
 | 
			
		||||
                  <span class="label label-warning">
 | 
			
		||||
                    Changed Rows
 | 
			
		||||
                    <span class="badge badge-warning">{{ lens.updated }}</span>
 | 
			
		||||
                    <span class="badge">{{ lens.updated }}</span>
 | 
			
		||||
                  </span>
 | 
			
		||||
                  <span class="label label-success">
 | 
			
		||||
                    Added Rows
 | 
			
		||||
                    <span class="badge badge-success">{{ lens.new }}</span>
 | 
			
		||||
                    <span class="badge">{{ lens.new }}</span>
 | 
			
		||||
                  </span>
 | 
			
		||||
                  <span class="label label-danger">
 | 
			
		||||
                    Deleted Rows
 | 
			
		||||
                    <span class="badge badge-danger">{{ lens.deleted }}</span>
 | 
			
		||||
                    <span class="badge">{{ lens.deleted }}</span>
 | 
			
		||||
                  </span>
 | 
			
		||||
                </div>
 | 
			
		||||
              </div>
 | 
			
		||||
@@ -519,8 +515,8 @@
 | 
			
		||||
          class="h-25vh d-flex flex-column justify-content-center align-items-center"
 | 
			
		||||
        >
 | 
			
		||||
          <span class="spinner"> Loading... </span>
 | 
			
		||||
          <div *ngIf="!loadingTable">
 | 
			
		||||
            <h3>Loading table</h3>
 | 
			
		||||
          <div *ngIf="!loadingTable" class="mt-10">
 | 
			
		||||
            <p cds-text="section">Loading table</p>
 | 
			
		||||
          </div>
 | 
			
		||||
        </div>
 | 
			
		||||
        <div class="tableCont">
 | 
			
		||||
 
 | 
			
		||||
@@ -1,179 +0,0 @@
 | 
			
		||||
.loader {
 | 
			
		||||
  display:flex;
 | 
			
		||||
  justify-content: center;
 | 
			
		||||
  height:75vh;
 | 
			
		||||
  align-items:center;
 | 
			
		||||
  flex-direction:column
 | 
			
		||||
}
 | 
			
		||||
.modalLarge {
 | 
			
		||||
    width: 50rem!important;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.addedRow {
 | 
			
		||||
  background:  rgb(146, 208, 154);
 | 
			
		||||
  border: 1px solid rgba(9, 77, 117, 0.2);
 | 
			
		||||
  border-radius: 5px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.deletedRow {
 | 
			
		||||
  background: rgb(230, 179, 179);
 | 
			
		||||
  border: 1px solid rgba(70, 71, 70, 0.2);
 | 
			
		||||
  border-radius: 5px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.updatedRow {
 | 
			
		||||
  background: #fafda8;
 | 
			
		||||
  border: 1px solid rgba(9, 117, 9, 0.2);
 | 
			
		||||
  border-radius: 5px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.table {
 | 
			
		||||
  border: 0px solid;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.ch {
 | 
			
		||||
  background: rgba(0,0,0,.1);
 | 
			
		||||
  border: 1px solid rgba(104, 100, 0, 0.4);
 | 
			
		||||
  border-radius: 5px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.ch:hover {
 | 
			
		||||
background: rgba(252, 135, 120, 0.4);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.tooltip .tooltip-content.tooltip-top-right, .tooltip.tooltip-top-right>.tooltip-content, .tooltip>.tooltip-content {
 | 
			
		||||
    font-size: .54167rem;
 | 
			
		||||
    font-weight: 400;
 | 
			
		||||
    letter-spacing: normal;
 | 
			
		||||
    background: #314351;
 | 
			
		||||
    border-radius: .125rem;
 | 
			
		||||
    color: #f0f1ec;;
 | 
			
		||||
    line-height: .75rem;
 | 
			
		||||
    margin: 0;
 | 
			
		||||
    padding: .375rem .5rem;
 | 
			
		||||
    width: 235px;
 | 
			
		||||
    position: absolute;
 | 
			
		||||
    top: auto;
 | 
			
		||||
    bottom: 100%;
 | 
			
		||||
    left: 12px;
 | 
			
		||||
    right: auto;
 | 
			
		||||
    border-bottom-left-radius: 0;
 | 
			
		||||
    margin-bottom: .66667rem;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.tooltip .tooltip-content.tooltip-top-right:before, .tooltip.tooltip-top-right>.tooltip-content:before, .tooltip>.tooltip-content:before {
 | 
			
		||||
    position: absolute;
 | 
			
		||||
    bottom: -.375rem;
 | 
			
		||||
    left: 0;
 | 
			
		||||
    top: auto;
 | 
			
		||||
    right: auto;
 | 
			
		||||
    content: "";
 | 
			
		||||
    border-left: .25rem solid #314351;
 | 
			
		||||
    border-top: .20833rem solid #314351;
 | 
			
		||||
    border-right: .25rem solid transparent;
 | 
			
		||||
    border-bottom: .20833rem solid transparent;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.table {
 | 
			
		||||
border: 0px solid;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.toggle-switch input[type=checkbox]:checked+label:before {
 | 
			
		||||
    border-color: #314351;
 | 
			
		||||
    background-color: #314351!important;
 | 
			
		||||
    transition: .15s ease-in;
 | 
			
		||||
    transition-property: border-color,background-color;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.tableCont {
 | 
			
		||||
  overflow:auto;
 | 
			
		||||
  margin: 15px 10px 10px 10px;
 | 
			
		||||
 | 
			
		||||
  td {
 | 
			
		||||
    word-break: break-word;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.approvalInfo {
 | 
			
		||||
  display: flex;
 | 
			
		||||
  justify-content: flex-end
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.approvalBack {
 | 
			
		||||
  display: flex;
 | 
			
		||||
  justify-content: flex-start;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@media screen and (max-width:768px) {
 | 
			
		||||
  .approvalInfo {
 | 
			
		||||
    display: flex;
 | 
			
		||||
    justify-content: center;
 | 
			
		||||
    margin-top: 15px;
 | 
			
		||||
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .approvalBack {
 | 
			
		||||
    display: flex;
 | 
			
		||||
    justify-content: center;
 | 
			
		||||
    margin-bottom: 15px;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .card {
 | 
			
		||||
    margin-top:0rem!important;
 | 
			
		||||
    min-height: calc(100vh - 0px)!important;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .table td.left, .table th.left {
 | 
			
		||||
    text-align: left;
 | 
			
		||||
    width: 150px!important;
 | 
			
		||||
    flex: 0
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.table td.left, .table th.left {
 | 
			
		||||
  text-align: left;
 | 
			
		||||
  flex: 1;
 | 
			
		||||
  width: 300px!important;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
.tooll {
 | 
			
		||||
    position: absolute;
 | 
			
		||||
    background: #e6b3b3;
 | 
			
		||||
    color: #314351;
 | 
			
		||||
    top: 0px;
 | 
			
		||||
    height: 36px;
 | 
			
		||||
    width: 100%;
 | 
			
		||||
    left: 0px;
 | 
			
		||||
    justify-content: center;
 | 
			
		||||
    align-items: center;
 | 
			
		||||
    display: flex;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#acceptBtn, #rejectBtn {
 | 
			
		||||
  width: 175px
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.formatted-values-toggle {
 | 
			
		||||
  min-width: 75px
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
clr-modal {
 | 
			
		||||
  ::ng-deep {
 | 
			
		||||
    .modal-body-wrapper {
 | 
			
		||||
      overflow: auto;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.rows-notice {
 | 
			
		||||
  display: flex;
 | 
			
		||||
  align-items: center;
 | 
			
		||||
  margin-right: 10px;
 | 
			
		||||
  color: #6a6a6a;
 | 
			
		||||
  font-size: 15px;
 | 
			
		||||
 | 
			
		||||
  clr-icon {
 | 
			
		||||
    margin: 0;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,6 +1,11 @@
 | 
			
		||||
import { ActivatedRoute } from '@angular/router'
 | 
			
		||||
import { SasStoreService } from '../../services/sas-store.service'
 | 
			
		||||
import { Component, AfterViewInit, OnDestroy } from '@angular/core'
 | 
			
		||||
import {
 | 
			
		||||
  Component,
 | 
			
		||||
  AfterViewInit,
 | 
			
		||||
  OnDestroy,
 | 
			
		||||
  ViewEncapsulation
 | 
			
		||||
} from '@angular/core'
 | 
			
		||||
import { Subscription } from 'rxjs'
 | 
			
		||||
import { Router } from '@angular/router'
 | 
			
		||||
import { EventService } from '../../services/event.service'
 | 
			
		||||
@@ -22,7 +27,8 @@ interface ChangesObj {
 | 
			
		||||
  styleUrls: ['./approve-details.component.scss'],
 | 
			
		||||
  host: {
 | 
			
		||||
    class: 'content-container'
 | 
			
		||||
  }
 | 
			
		||||
  },
 | 
			
		||||
  encapsulation: ViewEncapsulation.None
 | 
			
		||||
})
 | 
			
		||||
export class ApproveDetailsComponent implements AfterViewInit, OnDestroy {
 | 
			
		||||
  private _detailsSub: Subscription | undefined
 | 
			
		||||
@@ -349,13 +355,12 @@ export class ApproveDetailsComponent implements AfterViewInit, OnDestroy {
 | 
			
		||||
            this.params = param
 | 
			
		||||
            this.response = res
 | 
			
		||||
            this.calcDiff()
 | 
			
		||||
            this.callChangesInfo(this.tableId)
 | 
			
		||||
          })
 | 
			
		||||
          .catch((err: any) => err)
 | 
			
		||||
          .finally(() => {
 | 
			
		||||
            this.loadingTable = true
 | 
			
		||||
          })
 | 
			
		||||
 | 
			
		||||
        this.callChangesInfo(this.tableId)
 | 
			
		||||
      }
 | 
			
		||||
    )
 | 
			
		||||
    if (typeof this.router.snapshot.params['tableId'] === 'undefined') {
 | 
			
		||||
@@ -378,6 +383,7 @@ export class ApproveDetailsComponent implements AfterViewInit, OnDestroy {
 | 
			
		||||
        this.params = param
 | 
			
		||||
        this.response = res
 | 
			
		||||
        this.calcDiff()
 | 
			
		||||
        this.callChangesInfo(this.tableId)
 | 
			
		||||
      })
 | 
			
		||||
      .catch((err: any) => {
 | 
			
		||||
        this.acceptLoading = false
 | 
			
		||||
@@ -386,8 +392,6 @@ export class ApproveDetailsComponent implements AfterViewInit, OnDestroy {
 | 
			
		||||
        this.loadingTable = true
 | 
			
		||||
        this.setFocus()
 | 
			
		||||
      })
 | 
			
		||||
 | 
			
		||||
    this.callChangesInfo(this.tableId)
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  ngOnDestroy() {
 | 
			
		||||
 
 | 
			
		||||
@@ -89,6 +89,7 @@
 | 
			
		||||
            <clr-dg-cell class="p-0 d-flex justify-content-center">
 | 
			
		||||
              <button
 | 
			
		||||
                class="btn btn-success"
 | 
			
		||||
                aria-label="Download audit file"
 | 
			
		||||
                [id]="approveItem.tableId"
 | 
			
		||||
                (click)="
 | 
			
		||||
                  download(approveItem.tableId); $event.stopPropagation()
 | 
			
		||||
@@ -99,19 +100,11 @@
 | 
			
		||||
            </clr-dg-cell>
 | 
			
		||||
          </clr-dg-row>
 | 
			
		||||
 | 
			
		||||
          <clr-dg-footer class="d-flex justify-content-start">
 | 
			
		||||
            <span>items per page</span>
 | 
			
		||||
            <select [(ngModel)]="itemsNum">
 | 
			
		||||
              <option [ngValue]="3">3</option>
 | 
			
		||||
              <option [ngValue]="5">5</option>
 | 
			
		||||
              <option [ngValue]="10">10</option>
 | 
			
		||||
              <option [ngValue]="15">15</option>
 | 
			
		||||
            </select>
 | 
			
		||||
            <clr-dg-pagination
 | 
			
		||||
              #pagination
 | 
			
		||||
              [clrDgPageSize]="itemsNum"
 | 
			
		||||
              class="center"
 | 
			
		||||
            >
 | 
			
		||||
          <clr-dg-footer>
 | 
			
		||||
            <clr-dg-pagination #pagination [clrDgPageSize]="10">
 | 
			
		||||
              <clr-dg-page-size [clrPageSizeOptions]="[3, 5, 10, 15]"
 | 
			
		||||
                >Items per page</clr-dg-page-size
 | 
			
		||||
              >
 | 
			
		||||
              {{ pagination.firstItem + 1 }} - {{ pagination.lastItem + 1 }} of
 | 
			
		||||
              {{ pagination.totalItems }} approvals
 | 
			
		||||
            </clr-dg-pagination>
 | 
			
		||||
 
 | 
			
		||||
@@ -1,41 +0,0 @@
 | 
			
		||||
.column-center {
 | 
			
		||||
    display: flex;
 | 
			
		||||
    flex-direction: column;
 | 
			
		||||
    justify-content: center;
 | 
			
		||||
    align-items: center;
 | 
			
		||||
}
 | 
			
		||||
.datagrid .datagrid-column .datagrid-column-title{
 | 
			
		||||
  outline: none!important;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.links {
 | 
			
		||||
  font-weight: 700;cursor: pointer;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.tooltip.tooltip-bottom-left>.tooltip-content, .tooltip .tooltip-content.tooltip-bottom-left {
 | 
			
		||||
    background: #314351!important;
 | 
			
		||||
}
 | 
			
		||||
.tooltip.tooltip-bottom-left>.tooltip-content:before, .tooltip .tooltip-content.tooltip-bottom-left:before {
 | 
			
		||||
    border-right: .25rem solid #314351;
 | 
			
		||||
    border-bottom: .20833rem solid #314351;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.noBorder {
 | 
			
		||||
  border-bottom: 1px solid transparent!important;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.approvals-list-wrapper {
 | 
			
		||||
  height: 70vh;
 | 
			
		||||
  display: flex;
 | 
			
		||||
  justify-content: center;
 | 
			
		||||
  flex-direction: column;
 | 
			
		||||
  align-items: center;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.noapprovals-info-wrapper {
 | 
			
		||||
  display: flex;
 | 
			
		||||
  justify-content: center;
 | 
			
		||||
  flex-direction: column;
 | 
			
		||||
  align-items: center;
 | 
			
		||||
  height: calc(100vh - 200px);
 | 
			
		||||
}
 | 
			
		||||
@@ -1,4 +1,9 @@
 | 
			
		||||
import { Component, OnInit, ChangeDetectorRef } from '@angular/core'
 | 
			
		||||
import {
 | 
			
		||||
  Component,
 | 
			
		||||
  OnInit,
 | 
			
		||||
  ChangeDetectorRef,
 | 
			
		||||
  ViewEncapsulation
 | 
			
		||||
} from '@angular/core'
 | 
			
		||||
import { SasStoreService } from '../../services/sas-store.service'
 | 
			
		||||
import { Router } from '@angular/router'
 | 
			
		||||
import { SasService } from '../../services/sas.service'
 | 
			
		||||
@@ -20,7 +25,8 @@ interface ApproveData {
 | 
			
		||||
  styleUrls: ['./approve.component.scss'],
 | 
			
		||||
  host: {
 | 
			
		||||
    class: 'content-container'
 | 
			
		||||
  }
 | 
			
		||||
  },
 | 
			
		||||
  encapsulation: ViewEncapsulation.None
 | 
			
		||||
})
 | 
			
		||||
export class ApproveComponent implements OnInit {
 | 
			
		||||
  public approveList: Array<ApproveData> | undefined
 | 
			
		||||
 
 | 
			
		||||
@@ -24,19 +24,19 @@
 | 
			
		||||
              <a
 | 
			
		||||
                *ngIf="ind < 1"
 | 
			
		||||
                (click)="getTable(approveData[col])"
 | 
			
		||||
                class="cursor-pointer"
 | 
			
		||||
                class="cursor-pointer table-link"
 | 
			
		||||
                >{{ approveData[col] }}</a
 | 
			
		||||
              >
 | 
			
		||||
              <div *ngIf="ind < 2 && ind >= 1">
 | 
			
		||||
                <a
 | 
			
		||||
                  (click)="getBaseTable(approveData[col])"
 | 
			
		||||
                  class="cursor-pointer"
 | 
			
		||||
                  class="cursor-pointer table-link"
 | 
			
		||||
                  >VIEW</a
 | 
			
		||||
                >
 | 
			
		||||
                <span> / </span>
 | 
			
		||||
                <a
 | 
			
		||||
                  (click)="getEditTable(approveData[col])"
 | 
			
		||||
                  class="cursor-pointer"
 | 
			
		||||
                  class="cursor-pointer table-link"
 | 
			
		||||
                  >EDIT</a
 | 
			
		||||
                >
 | 
			
		||||
              </div>
 | 
			
		||||
@@ -47,7 +47,12 @@
 | 
			
		||||
      </table>
 | 
			
		||||
    </div>
 | 
			
		||||
    <div class="modal-footer">
 | 
			
		||||
      <button type="button" class="btn btn-outline" (click)="openModal = false">
 | 
			
		||||
      <button
 | 
			
		||||
        type="button"
 | 
			
		||||
        aria-label="Close modal"
 | 
			
		||||
        class="btn btn-outline"
 | 
			
		||||
        (click)="openModal = false"
 | 
			
		||||
      >
 | 
			
		||||
        OK
 | 
			
		||||
      </button>
 | 
			
		||||
    </div>
 | 
			
		||||
@@ -119,6 +124,7 @@
 | 
			
		||||
        <clr-dg-cell class="verCenter">{{ historyItem.reviewed }}</clr-dg-cell>
 | 
			
		||||
        <clr-dg-cell class="verCenter p-0 d-flex justify-content-center">
 | 
			
		||||
          <button
 | 
			
		||||
            aria-label="Download audit file"
 | 
			
		||||
            class="btn btn-success"
 | 
			
		||||
            (click)="download(historyItem.tableId); $event.stopPropagation()"
 | 
			
		||||
          >
 | 
			
		||||
 
 | 
			
		||||
@@ -1,37 +0,0 @@
 | 
			
		||||
.rejected {
 | 
			
		||||
  color: #f83126;
 | 
			
		||||
  font-weight: bold
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.accepted {
 | 
			
		||||
  color: #3fc424;
 | 
			
		||||
  font-weight: bold
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.hsCell {
 | 
			
		||||
  display: flex !important;
 | 
			
		||||
    flex-direction: column !important;
 | 
			
		||||
    justify-content: center !important;
 | 
			
		||||
    align-items: center !important;
 | 
			
		||||
    padding: 7px;
 | 
			
		||||
}
 | 
			
		||||
.btCell {
 | 
			
		||||
  display: flex !important;
 | 
			
		||||
  justify-content: center !important;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.verCenter {
 | 
			
		||||
    display: flex;
 | 
			
		||||
    align-items: center;
 | 
			
		||||
    word-break: break-all;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.load-more {
 | 
			
		||||
  input {
 | 
			
		||||
    width: 90px;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#noDataContainer {
 | 
			
		||||
  height: calc(100vh - 200px);
 | 
			
		||||
}
 | 
			
		||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user