1
0
mirror of https://github.com/sasjs/core.git synced 2025-12-15 08:14:34 +00:00

Compare commits

...

94 Commits

Author SHA1 Message Date
^
22f0cb67a5 fix: handling consecutive add+delete in mp_stripdiffs 2024-04-30 17:38:36 +01:00
^
e6da373853 fix: more dedup fixes on mp_stripdiffs 2024-04-30 14:04:15 +01:00
^
ed20bcaa5c fix: supporting long character strings in mp_stripdiffs.sas 2024-04-30 11:12:19 +01:00
^
96e8b096c5 fix: addressing bug with non-unique PK for reverting multiple loads at once in mp_stripdiffs.sas 2024-04-29 23:40:49 +01:00
^
7413266a8e fix: correcting name to _____DELETE_THIS_RECORD_____ in mp_stripdiffs 2024-04-29 20:14:35 +01:00
^
cf70c33bde fix: length of key_hash variable in mp_stripdiffs.sas 2024-04-29 19:54:16 +01:00
Allan Bowe
934629d46d Merge pull request #374 from sasjs/issue373
feat: mp_stripdiffs macro - closes #373
2024-04-25 10:49:21 +01:00
github-actions
16a3b63161 chore: updating all.sas 2024-04-25 09:49:00 +00:00
Allan Bowe
d7288b7fa1 Merge branch 'main' into issue373 2024-04-25 10:48:37 +01:00
github-actions
015749a9b2 chore: updating all.sas 2024-04-25 09:45:46 +00:00
^
556c7bdb28 feat: mp_stripdiffs macro - closes #373 2024-04-25 10:45:23 +01:00
^
602758c3c3 fix: ensuring consistent column names across invocations in output dataset 2024-04-11 14:16:25 +01:00
^
a244a0b27b chore: updating all.sas 2024-04-11 12:00:28 +01:00
^
3bb632d60d feat: new mx_getgroups.sas macro for cross-platform use 2024-04-11 11:58:45 +01:00
Allan Bowe
bdd348483c Merge pull request #372 from sasjs/issue371
Issue371
2024-02-23 10:29:25 +00:00
github-actions
92f575551d chore: updating all.sas 2024-02-23 10:26:33 +00:00
^
e616bc940f fix: partial short numeric support in mp_ds2csv 2024-02-23 10:26:01 +00:00
Allan Bowe
b7bca48129 Merge pull request #370 from sasjs/dependabot/npm_and_yarn/follow-redirects-1.15.4
chore(deps): bump follow-redirects from 1.15.2 to 1.15.4
2024-02-19 12:48:34 +00:00
dependabot[bot]
6a2dcbb23f chore(deps): bump follow-redirects from 1.15.2 to 1.15.4
Bumps [follow-redirects](https://github.com/follow-redirects/follow-redirects) from 1.15.2 to 1.15.4.
- [Release notes](https://github.com/follow-redirects/follow-redirects/releases)
- [Commits](https://github.com/follow-redirects/follow-redirects/compare/v1.15.2...v1.15.4)

---
updated-dependencies:
- dependency-name: follow-redirects
  dependency-type: indirect
...

Signed-off-by: dependabot[bot] <support@github.com>
2024-01-10 10:19:46 +00:00
Allan Bowe
6da578e336 Merge pull request #369 from sasjs/368-enable-filter-by-variable-name-in-mp_filter-series
feat: enable filter by variable name in mp filter series
2023-12-31 00:36:53 +00:00
github-actions
c874b31b63 chore: updating all.sas 2023-12-31 00:35:46 +00:00
zver
532e0d535a fix: tests for char support, #368 2023-12-31 00:35:11 +00:00
github-actions
ee5688f97f chore: updating all.sas 2023-12-31 00:08:50 +00:00
zver
359b007f85 chore: merge 2023-12-31 00:08:13 +00:00
zver
3294767c1b feat: enabling variable names for numeric fields. #368 2023-12-31 00:07:02 +00:00
github-actions
9d6f87c87a chore: updating all.sas 2023-12-30 22:43:02 +00:00
Allan
ec14b9cef8 fix: mp_loadformat updates by FMTROW
Previously, FMTROW was not being honoured when adding / deleting individual format records.  Updated tests and added additional validations to ensure FMTROW is provided correctly at the input stage.
2023-12-03 13:39:50 +00:00
Allan Bowe
94af8661b0 Merge pull request #367 from sasjs/all-contributors/add-andyjessen
docs: add andyjessen as a contributor for doc
2023-11-16 15:44:35 +00:00
Allan Bowe
c9e431142c Merge branch 'main' into all-contributors/add-andyjessen 2023-11-16 15:44:20 +00:00
Allan Bowe
2b2aa5eb58 Merge pull request #366 from andyjessen/fix-macro
Add macro trigger to usage example
2023-11-16 15:44:08 +00:00
allcontributors[bot]
1ac2b480a6 docs: update .all-contributorsrc [skip ci] 2023-11-16 15:43:53 +00:00
allcontributors[bot]
4e53544b66 docs: update README.md [skip ci] 2023-11-16 15:43:52 +00:00
github-actions
9b5f1cf170 chore: updating all.sas 2023-11-16 14:23:42 +00:00
andyjessen
703fe4ef38 Add macro trigger to usage example
This commit adds macro trigger to mf_isblank usage example.
2023-11-16 07:22:54 -07:00
Allan Bowe
f4a4263046 Merge pull request #365 from sasjs/issue363
Issue363
2023-11-08 21:31:38 +00:00
github-actions
02bf9c85db chore: updating all.sas 2023-11-08 21:30:00 +00:00
Allan
5835cfaa83 fix: HLO variable label updates, closes #364
Removed line breaks and reduced label length by moving the information to the core doc site and providing a link instead
2023-11-08 21:28:17 +00:00
Allan
b50521a8de fix: adding missing mp_md5 dependency. Closes #363 2023-11-08 21:10:32 +00:00
Allan
fccd6fcc44 fix: updating PR desc to include conventional commit reference 2023-10-18 10:16:02 +01:00
Allan Bowe
487ff5faa9 Merge pull request #362 from rudvfaden/main
fixed error from %put message when mdebug=0
2023-10-18 09:03:06 +01:00
Rud Faden
5efc20eacc fixed error from %put message when mdebug=0 2023-10-18 09:37:11 +02:00
Allan Bowe
cbd62fbfab Merge pull request #361 from sasjs/bumpfix
chore: avoiding vpn start
2023-10-17 16:39:29 +01:00
Allan Bowe
2808145302 Merge branch 'main' into bumpfix 2023-10-17 16:39:19 +01:00
Allan
815e5f3e0e chore: avoiding vpn start 2023-10-17 16:37:48 +01:00
Allan Bowe
843d6e5c2d Merge pull request #360 from sasjs/bumpfix
chore: Bumpfix
2023-10-17 16:37:23 +01:00
github-actions
b084f4e84b chore: updating all.sas 2023-10-17 15:37:03 +00:00
Allan Bowe
5b5116070e Merge branch 'main' into bumpfix 2023-10-17 16:36:39 +01:00
github-actions
a2002db838 chore: updating all.sas 2023-10-17 15:36:07 +00:00
Allan
dc6bcdd69e chore: merge msg 2023-10-17 16:34:40 +01:00
Allan
c97dc9a16d chore: commenting vpn due to package issues 2023-10-17 16:32:24 +01:00
github-actions
ef669db622 chore: updating all.sas 2023-10-17 15:15:30 +00:00
Allan Bowe
26499d2058 Merge pull request #359 from sasjs/bumpfix
chore: moving all.sas logic to another action
2023-10-17 16:15:15 +01:00
Allan Bowe
17e5d0f0e0 Merge branch 'main' into bumpfix 2023-10-17 16:15:04 +01:00
github-actions
fc9205e355 chore: updating all.sas 2023-10-17 15:14:17 +00:00
Allan
ce344fc8e2 chore: allow empty commit to avoid workflow error 2023-10-17 16:12:50 +01:00
Allan
40239c53d8 chore: removing utf char from pre-commit hook 2023-10-17 16:07:57 +01:00
Allan
814ecec94f chore: moving all.sas logic to another action 2023-10-17 16:05:32 +01:00
Allan Bowe
934c501fec Merge pull request #358 from sasjs/bumpfix
chore: adding cli dependency
2023-10-17 16:02:13 +01:00
Allan Bowe
091b2e28be Merge branch 'main' into bumpfix 2023-10-17 16:01:58 +01:00
Allan
d8ea29bf8c chore: -g param 2023-10-17 16:00:47 +01:00
Allan
993dec4610 chore: adding cli dependency 2023-10-17 16:00:05 +01:00
Allan Bowe
f905387d66 Merge pull request #357 from sasjs/bumpfix
fix: bumping semantic release and checkout actions to v4
2023-10-17 15:59:25 +01:00
Allan
7512423b04 fix: bumping semantic release and checkout actions to v4 2023-10-17 15:57:29 +01:00
Allan
50e6d416a4 fix: mentioning conv. commits in CONTRIBUTING.md and setting eol=lf in .gitattributes 2023-10-17 15:24:49 +01:00
Allan Bowe
18b6cadce6 Merge pull request #354 from sasjs/all-contributors/add-rudvfaden
fix: add rudvfaden as a contributor for code
2023-10-17 15:04:27 +01:00
Allan Bowe
413743bbe6 Merge branch 'main' into all-contributors/add-rudvfaden 2023-10-17 15:03:36 +01:00
Allan Bowe
fcafb1026e Merge pull request #353 from rudvfaden/main
added authdomain for odbc engine in MM_ASSIGNDIRECTLIB
2023-10-17 15:01:37 +01:00
allcontributors[bot]
b8f24264d4 docs: update .all-contributorsrc [skip ci] 2023-10-17 14:00:19 +00:00
allcontributors[bot]
5eb87a754e docs: update README.md [skip ci] 2023-10-17 14:00:14 +00:00
Rud Faden
3a5fd4bfc5 added .gitattributes 2023-10-17 15:54:27 +02:00
Rud Faden
b7ae9a2737 added backwards compebility 2023-10-17 15:45:44 +02:00
Rud Faden
4057ac4b2e fix lineending CRLF to LF 2023-10-17 14:00:42 +02:00
Rud Faden
fa0a6ab22d run build.py 2023-10-17 13:46:28 +02:00
Rud Faden
2ae7a60be5 added authdomain for odvc engine 2023-10-17 13:33:48 +02:00
Rud Faden
0a24f3ff7b add authdomain for odbc connection 2023-10-17 13:13:03 +02:00
Allan
592f477063 chore(docs): updated readme about non-ascii char recommendations 2023-10-11 22:42:25 +01:00
Allan Bowe
a91db81894 Merge pull request #352 from sasjs/dcissue50
fix: removing UTF 8 char to support LATIN9 environments
2023-10-11 22:18:30 +01:00
Allan
236e7cc4c0 fix: removing UTF 8 char to support LATIN9 environments
Source issue: https://git.datacontroller.io/dc/dc/issues/50
2023-10-11 22:12:19 +01:00
Allan Bowe
2b6882cb9c Merge pull request #351 from sasjs/issue350
feat: adding LogicalServerType option to mm_createstp.sas macro.
2023-10-05 16:09:41 +01:00
Allan
2a3071708a feat: adding LogicalServerType option to mm_createstp.sas macro. Closes #350 2023-10-05 16:06:11 +01:00
Allan Bowe
3890aefccf Merge pull request #349 from sasjs/hlo
feat: extended comment for hlo variable in mdds_sas_cntlout
2023-09-18 16:20:52 +01:00
Allan
f378a5637f feat: extended comment for hlo variable in mdds_sas_cntlout 2023-09-18 16:18:37 +01:00
Allan Bowe
121c692e09 chore: adding @paul-canals toolbox repo
also ordered the repos alphabetically
2023-09-16 11:24:22 +01:00
Allan
759d6bd144 chore: temporarily disabling auto-update due to un-reproducable pipeline issue 2023-08-18 12:30:27 +01:00
Allan
76d248f302 chore: updating main.yml to debug packages rebuild 2023-08-18 11:02:24 +01:00
Allan Bowe
f30e30c024 Merge pull request #348 from sasjs/docfixes
chore: missing doc updates
2023-08-18 08:36:33 +01:00
Allan Bowe
30637b5025 Merge branch 'main' into docfixes 2023-08-18 08:36:21 +01:00
Allan
9f16d090f5 chore: missing doc updates 2023-08-18 08:34:53 +01:00
Allan Bowe
74143bdf29 Merge pull request #347 from sasjs/docfixes
chore: updating documentation
2023-08-18 08:26:45 +01:00
Allan
1c4c9793f6 chore: missed some doxygen in/out tags 2023-08-18 08:25:19 +01:00
Allan
d42fd4ebac chore: updating documentation in preparation for upcoming sasjs snippets feature 2023-08-18 08:20:13 +01:00
Allan Bowe
d39b1be7a8 Merge pull request #346 from sasjs/gh-action0fix
chore(github): fixed token
2023-08-17 12:26:39 +01:00
Yury Shkoda
65d4c7969d chore(github): fixed token 2023-08-08 16:15:13 +03:00
Allan
fa152cb375 chore(docs): removing asterisk in mv_registerclient example 2023-07-26 21:29:02 +01:00
159 changed files with 2718 additions and 1061 deletions

View File

@@ -144,8 +144,27 @@
"contributions": [ "contributions": [
"doc" "doc"
] ]
},
{
"login": "rudvfaden",
"name": "Rud Faden",
"avatar_url": "https://avatars.githubusercontent.com/u/2445577?v=4",
"profile": "http://rudvfaden.github.io/",
"contributions": [
"code"
]
},
{
"login": "andyjessen",
"name": "andyjessen",
"avatar_url": "https://avatars.githubusercontent.com/u/62343929?v=4",
"profile": "https://github.com/andyjessen",
"contributions": [
"doc"
]
} }
], ],
"contributorsPerLine": 7, "contributorsPerLine": 7,
"skipCi": true "skipCi": true,
"commitType": "docs"
} }

View File

@@ -2,7 +2,7 @@
# Ensure lint is passing # Ensure lint is passing
LINT=`sasjs lint` LINT=`sasjs lint`
if [[ "$LINT" != *"All matched files use @sasjs/lint code style!" ]]; then if [[ "$LINT" != *"All matched files use @sasjs/lint code style!" ]]; then
echo "$LINT" echo "$LINT"
echo "To commit in spite of these warnings, use the -n parameter." echo "To commit in spite of these warnings, use the -n parameter."
exit 1 exit 1

1
.gitattributes vendored Normal file
View File

@@ -0,0 +1 @@
* text=auto eol=lf

View File

@@ -27,5 +27,6 @@ To contribute:
1. Create your feature branch (`git checkout -b myfeature`) 1. Create your feature branch (`git checkout -b myfeature`)
2. Make your change 2. Make your change
3. Update the `all.sas` file (`python3 build.py`) 3. Update the `all.sas` file (`python3 build.py`)
4. Push and make a PR 4. Commit using a [Conventional Commit](https://www.conventionalcommits.org)
5. Push and make a PR

View File

@@ -15,4 +15,4 @@ What code changes have been made to achieve the intent.
- [ ] Code is formatted correctly (`sasjs lint`). - [ ] Code is formatted correctly (`sasjs lint`).
- [ ] Any new functionality has been unit tested. - [ ] Any new functionality has been unit tested.
- [ ] All unit tests are passing (`sasjs test`). - [ ] All unit tests are passing (`sasjs test`).
- [ ] `all.sas` has been regenerated (`python3 build.py`) - [ ] The PR desc or underlying commits follow the [Conventional Commit](https://www.conventionalcommits.org) standard

View File

@@ -13,10 +13,11 @@ jobs:
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- name: Checkout - name: Checkout
uses: actions/checkout@v2 uses: actions/checkout@v4
- name: Install dependencies - name: Install dependencies
run: npm ci run: |
npm ci
- name: Check code style (aborts if errors found) - name: Check code style (aborts if errors found)
run: npx @sasjs/cli lint run: npx @sasjs/cli lint
@@ -36,14 +37,15 @@ jobs:
- name: Install Open VPN - name: Install Open VPN
run: | run: |
sudo apt install apt-transport-https #sudo apt install apt-transport-https
sudo wget https://swupdate.openvpn.net/repos/openvpn-repo-pkg-key.pub #sudo wget https://swupdate.openvpn.net/repos/openvpn-repo-pkg-key.pub
sudo apt-key add openvpn-repo-pkg-key.pub #sudo apt-key add openvpn-repo-pkg-key.pub
sudo wget -O /etc/apt/sources.list.d/openvpn3.list https://swupdate.openvpn.net/community/openvpn3/repos/openvpn3-jammy.list #sudo wget -O /etc/apt/sources.list.d/openvpn3.list https://swupdate.openvpn.net/community/openvpn3/repos/openvpn3-jammy.list
sudo apt update #sudo apt update
sudo apt install openvpn3=17~betaUb22042+jammy #sudo apt install openvpn3=17~betaUb22042+jammy
- name: Start Open VPN 3 - name: Start Open VPN 3
run: openvpn3 session-start --config .github/vpn/config.ovpn run: |
# openvpn3 session-start --config .github/vpn/config.ovpn
- name: Add credentials - name: Add credentials
run: | run: |
@@ -52,13 +54,13 @@ jobs:
echo "REFRESH_TOKEN=${{secrets.SAS9_4GL_IO_REFRESH_TOKEN}}" >> .env.server echo "REFRESH_TOKEN=${{secrets.SAS9_4GL_IO_REFRESH_TOKEN}}" >> .env.server
- name: Semantic Release - name: Semantic Release
uses: cycjimmy/semantic-release-action@v3 uses: cycjimmy/semantic-release-action@v4
env: env:
GITHUB_TOKEN: ${{ secrets.GH_TOKEN }} GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
NPM_TOKEN: ${{ secrets.NPM_TOKEN }} NPM_TOKEN: ${{ secrets.NPM_TOKEN }}
- name: SAS Packages Release - name: SAS Packages Release
run: | run: |
npx @sasjs/cli compile job -s sasjs/utils/create_sas_package.sas -o sasjsbuild -t server npx @sasjs/cli compile job -s sasjs/utils/create_sas_package.sas -o sasjsbuild -t server
# this part depends on https://github.com/sasjs/server/issues/307 # need long duration token per https://github.com/sasjs/server/issues/307
npx @sasjs/cli run sasjsbuild/jobs/utils/create_sas_package.sas -t server # npx @sasjs/cli run sasjsbuild/jobs/utils/create_sas_package.sas -t server

32
.github/workflows/notmain.yml vendored Normal file
View File

@@ -0,0 +1,32 @@
# This workflow will run tests using node and then publish a package to GitHub Packages when a release is created
# For more information see: https://help.github.com/actions/language-and-framework-guides/publishing-nodejs-packages
name: SASjs Core - Update all.sas
on:
push:
branches-ignore:
- main
jobs:
release:
runs-on: ubuntu-latest
steps:
- name: Checkout
uses: actions/checkout@v4
- name: Install dependencies
run: |
npm ci
npm i -g @sasjs/cli@latest
- name: Ensure all.sas is always up to date
run: |
git config user.name github-actions
git config user.email github-actions@github.com
python3 build.py
git add all.sas
git commit -m "chore: updating all.sas" --allow-empty
git push

1
.gitignore vendored
View File

@@ -13,3 +13,4 @@ mc_*
*.env* *.env*
~ ~

View File

@@ -212,7 +212,8 @@ When contributing to this library, it is therefore important to ensure that all
## General Notes ## General Notes
- All macros should be compatible with SAS versions from support level B and above (so currently 9.2 and later). If an earlier version is not supported, then the macro should say as such in the header documentation, and exit gracefully (eg `%if %sysevalf(&sysver<9.3) %then %return`). - All macros should be compatible with SAS versions from support level B and above (so currently 9.3 and later). If an earlier version is not supported, then the macro should say as such in the header documentation, and exit gracefully.
- It's [best to avoid](https://git.datacontroller.io/dc/dc/issues/50) special / non-ASCII characters for compatibility with the widest variety of SAS installations.
## Breaking Changes ## Breaking Changes
@@ -235,18 +236,19 @@ If you find this library useful, please leave a [star](https://github.com/sasjs/
The following repositories are also worth checking out: The following repositories are also worth checking out:
* [xieliaing/SAS](https://github.com/xieliaing/SAS)
* [SASJedi/sas-macros](https://github.com/SASJedi/sas-macros)
* [chris-swenson/sasmacros](https://github.com/chris-swenson/sasmacros) * [chris-swenson/sasmacros](https://github.com/chris-swenson/sasmacros)
* [greg-wotton/sas-programs](https://github.com/greg-wootton/sas-programs) * [greg-wotton/sas-programs](https://github.com/greg-wootton/sas-programs)
* [KatjaGlassConsulting/SMILE-SmartSASMacros](https://github.com/KatjaGlassConsulting/SMILE-SmartSASMacros) * [KatjaGlassConsulting/SMILE-SmartSASMacros](https://github.com/KatjaGlassConsulting/SMILE-SmartSASMacros)
* [paul-canals/toolbox](https://github.com/paul-canals/toolbox)
* [rogerjdeangelis](https://github.com/rogerjdeangelis) * [rogerjdeangelis](https://github.com/rogerjdeangelis)
* [SASJedi/sas-macros](https://github.com/SASJedi/sas-macros)
* [scottbass/sas](https://github.com/scottbass/SAS) * [scottbass/sas](https://github.com/scottbass/SAS)
* [xieliaing/SAS](https://github.com/xieliaing/SAS)
* [yabwon/sas_packages](https://github.com/yabwon/SAS_PACKAGES) * [yabwon/sas_packages](https://github.com/yabwon/SAS_PACKAGES)
## Contributors ✨ ## Contributors ✨
<!-- ALL-CONTRIBUTORS-BADGE:START - Do not remove or modify this section --> <!-- ALL-CONTRIBUTORS-BADGE:START - Do not remove or modify this section -->
[![All Contributors](https://img.shields.io/badge/all_contributors-13-orange.svg?style=flat-square)](#contributors-) [![All Contributors](https://img.shields.io/badge/all_contributors-15-orange.svg?style=flat-square)](#contributors-)
<!-- ALL-CONTRIBUTORS-BADGE:END --> <!-- ALL-CONTRIBUTORS-BADGE:END -->
Thanks goes to these wonderful people ([emoji key](https://allcontributors.org/docs/en/emoji-key)): Thanks goes to these wonderful people ([emoji key](https://allcontributors.org/docs/en/emoji-key)):
@@ -271,6 +273,10 @@ Thanks goes to these wonderful people ([emoji key](https://allcontributors.org/d
<td align="center" valign="top" width="14.28%"><a href="https://github.com/yabwon"><img src="https://avatars.githubusercontent.com/u/9314894?v=4?s=100" width="100px;" alt="Bart Jablonski"/><br /><sub><b>Bart Jablonski</b></sub></a><br /><a href="https://github.com/sasjs/core/commits?author=yabwon" title="Code">💻</a></td> <td align="center" valign="top" width="14.28%"><a href="https://github.com/yabwon"><img src="https://avatars.githubusercontent.com/u/9314894?v=4?s=100" width="100px;" alt="Bart Jablonski"/><br /><sub><b>Bart Jablonski</b></sub></a><br /><a href="https://github.com/sasjs/core/commits?author=yabwon" title="Code">💻</a></td>
<td align="center" valign="top" width="14.28%"><a href="https://bandism.net/"><img src="https://avatars.githubusercontent.com/u/22633385?v=4?s=100" width="100px;" alt="Ikko Ashimine"/><br /><sub><b>Ikko Ashimine</b></sub></a><br /><a href="https://github.com/sasjs/core/commits?author=eltociear" title="Code">💻</a></td> <td align="center" valign="top" width="14.28%"><a href="https://bandism.net/"><img src="https://avatars.githubusercontent.com/u/22633385?v=4?s=100" width="100px;" alt="Ikko Ashimine"/><br /><sub><b>Ikko Ashimine</b></sub></a><br /><a href="https://github.com/sasjs/core/commits?author=eltociear" title="Code">💻</a></td>
<td align="center" valign="top" width="14.28%"><a href="https://github.com/henrik-forsell"><img src="https://avatars.githubusercontent.com/u/109935936?v=4?s=100" width="100px;" alt="Henrik Forsell"/><br /><sub><b>Henrik Forsell</b></sub></a><br /><a href="https://github.com/sasjs/core/commits?author=henrik-forsell" title="Documentation">📖</a></td> <td align="center" valign="top" width="14.28%"><a href="https://github.com/henrik-forsell"><img src="https://avatars.githubusercontent.com/u/109935936?v=4?s=100" width="100px;" alt="Henrik Forsell"/><br /><sub><b>Henrik Forsell</b></sub></a><br /><a href="https://github.com/sasjs/core/commits?author=henrik-forsell" title="Documentation">📖</a></td>
<td align="center" valign="top" width="14.28%"><a href="http://rudvfaden.github.io/"><img src="https://avatars.githubusercontent.com/u/2445577?v=4?s=100" width="100px;" alt="Rud Faden"/><br /><sub><b>Rud Faden</b></sub></a><br /><a href="https://github.com/sasjs/core/commits?author=rudvfaden" title="Code">💻</a></td>
</tr>
<tr>
<td align="center" valign="top" width="14.28%"><a href="https://github.com/andyjessen"><img src="https://avatars.githubusercontent.com/u/62343929?v=4?s=100" width="100px;" alt="andyjessen"/><br /><sub><b>andyjessen</b></sub></a><br /><a href="https://github.com/sasjs/core/commits?author=andyjessen" title="Documentation">📖</a></td>
</tr> </tr>
</tbody> </tbody>
</table> </table>

1563
all.sas

File diff suppressed because it is too large Load Diff

View File

@@ -3,6 +3,11 @@
@brief Abort, ungracefully @brief Abort, ungracefully
@details Will abort with a straightforward %abort if the condition is true. @details Will abort with a straightforward %abort if the condition is true.
@param [in] mac= (mf_abort.sas) Name of calling macro (is printed to the log)
@param [in] msg= ( ) Additional string to print to the log
@param [in] iftrue= (%str(1=1)) Conditional logic under which to perform the
abort
<h4> Related Macros </h4> <h4> Related Macros </h4>
@li mp_abort.sas @li mp_abort.sas

View File

@@ -10,7 +10,7 @@
%mf_deletefile(&sasjswork/myfile.txt) %mf_deletefile(&sasjswork/myfile.txt)
@param filepath Full path to the target file @param [in] file Full path to the target file
@returns The return code from the fdelete() invocation @returns The return code from the fdelete() invocation

View File

@@ -10,7 +10,7 @@
expected results (depending on whether you 'expect' the result to be expected results (depending on whether you 'expect' the result to be
case insensitive in this context!) case insensitive in this context!)
@param libds library.dataset @param [in] libds library.dataset
@return output returns 1 or 0 @return output returns 1 or 0
<h4> Related Macros </h4> <h4> Related Macros </h4>

View File

@@ -4,7 +4,7 @@
@details You can probably do without this macro as it is just a one liner. @details You can probably do without this macro as it is just a one liner.
Mainly it is here as a convenient way to remember the syntax! Mainly it is here as a convenient way to remember the syntax!
@param fref the fileref to detect @param [in] fref the fileref to detect
@return output Returns 1 if found and 0 if not found. Note - it is possible @return output Returns 1 if found and 0 if not found. Note - it is possible
that the fileref is found, but the file does not (yet) exist. If you need that the fileref is found, but the file does not (yet) exist. If you need

View File

@@ -16,7 +16,7 @@
https://github.com/yabwon/SAS_PACKAGES/blob/main/packages/baseplus.md#functionexists-macro https://github.com/yabwon/SAS_PACKAGES/blob/main/packages/baseplus.md#functionexists-macro
). ).
@param [in] name (positional) - function name @param [in] name function name
@author Allan Bowe @author Allan Bowe
**/ **/

View File

@@ -6,8 +6,8 @@
%put %mf_existVarList(sashelp.class, age sex name dummyvar); %put %mf_existVarList(sashelp.class, age sex name dummyvar);
@param libds 2 part dataset or view reference @param [in] libds 2 part dataset or view reference
@param varlist space separated variable names @param [in] varlist space separated variable names
@version 9.2 @version 9.2
@author Allan Bowe @author Allan Bowe

View File

@@ -31,6 +31,7 @@
%put %mf_getapploc(/some/location/jobs/extract/somejob/); %put %mf_getapploc(/some/location/jobs/extract/somejob/);
%put %mf_getapploc(/some/location/tests/jobs/somejob/); %put %mf_getapploc(/some/location/tests/jobs/somejob/);
@param [in] pgm The _program value from which to extract the appLoc
@author Allan Bowe @author Allan Bowe
**/ **/

View File

@@ -6,8 +6,8 @@
%put Dataset label = %mf_getattrc(sashelp.class,LABEL); %put Dataset label = %mf_getattrc(sashelp.class,LABEL);
%put Member Type = %mf_getattrc(sashelp.class,MTYPE); %put Member Type = %mf_getattrc(sashelp.class,MTYPE);
@param libds library.dataset @param [in] libds library.dataset
@param attr full list in [documentation]( @param [in] attr full list in [documentation](
https://support.sas.com/documentation/cdl/en/lrdict/64316/HTML/default/viewer.htm#a000147794.htm) https://support.sas.com/documentation/cdl/en/lrdict/64316/HTML/default/viewer.htm#a000147794.htm)
@return output returns result of the attrc value supplied, or -1 and log @return output returns result of the attrc value supplied, or -1 and log
message if err. message if err.

View File

@@ -6,8 +6,8 @@
%put Number of observations=%mf_getattrn(sashelp.class,NLOBS); %put Number of observations=%mf_getattrn(sashelp.class,NLOBS);
%put Number of variables = %mf_getattrn(sashelp.class,NVARS); %put Number of variables = %mf_getattrn(sashelp.class,NVARS);
@param libds library.dataset @param [in] libds library.dataset
@param attr Common values are NLOBS and NVARS, full list in [documentation]( @param [in] attr Common values are NLOBS and NVARS, full list in [documentation](
http://support.sas.com/documentation/cdl/en/lrdict/64316/HTML/default/viewer.htm#a000212040.htm) http://support.sas.com/documentation/cdl/en/lrdict/64316/HTML/default/viewer.htm#a000212040.htm)
@return output returns result of the attrn value supplied, or -1 and log @return output returns result of the attrn value supplied, or -1 and log
message if err. message if err.

View File

@@ -7,8 +7,9 @@
%put %mf_getkeyvalue(someindex) %put %mf_getkeyvalue(someindex)
@param key Provide a key on which to perform the lookup @param [in] key Provide a key on which to perform the lookup
@param libds= define the target table which holds the parameters @param [in] libds= (work.mp_setkeyvalue) The library.dataset which holds the
parameters
@version 9.2 @version 9.2
@author Allan Bowe @author Allan Bowe

View File

@@ -12,7 +12,7 @@
@li SASJS @li SASJS
@li BASESAS @li BASESAS
@param switch the param for which to return a platform specific variable @param [in] switch the param for which to return a platform specific variable
<h4> SAS Macros </h4> <h4> SAS Macros </h4>
@li mf_mval.sas @li mf_mval.sas

View File

@@ -8,7 +8,7 @@
returns: returns:
> dbo > dbo
@param libref Library reference (also accepts a 2 level libds ref). @param [in] libref Library reference (also accepts a 2 level libds ref).
@return output returns the library schema for the FIRST library encountered @return output returns the library schema for the FIRST library encountered

View File

@@ -19,7 +19,7 @@
@param [in] prefix= (mclib) first part of the returned libref. As librefs can @param [in] prefix= (mclib) first part of the returned libref. As librefs can
be as long as 8 characters, a maximum length of 7 characters is premitted be as long as 8 characters, a maximum length of 7 characters is premitted
for this prefix. for this prefix.
@param [in] maxtries= Deprecated parameter. Remains here to ensure a @param [in] maxtries= (1000) Deprecated parameter. Remains here to ensure a
non-breaking change. Will be removed in v5. non-breaking change. Will be removed in v5.
@version 9.2 @version 9.2

View File

@@ -10,7 +10,7 @@
> MCc59c750610321d4c8bf75faadbcd22 > MCc59c750610321d4c8bf75faadbcd22
@param prefix= set a prefix for the new name @param [in] prefix= (MC) Sets a prefix for the new name
@version 9.3 @version 9.3
@author Allan Bowe @author Allan Bowe

View File

@@ -13,8 +13,6 @@
%let user= %mf_getUser(); %let user= %mf_getUser();
%put &user; %put &user;
@param type - do not use, may be deprecated in a future release
@return SYSUSERID (if workspace server) @return SYSUSERID (if workspace server)
@return _METAPERSON (if stored process server) @return _METAPERSON (if stored process server)
@return SYS_COMPUTE_SESSION_OWNER (if Viya compute session) @return SYS_COMPUTE_SESSION_OWNER (if Viya compute session)

View File

@@ -13,9 +13,9 @@
<h4> Related Macros </h4> <h4> Related Macros </h4>
@li mp_setkeyvalue.sas @li mp_setkeyvalue.sas
@param libds dataset to query @param [in] libds dataset to query
@param variable the variable which contains the value to return. @param [in] variable the variable which contains the value to return.
@param filter contents of where clause @param [in] filter= (1) contents of where clause
@version 9.2 @version 9.2
@author Allan Bowe @author Allan Bowe

View File

@@ -25,7 +25,8 @@
@param [in] libds Two part dataset (or view) reference. @param [in] libds Two part dataset (or view) reference.
@param [in] var Variable name for which a format should be returned @param [in] var Variable name for which a format should be returned
@param [in] force=(0) Set to 1 to supply a default if the variable has no format @param [in] force= (0) Set to 1 to supply a default if the variable has no
format
@returns outputs format @returns outputs format
@author Allan Bowe @author Allan Bowe

View File

@@ -18,8 +18,8 @@
8 8
NOTE: Variable renegade does not exist in test NOTE: Variable renegade does not exist in test
@param libds Two part dataset (or view) reference. @param [in] libds Two part dataset (or view) reference.
@param var Variable name for which a length should be returned @param [in] var Variable name for which a length should be returned
@returns outputs length @returns outputs length
@author Allan Bowe @author Allan Bowe

View File

@@ -21,8 +21,8 @@ returns:
> NOTE: Variable renegade does not exist in test > NOTE: Variable renegade does not exist in test
@param libds Two part dataset (or view) reference. @param [in] libds Two part dataset (or view) reference.
@param var Variable name for which a position should be returned @param [in] var Variable name for which a position should be returned
@author Allan Bowe @author Allan Bowe
@version 9.2 @version 9.2

View File

@@ -13,8 +13,8 @@ Usage:
@param libds Two part dataset (or view) reference. @param [in] libds Two part dataset (or view) reference.
@param var the variable name to be checked @param [in] var the variable name to be checked
@return output returns C or N depending on variable type. If variable @return output returns C or N depending on variable type. If variable
does not exist then a blank is returned and a note is written to the log. does not exist then a blank is returned and a note is written to the log.

View File

@@ -10,7 +10,7 @@
returns: returns:
> TEMP > TEMP
@param fref The fileref to check @param [in] fref The fileref to check
@returns The XENGINE value in sashelp.vextfl or 0 if not found. @returns The XENGINE value in sashelp.vextfl or 0 if not found.

View File

@@ -11,8 +11,8 @@
%put Now we have run %mf_increment(cnt) lines; %put Now we have run %mf_increment(cnt) lines;
%put There are %mf_increment(cnt) lines in total; %put There are %mf_increment(cnt) lines in total;
@param [in] MACRO_NAME the name of the macro variable to increment @param [in] macro_name The name of the macro variable to increment
@param [in] ITER= The amount to add or subtract to the macro @param [in] incr= (1) The amount to add or subtract to the macro
<h4> Related Files </h4> <h4> Related Files </h4>
@li mf_increment.test.sas @li mf_increment.test.sas

View File

@@ -7,12 +7,12 @@
Usage: Usage:
%put mf_isblank(&var); %put %mf_isblank(&var);
inspiration: inspiration:
https://support.sas.com/resources/papers/proceedings09/022-2009.pdf https://support.sas.com/resources/papers/proceedings09/022-2009.pdf
@param param VALUE to be checked @param [in] Param VALUE to be checked
@return output returns 1 (if blank) else 0 @return output returns 1 (if blank) else 0

View File

@@ -9,7 +9,7 @@
With thanks and full credit to Andrea Defronzo - With thanks and full credit to Andrea Defronzo -
https://www.linkedin.com/in/andrea-defronzo-b1a47460/ https://www.linkedin.com/in/andrea-defronzo-b1a47460/
@param path full path of the file/directory to be checked @param [in] path Full path of the file/directory to be checked
@return output returns 1 if path is a directory, 0 if it is not @return output returns 1 if path is a directory, 0 if it is not

View File

@@ -6,6 +6,10 @@
%put %mf_loc(POF); %*location of PlatformObjectFramework tools; %put %mf_loc(POF); %*location of PlatformObjectFramework tools;
@param [in] loc The item to locate, eg:
@li PLAATFORMOBJECTFRAMEWORK (or POF)
@li VIYACONFG
@version 9.2 @version 9.2
@author Allan Bowe @author Allan Bowe
**/ **/
@@ -15,7 +19,8 @@
%local root; %local root;
%if &loc=POF or &loc=PLATFORMOBJECTFRAMEWORK %then %do; %if &loc=POF or &loc=PLATFORMOBJECTFRAMEWORK %then %do;
%let root=%substr(%sysget(SASROOT),1,%index(%sysget(SASROOT),SASFoundation)-2); %let root=%sysget(SASROOT);
%let root=%substr(&root,1,%index(&root,SASFoundation)-2);
%let root=&root/SASPlatformObjectFramework/&sysver; %let root=&root/SASPlatformObjectFramework/&sysver;
%put Batch tools located at: &root; %put Batch tools located at: &root;
&root &root

View File

@@ -7,7 +7,7 @@ Usage:
%mf_mkdir(/some/path/name) %mf_mkdir(/some/path/name)
@param dir relative or absolute pathname. Unquoted. @param [in] dir Relative or absolute pathname. Unquoted.
@version 9.2 @version 9.2
**/ **/

View File

@@ -8,6 +8,8 @@
%if %mf_mval(maynotexist)=itdid %then %do; %if %mf_mval(maynotexist)=itdid %then %do;
@param [in] var The macro variable NAME to return the (possible) value for
@version 9.2 @version 9.2
@author Allan Bowe @author Allan Bowe
**/ **/

View File

@@ -9,7 +9,7 @@
<h4> SAS Macros </h4> <h4> SAS Macros </h4>
@li mf_getattrn.sas @li mf_getattrn.sas
@param libds library.dataset @param [in] libds library.dataset
@return output returns result of the attrn value supplied, or log message @return output returns result of the attrn value supplied, or log message
if err. if err.

View File

@@ -11,8 +11,8 @@
<h4> SAS Macros </h4> <h4> SAS Macros </h4>
@param basestr The string to be modified @param [in] basestr The string to be modified
@param trimstr The string to be removed from the end of `basestr`, if it @param [in] trimstr The string to be removed from the end of `basestr`, if it
exists exists
@return output returns result with the value of `trimstr` removed from the end @return output returns result with the value of `trimstr` removed from the end

View File

@@ -13,8 +13,8 @@
returns: returns:
> blah blaaah brah > blah blaaah brah
@param str1= string containing words to extract @param [in] str1= () string containing words to extract
@param str2= used to compare with the extract string @param [in] str2= () used to compare with the extract string
@warning CASE SENSITIVE! @warning CASE SENSITIVE!

View File

@@ -16,8 +16,8 @@
returns: returns:
> sss bram boo > sss bram boo
@param [in] str1= string containing words to extract @param [in] str1= () String containing words to extract
@param [in] str2= used to compare with the extract string @param [in] str2= () Used to compare with the extract string
@version 9.2 @version 9.2
@author Allan Bowe @author Allan Bowe

View File

@@ -21,8 +21,8 @@
@param [in] mode= (O) Available options are A or O as follows: @param [in] mode= (O) Available options are A or O as follows:
@li A APPEND mode, writes new records after the current end of the file. @li A APPEND mode, writes new records after the current end of the file.
@li O OUTPUT mode, writes new records from the beginning of the file. @li O OUTPUT mode, writes new records from the beginning of the file.
@param [in] l1= First line @param [in] l1= () First line
@param [in] l2= Second line (etc through to l10) @param [in] l2= () Second line (etc through to l10)
<h4> Related Macros </h4> <h4> Related Macros </h4>
@li mf_writefile.test.sas @li mf_writefile.test.sas

View File

@@ -27,14 +27,15 @@
currently investigating approaches to deal with this. currently investigating approaches to deal with this.
@param mac= (mp_abort.sas) To contain the name of the calling macro. Do not @param [in] mac= (mp_abort.sas) To contain the name of the calling macro. Do
use &sysmacroname as this will always resolve to MP_ABORT. not use &sysmacroname as this will always resolve to MP_ABORT.
@param msg= message to be returned @param [out] msg= message to be returned
@param iftrue= (1=1) Supply a condition for which the macro should be executed @param [in] iftrue= (1=1) Condition under which the macro should be executed
@param errds= (work.mp_abort_errds) There is no clean way to end a process @param [in] errds= (work.mp_abort_errds) There is no clean way to end a
within a %include called within a macro. Furthermore, there is no way to process within a %include called within a macro. Furthermore, there is no
test if a macro is called within a %include. To handle this particular way to test if a macro is called within a %include. To handle this
scenario, the %include should be switched for the mp_include.sas macro. particular scenario, the %include should be switched for the mp_include.sas
macro.
This provides an indicator that we are running a macro within a \%include This provides an indicator that we are running a macro within a \%include
(`_SYSINCLUDEFILEDEVICE`) and allows us to provide a dataset with the abort (`_SYSINCLUDEFILEDEVICE`) and allows us to provide a dataset with the abort
values (msg, mac). values (msg, mac).
@@ -45,8 +46,8 @@
@li msg (the message) @li msg (the message)
@li mac (the mac param) @li mac (the mac param)
@param mode= (REGULAR) If mode=INCLUDE then the &errds dataset is checked for @param [in] mode= (REGULAR) If mode=INCLUDE then the &errds dataset is checked
an abort status. for an abort status.
Valid values: Valid values:
@li REGULAR (default) @li REGULAR (default)
@li INCLUDE @li INCLUDE

View File

@@ -62,8 +62,8 @@
7998580.8415 7998580.8415
@param var The (data step, character) variable to modify @param [in] var The (data step, character) variable to modify
@param width= (8) The number of characters BEFORE the decimal point @param [in] width= (8) The number of characters BEFORE the decimal point
<h4> SAS Macros </h4> <h4> SAS Macros </h4>
@li mf_getuniquename.sas @li mf_getuniquename.sas

View File

@@ -17,8 +17,8 @@
%mp_appendfile(baseref=tmp1, appendrefs=tmp2 tmp3) %mp_appendfile(baseref=tmp1, appendrefs=tmp2 tmp3)
@param [in] baseref= Fileref of the base file (should exist) @param [in] baseref= (0) Fileref of the base file (should exist)
@param [in] appendrefs= One or more filerefs to be appended to the base @param [in] appendrefs= (0) One or more filerefs to be appended to the base
fileref. Space separated. fileref. Space separated.
@version 9.2 @version 9.2

View File

@@ -31,8 +31,8 @@
@param [in] inds The input library.dataset to test for values @param [in] inds The input library.dataset to test for values
@param [in] cols= The list of columns to check for @param [in] cols= (0) The list of columns to check for
@param [in] desc= (Testing observations) The user provided test description @param [in] desc= (0) The user provided test description
@param [in] test= (ALL) The test to apply. Valid values are: @param [in] test= (ALL) The test to apply. Valid values are:
@li ALL - Test is a PASS if ALL columns exist in &inds @li ALL - Test is a PASS if ALL columns exist in &inds
@li ANY - Test is a PASS if ANY of the columns exist in &inds @li ANY - Test is a PASS if ANY of the columns exist in &inds

View File

@@ -36,7 +36,7 @@
@param [in] indscol The input library.dataset.column to test for values @param [in] indscol The input library.dataset.column to test for values
@param [in] checkvals= A library.dataset.column value containing a UNIQUE @param [in] checkvals= (0) A library.dataset.column value containing a UNIQUE
list of values to be compared against the source (indscol). list of values to be compared against the source (indscol).
@param [in] desc= (Testing observations) The user provided test description @param [in] desc= (Testing observations) The user provided test description
@param [in] test= (ALLVALS) The test to apply. Valid values are: @param [in] test= (ALLVALS) The test to apply. Valid values are:

View File

@@ -28,8 +28,8 @@
put _infile_; put _infile_;
run; run;
@param [in] inref= Fileref of the input file (should exist) @param [in] inref= (0) Fileref of the input file (should exist)
@param [out] outref= Output fileref. If it does not exist, it is created. @param [out] outref= (0) Output fileref. If it does not exist, it is created.
@param [in] action= (ENCODE) The action to take. Valid values: @param [in] action= (ENCODE) The action to take. Valid values:
@li ENCODE - Convert the file to base64 format @li ENCODE - Convert the file to base64 format
@li DECODE - Decode the file from base64 format @li DECODE - Decode the file from base64 format

View File

@@ -22,16 +22,17 @@
%mp_binarycopy(inref=tmp1, outref=tmp2, mode=APPEND) %mp_binarycopy(inref=tmp1, outref=tmp2, mode=APPEND)
@param [in] inloc quoted "path/and/filename.ext" of the file to be copied @param [in] inloc= () quoted "path/and/filename.ext" of the file to be copied
@param [out] outloc quoted "path/and/filename.ext" of the file to be created @param [out] outloc= () quoted "path/and/filename.ext" of the file to create
@param [in] inref (____in) If provided, this fileref will take precedence over @param [in] inref= (____in) If provided, this fileref takes precedence over
the `inloc` parameter the `inloc` parameter
@param [out] outref (____in) If provided, this fileref will take precedence @param [out] outref= (____in) If provided, this fileref takes precedence
over the `outloc` parameter. It must already exist! over the `outloc` parameter. It must already exist!
@param [in] mode (CREATE) Valid values: @param [in] mode= (CREATE) Valid values:
@li CREATE - Create the file (even if it already exists) @li CREATE - Create the file (even if it already exists)
@li APPEND - Append to the file (don't overwrite) @li APPEND - Append to the file (don't overwrite)
@param iftrue= (1=1) Supply a condition for which the macro should be executed @param [in] iftrue= (1=1)
Supply a condition for which the macro should be executed
@returns nothing @returns nothing

View File

@@ -44,7 +44,7 @@
For more examples, see mp_chop.test.sas For more examples, see mp_chop.test.sas
@param [in] infile The QUOTED path to the file on which to perform the chop @param [in] infile The QUOTED path to the file on which to perform the chop
@param [in] matchvar= Macro variable NAME containing the string to split by @param [in] matchvar= () Macro variable NAME containing the string to split by
@param [in] matchpoint= (START) Valid values: @param [in] matchpoint= (START) Valid values:
@li START - chop at the beginning of the string in `matchvar`. @li START - chop at the beginning of the string in `matchvar`.
@li END - chop at the end of the string in `matchvar`. @li END - chop at the end of the string in `matchvar`.
@@ -54,7 +54,8 @@
@li FIRST - keep the section of the file before the chop @li FIRST - keep the section of the file before the chop
@li LAST - keep the section of the file after the chop @li LAST - keep the section of the file after the chop
@param [in] mdebug= (0) Set to 1 to provide macro debugging @param [in] mdebug= (0) Set to 1 to provide macro debugging
@param outfile= (0) Optional QUOTED path to the adjusted output file (avoids @param [out] outfile= (0)
Optional QUOTED path to the adjusted output file (avoids
overwriting the first file). overwriting the first file).
<h4> SAS Macros </h4> <h4> SAS Macros </h4>

View File

@@ -12,11 +12,12 @@
fileref mycsv "/path/your/csv"; fileref mycsv "/path/your/csv";
%mp_cleancsv(in=mycsv,out=/path/new.csv) %mp_cleancsv(in=mycsv,out=/path/new.csv)
@param in= (NOTPROVIDED) Provide path or fileref to input csv. If a period is @param [in] in= (NOTPROVIDED)
Provide path or fileref to input csv. If a period is
found, it is assumed to be a file. found, it is assumed to be a file.
@param out= (NOTPROVIDED) Output path or fileref to output csv. If a period @param [in] out= (NOTPROVIDED) Output path or fileref to output csv.
is found, it is assumed to be a file. If a period is found, it is assumed to be a file.
@param qchar= ('22'x) Quote char - hex code 22 is the double quote. @param [in] qchar= ('22'x) Quote char - hex code 22 is the double quote.
@version 9.2 @version 9.2
@author Allan Bowe @author Allan Bowe

View File

@@ -79,8 +79,8 @@ data &cntlout/nonote2err;
end; end;
/* create row marker. Data cannot be sorted without it! */ /* create row marker. Data cannot be sorted without it! */
if first.fmtname then fmtrow=0; if first.fmtname then fmtrow=1;
fmtrow+1; else fmtrow+1;
run; run;
proc sort; proc sort;

View File

@@ -18,11 +18,11 @@
%mp_deleteconstraints(inds=work.constraints,outds=dropped,execute=YES) %mp_deleteconstraints(inds=work.constraints,outds=dropped,execute=YES)
%mp_createconstraints(inds=work.constraints,outds=created,execute=YES) %mp_createconstraints(inds=work.constraints,outds=created,execute=YES)
@param inds= (work.mp_getconstraints) The input table containing the @param [in] inds= (work.mp_getconstraints) The input table containing the
constraint info constraint info
@param outds= (work.mp_createconstraints) A table containing the create @param [out] outds= (work.mp_createconstraints) A table containing the create
statements (create_statement column) statements (create_statement column)
@param execute= (NO) To actually create, use YES. @param [in] execute= (NO) To actually create, use YES.
<h4> Related Files </h4> <h4> Related Files </h4>
@li mp_getconstraints.sas @li mp_getconstraints.sas

View File

@@ -1,6 +1,6 @@
/** /**
@file mp_createwebservice.sas @file mp_createwebservice.sas
@brief Create a web service in SAS 9, Viya or SASjs Server @brief Create a web service in SAS 9, Viya or SASjs Server (legacy)
@details This is actually a wrapper for mx_createwebservice.sas, remaining @details This is actually a wrapper for mx_createwebservice.sas, remaining
for legacy purposes. For new apps, use mx_createwebservice.sas. for legacy purposes. For new apps, use mx_createwebservice.sas.

View File

@@ -19,11 +19,12 @@
%mp_csv2ds(inref=mycsv,outds=myds,baseds=sashelp.class) %mp_csv2ds(inref=mycsv,outds=myds,baseds=sashelp.class)
@param inref= fileref to the CSV @param [in] inref= (0) Fileref to the CSV
@param outds= output ds (lib.ds format) @param [out] outds= (0) Output ds (lib.ds format)
@param view= Set to YES or NO to determine whether the output should be @param [in] view= (NO) Set to YES or NO to determine whether the output
a view or not. Default is NO (not a view). should be a view or not. Default is NO (not a view).
@param baseds= Template dataset on which to create the input statement. @param [in] baseds= (0)
Template dataset on which to create the input statement.
Is used to determine types, lengths, and any informats. Is used to determine types, lengths, and any informats.
@version 9.2 @version 9.2

View File

@@ -17,9 +17,11 @@
%mp_getconstraints(lib=work,ds=example,outds=work.constraints) %mp_getconstraints(lib=work,ds=example,outds=work.constraints)
%mp_deleteconstraints(inds=work.constraints,outds=dropped,execute=YES) %mp_deleteconstraints(inds=work.constraints,outds=dropped,execute=YES)
@param inds= The input table containing the constraint info @param [in] inds= (mp_getconstraints)
@param outds= a table containing the drop statements (drop_statement column) The input table containing constraint info
@param execute= `YES|NO` - default is NO. To actually drop, use YES. @param [out] outds= (mp_deleteconstraints)
Table containing the drop statements (drop_statement column)
@param [in] execute= (NO) `YES|NO` - default is NO. To actually drop, use YES.
@version 9.2 @version 9.2

View File

@@ -15,7 +15,7 @@
%mp_deletefolder(&rootdir) %mp_deletefolder(&rootdir)
@param path Unquoted path to the folder to delete. @param [in] folder Unquoted path to the folder to delete.
<h4> SAS Macros </h4> <h4> SAS Macros </h4>
@li mf_getuniquename.sas @li mf_getuniquename.sas

View File

@@ -29,7 +29,7 @@
![](https://user-images.githubusercontent.com/4420615/188278365-2987db97-0594-4a39-ac81-dbacdef5cdc8.png) ![](https://user-images.githubusercontent.com/4420615/188278365-2987db97-0594-4a39-ac81-dbacdef5cdc8.png)
@param lib= (WORK) The libref in which to create the views @param [in] lib= (WORK) The libref in which to create the views
<h4> Related Files </h4> <h4> Related Files </h4>
@li mp_dictionary.test.sas @li mp_dictionary.test.sas

View File

@@ -7,11 +7,11 @@
%mp_distinctfmtvalues(libds=sashelp.class,var=age,outvar=age,outds=test) %mp_distinctfmtvalues(libds=sashelp.class,var=age,outvar=age,outds=test)
@param libds input dataset @param [in] libds= () input dataset
@param var variable to get distinct values for @param [in] var= (0) variable to get distinct values for
@param outvar variable to create. Default: `formatted_value` @param [out] outvar= (formatteed_value) variable to create.
@param outds dataset to create. Default: work.mp_distinctfmtvalues @param [out] outds= (work.mp_distinctfmtvalues) dataset to create.
@param varlen length of variable to create (default 200) @param [in] varlen= (2000) length of variable to create
@version 9.2 @version 9.2
@author Allan Bowe @author Allan Bowe

View File

@@ -118,13 +118,21 @@ data _null_;
header = cats(coalescec(varlabel(dsid,i),varnm),dlm); header = cats(coalescec(varlabel(dsid,i),varnm),dlm);
%end; %end;
%else %if &headerformat=SASJS %then %do; %else %if &headerformat=SASJS %then %do;
if vartype(dsid,i)='C' then header=cats(varnm,':$char',varlen(dsid,i),'.'); vlen=varlen(dsid,i);
if vartype(dsid,i)='C' then header=cats(varnm,':$char',vlen,'.');
else do; else do;
vfmt=coalescec(varfmt(dsid,i),'0'); vfmt=coalescec(varfmt(dsid,i),'0');
fmttype=mcf_getfmttype(vfmt); fmttype=mcf_getfmttype(vfmt);
if fmttype='DATE' then header=cats(varnm,':date9.'); if fmttype='DATE' then header=cats(varnm,':date9.');
else if fmttype='DATETIME' then header=cats(varnm,':E8601DT26.6'); else if fmttype='DATETIME' then header=cats(varnm,':E8601DT26.6');
else if fmttype='TIME' then header=cats(varnm,':TIME12.'); else if fmttype='TIME' then header=cats(varnm,':TIME12.');
/**
* there is not much point importing a short length numeric like this,
* eg with best4., as the resulting variable will still be stored as
* length 8. We need a length or format statement to ensure variable
* is creatd with the smaller length...
**/
else if vlen<8 then header=cats(varnm,':best',vlen,'.');
else header=cats(varnm,':best.'); else header=cats(varnm,':best.');
end; end;
%end; %end;
@@ -151,6 +159,7 @@ data _null_;
set &ds end=last; set &ds end=last;
%do i=1 %to &vcnt; %do i=1 %to &vcnt;
%let var=%scan(&varlist,&i); %let var=%scan(&varlist,&i);
%local vlen&i;
%if %mf_getvartype(&ds,&var)=C %then %do; %if %mf_getvartype(&ds,&var)=C %then %do;
%let dsv1=%mf_getuniquename(prefix=csvcol1_); %let dsv1=%mf_getuniquename(prefix=csvcol1_);
%let dsv2=%mf_getuniquename(prefix=csvcol2_); %let dsv2=%mf_getuniquename(prefix=csvcol2_);

View File

@@ -1,8 +1,20 @@
/** /**
@file @file
@brief A wrapper for mp_getddl.sas @brief Fetches DDL for a specific table
@details In the next release, this will be the main version. @details Uses mp_getddl under the hood
@param [in] libds library.dataset to create ddl for
@param [in] fref= (getddl) the fileref to which to _append_ the DDL. If it
does not exist, it will be created.
@param [in] flavour= (SAS) The type of DDL to create. Options:
@li SAS
@li TSQL
@param [in]showlog= (NO) Set to YES to show the DDL in the log
@param [in] schema= () Choose a preferred schema name (default is to use
actual schema, else libref)
@param [in] applydttm= (NO) For non SAS DDL, choose if columns are created with
native datetime2 format or regular decimal type
<h4> SAS Macros </h4> <h4> SAS Macros </h4>
@li mp_getddl.sas @li mp_getddl.sas

View File

@@ -49,8 +49,8 @@
https://support.sas.com/resources/papers/proceedings14/1549-2014.pdf) by https://support.sas.com/resources/papers/proceedings14/1549-2014.pdf) by
[Louise Hadden](https://www.linkedin.com/in/louisehadden/). [Louise Hadden](https://www.linkedin.com/in/louisehadden/).
@param libds The library.dataset to export the metadata for @param [in] libds The library.dataset to export the metadata for
@param outds= (work.dsmeta) The output table to contain the metadata @param [out] outds= (work.dsmeta) The output table to contain the metadata
<h4> Related Files </h4> <h4> Related Files </h4>
@li mp_dsmeta.test.sas @li mp_dsmeta.test.sas

View File

@@ -37,10 +37,12 @@
@param [in] targetds= The target dataset against which to verify VARIABLE_NM. @param [in] targetds= The target dataset against which to verify VARIABLE_NM.
This must be available (ie, the library must be assigned). This must be available (ie, the library must be assigned).
@param [out] abort= (YES) If YES will call mp_abort.sas on any exceptions @param [out] abort= (YES) If YES will call mp_abort.sas on any exceptions
@param [out] outds= The output table, which is a copy of the &inds. table @param [out] outds= (work.badrecords) The output table, which is a copy of the
plus a REASON_CD column, containing only bad records. If bad records found, &inds. table plus a REASON_CD column, containing only bad records.
the SYSCC value will be set to 1008 (general data problem). Downstream If bad records are found, the SYSCC value will be set to 1008
processes should check this table (and return code) before continuing. (a general data problem).
Downstream processes should check this table (and return code) before
continuing.
<h4> SAS Macros </h4> <h4> SAS Macros </h4>
@li mp_abort.sas @li mp_abort.sas
@@ -84,15 +86,14 @@
/** /**
* Sanitise the values based on valid value lists, then strip out * Sanitise the values based on valid value lists, then strip out
* quotes, commas, periods and spaces. * quotes, commas, periods and spaces.
* Only numeric values should remain
*/ */
%local reason_cd nobs; %local reason_cd nobs;
%let nobs=0; %let nobs=0;
data &outds; data &outds;
/*length GROUP_LOGIC SUBGROUP_LOGIC $3 SUBGROUP_ID 8 VARIABLE_NM $32 /*length GROUP_LOGIC SUBGROUP_LOGIC $3 SUBGROUP_ID 8 VARIABLE_NM $32
OPERATOR_NM $10 RAW_VALUE $4000;*/ OPERATOR_NM $10 RAW_VALUE $4000;*/
set &inds; set &inds end=last;
length reason_cd $4032 vtype $1 vnum dsid 8 tmp $4000; length reason_cd $4032 vtype vtype2 $1 vnum dsid 8 tmp $4000;
drop tmp; drop tmp;
/* quick check to ensure column exists */ /* quick check to ensure column exists */
@@ -108,7 +109,8 @@ data &outds;
end; end;
/* need to open the dataset to get the column type */ /* need to open the dataset to get the column type */
dsid=open("&targetds","i"); retain dsid;
if _n_=1 then dsid=open("&targetds","i");
if dsid>0 then do; if dsid>0 then do;
vnum=varnum(dsid,VARIABLE_NM); vnum=varnum(dsid,VARIABLE_NM);
if vnum<1 then do; if vnum<1 then do;
@@ -118,11 +120,19 @@ data &outds;
call symputx('reason_cd',reason_cd,'l'); call symputx('reason_cd',reason_cd,'l');
call symputx('nobs',_n_,'l'); call symputx('nobs',_n_,'l');
output; output;
return; goto endstep;
end; end;
/* now we can get the type */ /* now we can get the type */
else vtype=vartype(dsid,vnum); else vtype=vartype(dsid,vnum);
end; end;
else do;
REASON_CD=cats("Could not open &targetds");
putlog REASON_CD= dsid=;
call symputx('reason_cd',reason_cd,'l');
call symputx('nobs',_n_,'l');
output;
stop;
end;
/* closed list checks */ /* closed list checks */
if GROUP_LOGIC not in ('AND','OR') then do; if GROUP_LOGIC not in ('AND','OR') then do;
@@ -157,9 +167,8 @@ data &outds;
end; end;
/* special missing logic */ /* special missing logic */
if vtype='N' if vtype='N' & OPERATOR_NM in ('=','>','<','<=','>=','NE','GE','LE') then do;
and OPERATOR_NM in ('=','>','<','<=','>=','NE','GE','LE') if cats(upcase(raw_value)) in (
and cats(upcase(raw_value)) in (
'.','.A','.B','.C','.D','.E','.F','.G','.H','.I','.J','.K','.L','.M','.N' '.','.A','.B','.C','.D','.E','.F','.G','.H','.I','.J','.K','.L','.M','.N'
'.N','.O','.P','.Q','.R','.S','.T','.U','.V','.W','.X','.Y','.Z','._' '.N','.O','.P','.Q','.R','.S','.T','.U','.V','.W','.X','.Y','.Z','._'
) )
@@ -167,6 +176,32 @@ data &outds;
/* valid numeric - exit data step loop */ /* valid numeric - exit data step loop */
return; return;
end; end;
else if subpad(upcase(raw_value),1,1) in (
'A','B','C','D','E','F','G','H','I','J','K','L','M','N'
'N','O','P','Q','R','S','T','U','V','W','X','Y','Z','_'
)
then do;
/* check if the raw_value contains a valid variable NAME */
vnum=varnum(dsid,subpad(raw_value,1,32));
if vnum>0 then do;
/* now we can get the type */
vtype2=vartype(dsid,vnum);
/* check type matches */
if vtype2=vtype then do;
/* valid target var - exit loop */
return;
end;
else do;
REASON_CD=cats("Compared Type (",vtype2,") is not (",vtype,")");
putlog REASON_CD= dsid=;
call symputx('reason_cd',reason_cd,'l');
call symputx('nobs',_n_,'l');
output;
goto endstep;
end;
end;
end;
end;
/* special logic */ /* special logic */
if OPERATOR_NM in ('IN','NOT IN','BETWEEN') then do; if OPERATOR_NM in ('IN','NOT IN','BETWEEN') then do;
@@ -187,6 +222,32 @@ data &outds;
if vtype='N' then do i=1 to countc(raw_value1, ',')+1; if vtype='N' then do i=1 to countc(raw_value1, ',')+1;
tmp=scan(raw_value1,i,','); tmp=scan(raw_value1,i,',');
if cats(tmp) ne '.' and input(tmp, ?? 8.) eq . then do; if cats(tmp) ne '.' and input(tmp, ?? 8.) eq . then do;
if OPERATOR_NM ='BETWEEN' and subpad(upcase(tmp),1,1) in (
'A','B','C','D','E','F','G','H','I','J','K','L','M','N'
'N','O','P','Q','R','S','T','U','V','W','X','Y','Z','_'
)
then do;
/* check if the raw_value contains a valid variable NAME */
/* is not valid syntax for IN or NOT IN */
vnum=varnum(dsid,subpad(tmp,1,32));
if vnum>0 then do;
/* now we can get the type */
vtype2=vartype(dsid,vnum);
/* check type matches */
if vtype2=vtype then do;
/* valid target var - exit loop */
return;
end;
else do;
REASON_CD=cats("Compared Type (",vtype2,") is not (",vtype,")");
putlog REASON_CD= dsid=;
call symputx('reason_cd',reason_cd,'l');
call symputx('nobs',_n_,'l');
output;
goto endstep;
end;
end;
end;
REASON_CD='Non Numeric value provided'; REASON_CD='Non Numeric value provided';
putlog REASON_CD= OPERATOR_NM= raw_value= raw_value1= ; putlog REASON_CD= OPERATOR_NM= raw_value= raw_value1= ;
call symputx('reason_cd',reason_cd,'l'); call symputx('reason_cd',reason_cd,'l');
@@ -211,14 +272,42 @@ data &outds;
/* output records that contain values other than digits and spaces */ /* output records that contain values other than digits and spaces */
if notdigit(compress(raw_value3,' '))>0 then do; if notdigit(compress(raw_value3,' '))>0 then do;
if vtype='C' and subpad(upcase(raw_value),1,1) in (
'A','B','C','D','E','F','G','H','I','J','K','L','M','N'
'N','O','P','Q','R','S','T','U','V','W','X','Y','Z','_'
)
then do;
/* check if the raw_value contains a valid variable NAME */
vnum=varnum(dsid,subpad(raw_value,1,32));
if vnum>0 then do;
/* now we can get the type */
vtype2=vartype(dsid,vnum);
/* check type matches */
if vtype2=vtype then do;
/* valid target var - exit loop */
return;
end;
else do;
REASON_CD=cats("Compared Char Type (",vtype2,") is not (",vtype,")");
putlog REASON_CD= dsid=;
call symputx('reason_cd',reason_cd,'l');
call symputx('nobs',_n_,'l');
output;
goto endstep;
end;
end;
end;
putlog raw_value3= $hex32.; putlog raw_value3= $hex32.;
REASON_CD=cats('Invalid RAW_VALUE:',raw_value); REASON_CD=cats('Invalid RAW_VALUE:',raw_value);
putlog REASON_CD= raw_value= raw_value1= raw_value2= raw_value3=; putlog (_all_)(=);
call symputx('reason_cd',reason_cd,'l'); call symputx('reason_cd',reason_cd,'l');
call symputx('nobs',_n_,'l'); call symputx('nobs',_n_,'l');
output; output;
end; end;
endstep:
if last then rc=close(dsid);
run; run;

View File

@@ -51,8 +51,8 @@
> ) > )
@param [in] inds The input table with query values @param [in] inds The input table with query values
@param [out] outref= The output fileref to contain the filter clause. Will @param [out] outref= (filter) The output fileref to contain the filter clause.
be created (or replaced). Will be created (or replaced).
<h4> Related Macros </h4> <h4> Related Macros </h4>
@li mp_filtercheck.sas @li mp_filtercheck.sas

View File

@@ -44,9 +44,10 @@
mp_coretable.sas as follows: `mp_coretable(LOCKTABLE)`. mp_coretable.sas as follows: `mp_coretable(LOCKTABLE)`.
@param [in] maxkeytable= (0) Optional permanent reference table used for @param [in] maxkeytable= (0) Optional permanent reference table used for
retained key tracking. Described in mp_retainedkey.sas. retained key tracking. Described in mp_retainedkey.sas.
@param [in] mdebug= set to 1 to enable DEBUG messages @param [in] mdebug= (1) set to 1 to enable DEBUG messages
@param [out] outresult= The result table with the FILTER_RK @param [out] outresult= (work.result) The result table with the FILTER_RK
@param [out] outquery= The original query, taken as extract after table load @param [out] outquery= (work.query) The original query, taken as extract
after table load
<h4> SAS Macros </h4> <h4> SAS Macros </h4>

View File

@@ -12,14 +12,15 @@
%mp_getcols(sashelp.airline,outds=work.myds) %mp_getcols(sashelp.airline,outds=work.myds)
@param ds The dataset from which to obtain column metadata @param [in] ds The dataset from which to obtain column metadata
@param outds= (work.cols) The output dataset to create. Sample data: @param [out] outds= (work.cols) The output dataset to create. Sample data:
|NAME:$32.|LENGTH:best.|VARNUM:best.|LABEL:$256.|FMTNAME:$32.|FORMAT:$49.|TYPE:$1.|DDTYPE:$9.| |NAME:$32.|LENGTH:best.|VARNUM:best.|LABEL:$256.|FMTNAME:$32.|FORMAT:$49.|TYPE:$1.|DDTYPE:$9.|
|---|---|---|---|---|---|---|---| |---|---|---|---|---|---|---|---|
|`AIR `|`8 `|`2 `|`international airline travel (thousands) `|` `|`8. `|`N `|`NUMERIC `| |`AIR `|`8 `|`2 `|`international airline travel (thousands) `|` `|`8. `|`N `|`NUMERIC `|
|`DATE `|`8 `|`1 `|`DATE `|`MONYY `|`MONYY. `|`N `|`DATE `| |`DATE `|`8 `|`1 `|`DATE `|`MONYY `|`MONYY. `|`N `|`DATE `|
|`REGION `|`3 `|`3 `|`REGION `|` `|`$3. `|`C `|`CHARACTER `| |`REGION `|`3 `|`3 `|`REGION `|` `|`$3. `|`C `|`CHARACTER `|
<h4> Related Macros </h4> <h4> Related Macros </h4>
@li mf_getvarlist.sas @li mf_getvarlist.sas
@li mm_getcols.sas @li mm_getcols.sas

View File

@@ -23,10 +23,10 @@
@li mf_getquotedstr.sas @li mf_getquotedstr.sas
@li mp_getconstraints.sas @li mp_getconstraints.sas
@param liblist= Space seperated list of librefs to take as @param [in] liblist= (SASHELP) Space seperated list of librefs to take as
input (Default=SASHELP) input
@param outref= Fileref to contain the DBML (Default=getdbml) @param [out] outref= (getdbml) Fileref to contain the DBML
@param showlog= set to YES to show the DBML in the log (Default is NO) @param [in] showlog= (NO) set to YES to show the DBML in the log
@version 9.3 @version 9.3
@author Allan Bowe @author Allan Bowe

View File

@@ -22,16 +22,21 @@
@li mf_getvarcount.sas @li mf_getvarcount.sas
@li mp_getconstraints.sas @li mp_getconstraints.sas
@param lib libref of the library to create DDL for. Should be assigned. @param [in] libref Libref of the library to create DDL for. Should already
@param ds dataset to create ddl for (optional) be assigned.
@param fref= the fileref to which to _append_ the DDL. If it does not exist, @param [in] ds dataset to create ddl for (optional)
it will be created. @param [in] fref= (getddl) the fileref to which to _append_ the DDL. If it
@param flavour= The type of DDL to create (default=SAS). Supported=TSQL does not exist, it will be created.
@param showlog= Set to YES to show the DDL in the log @param [in] flavour= (SAS) The type of DDL to create. Options:
@param schema= Choose a preferred schema name (default is to use actual schema @li SAS
,else libref) @li TSQL
@param applydttm= for non SAS DDL, choose if columns are created with native
datetime2 format or regular decimal type @param [in]showlog= (NO) Set to YES to show the DDL in the log
@param [in] schema= () Choose a preferred schema name (default is to use
actual schema, else libref)
@param [in] applydttm= (NO) For non SAS DDL, choose if columns are created
with native datetime2 format or regular decimal type
@version 9.3 @version 9.3
@author Allan Bowe @author Allan Bowe
**/ **/

View File

@@ -22,15 +22,15 @@
list; list;
run; run;
@param file= (0) The file to perform the substitution on @param [in] file= (0) The file to perform the substitution on
@param patternvar= A macro variable containing the Lua @param [in] patternvar= A macro variable containing the Lua
[pattern](https://www.lua.org/pil/20.2.html) to search for. Due to the use [pattern](https://www.lua.org/pil/20.2.html) to search for. Due to the use
of special (magic) characters in Lua patterns, it is safer to pass the NAME of special (magic) characters in Lua patterns, it is safer to pass the NAME
of the macro variable containing the string, rather than the value itself. of the macro variable containing the string, rather than the value itself.
@param replacevar= The name of the macro variable containing the replacement @param [in] replacevar= ()
_string_. The name of the macro variable containing the replacement _string_.
@param outfile= (0) The file to write the output to. If zero, then the file @param [out] outfile= (0) The file to write the output to.
is overwritten in-place. If zero, then the file is overwritten in-place.
<h4> SAS Macros </h4> <h4> SAS Macros </h4>
@li ml_gsubfile.sas @li ml_gsubfile.sas

View File

@@ -18,7 +18,7 @@
%mp_guesspk(sashelp.class,outds=classpks) %mp_guesspk(sashelp.class,outds=classpks)
@param [in] baseds The dataset to analyse @param [in] baseds The dataset to analyse
@param [out] outds= The output dataset to contain the possible PKs @param [out] outds= (mp_guesspk) Output dataset to contain the possible PKs
@param [in] max_guesses= (3) The total number of possible primary keys to @param [in] max_guesses= (3) The total number of possible primary keys to
generate. A table may have multiple (unlikely) PKs, so no need to list them generate. A table may have multiple (unlikely) PKs, so no need to list them
all. all.

View File

@@ -24,7 +24,7 @@
@li mp_hashdirectory.sas @li mp_hashdirectory.sas
@param [in] libds dataset to hash @param [in] libds dataset to hash
@param [in] salt= Provide a salt (could be, for instance, the dataset name) @param [in] salt= () Provide a salt (could be, for instance, the dataset name)
@param [in] iftrue= (1=1) A condition under which the macro should be executed @param [in] iftrue= (1=1) A condition under which the macro should be executed
@param [out] outds= (work._data_) The output dataset to create. This @param [out] outds= (work._data_) The output dataset to create. This
will contain one column (hashkey) with one observation (a $hex32. will contain one column (hashkey) with one observation (a $hex32.

View File

@@ -24,19 +24,19 @@
@li mf_trimstr.sas @li mf_trimstr.sas
@li mp_ds2cards.sas @li mp_ds2cards.sas
@param [in] lib= Library in which to convert all datasets @param [in] lib= () Library in which to convert all datasets
@param [out] outloc= Location in which to store output. Defaults to WORK @param [out] outloc= (%sysfunc(pathname(work))) Location in which to store
library. No quotes. output. No quotes.
@param [out] outfile= Optional output file NAME - if provided, then will create @param [out] outfile= (0) Optional output file NAME - if provided, then
a single output file instead of one file per input table. will create a single output file instead of one file per input table.
@param [in] maxobs= limit output to the first <code>maxobs</code> observations @param [in] maxobs= (max) limit output to the first <code>maxobs</code> rows
@version 9.2 @version 9.2
@author Allan Bowe @author Allan Bowe
**/ **/
%macro mp_lib2cards(lib= %macro mp_lib2cards(lib=
,outloc=%sysfunc(pathname(work)) /* without trailing slash */ ,outloc=%sysfunc(pathname(work))
,maxobs=max ,maxobs=max
,random_sample=NO ,random_sample=NO
,outfile=0 ,outfile=0

View File

@@ -9,6 +9,9 @@
format, to prevent loss of data - UNLESS the input dataset contains a marker format, to prevent loss of data - UNLESS the input dataset contains a marker
column, specifying that a particular row needs to be deleted (`delete_col=`). column, specifying that a particular row needs to be deleted (`delete_col=`).
Positions of formats are made using the FMTROW variable - this must be present
and unique (on TYPE / FMTNAME / FMTROW).
This macro can also be used to identify which records would be (or were) This macro can also be used to identify which records would be (or were)
considered new, modified or deleted (`loadtarget=`) by creating the following considered new, modified or deleted (`loadtarget=`) by creating the following
tables: tables:
@@ -17,7 +20,7 @@
@li work.outds_del @li work.outds_del
@li work.outds_mod @li work.outds_mod
For example usage, see mp_loadformat.test.sas For example usage, see test (under Related Macros)
@param [in] libcat The format catalog to be loaded @param [in] libcat The format catalog to be loaded
@param [in] libds The staging table to load @param [in] libds The staging table to load
@@ -34,12 +37,15 @@
@param [in] mdebug= (0) Set to 1 to enable DEBUG messages and preserve outputs @param [in] mdebug= (0) Set to 1 to enable DEBUG messages and preserve outputs
<h4> SAS Macros </h4> <h4> SAS Macros </h4>
@li mf_existds.sas
@li mf_existvar.sas
@li mf_getuniquename.sas @li mf_getuniquename.sas
@li mf_nobs.sas @li mf_nobs.sas
@li mp_abort.sas @li mp_abort.sas
@li mp_aligndecimal.sas @li mp_aligndecimal.sas
@li mp_cntlout.sas @li mp_cntlout.sas
@li mp_lockanytable.sas @li mp_lockanytable.sas
@li mp_md5.sas
@li mp_storediffs.sas @li mp_storediffs.sas
<h4> Related Macros </h4> <h4> Related Macros </h4>
@@ -83,6 +89,16 @@
%let libcat=%scan(&libcat,1,-); %let libcat=%scan(&libcat,1,-);
/* perform input validations */ /* perform input validations */
%mp_abort(
iftrue=(%mf_existds(&libds)=0)
,mac=&sysmacroname
,msg=%str(&libds could not be found)
)
%mp_abort(
iftrue=(%mf_existvar(&libds,FMTROW)=0)
,mac=&sysmacroname
,msg=%str(FMTROW not found in &libds)
)
%let err=0; %let err=0;
%let msg=0; %let msg=0;
data _null_; data _null_;
@@ -103,13 +119,6 @@ data _null_;
stop; stop;
end; end;
end; end;
else if name='LIBDS' then do;
if exist(value) le 0 then do;
call symputx('msg',"Unable to open staging table: "!!value);
call symputx('err',1);
stop;
end;
end;
else if (name=:'OUTDS' or name in ('DELETE_COL','LOCKLIBDS','AUDITLIBDS')) else if (name=:'OUTDS' or name in ('DELETE_COL','LOCKLIBDS','AUDITLIBDS'))
and missing(value) then do; and missing(value) then do;
call symputx('msg',"missing value in var: "!!name); call symputx('msg',"missing value in var: "!!name);
@@ -117,6 +126,14 @@ data _null_;
stop; stop;
end; end;
run; run;
data _null_;
set &libds;
if missing(fmtrow) then do;
call symputx('msg',"missing fmtrow in format: "!!FMTNAME);
call symputx('err',1);
stop;
end;
run;
%mp_abort( %mp_abort(
iftrue=(&err ne 0) iftrue=(&err ne 0)
@@ -124,6 +141,15 @@ run;
,msg=%str(&msg) ,msg=%str(&msg)
) )
%local cnt;
proc sql noprint;
select count(distinct catx('|',type,fmtname,fmtrow)) into: cnt from &libds;
%mp_abort(
iftrue=(&cnt ne %mf_nobs(&libds))
,mac=&sysmacroname
,msg=%str(Non-unique primary key on &libds)
)
/** /**
* First, extract only relevant formats from the catalog * First, extract only relevant formats from the catalog
*/ */
@@ -177,12 +203,6 @@ data &inlibds/nonote2err;
%mp_aligndecimal(end,width=16) %mp_aligndecimal(end,width=16)
end; end;
/* update row marker - retain new var as fmtrow may already be in libds */
if first.fmtname then row=1;
else row+1;
drop row;
fmtrow=row;
fmthash=%mp_md5(cvars=&cvars, nvars=&nvars); fmthash=%mp_md5(cvars=&cvars, nvars=&nvars);
run; run;

View File

@@ -28,8 +28,8 @@
@li Global option: `options dsoptions=nonote2err;` @li Global option: `options dsoptions=nonote2err;`
@li Data step option: `data YOURLIB.YOURDATASET /nonote2err;` @li Data step option: `data YOURLIB.YOURDATASET /nonote2err;`
@param cvars= Space seperated list of character variables @param [in] cvars= () Space seperated list of character variables
@param nvars= Space seperated list of numeric variables @param [in] nvars= () Space seperated list of numeric variables
<h4> Related Programs </h4> <h4> Related Programs </h4>
@li mp_init.sas @li mp_init.sas

View File

@@ -11,9 +11,9 @@
%mp_perflog(finished) %mp_perflog(finished)
@param label Provide label to go into the control dataset @param [in] label Provide label to go into the control dataset
@param libds= Provide a dataset in which to store performance stats. Default @param [in] libds= (work.mp_perflog) Provide a dataset in which to store
name is <code>work.mp_perflog</code>; performance stats. Default name is <code>work.mp_perflog</code>;
@version 9.2 @version 9.2
@author Allan Bowe @author Allan Bowe

View File

@@ -34,12 +34,12 @@
Credit is made to `data _null_` for authoring this very helpful paper: Credit is made to `data _null_` for authoring this very helpful paper:
https://www.lexjansen.com/pharmasug/2008/cc/CC08.pdf https://www.lexjansen.com/pharmasug/2008/cc/CC08.pdf
@param action Either FETCH a current or previous record, or INITialise. @param [in] action Either FETCH a current or previous record, or INITialise.
@param record The relative (to current) position of the previous observation @param [in] record The relative (to current) position of the previous row
to return. to return.
@param history= The number of records to retain in the hash table. Default=5 @param [in] history= (5) The number of records to retain in the hash table.
@param prefix= the prefix to give to the variables used to store the hash name @param [in] prefix= (mp_prevobs) The prefix to give to the variables used to
and index. Default=mp_prevobs store the hash name and index.
@version 9.2 @version 9.2
@author Allan Bowe @author Allan Bowe

View File

@@ -23,12 +23,13 @@
,childvar=c ,childvar=c
) )
@param base_ds= base table containing hierarchy (not modified) @param [in] base_ds= base table containing hierarchy (not modified)
@param outds= the output dataset to create with the generated hierarchy @param [out] outds= the output dataset to create with the generated hierarchy
@param matchval= the ultimate parent from which to filter @param [in] matchval= the ultimate parent from which to filter
@param parentvar= name of the parent variable @param [in] parentvar= name of the parent variable
@param childvar= name of the child variable (should be same type as parent) @param [in] childvar= () name of the child variable (should be same type as
@param mdebug= set to 1 to prevent temp tables being dropped parent)
@param [in] mdebug= set to 1 to prevent temp tables being dropped
@returns outds contains the following variables: @returns outds contains the following variables:

View File

@@ -32,10 +32,11 @@
Note - if you are running a version of SAS that will allow the io package in Note - if you are running a version of SAS that will allow the io package in
LUA, you can also use this macro: mp_gsubfile.sas LUA, you can also use this macro: mp_gsubfile.sas
@param infile The QUOTED path to the file on which to perform the substitution @param [in] infile The QUOTED path to the file on which to perform the
@param findvar= Macro variable NAME containing the string to search for substitution
@param replacevar= Macro variable NAME containing the replacement string @param [in] findvar= Macro variable NAME containing the string to search for
@param outfile= (0) Optional QUOTED path to the adjusted output file (to @param [in] replacevar= Macro variable NAME containing the replacement string
@param [out] outfile= (0) Optional QUOTED path to the adjusted output file (to
avoid overwriting the first file). avoid overwriting the first file).
<h4> SAS Macros </h4> <h4> SAS Macros </h4>

View File

@@ -31,9 +31,9 @@
%mp_runddl(/some/rootlib, exc=LIBREF3) * same as above ; %mp_runddl(/some/rootlib, exc=LIBREF3) * same as above ;
@param path location of the DDL folder structure @param [in] path location of the DDL folder structure
@param inc= list of librefs to include @param [in] inc= list of librefs to include
@param exc= list of librefs to exclude (takes precedence over inc=) @param [in] exc= list of librefs to exclude (takes precedence over inc=)
@version 9.3 @version 9.3
@author Allan Bowe @author Allan Bowe

View File

@@ -9,12 +9,13 @@
%mp_searchcols(libs=sashelp work, cols=name sex age) %mp_searchcols(libs=sashelp work, cols=name sex age)
@param libs=(SASHELP) Space separated list of libraries to search for columns @param [in] libs= (SASHELP)
@param cols= Space separated list of column names to search for (not case Space separated list of libraries to search for columns
sensitive) @param [in] cols=
@param outds=(mp_searchcols) the table to create with the results. Will have Space separated list of column names to search for (not case sensitive)
one line per table match. @param [out] outds= (mp_searchcols)
@param match=(ANY) The match type. Valid values: The table to create with the results. Will have one line per table match.
@param [in] match= (ANY) The match type. Valid values:
@li ANY - The table contains at least one of the columns @li ANY - The table contains at least one of the columns
@li WILD - The table contains a column with a name that partially matches @li WILD - The table contains a column with a name that partially matches

View File

@@ -197,6 +197,7 @@
@li mp_coretable.sas @li mp_coretable.sas
@li mp_stackdiffs.test.sas @li mp_stackdiffs.test.sas
@li mp_storediffs.sas @li mp_storediffs.sas
@li mp_stripdiffs.sas
@todo The current approach assumes that a variable called KEY_HASH is not on @todo The current approach assumes that a variable called KEY_HASH is not on
the base table. This part will need to be refactored (eg using the base table. This part will need to be refactored (eg using

View File

@@ -64,6 +64,7 @@
<h4> Related Macros </h4> <h4> Related Macros </h4>
@li mp_stackdiffs.sas @li mp_stackdiffs.sas
@li mp_storediffs.test.sas @li mp_storediffs.test.sas
@li mp_stripdiffs.sas
@version 9.2 @version 9.2
@author Allan Bowe @author Allan Bowe
@@ -183,7 +184,7 @@ data &ds4;
run; run;
%if "&loadref"="0" %then %let loadref=%sysfunc(uuidgen()); %if "&loadref"="0" %then %let loadref=%sysfunc(uuidgen());
%if &processed_dttm=0 %then %let processed_dttm=%sysfunc(datetime()); %if &processed_dttm=0 %then %let processed_dttm=%sysfunc(datetime(),8.6);
%let libds=%upcase(&libds); %let libds=%upcase(&libds);
/* join orig vals for modified & deleted */ /* join orig vals for modified & deleted */

View File

@@ -21,9 +21,10 @@
%mp_stprequests(status_cd=INIT, libds=YOURLIB.DATASET ) %mp_stprequests(status_cd=INIT, libds=YOURLIB.DATASET )
@param status_cd= Use INIT for INIT and TERM for TERM events @param [in] status_cd= Use INIT for INIT and TERM for TERM events
@param libds= Location of base table (library.dataset). To minimise risk @param [in] libds= (somelib.stp_requests) Location of base table
of table locks, we HIGHLY recommend using a database (NOT a SAS dataset). (library.dataset). To minimise risk of table locks, we HIGHLY recommend
using a database (NOT a SAS dataset).
THE LIBRARY SHOULD BE ASSIGNED ALREADY - eg in autoexec or earlier in the THE LIBRARY SHOULD BE ASSIGNED ALREADY - eg in autoexec or earlier in the
init program proper. init program proper.

252
base/mp_stripdiffs.sas Normal file
View File

@@ -0,0 +1,252 @@
/**
@file
@brief Generates a stage dataset to revert diffs tracked in an audit table
@details A big benefit of tracking data changes in an audit table is that
those changes can be subsequently reverted if necessary!
This macro prepares a staging dataset containing those differences - eg for:
@li deleted rows - these are re-inserted
@li changed rows - differences are reverted
@li added rows - marked with `_____DELETE__THIS__RECORD_____="YES"`
These changes are NOT applied to the base table - a staging dataset is
simply prepared for an ETL process to action. In Data Controller, this
dataset is used directly as an input to the APPROVE process (so that the
reversion diffs can be reviewed prior to being applied).
@param [in] libds Base library.dataset (will not be modified). The library
must be assigned.
@param [in] loadref Unique identifier for the version to be reverted. This
change, plus ALL SUBSEQUENT CHANGES, will be reverted in the output table.
@param [in] difftable The dataset containing the diffs. Definition available
in mddl_dc_difftable.sas
@param [out] outds= (work.mp_stripdiffs) Output table containing the diffs.
Has the same format as the base datset, plus a
`_____DELETE__THIS__RECORD_____` variable.
@param [in] mdebug= set to 1 to enable DEBUG messages and preserve outputs
<h4> SAS Macros </h4>
@li mf_getuniquefileref.sas
@li mf_getuniquename.sas
@li mf_getvarlist.sas
@li mf_islibds.sas
@li mf_wordsinstr1butnotstr2.sas
@li mp_abort.sas
<h4> Related Macros </h4>
@li mddl_dc_difftable.sas
@li mp_stackdiffs.sas
@li mp_storediffs.sas
@li mp_stripdiffs.test.sas
@version 9.2
@author Allan Bowe
**/
/** @cond */
%macro mp_stripdiffs(libds
,loadref
,difftable
,outds=work.mp_stripdiffs
,mdebug=0
)/*/STORE SOURCE*/;
%local dbg;
%if &mdebug=1 %then %do;
%put &sysmacroname entry vars:;
%put _local_;
%end;
%else %let dbg=*;
%let libds=%upcase(&libds);
/* safety checks */
%mp_abort(iftrue= (&syscc ne 0)
,mac=&sysmacroname
,msg=%str(SYSCC=&syscc on entry. Clean session required!)
)
%let libds=%upcase(&libds);
%mp_abort(iftrue= (%mf_islibds(&libds)=0)
,mac=&sysmacroname
,msg=%str(Invalid library.dataset reference - %superq(libds))
)
/* set up unique and temporary vars */
%local ds1 ds2 ds3 ds4 ds5 fref1;
%let fref1=%mf_getuniquefileref();
/* get timestamp of the diff to be reverted */
%local ts;
proc sql noprint;
select put(processed_dttm,datetime19.6) into: ts
from &difftable where load_ref="&loadref";
%mp_abort(iftrue= (&sqlobs=0)
,mac=&sysmacroname
,msg=%str(Load ref %superq(loadref) not found!)
)
/* extract diffs for this base table from this timestamp onwards */
%let ds1=%upcase(work.%mf_getuniquename(prefix=mpsd_diffs));
create table &ds1 (drop=libref dsn) as
select * from &difftable
where upcase(cats(libref))="%scan(&libds,1,.)"
and upcase(cats(dsn))="%scan(&libds,2,.)"
and processed_dttm ge "&ts"dt
order by processed_dttm desc, key_hash, is_pk;
/* extract key values only */
%let ds2=%upcase(work.%mf_getuniquename(prefix=mpsd_pks));
%local keyhash processed;
%let keyhash=%upcase(%mf_getuniquename(prefix=mpsdvar_keyhash));
%let processed=%upcase(%mf_getuniquename(prefix=mpsdvar_processed));
create table &ds2 as
select key_hash as &keyhash,
tgtvar_nm,
tgtvar_type,
coalescec(oldval_char,newval_char) as charval,
coalesce(oldval_num, newval_num) as numval,
processed_dttm as &processed
from &ds1
where is_pk=1
order by &keyhash, &processed;
/* grab pk values */
%local pk;
select distinct upcase(tgtvar_nm) into: pk separated by ' ' from &ds2;
%let ds3=%upcase(work.%mf_getuniquename(prefix=mpsd_keychar));
proc transpose data=&ds2(where=(tgtvar_type='C'))
out=&ds3(drop=_name_);
by &keyhash &processed;
id TGTVAR_NM;
var charval;
run;
%let ds4=%upcase(work.%mf_getuniquename(prefix=mpsd_keynum));
proc transpose data=&ds2(where=(tgtvar_type='N'))
out=&ds4(drop=_name_);
by &keyhash &processed;
id TGTVAR_NM;
var numval;
run;
/* shorten the lengths */
%mp_ds2squeeze(&ds3,outds=&ds3)
%mp_ds2squeeze(&ds4,outds=&ds4)
/* now merge to get all key values and de-dup */
%let ds5=%upcase(work.%mf_getuniquename(prefix=mpsd_merged));
data &ds5;
length &keyhash $32 &processed 8;
merge &ds3 &ds4;
by &keyhash &processed;
if not missing(&keyhash);
run;
proc sort data=&ds5 nodupkey;
by &pk;
run;
/* join to base table for preliminary stage DS */
proc sql;
create table &outds as select "No " as _____DELETE__THIS__RECORD_____
%do x=1 %to %sysfunc(countw(&pk,%str( )));
,a.%scan(&pk,&x,%str( ))
%end;
%local notpkcols;
%let notpkcols=%upcase(%mf_getvarlist(&libds));
%let notpkcols=%mf_wordsinstr1butnotstr2(str1=&notpkcols,str2=&pk);
%do x=1 %to %sysfunc(countw(&notpkcols,%str( )));
,b.%scan(&notpkcols,&x,%str( ))
%end;
from &ds5 a
left join &libds b
on 1=1
%do x=1 %to %sysfunc(countw(&pk,%str( )));
and a.%scan(&pk,&x,%str( ))=b.%scan(&pk,&x,%str( ))
%end;
;
/* create SAS code to apply to stage_ds */
data _null_;
set &ds1;
file &fref1 lrecl=33000;
length charval $32767;
if _n_=1 then put 'proc sql noprint;';
by descending processed_dttm key_hash is_pk;
if move_type='M' then do;
if first.key_hash then do;
put "update &outds set " @@;
end;
if IS_PK=0 then do;
put " " tgtvar_nm '=' @@;
cnt=count(oldval_char,'"');
charval=quote(trim(substr(oldval_char,1,32765-cnt)));
if tgtvar_type='C' then put charval @@;
else put oldval_num @@;
if not last.is_pk then put ',';
end;
else do;
if first.is_pk then put " where 1=1 " @@;
put " and " tgtvar_nm '=' @@;
cnt=count(oldval_char,'"');
charval=quote(trim(substr(oldval_char,1,32765-cnt)));
if tgtvar_type='C' then put charval @@;
else put oldval_num @@;
end;
end;
else if move_type='A' then do;
if first.key_hash then do;
put "update &outds set _____DELETE__THIS__RECORD_____='Yes' where 1=1 "@@;
end;
/* gating if - as only need PK now */
if is_pk=1;
put ' AND ' tgtvar_nm '=' @@;
cnt=count(newval_char,'"');
charval=quote(trim(substr(newval_char,1,32765-cnt)));
if tgtvar_type='C' then put charval @@;
else put newval_num @@;
end;
else if move_type='D' then do;
if first.key_hash then do;
put "update &outds set _____DELETE__THIS__RECORD_____='No' " @@;
end;
if IS_PK=0 then do;
put " ," tgtvar_nm '=' @@;
cnt=count(oldval_char,'"');
charval=quote(trim(substr(oldval_char,1,32765-cnt)));
if tgtvar_type='C' then put charval @@;
else put oldval_num @@;
end;
else do;
if first.is_pk then put " where 1=1 " @@;
put " and " tgtvar_nm '=' @@;
cnt=count(oldval_char,'"');
charval=quote(trim(substr(oldval_char,1,32765-cnt)));
if tgtvar_type='C' then put charval @@;
else put oldval_num @@;
end;
end;
if last.key_hash then put ';';
run;
/* apply the modification statements */
%inc &fref1/source2 lrecl=33000;
%if &mdebug=0 %then %do;
proc sql;
drop table &ds1, &ds2, &ds3, &ds4, &ds5;
file &fref1 clear;
%end;
%else %do;
data _null_;
infile &fref1;
input;
if _n_=1 then putlog "Contents of SQL adjustments";
putlog _infile_;
run;
%end;
%mend mp_stripdiffs;
/** @endcond */

View File

@@ -8,8 +8,8 @@
duration=60*5 duration=60*5
) )
@param [in] duration= the time in seconds which the job should run for. Actual @param [in] duration= (30) The time in seconds which the job should run for.
time may vary, as the check is done in between steps. Default = 30 (seconds). Actual time may vary, as the check is done in between steps.
<h4> SAS Macros </h4> <h4> SAS Macros </h4>
@li mf_getuniquelibref.sas @li mf_getuniquelibref.sas

View File

@@ -11,9 +11,9 @@
,outds=work.results ,outds=work.results
) )
@param lib= (WORK) The library in which to create the table @param [in] lib= (WORK) The library in which to create the table
@param size= (0.1) The size in GB of the table to create @param [in] size= (0.1) The size in GB of the table to create
@param outds= (WORK.RESULTS) The output dataset to be created. @param [out] outds= (WORK.RESULTS) The output dataset to be created.
<h4> SAS Macros </h4> <h4> SAS Macros </h4>
@li mf_getuniquename.sas @li mf_getuniquename.sas

View File

@@ -15,8 +15,8 @@ https://communities.sas.com/t5/SAS-Programming/listing-all-files-within-a-direct
https://communities.sas.com/t5/SAS-Programming/listing-all-files-of-all-types-from-all-subdirectories/m-p/334113/highlight/true#M75419 https://communities.sas.com/t5/SAS-Programming/listing-all-files-of-all-types-from-all-subdirectories/m-p/334113/highlight/true#M75419
@param dir= Directory to be scanned (default=/tmp) @param [in] dir= (/tmp) Directory to be scanned
@param outds= Dataset to create (default=work.mp_tree) @param [out] outds= (work.mp_tree) Dataset to create
@returns outds contains the following variables: @returns outds contains the following variables:

View File

@@ -13,11 +13,12 @@
%mp_unzip(ziploc="/some/file.zip",outdir=/some/folder) %mp_unzip(ziploc="/some/file.zip",outdir=/some/folder)
More info: https://blogs.sas.com/content/sasdummy/2015/05/11/using-filename-zip-to-unzip-and-read-data-files-in-sas/ More info:
https://blogs.sas.com/content/sasdummy/2015/05/11/using-filename-zip-to-unzip-and-read-data-files-in-sas/
@param ziploc= Fileref or quoted full path to zip file ("/path/to/file.zip") @param [in] ziploc= Fileref or quoted full path, eg: "/path/to/file.zip"
@param outdir= (%sysfunc(pathname(work))) Directory in which to write the @param [out] outdir= (%sysfunc(pathname(work))) Directory in which to write
outputs (created if non existant) the outputs (created if needed)
<h4> SAS Macros </h4> <h4> SAS Macros </h4>
@li mf_mkdir.sas @li mf_mkdir.sas

View File

@@ -14,9 +14,9 @@
proc sql; proc sql;
describe table example; describe table example;
@param libds the library.dataset to be modified @param [in] libds the library.dataset to be modified
@param var The variable to modify @param [in] var The variable to modify
@param len The new length to apply @param [in] len The new length to apply
<h4> SAS Macros </h4> <h4> SAS Macros </h4>
@li mf_existds.sas @li mf_existds.sas

View File

@@ -57,7 +57,7 @@
%end; %end;
%else %if &rule=ISNUM %then %do; %else %if &rule=ISNUM %then %do;
/* /*
credit SØREN LASSEN credit SOREN LASSEN
https://sasmacro.blogspot.com/2009/06/welcome-isnum-macro.html https://sasmacro.blogspot.com/2009/06/welcome-isnum-macro.html
*/ */
&tempcol=input(&incol,?? best32.); &tempcol=input(&incol,?? best32.);

View File

@@ -15,14 +15,14 @@
<h4> SAS Macros </h4> <h4> SAS Macros </h4>
@li mp_dirlist.sas @li mp_dirlist.sas
@param in= unquoted filepath, dataset of files or directory to zip @param [in] in= unquoted filepath, dataset of files or directory to zip
@param type= (FILE) Valid values: @param [in] type= (FILE) Valid values:
@li FILE - /full/path/and/filename.extension to a particular file @li FILE - /full/path/and/filename.extension to a particular file
@li DATASET - a dataset containing a list of files to zip (see `incol`) @li DATASET - a dataset containing a list of files to zip (see `incol`)
@li DIRECTORY - a directory to zip @li DIRECTORY - a directory to zip
@param outname= (FILE) Output file to create, _without_ .zip extension @param [out] outname= (FILE) Output file to create, _without_ .zip extension
@param outpath= (%sysfunc(pathname(WORK))) Parent folder for output zip file @param [out] outpath= (%sysfunc(pathname(WORK))) Parent folder for zip file
@param incol= if DATASET input, say which column contains the filepath @param [in] incol= () If DATASET input, say which column contains the filepath
<h4> Related Macros </h4> <h4> Related Macros </h4>
@li mp_unzip.sas @li mp_unzip.sas

View File

@@ -6,6 +6,22 @@
(given various practical restrictions) are described here to enable (given various practical restrictions) are described here to enable
consistency when dealing with format data. consistency when dealing with format data.
The HLO variable may have a number of values, documented here due to the
256 char label description length limit:
F=Standard format/informat.
H=Range ending value is HIGH.
I=Numeric informat.
J=Justification for an informat.
L=Range starting value is LOW.
M=MultiLabel.
N=Format or informat has no ranges, including no OTHER= range.
O=Range is OTHER.
R=ROUND option is in effect.
S=Specifies that NOTSORTED is in effect.
U=Specifies that the UPCASE option for an informat be used.
**/ **/
@@ -13,9 +29,11 @@
proc sql; proc sql;
create table &libds( create table &libds(
TYPE char(1) label='Type of format - either N (num fmt), C (char fmt), I (num infmt) or J (char infmt)' TYPE char(1) label=
'Format Type: either N (num fmt), C (char fmt), I (num infmt) or J (char infmt)'
,FMTNAME char(32) label='Format name' ,FMTNAME char(32) label='Format name'
,FMTROW num label='CALCULATED Position of record by FMTNAME (reqd for multilabel formats)' ,FMTROW num label=
'CALCULATED Position of record by FMTNAME (reqd for multilabel formats)'
,START char(32767) label='Starting value for format' ,START char(32767) label='Starting value for format'
/* /*
Keep lengths of START and END the same to avoid this err: Keep lengths of START and END the same to avoid this err:
@@ -35,7 +53,8 @@
,NOEDIT num length=3 label='Is picture string noedit?' ,NOEDIT num length=3 label='Is picture string noedit?'
,SEXCL char(1) label='Start exclusion' ,SEXCL char(1) label='Start exclusion'
,EEXCL char(1) label='End exclusion' ,EEXCL char(1) label='End exclusion'
,HLO char(13) label='Additional information. M=MultiLabel' ,HLO char(13) label=
'More info: https://core.sasjs.io/mddl__sas__cntlout_8sas_source.html'
,DECSEP char(1) label='Decimal separator' ,DECSEP char(1) label='Decimal separator'
,DIG3SEP char(1) label='Three-digit separator' ,DIG3SEP char(1) label='Three-digit separator'
,DATATYPE char(8) label='Date/time/datetime?' ,DATATYPE char(8) label='Date/time/datetime?'

View File

@@ -13,9 +13,9 @@
,group=someGroup) ,group=someGroup)
@param user= the user name (not displayname) @param [in] user= the user name (not displayname)
@param group= the group to which to add the user @param [in] group= the group to which to add the user
@param mdebug= (0) set to 1 to show debug info in log @param [in] mdebug= (0) set to 1 to show debug info in log
<h4> Related Files </h4> <h4> Related Files </h4>
@li ms_adduser2group.sas @li ms_adduser2group.sas

View File

@@ -18,12 +18,13 @@
@li mf_getengine.sas @li mf_getengine.sas
@li mp_abort.sas @li mp_abort.sas
@param libref the libref (not name) of the metadata library @param [in] libref the libref (not name) of the metadata library
@param open_passthrough= provide an alias to produce the CONNECT TO statement @param [in] open_passthrough= () Provide an alias to produce the CONNECT TO
for the relevant external database statement for the relevant external database
@param sql_options= an override default output fileref to avoid naming clash @param [in] sql_options= () Add any options to add to proc sql statement,
@param mDebug= set to 1 to show debug messages in the log eg outobs= (only valid for pass through)
@param mAbort= set to 1 to call %mp_abort(). @param [in] mDebug= (0) set to 1 to show debug messages in the log
@param [in] mAbort= (0) set to 1 to call %mp_abort().
@returns libname statement @returns libname statement
@@ -33,12 +34,9 @@
**/ **/
%macro mm_assigndirectlib( %macro mm_assigndirectlib(
libref /* libref to assign from metadata */ libref
,open_passthrough= /* provide an alias to produce the ,open_passthrough=
CONNECT TO statement for the ,sql_options=
relevant external database */
,sql_options= /* add any options to add to proc sql statement eg outobs=
(only valid for pass through) */
,mDebug=0 ,mDebug=0
,mAbort=0 ,mAbort=0
)/*/STORE SOURCE*/; )/*/STORE SOURCE*/;
@@ -212,9 +210,9 @@ run;
%end; %end;
%end; %end;
%else %if &engine=ODBC %then %do; %else %if &engine=ODBC %then %do;
&mD.%put NOTE: Retrieving ODBC connection details; %&mD.put NOTE: Retrieving ODBC connection details;
data _null_; data _null_;
length connx_uri conprop_uri value datasource up_uri schema $256.; length connx_uri conprop_uri value datasource up_uri schema domprop_uri authdomain $256.;
call missing (of _all_); call missing (of _all_);
/* get source connection ID */ /* get source connection ID */
rc=metadata_getnasn("&liburi",'LibraryConnection',1,connx_uri); rc=metadata_getnasn("&liburi",'LibraryConnection',1,connx_uri);
@@ -229,6 +227,13 @@ run;
rc2=-1; rc2=-1;
end; end;
end; end;
/* get auth domain */
autrc=metadata_getnasn(connx_uri,"Domain",1,domprop_uri);
arc=metadata_getattr(domprop_uri,"Name",authdomain);
if not missing(authdomain) then authdomain=cats('AUTHDOMAIN=',authdomain);
call symputx('authdomain',authdomain,'l');
/* get SCHEMA */ /* get SCHEMA */
rc6=metadata_getnasn("&liburi",'UsingPackages',1,up_uri); rc6=metadata_getnasn("&liburi",'UsingPackages',1,up_uri);
rc7=metadata_getattr(up_uri,'SchemaName',schema); rc7=metadata_getattr(up_uri,'SchemaName',schema);
@@ -245,7 +250,7 @@ run;
(INSERT_SQL=YES DATASRC=&sql_dsn. CONNECTION=global); (INSERT_SQL=YES DATASRC=&sql_dsn. CONNECTION=global);
%end; %end;
%else %do; %else %do;
libname &libref ODBC DATASRC=&sql_dsn SCHEMA=&sql_schema; libname &libref ODBC DATASRC=&sql_dsn SCHEMA=&sql_schema &authdomain;
%end; %end;
%end; %end;
%else %if &engine=POSTGRES %then %do; %else %if &engine=POSTGRES %then %do;

View File

@@ -14,8 +14,8 @@
@li mp_abort.sas @li mp_abort.sas
@param [in] libref The libref (not name) of the metadata library @param [in] libref The libref (not name) of the metadata library
@param [in] mAbort= If not assigned, HARD will call %mp_abort(), SOFT will @param [in] mAbort= (HARD) If not assigned, HARD will call %mp_abort(), SOFT
silently return will silently return
@returns libname statement @returns libname statement

View File

@@ -21,21 +21,24 @@
@li mp_abort.sas @li mp_abort.sas
@li mf_verifymacvars.sas @li mf_verifymacvars.sas
@param tree= The metadata folder uri, or the metadata path, in which to @param [in] tree= The metadata folder uri, or the metadata path, in which to
create the object. This must exist. create the object. This must exist.
@param name= Application object name. Avoid spaces. @param [in] name= Application object name. Avoid spaces.
@param ClassIdentifier= the class of applications to which this app belongs @param [in] ClassIdentifier=
@param params= name=value pairs which will become public properties of the The class of applications to which this app belongs
@param [in] params=
name=value pairs which will become public properties of the
application object. These are delimited using &#x0a; (newline character) application object. These are delimited using &#x0a; (newline character)
@param desc= Application description (optional). Avoid ampersands as these @param [in] desc= Application description (optional).
Avoid ampersands as these
are illegal characters (unless they are escapted- eg &amp;) are illegal characters (unless they are escapted- eg &amp;)
@param version= version number of application @param [in] version= version number of application
@param frefin= fileref to use (enables change if there is a conflict). The @param [in] frefin= fileref to use (enables change if there is a conflict).
filerefs are left open, to enable inspection after running the The filerefs are left open, to enable inspection after running the
macro (or importing into an xmlmap if needed). macro (or importing into an xmlmap if needed).
@param frefout= fileref to use (enables change if there is a conflict) @param [out] frefout= fileref to use (enables change if there is a conflict)
@param mDebug= set to 1 to show debug messages in the log @param [in] mDebug= set to 1 to show debug messages in the log
@author Allan Bowe @author Allan Bowe

View File

@@ -22,12 +22,12 @@
@li mm_gettables.sas @li mm_gettables.sas
@li mm_getcols.sas @li mm_getcols.sas
@param libds= library.dataset metadata source. Note - table names in metadata @param [in] libds= library.dataset metadata source. Note - table names in metadata
can be longer than 32 chars (just fyi, not an issue here) can be longer than 32 chars (just fyi, not an issue here)
@param tableuri= Metadata URI of the table to be created @param [in] tableuri= Metadata URI of the table to be created
@param outds= (work.mm_createdataset) The dataset to create. The table name @param [out] outds= (work.mm_createdataset) The dataset to create. The table
needs to be 32 chars or less as per SAS naming rules. name needs to be 32 chars or less as per SAS naming rules.
@param mdebug= (0) Set to 1 to enable DEBUG messages @param [in] mdebug= (0) Set to 1 to enable DEBUG messages
@version 9.4 @version 9.4
@author Allan Bowe @author Allan Bowe

View File

@@ -17,15 +17,15 @@
@li mf_verifymacvars.sas @li mf_verifymacvars.sas
@param tree= The metadata folder uri, or the metadata path, in which to @param [in] tree= The metadata folder uri, or the metadata path, in which to
create the document. This must exist. create the document. This must exist.
@param name= Document object name. Avoid spaces. @param [in] name= Document object name. Avoid spaces.
@param desc= Document description (optional) @param [in] desc= Document description (optional)
@param textrole= TextRole property (optional) @param [in] textrole= TextRole property (optional)
@param frefin= fileref to use (enables change if there is a conflict) @param [in] frefin= fileref to use (enables change if there is a conflict)
@param frefout= fileref to use (enables change if there is a conflict) @param [out] frefout= fileref to use (enables change if there is a conflict)
@param mDebug= set to 1 to show debug messages in the log @param [in] mDebug= set to 1 to show debug messages in the log
@author Allan Bowe @author Allan Bowe

View File

@@ -24,30 +24,33 @@
@li mm_createfolder.sas @li mm_createfolder.sas
@param libname= Library name (as displayed to user, 256 chars). Duplicates @param [in] libname= (My New Library) Library name (as displayed to user,
are not created (case sensitive). 256 chars). Duplicates are not created (case sensitive).
@param libref= Library libref (8 chars). Duplicate librefs are not created, @param [in] libref= (mynewlib) Library libref (8 chars). Duplicate
HOWEVER- the check is not case sensitive - if *libref* exists, *LIBREF* librefs are not created, HOWEVER- the check is not case sensitive - if
will still be created. Librefs created will always be uppercased. *libref* exists, *LIBREF* will still be created.
@param engine= Library engine (currently only BASE supported) Librefs created will always be uppercased.
@param tree= The metadata folder uri, or the metadata path, in which to @param [in] engine= Library engine (currently only BASE supported)
@param [in] tree= The metadata folder uri, or the metadata path, in which to
create the library. create the library.
@param servercontext= The SAS server against which the library is registered. @param [in] servercontext= (SASApp) The SAS server against which
@param IsPreassigned= set to 1 if the library should be pre-assigned. the library is registered.
@param [in] IsPreassigned= set to 1 if the library should be pre-assigned.
@param libdesc= Library description (optional) @param [in] libdesc= Library description (optional)
@param directory= Required for the BASE engine. The metadata directory objects @param [in] directory= (/tmp/somelib) Required for the BASE engine.
are searched to find an existing one with a matching physical path. The metadata directory objects are searched to find an existing
one with a matching physical path.
If more than one uri found with that path, then the first one will be used. If more than one uri found with that path, then the first one will be used.
If no URI is found, a new directory object will be created. The physical If no URI is found, a new directory object will be created. The physical
path will also be created, if it doesn't exist. path will also be created, if it doesn't exist.
@param mDebug= set to 1 to show debug messages in the log @param [in] mDebug= set to 1 to show debug messages in the log
@param frefin= fileref to use (enables change if there is a conflict). The @param [in] frefin= fileref to use (enables change if there is a conflict).
filerefs are left open, to enable inspection after running the The filerefs are left open, to enable inspection after running the
macro (or importing into an xmlmap if needed). macro (or importing into an xmlmap if needed).
@param frefout= fileref to use (enables change if there is a conflict) @param [out] frefout= fileref to use (enables change if there is a conflict)
@version 9.3 @version 9.3

View File

@@ -40,29 +40,38 @@
,Server=SASApp ,Server=SASApp
,stptype=2) ,stptype=2)
@param stpname= Stored Process name. Avoid spaces - testing has shown that @param [in] stpname= (SASjs Default STP) Stored Process name.
Avoid spaces - testing has shown that
the check to avoid creating multiple STPs in the same folder with the same the check to avoid creating multiple STPs in the same folder with the same
name does not work when the name contains spaces. name does not work when the name contains spaces.
@param stpdesc= Stored Process description (optional) @param [in] stpdesc= Stored Process description (optional)
@param filename= the name of the .sas program to run @param [in] filename= the name of the .sas program to run
@param directory= The directory uri, or the actual path to the sas program @param [in] directory= (SASEnvironment/sascode)
(no trailing slash). If more than uri is found with that path, then the The directory uri or the actual path to the sas program (no trailing slash).
first one will be used. If more than uri is found with that path, then the first one will be used.
@param tree= The metadata folder uri, or the metadata path, in which to @param [in] tree= The metadata folder uri, or the metadata path, in which to
create the STP. create the STP.
@param server= The server which will run the STP. Server name or uri is fine. @param [in] server= (SASApp) The server which will run the STP.
@param outds= The two level name of the output dataset. Will contain all the Server name or uri is fine.
meta uris. Defaults to work.mm_createstp. @param [out] outds= (work.mm_createstp)
@param mDebug= set to 1 to show debug messages in the log The two level name of the output dataset. Will contain all the meta uris.
@param stptype= Default is 1 (STP code saved on filesystem). Set to 2 if @param [in] mDebug= set to 1 to show debug messages in the log
@param [in] stptype= Default is 1 (STP code saved on filesystem). Set to 2 if
source code is to be saved in metadata (9.3 and above feature). source code is to be saved in metadata (9.3 and above feature).
@param minify= set to YES to strip comments / blank lines etc @param [in] minify= set to YES to strip comments / blank lines etc
@param frefin= fileref to use (enables change if there is a conflict). The @param [in] frefin= (mm_in) fileref to use (enables change if there is
filerefs are left open, to enable inspection after running the a conflict).
The filerefs are left open, to enable inspection after running the
macro (or importing into an xmlmap if needed). macro (or importing into an xmlmap if needed).
@param frefout= fileref to use (enables change if there is a conflict) @param [out] frefout= (mm_out) fileref to use (enables change if there is
@param repo= ServerContext is tied to a repo, if you are not using the a conflict)
@param [in] repo= ServerContext is tied to a repo, if you are not using the
foundation repo then select a different one here foundation repo then select a different one here
@param [in] LogicalServerType= (Sps) Server Type to use. Valid options:
@li Any - Uses the default server.
@li Sps - Stored Process Server, best choice for web app development. Runs
under a system account identity (eg sassrv).
@li Wks - Workspace Server. Runs under the end user identity.
@returns outds dataset containing the following columns: @returns outds dataset containing the following columns:
- stpuri - stpuri
@@ -88,7 +97,7 @@
**/ **/
%macro mm_createstp( %macro mm_createstp(
stpname=Macro People STP stpname=SASjs Default STP
,stpdesc=This stp was created automatically by the mm_createstp macro ,stpdesc=This stp was created automatically by the mm_createstp macro
,filename=mm_createstp.sas ,filename=mm_createstp.sas
,directory=SASEnvironment/SASCode ,directory=SASEnvironment/SASCode
@@ -102,6 +111,7 @@
,minify=NO ,minify=NO
,frefin=mm_in ,frefin=mm_in
,frefout=mm_out ,frefout=mm_out
,LogicalServerType=Sps
)/*/STORE SOURCE*/; )/*/STORE SOURCE*/;
%local mD; %local mD;
@@ -118,6 +128,17 @@
%mp_dropmembers(%scan(&outds,2,.)) %mp_dropmembers(%scan(&outds,2,.))
/* check LogicalServerType validity */
%mp_abort(
iftrue=(
&LogicalServerType ne Sps
and &LogicalServerType ne Wks
and &LogicalServerType ne Any
)
,mac=&sysmacroname
,msg=%str(Invalid value for LogicalServerType (&LogicalServerType))
)
/** /**
* check tree exists * check tree exists
*/ */
@@ -350,8 +371,9 @@ run;
' <TextStore IsHidden="0" Name="Stored Process" UsageVersion="0" '/ ' <TextStore IsHidden="0" Name="Stored Process" UsageVersion="0" '/
' TextRole="StoredProcessConfiguration" TextType="XML" '/ ' TextRole="StoredProcessConfiguration" TextType="XML" '/
' StoredText="&lt;?xml version=&quot;1.0&quot; encoding=&quot;UTF-8&qu'@@ ' StoredText="&lt;?xml version=&quot;1.0&quot; encoding=&quot;UTF-8&qu'@@
'ot;?&gt;&lt;StoredProcess&gt;&lt;ServerContext LogicalServerType=&quot;S'@@ 'ot;?&gt;&lt;StoredProcess&gt;&lt;ServerContext LogicalServerType=&quot;'@@
'ps&quot; OtherAllowed=&quot;false&quot;/&gt;&lt;ResultCapabilities Packa'@@ "&LogicalServerType"@@
'&quot; OtherAllowed=&quot;false&quot;/&gt;&lt;ResultCapabilities Packa'@@
'ge=&quot;' @@ "&package" @@ '&quot; Streaming=&quot;' @@ "&streaming" @@ 'ge=&quot;' @@ "&package" @@ '&quot; Streaming=&quot;' @@ "&streaming" @@
'&quot;/&gt;&lt;OutputParameters/&gt;&lt;/StoredProcess&gt;" />' / '&quot;/&gt;&lt;OutputParameters/&gt;&lt;/StoredProcess&gt;" />' /
" </Notes> "/ " </Notes> "/

Some files were not shown because too many files have changed in this diff Show More