mirror of
https://github.com/sasjs/core.git
synced 2025-12-12 23:14:35 +00:00
Compare commits
223 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
86f7876f50 | ||
|
|
46c96bc7ec | ||
|
|
cba3f5972b | ||
|
|
ed48c49964 | ||
|
|
203ff3f80d | ||
|
|
cfe90a8d0d | ||
|
|
0749ea0819 | ||
|
|
e09a39e748 | ||
|
|
20dcefaefd | ||
|
|
4c8347516a | ||
|
|
e497d226a0 | ||
|
|
ccf8f1acc0 | ||
|
|
fe9a2ed979 | ||
|
|
078815e83e | ||
|
|
bb80c7af5a | ||
|
|
842662aae1 | ||
|
|
876fac2332 | ||
|
|
427deca350 | ||
|
|
07bde4b25c | ||
|
|
80b06af581 | ||
|
|
3c026811e9 | ||
|
|
cf547ce7e4 | ||
|
|
6952c79899 | ||
|
|
09e3f63da7 | ||
|
|
d6956f4122 | ||
|
|
6fca73e7da | ||
|
|
880df4138c | ||
|
|
badf5b5761 | ||
|
|
b174aa25b3 | ||
|
|
bc6eac6977 | ||
|
|
2d4d595e5d | ||
|
|
7111fe14fb | ||
|
|
8499e38c55 | ||
|
|
682d80b1b8 | ||
|
|
4fe6f233f2 | ||
|
|
6ba3588eff | ||
|
|
53aa403630 | ||
|
|
cba9255732 | ||
|
|
a7b78c73c4 | ||
|
|
85e0b6a4a9 | ||
|
|
3c7e762eeb | ||
|
|
9a1f7d0985 | ||
|
|
dfd60200fb | ||
|
|
713f7544cd | ||
|
|
de4e96ab01 | ||
|
|
3e7b15c7db | ||
|
|
eb2ccfbbca | ||
|
|
70e508e583 | ||
|
|
8b0acf2eae | ||
|
|
d254870439 | ||
|
|
303225cb85 | ||
|
|
90de167643 | ||
|
|
8ee997de8e | ||
|
|
e27f6ac716 | ||
|
|
ec4de95fcf | ||
|
|
df0fa95519 | ||
|
|
2fe7fba79b | ||
|
|
e40234ee29 | ||
|
|
a287cc27a7 | ||
|
|
921186eb74 | ||
|
|
6fd215ceff | ||
|
|
0297509aa0 | ||
|
|
c5a8bc745d | ||
|
|
36aa466561 | ||
|
|
009485e5b9 | ||
|
|
eb01c8772d | ||
|
|
e3fb69928c | ||
|
|
65afa14466 | ||
|
|
0176b19616 | ||
|
|
9f3dfd9a59 | ||
|
|
513ea354ab | ||
|
|
7b686e11c9 | ||
|
|
3997000266 | ||
|
|
6e177d4cae | ||
|
|
3554991ff8 | ||
|
|
58d2d6382a | ||
|
|
67f28a366c | ||
|
|
64f53acce2 | ||
|
|
2e790f69a1 | ||
|
|
e62011d97e | ||
|
|
cd8d16d09f | ||
|
|
9c61965d4b | ||
|
|
61b8cb5dea | ||
|
|
899f6d9558 | ||
|
|
899de27617 | ||
|
|
322c488e72 | ||
|
|
5d5e66a1c5 | ||
|
|
5f4e9d541d | ||
|
|
306ea93be2 | ||
|
|
3fd83a3160 | ||
|
|
56c1397547 | ||
|
|
90adf8dcdd | ||
|
|
6e0fe0ff25 | ||
|
|
794ceec33c | ||
|
|
11d073c10a | ||
|
|
c160b5058b | ||
|
|
2f49738cf9 | ||
|
|
bfe4b1ec8b | ||
|
|
6224844915 | ||
|
|
81a17bc0c2 | ||
|
|
f4c2be7411 | ||
|
|
16489a9494 | ||
|
|
0e03b06a4b | ||
|
|
c3b89c7f7d | ||
|
|
142b46570d | ||
|
|
f7fac50108 | ||
|
|
f7078957cf | ||
|
|
f258d4f2f1 | ||
|
|
ae5fbcf857 | ||
|
|
2579b4c929 | ||
|
|
b69c3b7a78 | ||
|
|
67df4dffeb | ||
|
|
9cf2cc3c96 | ||
|
|
dd94215c3b | ||
|
|
1fd1a8e7ce | ||
|
|
90a831f59b | ||
|
|
9fb218f0be | ||
|
|
bdd22abc55 | ||
|
|
75f712a305 | ||
|
|
e3991c46e2 | ||
|
|
ccc9dfa4aa | ||
|
|
a37a72b7db | ||
|
|
724d3b91a0 | ||
|
|
887c797e13 | ||
|
|
0fd1e470e8 | ||
|
|
13ecab8390 | ||
|
|
15d9db822b | ||
|
|
dd355d1ddf | ||
|
|
c6dcf919e2 | ||
|
|
42541373af | ||
|
|
208c88f5a4 | ||
|
|
5605bc74df | ||
|
|
4bec574011 | ||
|
|
8cfa37ce8b | ||
|
|
351ceeb357 | ||
|
|
259bcc0173 | ||
|
|
db195a8311 | ||
|
|
4307bfb1b5 | ||
|
|
df46ee6939 | ||
|
|
70b9b71104 | ||
|
|
cd33355418 | ||
|
|
77d1cdb753 | ||
|
|
545218e3b9 | ||
|
|
cb07305a87 | ||
|
|
76a39cad20 | ||
|
|
ebd567af48 | ||
|
|
a9c418e3f2 | ||
|
|
e143acd67d | ||
|
|
84eb2f1845 | ||
|
|
b075e5d5d5 | ||
|
|
a08f6aeea2 | ||
|
|
469bd574ac | ||
|
|
c41918c0a8 | ||
|
|
0361ca574d | ||
|
|
c75c169b80 | ||
|
|
eac47bd5db | ||
|
|
d302ef266d | ||
|
|
fdfe9b8250 | ||
|
|
9b1f0d7bcb | ||
|
|
98b1c44283 | ||
|
|
ce026f19b5 | ||
|
|
8e723d06b0 | ||
|
|
a6d84cc65a | ||
|
|
536ce8e95d | ||
|
|
bc1d9e619b | ||
|
|
1062a97cfe | ||
|
|
51db64c90a | ||
|
|
7c4278c3f9 | ||
|
|
6c6b55dcea | ||
|
|
66b0c9e77e | ||
|
|
caf3b95269 | ||
|
|
3866b97416 | ||
|
|
d687658687 | ||
|
|
9f815c73e9 | ||
|
|
a13c782074 | ||
|
|
f2991cfd63 | ||
|
|
8eb4f0844c | ||
|
|
f90dc069dc | ||
|
|
436b430389 | ||
|
|
6667b91ced | ||
|
|
bce56d8105 | ||
|
|
2ec440b321 | ||
|
|
3d2ad531cf | ||
|
|
09136cfdbb | ||
|
|
0ca16f3d04 | ||
|
|
1e72f13f2d | ||
|
|
5e8e8e02d3 | ||
|
|
b2e2c7c798 | ||
|
|
b29dd38188 | ||
|
|
2e122c2ada | ||
|
|
8b68c3bb27 | ||
|
|
8c7523deda | ||
|
|
e8f656f48a | ||
|
|
1eb90202b9 | ||
|
|
82108f4b97 | ||
|
|
ab1030afb1 | ||
|
|
26292740bb | ||
|
|
96cc131305 | ||
|
|
724cd72876 | ||
|
|
fa5d9ef744 | ||
|
|
dc63b4adf5 | ||
|
|
3f20ca03dd | ||
|
|
3a826dccf1 | ||
|
|
a1ce68ce56 | ||
|
|
a45384aacb | ||
|
|
032c4f318e | ||
|
|
5faaa4a4cd | ||
|
|
4e41182521 | ||
|
|
7185032680 | ||
|
|
c9d8df0a48 | ||
|
|
d93693ba55 | ||
|
|
d49b21f3f1 | ||
|
|
a45d280a51 | ||
|
|
2536e299ad | ||
|
|
8b5238230b | ||
|
|
0ce7efee3e | ||
|
|
357677e45c | ||
|
|
a4a332926e | ||
|
|
0a29006914 | ||
|
|
0885bad859 | ||
|
|
42bd1750bd | ||
|
|
58784b2f28 | ||
|
|
e125aace9b |
@@ -117,6 +117,15 @@
|
|||||||
"contributions": [
|
"contributions": [
|
||||||
"bug"
|
"bug"
|
||||||
]
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "yabwon",
|
||||||
|
"name": "Bart Jablonski",
|
||||||
|
"avatar_url": "https://avatars.githubusercontent.com/u/9314894?v=4",
|
||||||
|
"profile": "https://github.com/yabwon",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"contributorsPerLine": 7,
|
"contributorsPerLine": 7,
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
#!/bin/sh
|
#!/bin/bash
|
||||||
sasjs lint
|
sasjs lint
|
||||||
|
|
||||||
# Avoid commits to the master branch
|
# Avoid commits to the master branch
|
||||||
|
|||||||
2
.github/CONTRIBUTING.md
vendored
2
.github/CONTRIBUTING.md
vendored
@@ -12,7 +12,7 @@ This repository makes use of the [SASjs](https://sasjs.io) framework for code or
|
|||||||
* [VSCode](https://sasjs.io/windows/#vscode) - feature packed IDE for code editing (warning - highly effective!)
|
* [VSCode](https://sasjs.io/windows/#vscode) - feature packed IDE for code editing (warning - highly effective!)
|
||||||
* [GIT](https://sasjs.io/windows/#git) - a safety net you cannot (and should not) do without.
|
* [GIT](https://sasjs.io/windows/#git) - a safety net you cannot (and should not) do without.
|
||||||
|
|
||||||
For generating the documentation (`sasjs doc`) it is also necessary to install [doxygen](https://www.doxygen.nl/manual/install.html).
|
For generating the documentation (`sasjs doc`) it is also necessary to install [doxygen](https://www.doxygen.nl/manual/install.html) and GraphViz (`sudo port install graphviz` on mac, or `sudo apt-get install graphviz` on Ubuntu).
|
||||||
|
|
||||||
|
|
||||||
To get configured:
|
To get configured:
|
||||||
|
|||||||
3
.github/FUNDING.yml
vendored
Normal file
3
.github/FUNDING.yml
vendored
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
# These are supported funding model platforms
|
||||||
|
|
||||||
|
github: [sasjs]
|
||||||
9
.github/dependabot.yml
vendored
Normal file
9
.github/dependabot.yml
vendored
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
version: 2
|
||||||
|
updates:
|
||||||
|
- package-ecosystem: npm
|
||||||
|
directory: '/'
|
||||||
|
schedule:
|
||||||
|
interval: monthly
|
||||||
|
open-pull-requests-limit: 3
|
||||||
|
allow:
|
||||||
|
- dependency-type: "production"
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
FROM gitpod/workspace-full
|
FROM gitpod/workspace-full
|
||||||
|
|
||||||
RUN sudo apt-get update \
|
RUN sudo apt-get update \
|
||||||
&& sudo apt-get install -y \
|
&& sudo apt-get install -y doxygen \
|
||||||
doxygen \
|
&& sudo apt-get install -y graphviz \
|
||||||
&& sudo rm -rf /var/lib/apt/lists/*
|
&& sudo rm -rf /var/lib/apt/lists/*
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
tasks:
|
tasks:
|
||||||
- init: nvm install --lts && npm i -g @sasjs/cli
|
- init: |
|
||||||
|
nvm install --lts
|
||||||
|
npm i -g @sasjs/cli
|
||||||
|
|
||||||
image:
|
image:
|
||||||
file: .gitpod.dockerfile
|
file: .gitpod.dockerfile
|
||||||
|
|||||||
@@ -6,7 +6,7 @@
|
|||||||
"hasMacroParentheses": true,
|
"hasMacroParentheses": true,
|
||||||
"noNestedMacros": false,
|
"noNestedMacros": false,
|
||||||
"noSpacesInFileNames": true,
|
"noSpacesInFileNames": true,
|
||||||
"maxLineLength": 230,
|
"maxLineLength": 300,
|
||||||
"lowerCaseFileNames": true,
|
"lowerCaseFileNames": true,
|
||||||
"noTabIndentation": true,
|
"noTabIndentation": true,
|
||||||
"indentationMultiple": 2
|
"indentationMultiple": 2
|
||||||
|
|||||||
51
README.md
51
README.md
@@ -31,19 +31,28 @@ Documentation: https://core.sasjs.io
|
|||||||
|
|
||||||
## Components
|
## Components
|
||||||
|
|
||||||
### BASE library (SAS9/Viya)
|
### BASE library (All Platforms)
|
||||||
|
|
||||||
- OS independent
|
- OS independent
|
||||||
- Not metadata aware
|
- Works on all SAS Platforms
|
||||||
- No X command
|
- No X command
|
||||||
- Prefixes: _mf_, _mp_
|
- Prefixes: _mf_, _mp_
|
||||||
|
|
||||||
#### FCMP library (SAS9/Viya)
|
### DDL library (All Platforms)
|
||||||
|
|
||||||
|
- OS independent
|
||||||
|
- Works on all SAS Platforms
|
||||||
|
- No X command
|
||||||
|
- Prefixes: _mddl_(lib)_ -> where lib can be "SAS" (in relation to a SAS component) or "DC" (in relation to a Data Controller component)
|
||||||
|
|
||||||
|
This library will not be used for storing data entries (such as formats or datalines). Where this becomes necessary in the future, a new repo will be created, in order to keep the NPM bundle size down (for the benefit of those looking to embed purely macros in their applications).
|
||||||
|
|
||||||
|
### FCMP library (All Platforms)
|
||||||
|
|
||||||
- Function and macro names are identical, except for special cases
|
- Function and macro names are identical, except for special cases
|
||||||
- Prefixes: _mcf_
|
- Prefixes: _mcf_
|
||||||
|
|
||||||
The fcmp macros are used to generate fcmp functions, and can be used with or
|
The fcmp macros are used to generate fcmp functions, and can be used with or without the `proc fcmp` wrapper.
|
||||||
without the `proc fcmp` wrapper.
|
|
||||||
|
|
||||||
### META library (SAS9 only)
|
### META library (SAS9 only)
|
||||||
|
|
||||||
@@ -81,7 +90,7 @@ Macros used for interfacing with SAS Viya.
|
|||||||
|
|
||||||
Wait - this is a macro library - what is LUA doing here? Well, it is a little known fact that you CAN run LUA within a SAS Macro. It has to be written to a text file with a `.lua` extension, from where you can `%include` it. So, without using the `proc lua` wrapper.
|
Wait - this is a macro library - what is LUA doing here? Well, it is a little known fact that you CAN run LUA within a SAS Macro. It has to be written to a text file with a `.lua` extension, from where you can `%include` it. So, without using the `proc lua` wrapper.
|
||||||
|
|
||||||
To contribute, simply write your freeform LUA in the LUA folder. Then run the `build.py`, which will convert your LUA into a data step with put statements, and create the macro wrapper with a `ml_` prefix. You can then use your module in any program by running:
|
To contribute, simply write your freeform LUA in the LUA folder. Then run the `build.py`, which will convert all files with a ".lua" extension into a macro wrapper with an `ml_` prefix (embedding the necessary data step put statements). You can then use your module in any program by running:
|
||||||
|
|
||||||
```sas
|
```sas
|
||||||
/* compile the lua module */
|
/* compile the lua module */
|
||||||
@@ -95,8 +104,7 @@ endsubmit;
|
|||||||
run;
|
run;
|
||||||
```
|
```
|
||||||
|
|
||||||
- X command enabled
|
- Prefixes: _ml_
|
||||||
- Prefixes: _mmw_,_mmu_,_mmx_
|
|
||||||
|
|
||||||
## Installation
|
## Installation
|
||||||
|
|
||||||
@@ -125,14 +133,16 @@ filename mc url "https://raw.githubusercontent.com/sasjs/core/main/all.sas";
|
|||||||
- macro names must be lowercase
|
- macro names must be lowercase
|
||||||
- one macro per file
|
- one macro per file
|
||||||
- prefixes:
|
- prefixes:
|
||||||
|
- _mcf_ for macro compiled functions (proc fcmp)
|
||||||
|
- _mddl_ for macros containing DDL (Data Definition Language)
|
||||||
- _mf_ for macro functions (can be used in open code).
|
- _mf_ for macro functions (can be used in open code).
|
||||||
- _ml_ for macros that are used to compile LUA modules
|
- _ml_ for macros that are used to compile LUA modules
|
||||||
- _mm_ for metadata macros (interface with the metadata server).
|
- _mm_ for metadata macros (interface with the metadata server).
|
||||||
- _mmx_ for macros that use metadata and are XCMD enabled
|
- _mmx_ for macros that use metadata and are XCMD enabled (working on both windows and unix)
|
||||||
- _mp_ for macro procedures (which generate sas code)
|
- _mp_ for macro procedures (which generate sas code)
|
||||||
- _ms_ for macro procedures that will only work with [@sasjs/server](https://github.com/sasjs/server)
|
- _ms_ for macro procedures that will only work with [@sasjs/server](https://github.com/sasjs/server)
|
||||||
- _mv_ for macro procedures that will only work in Viya
|
- _mv_ for macro procedures that will only work in Viya
|
||||||
- _mx_ for macros that are XCMD enabled
|
- _mx_ for macros that are XCMD enabled (working on both windows and unix)
|
||||||
- follow verb-noun convention
|
- follow verb-noun convention
|
||||||
- unix style line endings (lf)
|
- unix style line endings (lf)
|
||||||
- individual lines should be no more than 80 characters long
|
- individual lines should be no more than 80 characters long
|
||||||
@@ -166,7 +176,7 @@ SAS code can contain one of two types of dependency - SAS Macros, and SAS Includ
|
|||||||
@li someprogram.sas FREFTWO
|
@li someprogram.sas FREFTWO
|
||||||
```
|
```
|
||||||
|
|
||||||
The CLI can then extract all the dependencies and insert as precode (SAS Macros) or in a temp engine fileref (SAS Includes) when creating SAS Jobs and Services.
|
The CLI can then extract all the dependencies and insert as precode (SAS Macros) or in a temp engine fileref (SAS Includes) when creating SAS Jobs and Services (and Tests).
|
||||||
|
|
||||||
When contributing to this library, it is therefore important to ensure that all dependencies are listed in the header in this format.
|
When contributing to this library, it is therefore important to ensure that all dependencies are listed in the header in this format.
|
||||||
|
|
||||||
@@ -180,14 +190,27 @@ When contributing to this library, it is therefore important to ensure that all
|
|||||||
- The closing `%mend;` should **not** contain the macro name.
|
- The closing `%mend;` should **not** contain the macro name.
|
||||||
- All macros should be defined with brackets, even if no variables are needed - ie `%macro x();` not `%macro x;`
|
- All macros should be defined with brackets, even if no variables are needed - ie `%macro x();` not `%macro x;`
|
||||||
- Mandatory parameters should be positional, all optional parameters should be keyword (var=) style.
|
- Mandatory parameters should be positional, all optional parameters should be keyword (var=) style.
|
||||||
- All dataset references must be 2 level (eg `work.blah`, not `blah`). This is to avoid contention when options [DATASTMTCHK](https://support.sas.com/documentation/cdl/en/lrdict/64316/HTML/default/viewer.htm#a000279064.htm)=ALLKEYWORDS is in effect.
|
- All dataset references must be 2 level (eg `work.blah`, not `blah`). This is to avoid contention when options [DATASTMTCHK](https://support.sas.com/documentation/cdl/en/lrdict/64316/HTML/default/viewer.htm#a000279064.htm)=ALLKEYWORDS is in effect, or the [USER](https://documentation.sas.com/doc/en/pgmsascdc/9.4_3.5/lrcon/n18m1vkqmeo4esn1moikt23zhp8s.htm) library is active.
|
||||||
- Avoid naming collisions! All macro variables should be local scope. Use system generated work tables where possible - eg `data ; set sashelp.class; run; data &output; set &syslast; run;`
|
- Avoid naming collisions! All macro variables should be local scope. Use system generated work tables where possible - eg `data ; set sashelp.class; run; data &output; set &syslast; run;`
|
||||||
|
- Where global macro variables are absolutely necessary, they should make use of `&sasjs_prefix` - see mp_init.sas
|
||||||
- The use of `quit;` for `proc sql` is optional unless you are looking to benefit from the timing statistics.
|
- The use of `quit;` for `proc sql` is optional unless you are looking to benefit from the timing statistics.
|
||||||
|
- Use [sasjs lint](https://github.com/sasjs/lint)!
|
||||||
|
|
||||||
## 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.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`).
|
||||||
|
|
||||||
|
## Breaking Changes
|
||||||
|
|
||||||
|
We are currently on major release v4. Breaking changes should be marked with the [deprecated](https://www.doxygen.nl/manual/commands.html#cmddeprecated) doxygen tag. The following changes are planned when the next major/breaking release (v5) becomes necessary:
|
||||||
|
|
||||||
|
* mf_getuniquelibref.sas to have the deprecated maxtried parameter removed (no longer needed)
|
||||||
|
* mp_testservice.sas to be renamed as mp_execute.sas (as it doesn't actually test anything)
|
||||||
|
* `insert_cmplib` option of mcf_xxx macros will be deprecated (the option is now checked automatically with value inserted only if needed)
|
||||||
|
* mcf_xxx macros to have `wrap=` option defaulted to YES for convenience. Set this option explicitly to avoid issues.
|
||||||
|
* mp_getddl.sas to be renamed to mp_ds2ddl.sas (consistent with other ds2xxx macros). A wrapper macro is already in place, and you are able to use this immediately. The default for SHOWLOG will also be YES instead of NO.
|
||||||
|
* mp_coretable.sas will be replaced by the standalone macros in the `ddl` folder (which are already available)
|
||||||
|
|
||||||
## Star Gazing
|
## Star Gazing
|
||||||
|
|
||||||
If you find this library useful, please leave a [star](https://github.com/sasjs/core/stargazers) and help us grow our star graph!
|
If you find this library useful, please leave a [star](https://github.com/sasjs/core/stargazers) and help us grow our star graph!
|
||||||
@@ -195,10 +218,9 @@ If you find this library useful, please leave a [star](https://github.com/sasjs/
|
|||||||

|

|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
## Contributors ✨
|
## Contributors ✨
|
||||||
<!-- ALL-CONTRIBUTORS-BADGE:START - Do not remove or modify this section -->
|
<!-- ALL-CONTRIBUTORS-BADGE:START - Do not remove or modify this section -->
|
||||||
[](#contributors-)
|
[](#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)):
|
||||||
|
|
||||||
@@ -219,6 +241,7 @@ Thanks goes to these wonderful people ([emoji key](https://allcontributors.org/d
|
|||||||
<td align="center"><a href="https://github.com/kkchandok"><img src="https://avatars.githubusercontent.com/u/46090627?v=4?s=100" width="100px;" alt=""/><br /><sub><b>kkchandok</b></sub></a><br /><a href="#ideas-kkchandok" title="Ideas, Planning, & Feedback">🤔</a></td>
|
<td align="center"><a href="https://github.com/kkchandok"><img src="https://avatars.githubusercontent.com/u/46090627?v=4?s=100" width="100px;" alt=""/><br /><sub><b>kkchandok</b></sub></a><br /><a href="#ideas-kkchandok" title="Ideas, Planning, & Feedback">🤔</a></td>
|
||||||
<td align="center"><a href="https://github.com/VladislavParhomchik"><img src="https://avatars.githubusercontent.com/u/83717836?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Vladislav Parhomchik</b></sub></a><br /><a href="https://github.com/sasjs/core/commits?author=VladislavParhomchik" title="Tests">⚠️</a> <a href="https://github.com/sasjs/core/pulls?q=is%3Apr+reviewed-by%3AVladislavParhomchik" title="Reviewed Pull Requests">👀</a></td>
|
<td align="center"><a href="https://github.com/VladislavParhomchik"><img src="https://avatars.githubusercontent.com/u/83717836?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Vladislav Parhomchik</b></sub></a><br /><a href="https://github.com/sasjs/core/commits?author=VladislavParhomchik" title="Tests">⚠️</a> <a href="https://github.com/sasjs/core/pulls?q=is%3Apr+reviewed-by%3AVladislavParhomchik" title="Reviewed Pull Requests">👀</a></td>
|
||||||
<td align="center"><a href="https://github.com/vznesh"><img src="https://avatars.githubusercontent.com/u/28916792?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Vignesh T.</b></sub></a><br /><a href="https://github.com/sasjs/core/issues?q=author%3Avznesh" title="Bug reports">🐛</a></td>
|
<td align="center"><a href="https://github.com/vznesh"><img src="https://avatars.githubusercontent.com/u/28916792?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Vignesh T.</b></sub></a><br /><a href="https://github.com/sasjs/core/issues?q=author%3Avznesh" title="Bug reports">🐛</a></td>
|
||||||
|
<td align="center"><a href="https://github.com/yabwon"><img src="https://avatars.githubusercontent.com/u/9314894?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Bart Jablonski</b></sub></a><br /><a href="https://github.com/sasjs/core/commits?author=yabwon" title="Code">💻</a></td>
|
||||||
</tr>
|
</tr>
|
||||||
</table>
|
</table>
|
||||||
|
|
||||||
|
|||||||
@@ -11,7 +11,7 @@
|
|||||||
@cond
|
@cond
|
||||||
**/
|
**/
|
||||||
|
|
||||||
%macro mf_abort(mac=mf_abort.sas, type=deprecated, msg=, iftrue=%str(1=1)
|
%macro mf_abort(mac=mf_abort.sas, msg=, iftrue=%str(1=1)
|
||||||
)/*/STORE SOURCE*/;
|
)/*/STORE SOURCE*/;
|
||||||
|
|
||||||
%if not(%eval(%unquote(&iftrue))) %then %return;
|
%if not(%eval(%unquote(&iftrue))) %then %return;
|
||||||
|
|||||||
53
base/mf_dedup.sas
Normal file
53
base/mf_dedup.sas
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
/**
|
||||||
|
@file
|
||||||
|
@brief de-duplicates a macro string
|
||||||
|
@details Removes all duplicates from a string of words. A delimeter can be
|
||||||
|
chosen. Is inspired heavily by this excellent [macro](
|
||||||
|
https://github.com/scottbass/SAS/blob/master/Macro/dedup_mstring.sas) from
|
||||||
|
[Scott Base](https://www.linkedin.com/in/scottbass). Case sensitive.
|
||||||
|
|
||||||
|
Usage:
|
||||||
|
|
||||||
|
%let str=One two one two and through and through;
|
||||||
|
%put %mf_dedup(&str);
|
||||||
|
%put %mf_dedup(&str,outdlm=%str(,));
|
||||||
|
|
||||||
|
Which returns:
|
||||||
|
|
||||||
|
> One two one and through
|
||||||
|
> One,two,one,and,through
|
||||||
|
|
||||||
|
@param [in] str String to be deduplicated
|
||||||
|
@param [in] indlm= ( ) Delimeter of the input string
|
||||||
|
@param [out] outdlm= ( ) Delimiter of the output string
|
||||||
|
|
||||||
|
<h4> Related Macros </h4>
|
||||||
|
@li mf_trimstr.sas
|
||||||
|
|
||||||
|
@version 9.2
|
||||||
|
@author Allan Bowe
|
||||||
|
**/
|
||||||
|
|
||||||
|
%macro mf_dedup(str
|
||||||
|
,indlm=%str( )
|
||||||
|
,outdlm=%str( )
|
||||||
|
)/*/STORE SOURCE*/;
|
||||||
|
|
||||||
|
%local num word i pos out;
|
||||||
|
|
||||||
|
%* loop over each token, searching the target for that token ;
|
||||||
|
%let num=%sysfunc(countc(%superq(str),%str(&indlm)));
|
||||||
|
%do i=1 %to %eval(&num+1);
|
||||||
|
%let word=%scan(%superq(str),&i,%str(&indlm));
|
||||||
|
%let pos=%sysfunc(indexw(&out,&word,%str(&outdlm)));
|
||||||
|
%if (&pos eq 0) %then %do;
|
||||||
|
%if (&i gt 1) %then %let out=&out%str(&outdlm);
|
||||||
|
%let out=&out&word;
|
||||||
|
%end;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%unquote(&out)
|
||||||
|
|
||||||
|
%mend mf_dedup;
|
||||||
|
|
||||||
|
|
||||||
@@ -5,7 +5,7 @@
|
|||||||
Run without arguments to see a list of detectable features.
|
Run without arguments to see a list of detectable features.
|
||||||
Note - this list is based on known versions of SAS rather than
|
Note - this list is based on known versions of SAS rather than
|
||||||
actual feature detection, as that is tricky / impossible to do
|
actual feature detection, as that is tricky / impossible to do
|
||||||
without generating errors in most cases.
|
without generating errs in most cases.
|
||||||
|
|
||||||
%put %mf_existfeature(PROCLUA);
|
%put %mf_existfeature(PROCLUA);
|
||||||
|
|
||||||
|
|||||||
@@ -1,14 +1,18 @@
|
|||||||
/**
|
/**
|
||||||
@file
|
@file
|
||||||
@brief Checks if a variable exists in a data set.
|
@brief Checks if a variable exists in a data set.
|
||||||
@details Returns 0 if the variable does NOT exist, and return the position of
|
@details Returns 0 if the variable does NOT exist, and the position of the var
|
||||||
the var if it does.
|
if it does.
|
||||||
Usage:
|
Usage:
|
||||||
|
|
||||||
%put %mf_existvar(work.someds, somevar)
|
%put %mf_existvar(work.someds, somevar)
|
||||||
|
|
||||||
|
@param [in] libds 2 part dataset or view reference
|
||||||
|
@param [in] var variable name
|
||||||
|
|
||||||
|
<h4> Related Macros </h4>
|
||||||
|
@li mf_existvar.test.sas
|
||||||
|
|
||||||
@param libds (positional) - 2 part dataset or view reference
|
|
||||||
@param var (positional) - variable name
|
|
||||||
@version 9.2
|
@version 9.2
|
||||||
@author Allan Bowe
|
@author Allan Bowe
|
||||||
**/
|
**/
|
||||||
|
|||||||
@@ -15,6 +15,7 @@
|
|||||||
@li /data
|
@li /data
|
||||||
@li /jobs
|
@li /jobs
|
||||||
@li /services
|
@li /services
|
||||||
|
@li /tests
|
||||||
@li /tests/jobs
|
@li /tests/jobs
|
||||||
@li /tests/services
|
@li /tests/services
|
||||||
@li /tests/macros
|
@li /tests/macros
|
||||||
@@ -46,9 +47,13 @@
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* First check we are not in the tests/macros folder (which has no subfolders)
|
* First check we are not in the tests/macros folder (which has no subfolders)
|
||||||
|
* or specifically in the testsetup or testteardown services
|
||||||
*/
|
*/
|
||||||
%if %index(&pgm,/tests/macros/) %then %do;
|
%if %index(&pgm,/tests/macros/)
|
||||||
%let root=%substr(&pgm,1,%index(&pgm,/tests/macros)-1);
|
or %index(&pgm,/tests/testsetup)
|
||||||
|
or %index(&pgm,/tests/testteardown)
|
||||||
|
%then %do;
|
||||||
|
%let root=%substr(&pgm,1,%index(&pgm,/tests)-1);
|
||||||
&root
|
&root
|
||||||
%return;
|
%return;
|
||||||
%end;
|
%end;
|
||||||
|
|||||||
@@ -10,7 +10,7 @@
|
|||||||
@param attr full list in [documentation](
|
@param 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 error.
|
message if err.
|
||||||
|
|
||||||
@version 9.2
|
@version 9.2
|
||||||
@author Allan Bowe
|
@author Allan Bowe
|
||||||
|
|||||||
@@ -10,7 +10,7 @@
|
|||||||
@param attr Common values are NLOBS and NVARS, full list in [documentation](
|
@param 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 error.
|
message if err.
|
||||||
|
|
||||||
@version 9.2
|
@version 9.2
|
||||||
@author Allan Bowe
|
@author Allan Bowe
|
||||||
|
|||||||
@@ -12,9 +12,10 @@
|
|||||||
contributors of Chris Hemedingers blog [post](
|
contributors of Chris Hemedingers blog [post](
|
||||||
http://blogs.sas.com/content/sasdummy/2013/06/04/find-a-sas-library-engine/)
|
http://blogs.sas.com/content/sasdummy/2013/06/04/find-a-sas-library-engine/)
|
||||||
|
|
||||||
@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 engine for the FIRST library encountered.
|
@return output returns the library engine (uppercase) for the FIRST library
|
||||||
|
encountered.
|
||||||
|
|
||||||
@warning will only return the FIRST library engine - for concatenated
|
@warning will only return the FIRST library engine - for concatenated
|
||||||
libraries, with different engines, inconsistent results may be encountered.
|
libraries, with different engines, inconsistent results may be encountered.
|
||||||
@@ -46,7 +47,7 @@
|
|||||||
%let rc= %sysfunc(close(&dsid));
|
%let rc= %sysfunc(close(&dsid));
|
||||||
%end;
|
%end;
|
||||||
|
|
||||||
&engine
|
%upcase(&engine)
|
||||||
|
|
||||||
%mend mf_getengine;
|
%mend mf_getengine;
|
||||||
|
|
||||||
|
|||||||
@@ -5,18 +5,19 @@
|
|||||||
|
|
||||||
%put %mf_getfilesize(fpath=C:\temp\myfile.txt);
|
%put %mf_getfilesize(fpath=C:\temp\myfile.txt);
|
||||||
|
|
||||||
or
|
or, provide a libds value as follows:
|
||||||
|
|
||||||
data x;do x=1 to 100000;y=x;output;end;run;
|
data x;do x=1 to 100000;y=x;output;end;run;
|
||||||
%put %mf_getfilesize(libds=work.x,format=yes);
|
%put %mf_getfilesize(libds=work.x,format=yes);
|
||||||
|
|
||||||
gives:
|
Which gives:
|
||||||
|
|
||||||
2mb
|
> 2mb
|
||||||
|
|
||||||
|
@param [in] fpath= Full path and filename. Provide this OR the libds value.
|
||||||
|
@param [in] libds= (0) Library.dataset value (assumes library is BASE engine)
|
||||||
|
@param [in] format= (NO) Set to yes to apply sizekmg. format
|
||||||
|
|
||||||
@param fpath= full path and filename. Provide this OR the libds value.
|
|
||||||
@param libds= library.dataset value (assumes library is BASE engine)
|
|
||||||
@param format= set to yes to apply sizekmg. format
|
|
||||||
@returns bytes
|
@returns bytes
|
||||||
|
|
||||||
@version 9.2
|
@version 9.2
|
||||||
@@ -26,16 +27,32 @@
|
|||||||
%macro mf_getfilesize(fpath=,libds=0,format=NO
|
%macro mf_getfilesize(fpath=,libds=0,format=NO
|
||||||
)/*/STORE SOURCE*/;
|
)/*/STORE SOURCE*/;
|
||||||
|
|
||||||
%if &libds ne 0 %then %do;
|
%local rc fid fref bytes dsid lib vnum;
|
||||||
%let fpath=%sysfunc(pathname(%scan(&libds,1,.)))/%scan(&libds,2,.).sas7bdat;
|
|
||||||
%end;
|
|
||||||
|
|
||||||
%local rc fid fref bytes;
|
%if &libds ne 0 %then %do;
|
||||||
%let rc=%sysfunc(filename(fref,&fpath));
|
%let libds=%upcase(&libds);
|
||||||
%let fid=%sysfunc(fopen(&fref));
|
%if %index(&libds,.)=0 %then %let lib=WORK;
|
||||||
%let bytes=%sysfunc(finfo(&fid,File Size (bytes)));
|
%else %let lib=%scan(&libds,1,.);
|
||||||
%let rc=%sysfunc(fclose(&fid));
|
%let dsid=%sysfunc(open(
|
||||||
%let rc=%sysfunc(filename(fref));
|
sashelp.vtable(where=(libname="&lib" and memname="%scan(&libds,-1,.)")
|
||||||
|
keep=libname memname filesize
|
||||||
|
)
|
||||||
|
));
|
||||||
|
%if (&dsid ^= 0) %then %do;
|
||||||
|
%let vnum=%sysfunc(varnum(&dsid,FILESIZE));
|
||||||
|
%let rc=%sysfunc(fetch(&dsid));
|
||||||
|
%let bytes=%sysfunc(getvarn(&dsid,&vnum));
|
||||||
|
%let rc= %sysfunc(close(&dsid));
|
||||||
|
%end;
|
||||||
|
%else %put &sysmacroname: &libds could not be opened! %sysfunc(sysmsg());
|
||||||
|
%end;
|
||||||
|
%else %do;
|
||||||
|
%let rc=%sysfunc(filename(fref,&fpath));
|
||||||
|
%let fid=%sysfunc(fopen(&fref));
|
||||||
|
%let bytes=%sysfunc(finfo(&fid,File Size (bytes)));
|
||||||
|
%let rc=%sysfunc(fclose(&fid));
|
||||||
|
%let rc=%sysfunc(filename(fref));
|
||||||
|
%end;
|
||||||
|
|
||||||
%if &format=NO %then %do;
|
%if &format=NO %then %do;
|
||||||
&bytes
|
&bytes
|
||||||
|
|||||||
61
base/mf_getfmtlist.sas
Normal file
61
base/mf_getfmtlist.sas
Normal file
@@ -0,0 +1,61 @@
|
|||||||
|
/**
|
||||||
|
@file
|
||||||
|
@brief Returns a distinct list of formats from a table
|
||||||
|
@details Reads the dataset header and returns a distinct list of formats
|
||||||
|
applied.
|
||||||
|
|
||||||
|
%put NOTE- %mf_getfmtlist(sashelp.prdsale);
|
||||||
|
%put NOTE- %mf_getfmtlist(sashelp.shoes);
|
||||||
|
%put NOTE- %mf_getfmtlist(sashelp.demographics);
|
||||||
|
|
||||||
|
returns:
|
||||||
|
|
||||||
|
> DOLLAR $CHAR W MONNAME
|
||||||
|
> $CHAR BEST DOLLAR
|
||||||
|
> BEST Z $CHAR COMMA PERCENTN
|
||||||
|
|
||||||
|
|
||||||
|
@param [in] libds Two part library.dataset reference.
|
||||||
|
|
||||||
|
<h4> SAS Macros </h4>
|
||||||
|
@li mf_getfmtname.sas
|
||||||
|
|
||||||
|
@version 9.2
|
||||||
|
@author Allan Bowe
|
||||||
|
|
||||||
|
**/
|
||||||
|
|
||||||
|
%macro mf_getfmtlist(libds
|
||||||
|
)/*/STORE SOURCE*/;
|
||||||
|
/* declare local vars */
|
||||||
|
%local out dsid nvars x rc fmt;
|
||||||
|
|
||||||
|
/* open dataset in macro */
|
||||||
|
%let dsid=%sysfunc(open(&libds));
|
||||||
|
|
||||||
|
/* continue if dataset exists */
|
||||||
|
%if &dsid %then %do;
|
||||||
|
/* loop each variable in the dataset */
|
||||||
|
%let nvars=%sysfunc(attrn(&dsid,NVARS));
|
||||||
|
%do x=1 %to &nvars;
|
||||||
|
/* grab format and check it exists */
|
||||||
|
%let fmt=%sysfunc(varfmt(&dsid,&x));
|
||||||
|
%if %quote(&fmt) ne %quote() %then %let fmt=%mf_getfmtname(&fmt);
|
||||||
|
%else %do;
|
||||||
|
/* assign default format depending on variable type */
|
||||||
|
%if %sysfunc(vartype(&dsid, &x))=C %then %let fmt=$CHAR;
|
||||||
|
%else %let fmt=BEST;
|
||||||
|
%end;
|
||||||
|
/* concatenate unique list of formats */
|
||||||
|
%if %sysfunc(indexw(&out,&fmt,%str( )))=0 %then %let out=&out &fmt;
|
||||||
|
%end;
|
||||||
|
%let rc=%sysfunc(close(&dsid));
|
||||||
|
%end;
|
||||||
|
%else %do;
|
||||||
|
%put &sysmacroname: Unable to open &libds (rc=&dsid);
|
||||||
|
%put &sysmacroname: SYSMSG= %sysfunc(sysmsg());
|
||||||
|
%let rc=%sysfunc(close(&dsid));
|
||||||
|
%end;
|
||||||
|
/* send them out without spaces or quote markers */
|
||||||
|
%do;%unquote(&out)%end;
|
||||||
|
%mend mf_getfmtlist;
|
||||||
44
base/mf_getfmtname.sas
Normal file
44
base/mf_getfmtname.sas
Normal file
@@ -0,0 +1,44 @@
|
|||||||
|
/**
|
||||||
|
@file
|
||||||
|
@brief Extracts a format name from a fully defined format
|
||||||
|
@details Converts formats in like $thi3. and th13.2 $THI and TH.
|
||||||
|
Usage:
|
||||||
|
|
||||||
|
%put %mf_getfmtname(8.);
|
||||||
|
%put %mf_getfmtname($4.);
|
||||||
|
%put %mf_getfmtname(comma14.10);
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
|
||||||
|
> W
|
||||||
|
> $CHAR
|
||||||
|
> COMMA
|
||||||
|
|
||||||
|
Note that system defaults are inferred from the values provided.
|
||||||
|
|
||||||
|
@param [in] fmt The fully defined format. If left blank, nothing is returned.
|
||||||
|
|
||||||
|
@returns The name (without width or decimal) of the format.
|
||||||
|
|
||||||
|
@version 9.2
|
||||||
|
@author Allan Bowe
|
||||||
|
|
||||||
|
**/
|
||||||
|
|
||||||
|
%macro mf_getfmtname(fmt
|
||||||
|
)/*/STORE SOURCE*/ /minoperator mindelimiter=' ';
|
||||||
|
|
||||||
|
%local out dsid nvars x rc fmt;
|
||||||
|
|
||||||
|
/* extract actual format name from the format definition */
|
||||||
|
%let fmt=%scan(&fmt,1,.);
|
||||||
|
%do %while(%substr(&fmt,%length(&fmt),1) in 1 2 3 4 5 6 7 8 9 0);
|
||||||
|
%if %length(&fmt)=1 %then %let fmt=W;
|
||||||
|
%else %let fmt=%substr(&fmt,1,%length(&fmt)-1);
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%if &fmt=$ %then %let fmt=$CHAR;
|
||||||
|
|
||||||
|
/* send them out without spaces or quote markers */
|
||||||
|
%do;%unquote(%upcase(&fmt))%end;
|
||||||
|
%mend mf_getfmtname;
|
||||||
@@ -22,6 +22,12 @@
|
|||||||
)/*/STORE SOURCE*/;
|
)/*/STORE SOURCE*/;
|
||||||
%local a b c;
|
%local a b c;
|
||||||
%if &switch.NONE=NONE %then %do;
|
%if &switch.NONE=NONE %then %do;
|
||||||
|
%if %symexist(sasjsprocessmode) %then %do;
|
||||||
|
%if &sasjsprocessmode=Stored Program %then %do;
|
||||||
|
SASJS
|
||||||
|
%return;
|
||||||
|
%end;
|
||||||
|
%end;
|
||||||
%if %symexist(sysprocessmode) %then %do;
|
%if %symexist(sysprocessmode) %then %do;
|
||||||
%if "&sysprocessmode"="SAS Object Server"
|
%if "&sysprocessmode"="SAS Object Server"
|
||||||
or "&sysprocessmode"= "SAS Compute Server" %then %do;
|
or "&sysprocessmode"= "SAS Compute Server" %then %do;
|
||||||
|
|||||||
@@ -16,8 +16,8 @@
|
|||||||
> "these","words","are","double","quoted"
|
> "these","words","are","double","quoted"
|
||||||
|
|
||||||
@param [in] in_str The unquoted, spaced delimited string to transform
|
@param [in] in_str The unquoted, spaced delimited string to transform
|
||||||
@param [in] dlm= The delimeter to be applied to the output (default comma)
|
@param [in] dlm= (,) The delimeter to be applied to the output (default comma)
|
||||||
@param [in] indlm= (,) The delimeter used for the input (default is space)
|
@param [in] indlm= ( ) The delimeter used for the input (default is space)
|
||||||
@param [in] quote= (S) The quote mark to apply (S=Single, D=Double, N=None).
|
@param [in] quote= (S) The quote mark to apply (S=Single, D=Double, N=None).
|
||||||
If any other value than uppercase S or D is supplied, then that value will
|
If any other value than uppercase S or D is supplied, then that value will
|
||||||
be used as the quoting character.
|
be used as the quoting character.
|
||||||
@@ -28,7 +28,10 @@
|
|||||||
**/
|
**/
|
||||||
|
|
||||||
|
|
||||||
%macro mf_getquotedstr(IN_STR,DLM=%str(,),QUOTE=S,indlm=%str( )
|
%macro mf_getquotedstr(IN_STR
|
||||||
|
,DLM=%str(,)
|
||||||
|
,QUOTE=S
|
||||||
|
,indlm=%str( )
|
||||||
)/*/STORE SOURCE*/;
|
)/*/STORE SOURCE*/;
|
||||||
/* credit Rowland Hale - byte34 is double quote, 39 is single quote */
|
/* credit Rowland Hale - byte34 is double quote, 39 is single quote */
|
||||||
%if "e=S %then %let quote=%qsysfunc(byte(39));
|
%if "e=S %then %let quote=%qsysfunc(byte(39));
|
||||||
|
|||||||
@@ -14,27 +14,44 @@
|
|||||||
|
|
||||||
> mclib3
|
> mclib3
|
||||||
|
|
||||||
@param prefix= first part of libref. Remember that librefs can only be 8 characters,
|
A blank value is returned if no usable libname is determined.
|
||||||
so a 7 letter prefix would mean that maxtries should be 10.
|
|
||||||
@param maxtries= the last part of the libref. Provide an integer value.
|
@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
|
||||||
|
for this prefix.
|
||||||
|
@param [in] maxtries= Deprecated parameter. Remains here to ensure a
|
||||||
|
non-breaking change. Will be removed in v5.
|
||||||
|
|
||||||
@version 9.2
|
@version 9.2
|
||||||
@author Allan Bowe
|
@author Allan Bowe
|
||||||
**/
|
**/
|
||||||
|
|
||||||
|
|
||||||
%macro mf_getuniquelibref(prefix=mclib,maxtries=1000);
|
%macro mf_getuniquelibref(prefix=mclib,maxtries=1000);
|
||||||
%local x libref;
|
%local x;
|
||||||
%let x=0;
|
|
||||||
%do x=0 %to &maxtries;
|
%if ( %length(&prefix) gt 7 ) %then %do;
|
||||||
%if %sysfunc(libref(&prefix&x)) ne 0 %then %do;
|
%put %str(ERR)OR: The prefix parameter cannot exceed 7 characters.;
|
||||||
%let libref=&prefix&x;
|
0
|
||||||
%let rc=%sysfunc(libname(&libref,%sysfunc(pathname(work))));
|
|
||||||
%if &rc %then %put %sysfunc(sysmsg());
|
|
||||||
&prefix&x
|
|
||||||
%*put &sysmacroname: Libref &libref assigned as WORK and returned;
|
|
||||||
%return;
|
%return;
|
||||||
%end;
|
%end;
|
||||||
|
%else %if (%sysfunc(NVALID(&prefix,v7))=0) %then %do;
|
||||||
|
%put %str(ERR)OR: Invalid prefix (&prefix);
|
||||||
|
0
|
||||||
|
%return;
|
||||||
%end;
|
%end;
|
||||||
%put unable to find available libref in range &prefix.0-&maxtries;
|
|
||||||
|
/* Set maxtries equal to '10 to the power of [# unused characters] - 1' */
|
||||||
|
%let maxtries=%eval(10**(8-%length(&prefix))-1);
|
||||||
|
|
||||||
|
%do x = 0 %to &maxtries;
|
||||||
|
%if %sysfunc(libref(&prefix&x)) ne 0 %then %do;
|
||||||
|
&prefix&x
|
||||||
|
%return;
|
||||||
|
%end;
|
||||||
|
%let x = %eval(&x + 1);
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%put %str(ERR)OR: No usable libref in range &prefix.0-&maxtries;
|
||||||
|
%put %str(ERR)OR- Try reducing the prefix or deleting some libraries!;
|
||||||
|
0
|
||||||
%mend mf_getuniquelibref;
|
%mend mf_getuniquelibref;
|
||||||
@@ -70,5 +70,5 @@
|
|||||||
%put &sysmacroname: SYSMSG= %sysfunc(sysmsg());
|
%put &sysmacroname: SYSMSG= %sysfunc(sysmsg());
|
||||||
%let rc=%sysfunc(close(&dsid));
|
%let rc=%sysfunc(close(&dsid));
|
||||||
%end;
|
%end;
|
||||||
&outvar
|
%do;%unquote(&outvar)%end;
|
||||||
%mend mf_getvarlist;
|
%mend mf_getvarlist;
|
||||||
33
base/mf_isint.sas
Normal file
33
base/mf_isint.sas
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
/**
|
||||||
|
@file
|
||||||
|
@brief Returns 1 if the variable contains only digits 0-9, else 0
|
||||||
|
@details Note that numerics containing any punctuation (including decimals
|
||||||
|
or exponents) will be flagged zero.
|
||||||
|
|
||||||
|
If you'd like support for this, then do raise an issue (or even better, a
|
||||||
|
pull request!)
|
||||||
|
|
||||||
|
Usage:
|
||||||
|
|
||||||
|
%put %mf_isint(1) returns 1;
|
||||||
|
%put %mf_isint(1.1) returns 0;
|
||||||
|
%put %mf_isint(%str(1,1)) returns 0;
|
||||||
|
|
||||||
|
@param [in] arg input value to check
|
||||||
|
|
||||||
|
@version 9.2
|
||||||
|
**/
|
||||||
|
|
||||||
|
%macro mf_isint(arg
|
||||||
|
)/*/STORE SOURCE*/;
|
||||||
|
/* remove minus sign if exists */
|
||||||
|
|
||||||
|
%local val;
|
||||||
|
%if "%substr(%str(&arg),1,1)"="-" %then %let val=%substr(%str(&arg),2);
|
||||||
|
%else %let val=&arg;
|
||||||
|
|
||||||
|
/* check remaining chars */
|
||||||
|
%if %sysfunc(findc(%str(&val),,kd)) %then %do;0%end;
|
||||||
|
%else %do;1%end;
|
||||||
|
|
||||||
|
%mend mf_isint;
|
||||||
40
base/mf_islibds.sas
Normal file
40
base/mf_islibds.sas
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
/**
|
||||||
|
@file
|
||||||
|
@brief Checks whether a string follows correct library.dataset format
|
||||||
|
@details Many macros in the core library accept a library.dataset parameter
|
||||||
|
referred to as 'libds'. This macro validates the structure of that parameter,
|
||||||
|
eg:
|
||||||
|
|
||||||
|
@li 8 character libref?
|
||||||
|
@li 32 character dataset?
|
||||||
|
@li contains a period?
|
||||||
|
|
||||||
|
It does NOT check whether the dataset exists, or if the library is assigned.
|
||||||
|
|
||||||
|
Usage:
|
||||||
|
|
||||||
|
%put %mf_islibds(work.something)=1;
|
||||||
|
%put %mf_islibds(nolib)=0;
|
||||||
|
%put %mf_islibds(badlibref.ds)=0;
|
||||||
|
%put %mf_islibds(w.t.f)=0;
|
||||||
|
|
||||||
|
@param [in] libds The string to be checked
|
||||||
|
|
||||||
|
@return output Returns 1 if libds is valid, 0 if it is not
|
||||||
|
|
||||||
|
<h4> Related Macros </h4>
|
||||||
|
@li mf_islibds.test.sas
|
||||||
|
@li mp_validatecol.sas
|
||||||
|
|
||||||
|
@version 9.2
|
||||||
|
**/
|
||||||
|
|
||||||
|
%macro mf_islibds(libds
|
||||||
|
)/*/STORE SOURCE*/;
|
||||||
|
|
||||||
|
%local regex;
|
||||||
|
%let regex=%sysfunc(prxparse(%str(/^[_a-z]\w{0,7}\.[_a-z]\w{0,31}$/i)));
|
||||||
|
|
||||||
|
%sysfunc(prxmatch(®ex,&libds))
|
||||||
|
|
||||||
|
%mend mf_islibds;
|
||||||
@@ -51,7 +51,7 @@ Usage:
|
|||||||
%end;
|
%end;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Now create the directory. Complain loudly of any errors.
|
Now create the directory. Complain loudly of any errs.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
%let dname = %sysfunc(dcreate(&child, &parent));
|
%let dname = %sysfunc(dcreate(&child, &parent));
|
||||||
|
|||||||
@@ -12,7 +12,7 @@
|
|||||||
@param libds library.dataset
|
@param 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 error.
|
if err.
|
||||||
|
|
||||||
|
|
||||||
@version 9.2
|
@version 9.2
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/**
|
/**
|
||||||
@file
|
@file
|
||||||
@brief Checks if a set of macro variables exist / contain values.
|
@brief Checks if a set of macro variables exist AND contain values.
|
||||||
@details Writes ERROR to log if abortType is SOFT, else will call %mf_abort.
|
@details Writes ERROR to log if abortType is SOFT, else will call %mf_abort.
|
||||||
Usage:
|
Usage:
|
||||||
|
|
||||||
@@ -14,10 +14,11 @@
|
|||||||
<h4> SAS Macros </h4>
|
<h4> SAS Macros </h4>
|
||||||
@li mf_abort.sas
|
@li mf_abort.sas
|
||||||
|
|
||||||
@param verifyvars space separated list of macro variable names
|
@param [in] verifyvars Space separated list of macro variable names
|
||||||
@param makeupcase= set to YES to convert all variable VALUES to
|
@param [in] makeupcase= (NO) Set to YES to convert all variable VALUES to
|
||||||
uppercase.
|
uppercase.
|
||||||
@param mAbort= Abort Type. Default is SOFT (writes err to log).
|
@param [in] mAbort= (SOFT) Abort Type. When SOFT, simply writes an err
|
||||||
|
message to the log.
|
||||||
Set to any other value to call mf_abort (which can be configured to abort in
|
Set to any other value to call mf_abort (which can be configured to abort in
|
||||||
various fashions according to context).
|
various fashions according to context).
|
||||||
|
|
||||||
@@ -58,8 +59,14 @@
|
|||||||
|
|
||||||
%goto exit_success;
|
%goto exit_success;
|
||||||
%exit_err:
|
%exit_err:
|
||||||
%if &mAbort=SOFT %then %put %str(ERR)OR: &abortmsg;
|
%put %str(ERR)OR: &abortmsg;
|
||||||
%else %mf_abort(mac=mf_verifymacvars,type=&mabort,msg=&abortmsg);
|
%mf_abort(iftrue=(&mabort ne SOFT),
|
||||||
|
mac=mf_verifymacvars,
|
||||||
|
msg=%str(&abortmsg)
|
||||||
|
)
|
||||||
|
0
|
||||||
|
%return;
|
||||||
%exit_success:
|
%exit_success:
|
||||||
|
1
|
||||||
|
|
||||||
%mend mf_verifymacvars;
|
%mend mf_verifymacvars;
|
||||||
|
|||||||
53
base/mf_wordsinstr1andstr2.sas
Normal file
53
base/mf_wordsinstr1andstr2.sas
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
/**
|
||||||
|
@file
|
||||||
|
@brief Returns words that are in both string 1 and string 2
|
||||||
|
@details Compares two space separated strings and returns the words that are
|
||||||
|
in both.
|
||||||
|
Usage:
|
||||||
|
|
||||||
|
%put %mf_wordsInStr1andStr2(
|
||||||
|
Str1=blah sss blaaah brah bram boo
|
||||||
|
,Str2= blah blaaah brah ssss
|
||||||
|
);
|
||||||
|
|
||||||
|
returns:
|
||||||
|
> blah blaaah brah
|
||||||
|
|
||||||
|
@param str1= string containing words to extract
|
||||||
|
@param str2= used to compare with the extract string
|
||||||
|
|
||||||
|
@warning CASE SENSITIVE!
|
||||||
|
|
||||||
|
@version 9.2
|
||||||
|
@author Allan Bowe
|
||||||
|
|
||||||
|
**/
|
||||||
|
|
||||||
|
%macro mf_wordsInStr1andStr2(
|
||||||
|
Str1= /* string containing words to extract */
|
||||||
|
,Str2= /* used to compare with the extract string */
|
||||||
|
)/*/STORE SOURCE*/;
|
||||||
|
|
||||||
|
%local count_base count_extr i i2 extr_word base_word match outvar;
|
||||||
|
%if %length(&str1)=0 or %length(&str2)=0 %then %do;
|
||||||
|
%put base string (str1)= &str1;
|
||||||
|
%put compare string (str2) = &str2;
|
||||||
|
%return;
|
||||||
|
%end;
|
||||||
|
%let count_base=%sysfunc(countw(&Str2));
|
||||||
|
%let count_extr=%sysfunc(countw(&Str1));
|
||||||
|
|
||||||
|
%do i=1 %to &count_extr;
|
||||||
|
%let extr_word=%scan(&Str1,&i,%str( ));
|
||||||
|
%let match=0;
|
||||||
|
%do i2=1 %to &count_base;
|
||||||
|
%let base_word=%scan(&Str2,&i2,%str( ));
|
||||||
|
%if &extr_word=&base_word %then %let match=1;
|
||||||
|
%end;
|
||||||
|
%if &match=1 %then %let outvar=&outvar &extr_word;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
&outvar
|
||||||
|
|
||||||
|
%mend mf_wordsInStr1andStr2;
|
||||||
|
|
||||||
@@ -3,6 +3,9 @@
|
|||||||
@brief Returns words that are in string 1 but not in string 2
|
@brief Returns words that are in string 1 but not in string 2
|
||||||
@details Compares two space separated strings and returns the words that are
|
@details Compares two space separated strings and returns the words that are
|
||||||
in the first but not in the second.
|
in the first but not in the second.
|
||||||
|
|
||||||
|
Note - case sensitive!
|
||||||
|
|
||||||
Usage:
|
Usage:
|
||||||
|
|
||||||
%let x= %mf_wordsInStr1ButNotStr2(
|
%let x= %mf_wordsInStr1ButNotStr2(
|
||||||
@@ -13,10 +16,8 @@
|
|||||||
returns:
|
returns:
|
||||||
> sss bram boo
|
> sss bram boo
|
||||||
|
|
||||||
@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!
|
|
||||||
|
|
||||||
@version 9.2
|
@version 9.2
|
||||||
@author Allan Bowe
|
@author Allan Bowe
|
||||||
@@ -30,7 +31,6 @@
|
|||||||
|
|
||||||
%local count_base count_extr i i2 extr_word base_word match outvar;
|
%local count_base count_extr i i2 extr_word base_word match outvar;
|
||||||
%if %length(&str1)=0 or %length(&str2)=0 %then %do;
|
%if %length(&str1)=0 or %length(&str2)=0 %then %do;
|
||||||
%put %str(WARN)ING: empty string provided!;
|
|
||||||
%put base string (str1)= &str1;
|
%put base string (str1)= &str1;
|
||||||
%put compare string (str2) = &str2;
|
%put compare string (str2) = &str2;
|
||||||
%return;
|
%return;
|
||||||
|
|||||||
67
base/mf_writefile.sas
Normal file
67
base/mf_writefile.sas
Normal file
@@ -0,0 +1,67 @@
|
|||||||
|
/**
|
||||||
|
@file
|
||||||
|
@brief Creates a text file using pure macro
|
||||||
|
@details Creates a text file of up to 10 lines. If further lines are
|
||||||
|
desired, feel free to [create an issue](
|
||||||
|
https://github.com/sasjs/core/issues/new), or make a pull request!
|
||||||
|
|
||||||
|
The use of PARMBUFF was considered for this macro, but it would have made
|
||||||
|
things problematic for writing lines containing commas.
|
||||||
|
|
||||||
|
Usage:
|
||||||
|
|
||||||
|
%mf_writefile(&sasjswork/myfile.txt,l1=some content,l2=more content)
|
||||||
|
data _null_;
|
||||||
|
infile "&sasjswork/myfile.txt";
|
||||||
|
input;
|
||||||
|
list;
|
||||||
|
run;
|
||||||
|
|
||||||
|
@param [in] fpath Full path to file to be created or appended to
|
||||||
|
@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 O OUTPUT mode, writes new records from the beginning of the file.
|
||||||
|
@param [in] l1= First line
|
||||||
|
@param [in] l2= Second line (etc through to l10)
|
||||||
|
|
||||||
|
<h4> Related Macros </h4>
|
||||||
|
@li mf_writefile.test.sas
|
||||||
|
|
||||||
|
@version 9.2
|
||||||
|
@author Allan Bowe
|
||||||
|
**/
|
||||||
|
/** @cond */
|
||||||
|
|
||||||
|
%macro mf_writefile(fpath,mode=O,l1=,l2=,l3=,l4=,l5=,l6=,l7=,l8=,l9=,l10=
|
||||||
|
)/*/STORE SOURCE*/;
|
||||||
|
%local fref rc fid i total_lines;
|
||||||
|
|
||||||
|
/* find number of lines by reference to first non-blank param */
|
||||||
|
%do i=10 %to 1 %by -1;
|
||||||
|
%if %str(&&l&i) ne %str() %then %goto continue;
|
||||||
|
%end;
|
||||||
|
%continue:
|
||||||
|
%let total_lines=&i;
|
||||||
|
|
||||||
|
%if %sysfunc(filename(fref,&fpath)) ne 0 %then %do;
|
||||||
|
%put &=fref &=fpath;
|
||||||
|
%put %str(ERR)OR: %sysfunc(sysmsg());
|
||||||
|
%return;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%let fid=%sysfunc(fopen(&fref,&mode));
|
||||||
|
|
||||||
|
%if &fid=0 %then %do;
|
||||||
|
%put %str(ERR)OR: %sysfunc(sysmsg());
|
||||||
|
%return;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%do i=1 %to &total_lines;
|
||||||
|
%let rc=%sysfunc(fput(&fid, &&l&i));
|
||||||
|
%let rc=%sysfunc(fwrite(&fid));
|
||||||
|
%end;
|
||||||
|
%let rc=%sysfunc(fclose(&fid));
|
||||||
|
%let rc=%sysfunc(filename(&fref));
|
||||||
|
|
||||||
|
%mend mf_writefile;
|
||||||
|
/** @endcond */
|
||||||
@@ -66,7 +66,10 @@
|
|||||||
%if %length(&mac)>0 %then %put NOTE- called by &mac;
|
%if %length(&mac)>0 %then %put NOTE- called by &mac;
|
||||||
%put NOTE - &msg;
|
%put NOTE - &msg;
|
||||||
|
|
||||||
%if %symexist(_SYSINCLUDEFILEDEVICE) %then %do;
|
%if %symexist(_SYSINCLUDEFILEDEVICE)
|
||||||
|
/* abort cancel FILE does not restart outside the INCLUDE on Viya 3.5 */
|
||||||
|
and "&SYSPROCESSNAME " ne "Compute Server "
|
||||||
|
%then %do;
|
||||||
%if "*&_SYSINCLUDEFILEDEVICE*" ne "**" %then %do;
|
%if "*&_SYSINCLUDEFILEDEVICE*" ne "**" %then %do;
|
||||||
data &errds;
|
data &errds;
|
||||||
iftrue='1=1';
|
iftrue='1=1';
|
||||||
@@ -110,8 +113,8 @@
|
|||||||
%if %symexist(SYSPRINTTOLOG) %then %let logloc=&SYSPRINTTOLOG;
|
%if %symexist(SYSPRINTTOLOG) %then %let logloc=&SYSPRINTTOLOG;
|
||||||
%else %let logloc=%qsysfunc(getoption(LOG));
|
%else %let logloc=%qsysfunc(getoption(LOG));
|
||||||
proc printto log=log;run;
|
proc printto log=log;run;
|
||||||
|
%let logline=0;
|
||||||
%if %length(&logloc)>0 %then %do;
|
%if %length(&logloc)>0 %then %do;
|
||||||
%let logline=0;
|
|
||||||
data _null_;
|
data _null_;
|
||||||
infile &logloc lrecl=5000;
|
infile &logloc lrecl=5000;
|
||||||
input; putlog _infile_;
|
input; putlog _infile_;
|
||||||
@@ -160,7 +163,10 @@
|
|||||||
file _webout mod lrecl=32000 encoding='utf-8';
|
file _webout mod lrecl=32000 encoding='utf-8';
|
||||||
length msg $32767 ;
|
length msg $32767 ;
|
||||||
sasdatetime=datetime();
|
sasdatetime=datetime();
|
||||||
msg=cats(symget('msg'),'\n\nLog Extract:\n',symget('logmsg'));
|
msg=symget('msg');
|
||||||
|
%if &logline>0 %then %do;
|
||||||
|
msg=cats(msg,'\n\nLog Extract:\n',symget('logmsg'));
|
||||||
|
%end;
|
||||||
/* escape the quotes */
|
/* escape the quotes */
|
||||||
msg=tranwrd(msg,'"','\"');
|
msg=tranwrd(msg,'"','\"');
|
||||||
/* ditch the CRLFs as chrome complains */
|
/* ditch the CRLFs as chrome complains */
|
||||||
@@ -170,7 +176,8 @@
|
|||||||
if symexist('_debug') then debug=quote(trim(symget('_debug')));
|
if symexist('_debug') then debug=quote(trim(symget('_debug')));
|
||||||
else debug='""';
|
else debug='""';
|
||||||
put '>>weboutBEGIN<<';
|
put '>>weboutBEGIN<<';
|
||||||
put '{"START_DTTM" : "' "%sysfunc(datetime(),datetime20.3)" '"';
|
put '{"SYSDATE" : "' "&SYSDATE" '"';
|
||||||
|
put ',"SYSTIME" : "' "&SYSTIME" '"';
|
||||||
put ',"sasjsAbort" : [{';
|
put ',"sasjsAbort" : [{';
|
||||||
put ' "MSG":' msg ;
|
put ' "MSG":' msg ;
|
||||||
put ' ,"MAC": "' "&mac" '"}]';
|
put ' ,"MAC": "' "&mac" '"}]';
|
||||||
@@ -199,7 +206,7 @@
|
|||||||
put ',"SYSVLONG" : ' sysvlong;
|
put ',"SYSVLONG" : ' sysvlong;
|
||||||
syswarningtext=quote(trim(symget('syswarningtext')));
|
syswarningtext=quote(trim(symget('syswarningtext')));
|
||||||
put ",""SYSWARNINGTEXT"" : " syswarningtext;
|
put ",""SYSWARNINGTEXT"" : " syswarningtext;
|
||||||
put ',"END_DTTM" : "' "%sysfunc(datetime(),datetime20.3)" '" ';
|
put ',"END_DTTM" : "' "%sysfunc(datetime(),E8601DT26.6)" '" ';
|
||||||
put "}" @;
|
put "}" @;
|
||||||
put '>>weboutEND<<';
|
put '>>weboutEND<<';
|
||||||
run;
|
run;
|
||||||
|
|||||||
181
base/mp_applyformats.sas
Normal file
181
base/mp_applyformats.sas
Normal file
@@ -0,0 +1,181 @@
|
|||||||
|
/**
|
||||||
|
@file
|
||||||
|
@brief Apply a set of formats to a table
|
||||||
|
@details Applies a set of formats to the metadata of one or more SAS datasets.
|
||||||
|
Can be used to migrate formats from one table to another. The input table
|
||||||
|
must contain the following columns:
|
||||||
|
|
||||||
|
@li lib - the libref of the table to be updated
|
||||||
|
@li ds - the dataset to be updated
|
||||||
|
@li var - the variable to be updated
|
||||||
|
@li fmt - the format to apply. Missing or default ($CHAR, 8.) formats are
|
||||||
|
ignored.
|
||||||
|
|
||||||
|
The macro will abort in the following scenarios:
|
||||||
|
|
||||||
|
@li Libref not assigned
|
||||||
|
@li Dataset does not exist
|
||||||
|
@li Input table contains null or invalid values
|
||||||
|
|
||||||
|
Example usage:
|
||||||
|
|
||||||
|
data work.example;
|
||||||
|
set sashelp.prdsale;
|
||||||
|
format _all_ clear;
|
||||||
|
run;
|
||||||
|
|
||||||
|
%mp_getcols(sashelp.prdsale,outds=work.cols)
|
||||||
|
|
||||||
|
data work.cols2;
|
||||||
|
set work.cols;
|
||||||
|
lib='WORK';
|
||||||
|
ds='EXAMPLE';
|
||||||
|
var=name;
|
||||||
|
fmt=format;
|
||||||
|
keep lib ds var fmt;
|
||||||
|
run;
|
||||||
|
|
||||||
|
%mp_applyformats(work.cols2)
|
||||||
|
|
||||||
|
@param [in] inds The input dataset containing the formats to apply (and where
|
||||||
|
to apply them). Example structure:
|
||||||
|
|LIB:$8.|DS:$32.|VAR:$32.|FMT:$49.|
|
||||||
|
|---|---|---|---|
|
||||||
|
|`WORK `|`EXAMPLE `|`ACTUAL `|`DOLLAR12.2 `|
|
||||||
|
|`WORK `|`EXAMPLE `|`COUNTRY `|`$CHAR10. `|
|
||||||
|
|`WORK `|`EXAMPLE `|`DIVISION `|`$CHAR10. `|
|
||||||
|
|`WORK `|`EXAMPLE `|`MONTH `|`MONNAME3. `|
|
||||||
|
|`WORK `|`EXAMPLE `|`PREDICT `|`DOLLAR12.2 `|
|
||||||
|
|`WORK `|`EXAMPLE `|`PRODTYPE `|`$CHAR10. `|
|
||||||
|
|`WORK `|`EXAMPLE `|`PRODUCT `|`$CHAR10. `|
|
||||||
|
|`WORK `|`EXAMPLE `|`QUARTER `|`8. `|
|
||||||
|
|`WORK `|`EXAMPLE `|`REGION `|`$CHAR10. `|
|
||||||
|
|`WORK `|`EXAMPLE `|`YEAR `|`8. `|
|
||||||
|
|
||||||
|
@param [out] errds= (0) Provide a libds reference here to export the
|
||||||
|
error messages to a table. In this case, they will not be printed to the
|
||||||
|
log.
|
||||||
|
|
||||||
|
<h4> SAS Macros </h4>
|
||||||
|
@li mf_getengine.sas
|
||||||
|
@li mf_getuniquefileref.sas
|
||||||
|
@li mf_getuniquename.sas
|
||||||
|
@li mf_nobs.sas
|
||||||
|
@li mp_validatecol.sas
|
||||||
|
|
||||||
|
|
||||||
|
<h4> Related Macros </h4>
|
||||||
|
@li mp_getformats.sas
|
||||||
|
|
||||||
|
@version 9.2
|
||||||
|
@author Allan Bowe
|
||||||
|
|
||||||
|
**/
|
||||||
|
|
||||||
|
%macro mp_applyformats(inds,errds=0
|
||||||
|
)/*/STORE SOURCE*/;
|
||||||
|
%local outds liblist i engine lib msg ;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Validations
|
||||||
|
*/
|
||||||
|
proc sort data=&inds;
|
||||||
|
by lib ds var fmt;
|
||||||
|
run;
|
||||||
|
|
||||||
|
%if &errds=0 %then %let outds=%mf_getuniquename(prefix=mp_applyformats);
|
||||||
|
%else %let outds=&errds;
|
||||||
|
|
||||||
|
data &outds;
|
||||||
|
set &inds;
|
||||||
|
where fmt not in ('','.', '$', '$CHAR.','8.');
|
||||||
|
length msg $128;
|
||||||
|
by lib ds var fmt;
|
||||||
|
if libref(lib) ne 0 then do;
|
||||||
|
msg=catx(' ','libref',lib,'is not assigned!');
|
||||||
|
%if &errds=0 %then %do;
|
||||||
|
putlog "%str(ERR)OR: " msg;
|
||||||
|
%end;
|
||||||
|
output;
|
||||||
|
return;
|
||||||
|
end;
|
||||||
|
if exist(cats(lib,'.',ds)) ne 1 then do;
|
||||||
|
msg=catx(' ','libds',lib,'.',ds,'does not exist!');
|
||||||
|
%if &errds=0 %then %do;
|
||||||
|
putlog "%str(ERR)OR: " msg;
|
||||||
|
%end;
|
||||||
|
output;
|
||||||
|
return;
|
||||||
|
end;
|
||||||
|
%mp_validatecol(fmt,FORMAT,is_fmt)
|
||||||
|
if is_fmt=0 then do;
|
||||||
|
msg=catx(' ','format',fmt,'on libds',lib,'.',ds,'.',var,'is not valid!');
|
||||||
|
%if &errds=0 %then %do;
|
||||||
|
putlog "%str(ERR)OR: " msg;
|
||||||
|
%end;
|
||||||
|
output;
|
||||||
|
return;
|
||||||
|
end;
|
||||||
|
|
||||||
|
if first.ds then do;
|
||||||
|
retain dsid;
|
||||||
|
dsid=open(cats(lib,'.',ds));
|
||||||
|
if dsid=0 then do;
|
||||||
|
msg=catx(' ','libds',lib,'.',ds,' could not be opened!');
|
||||||
|
%if &errds=0 %then %do;
|
||||||
|
putlog "%str(ERR)OR: " msg;
|
||||||
|
%end;
|
||||||
|
output;
|
||||||
|
return;
|
||||||
|
end;
|
||||||
|
if varnum(dsid,var)<1 then do;
|
||||||
|
msg=catx(' ','Variable',lib,'.',ds,'.',var,' was not found!');
|
||||||
|
%if &errds=0 %then %do;
|
||||||
|
putlog "%str(ERR)OR: " msg;
|
||||||
|
%end;
|
||||||
|
output;
|
||||||
|
end;
|
||||||
|
end;
|
||||||
|
if last.ds then rc=close(dsid);
|
||||||
|
run;
|
||||||
|
|
||||||
|
proc sql noprint;
|
||||||
|
select distinct lib into: liblist separated by ' ' from &inds;
|
||||||
|
%put &=liblist;
|
||||||
|
%if %length(&liblist)>0 %then %do i=1 %to %sysfunc(countw(&liblist));
|
||||||
|
%let lib=%scan(&liblist,1);
|
||||||
|
%let engine=%mf_getengine(&lib);
|
||||||
|
%if &engine ne V9 and &engine ne BASE %then %do;
|
||||||
|
%let msg=&lib has &engine engine - formats cannot be applied;
|
||||||
|
insert into &outds set lib="&lib",ds="_all_",var="_all", msg="&msg" ;
|
||||||
|
%if &errds=0 %then %put %str(ERR)OR: &msg;
|
||||||
|
%end;
|
||||||
|
%end;
|
||||||
|
quit;
|
||||||
|
|
||||||
|
%if %mf_nobs(&outds)>0 %then %return;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Validations complete - now apply the actual formats!
|
||||||
|
*/
|
||||||
|
%let fref=%mf_getuniquefileref();
|
||||||
|
data _null_;
|
||||||
|
set &inds;
|
||||||
|
by lib ds var fmt;
|
||||||
|
where fmt not in ('','.', '$', '$CHAR.','8.');
|
||||||
|
file &fref;
|
||||||
|
if first.lib then put 'proc datasets nolist lib=' lib ';';
|
||||||
|
if first.ds then put ' modify ' ds ';';
|
||||||
|
put ' format ' var fmt ';';
|
||||||
|
if last.ds then put ' run;';
|
||||||
|
if last.lib then put 'quit;';
|
||||||
|
run;
|
||||||
|
|
||||||
|
%inc &fref/source2;
|
||||||
|
|
||||||
|
%if &errds=0 %then %do;
|
||||||
|
proc sql;
|
||||||
|
drop table &outds;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%mend mp_applyformats;
|
||||||
@@ -12,10 +12,6 @@
|
|||||||
|
|
||||||
%mp_assertdsobs(sashelp.class,test=ATMOST 20) %* pass if <21 obs present;
|
%mp_assertdsobs(sashelp.class,test=ATMOST 20) %* pass if <21 obs present;
|
||||||
|
|
||||||
<h4> SAS Macros </h4>
|
|
||||||
@li mf_nobs.sas
|
|
||||||
@li mp_abort.sas
|
|
||||||
|
|
||||||
|
|
||||||
@param [in] inds input dataset to test for presence of observations
|
@param [in] inds input dataset to test for presence of observations
|
||||||
@param [in] desc= (Testing observations) The user provided test description
|
@param [in] desc= (Testing observations) The user provided test description
|
||||||
@@ -33,6 +29,11 @@
|
|||||||
|---|---|---|
|
|---|---|---|
|
||||||
|User Provided description|PASS|Dataset &inds has XX obs|
|
|User Provided description|PASS|Dataset &inds has XX obs|
|
||||||
|
|
||||||
|
<h4> SAS Macros </h4>
|
||||||
|
@li mf_getuniquename.sas
|
||||||
|
@li mf_nobs.sas
|
||||||
|
@li mp_abort.sas
|
||||||
|
|
||||||
<h4> Related Macros </h4>
|
<h4> Related Macros </h4>
|
||||||
@li mp_assertcolvals.sas
|
@li mp_assertcolvals.sas
|
||||||
@li mp_assert.sas
|
@li mp_assert.sas
|
||||||
@@ -49,9 +50,10 @@
|
|||||||
outds=work.test_results
|
outds=work.test_results
|
||||||
)/*/STORE SOURCE*/;
|
)/*/STORE SOURCE*/;
|
||||||
|
|
||||||
%local nobs;
|
%local nobs ds;
|
||||||
%let nobs=%mf_nobs(&inds);
|
%let nobs=%mf_nobs(&inds);
|
||||||
%let test=%upcase(&test);
|
%let test=%upcase(&test);
|
||||||
|
%let ds=%mf_getuniquename(prefix=mp_assertdsobs);
|
||||||
|
|
||||||
%if %substr(&test.xxxxx,1,6)=EQUALS %then %do;
|
%if %substr(&test.xxxxx,1,6)=EQUALS %then %do;
|
||||||
%let val=%scan(&test,2,%str( ));
|
%let val=%scan(&test,2,%str( ));
|
||||||
@@ -84,7 +86,7 @@
|
|||||||
)
|
)
|
||||||
%end;
|
%end;
|
||||||
|
|
||||||
data;
|
data &ds;
|
||||||
length test_description $256 test_result $4 test_comments $256;
|
length test_description $256 test_result $4 test_comments $256;
|
||||||
test_description=symget('desc');
|
test_description=symget('desc');
|
||||||
test_result='FAIL';
|
test_result='FAIL';
|
||||||
@@ -110,9 +112,6 @@
|
|||||||
%end;
|
%end;
|
||||||
run;
|
run;
|
||||||
|
|
||||||
%local ds;
|
|
||||||
%let ds=&syslast;
|
|
||||||
|
|
||||||
proc append base=&outds data=&ds;
|
proc append base=&outds data=&ds;
|
||||||
run;
|
run;
|
||||||
|
|
||||||
|
|||||||
147
base/mp_assertscope.sas
Normal file
147
base/mp_assertscope.sas
Normal file
@@ -0,0 +1,147 @@
|
|||||||
|
/**
|
||||||
|
@file
|
||||||
|
@brief Used to capture scope leakage of macro variables
|
||||||
|
@details
|
||||||
|
|
||||||
|
A common 'difficult to detect' bug in macros is where a nested macro
|
||||||
|
over-writes variables in a higher level macro.
|
||||||
|
|
||||||
|
This assertion takes a snapshot of the macro variables before and after
|
||||||
|
a macro invocation. Differences are captured in the `&outds` table. This
|
||||||
|
makes it easy to detect whether any macro variables were modified or
|
||||||
|
changed.
|
||||||
|
|
||||||
|
The following variables are NOT tested (as they are known, global variables
|
||||||
|
used in SASjs):
|
||||||
|
|
||||||
|
@li &sasjs_prefix._FUNCTIONS
|
||||||
|
|
||||||
|
Global variables are initialised in mp_init.sas - which will also trigger
|
||||||
|
"strict mode" in your SAS session. Whilst this is a default in SASjs
|
||||||
|
produced apps, if you prefer not to use this mode, simply instantiate the
|
||||||
|
following variable to prevent the macro from running: `SASJS_PREFIX`
|
||||||
|
|
||||||
|
Example usage:
|
||||||
|
|
||||||
|
%mp_assertscope(SNAPSHOT)
|
||||||
|
|
||||||
|
%let oops=I did it again;
|
||||||
|
|
||||||
|
%mp_assertscope(COMPARE,
|
||||||
|
desc=Checking macro variables against previous snapshot
|
||||||
|
)
|
||||||
|
|
||||||
|
This macro is designed to work alongside `sasjs test` - for more information
|
||||||
|
about this facility, visit [cli.sasjs.io/test](https://cli.sasjs.io/test).
|
||||||
|
|
||||||
|
@param [in] action (SNAPSHOT) The action to take. Valid values:
|
||||||
|
@li SNAPSHOT - take a copy of the current macro variables
|
||||||
|
@li COMPARE - compare the current macro variables against previous values
|
||||||
|
@param [in] scope= (GLOBAL) The scope of the variables to be checked. This
|
||||||
|
corresponds to the values in the SCOPE column in `sashelp.vmacro`.
|
||||||
|
@param [in] desc= (Testing scope leakage) The user provided test description
|
||||||
|
@param [in] ignorelist= Provide a list of macro variable names to ignore from
|
||||||
|
the comparison
|
||||||
|
@param [in,out] scopeds= (work.mp_assertscope) The dataset to contain the
|
||||||
|
scope snapshot
|
||||||
|
@param [out] outds= (work.test_results) The output dataset to contain the
|
||||||
|
results. If it does not exist, it will be created, with the following format:
|
||||||
|
|TEST_DESCRIPTION:$256|TEST_RESULT:$4|TEST_COMMENTS:$256|
|
||||||
|
|---|---|---|
|
||||||
|
|User Provided description|PASS|No out of scope variables created or modified|
|
||||||
|
|
||||||
|
<h4> SAS Macros </h4>
|
||||||
|
@li mf_getquotedstr.sas
|
||||||
|
@li mp_init.sas
|
||||||
|
|
||||||
|
<h4> Related Macros </h4>
|
||||||
|
@li mp_assert.sas
|
||||||
|
@li mp_assertcols.sas
|
||||||
|
@li mp_assertcolvals.sas
|
||||||
|
@li mp_assertdsobs.sas
|
||||||
|
@li mp_assertscope.test.sas
|
||||||
|
|
||||||
|
@version 9.2
|
||||||
|
@author Allan Bowe
|
||||||
|
|
||||||
|
**/
|
||||||
|
|
||||||
|
%macro mp_assertscope(action,
|
||||||
|
desc=Testing Scope Leakage,
|
||||||
|
scope=GLOBAL,
|
||||||
|
scopeds=work.mp_assertscope,
|
||||||
|
ignorelist=,
|
||||||
|
outds=work.test_results
|
||||||
|
)/*/STORE SOURCE*/;
|
||||||
|
%local ds test_result test_comments del add mod ilist;
|
||||||
|
%let ilist=%upcase(&sasjs_prefix._FUNCTIONS SYS_PROCHTTP_STATUS_CODE
|
||||||
|
SYS_PROCHTTP_STATUS_CODE SYS_PROCHTTP_STATUS_PHRASE &ignorelist);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* this sets up the global vars, it will also enter STRICT mode. If this
|
||||||
|
* behaviour is not desired, simply initiate the following global macro
|
||||||
|
* variable to prevent the macro from running: SASJS_PREFIX
|
||||||
|
*/
|
||||||
|
%mp_init()
|
||||||
|
|
||||||
|
/* get current variables */
|
||||||
|
%if &action=SNAPSHOT %then %do;
|
||||||
|
proc sql;
|
||||||
|
create table &scopeds as
|
||||||
|
select name,offset,value
|
||||||
|
from dictionary.macros
|
||||||
|
where scope="&scope" and upcase(name) not in (%mf_getquotedstr(&ilist))
|
||||||
|
order by name,offset;
|
||||||
|
%end;
|
||||||
|
%else %if &action=COMPARE %then %do;
|
||||||
|
|
||||||
|
proc sql;
|
||||||
|
create table _data_ as
|
||||||
|
select name,offset,value
|
||||||
|
from dictionary.macros
|
||||||
|
where scope="&scope" and upcase(name) not in (%mf_getquotedstr(&ilist))
|
||||||
|
order by name,offset;
|
||||||
|
|
||||||
|
%let ds=&syslast;
|
||||||
|
|
||||||
|
proc compare
|
||||||
|
base=&scopeds(where=(upcase(name) not in (%mf_getquotedstr(&ilist))))
|
||||||
|
compare=&ds;
|
||||||
|
run;
|
||||||
|
|
||||||
|
%if &sysinfo=0 %then %do;
|
||||||
|
%let test_result=PASS;
|
||||||
|
%let test_comments=&scope Variables Unmodified;
|
||||||
|
%end;
|
||||||
|
%else %do;
|
||||||
|
proc sql noprint undo_policy=none;
|
||||||
|
select distinct name into: del separated by ' ' from &scopeds
|
||||||
|
where name not in (select name from &ds);
|
||||||
|
select distinct name into: add separated by ' ' from &ds
|
||||||
|
where name not in (select name from &scopeds);
|
||||||
|
select distinct a.name into: mod separated by ' '
|
||||||
|
from &scopeds a
|
||||||
|
inner join &ds b
|
||||||
|
on a.name=b.name
|
||||||
|
and a.offset=b.offset
|
||||||
|
where a.value ne b.value;
|
||||||
|
%let test_result=FAIL;
|
||||||
|
%let test_comments=%str(Mod:(&mod) Add:(&add) Del:(&del));
|
||||||
|
%end;
|
||||||
|
|
||||||
|
|
||||||
|
data ;
|
||||||
|
length test_description $256 test_result $4 test_comments $256;
|
||||||
|
test_description=symget('desc');
|
||||||
|
test_comments=symget('test_comments');
|
||||||
|
test_result=symget('test_result');
|
||||||
|
run;
|
||||||
|
|
||||||
|
%let ds=&syslast;
|
||||||
|
proc append base=&outds data=&ds;
|
||||||
|
run;
|
||||||
|
proc sql;
|
||||||
|
drop table &ds;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%mend mp_assertscope;
|
||||||
85
base/mp_cntlout.sas
Normal file
85
base/mp_cntlout.sas
Normal file
@@ -0,0 +1,85 @@
|
|||||||
|
/**
|
||||||
|
@file mp_cntlout.sas
|
||||||
|
@brief Creates a cntlout dataset in a consistent format
|
||||||
|
@details The dataset produced by proc format in the cntlout option will vary
|
||||||
|
according to its contents.
|
||||||
|
|
||||||
|
When dealing with formats from an ETL perspective (eg in [Data Controller for
|
||||||
|
SAS](https://datacontroller.io)), it is important that the output dataset
|
||||||
|
has a consistent model (and compariable values).
|
||||||
|
|
||||||
|
This macro makes use of mddl_sas_cntlout.sas to provide the consistent model,
|
||||||
|
and will left-align the start and end values when dealing with numeric ranges
|
||||||
|
to enable consistency when checking for differences.
|
||||||
|
|
||||||
|
usage:
|
||||||
|
|
||||||
|
%mp_cntlout(libcat=yourlib.cat,cntlout=work.formatexport)
|
||||||
|
|
||||||
|
@param [in] libcat The library.catalog reference
|
||||||
|
@param [in] fmtlist= (0) provide a space separated list of specific formats to
|
||||||
|
extract
|
||||||
|
@param [in] iftrue= (1=1) A condition under which the macro should be executed
|
||||||
|
@param [out] cntlout= (work.fmtextract) Libds reference for the output dataset
|
||||||
|
|
||||||
|
<h4> SAS Macros </h4>
|
||||||
|
@li mddl_sas_cntlout.sas
|
||||||
|
@li mf_getuniquename.sas
|
||||||
|
|
||||||
|
<h4> Related Macros </h4>
|
||||||
|
@li mf_getvarformat.sas
|
||||||
|
@li mp_getformats.sas
|
||||||
|
@li mp_loadformat.sas
|
||||||
|
@li mp_ds2fmtds.sas
|
||||||
|
|
||||||
|
@version 9.2
|
||||||
|
@author Allan Bowe
|
||||||
|
@cond
|
||||||
|
**/
|
||||||
|
|
||||||
|
%macro mp_cntlout(
|
||||||
|
iftrue=(1=1)
|
||||||
|
,libcat=
|
||||||
|
,cntlout=work.fmtextract
|
||||||
|
,fmtlist=0
|
||||||
|
)/*/STORE SOURCE*/;
|
||||||
|
%local ddlds cntlds i;
|
||||||
|
|
||||||
|
%if not(%eval(%unquote(&iftrue))) %then %return;
|
||||||
|
|
||||||
|
%let ddlds=%mf_getuniquename();
|
||||||
|
%let cntlds=%mf_getuniquename();
|
||||||
|
|
||||||
|
%mddl_sas_cntlout(libds=&ddlds)
|
||||||
|
|
||||||
|
%if %index(&libcat,-)>0 and %scan(&libcat,2,-)=FC %then %do;
|
||||||
|
%let libcat=%scan(&libcat,1,-);
|
||||||
|
%end;
|
||||||
|
|
||||||
|
proc format lib=&libcat cntlout=&cntlds;
|
||||||
|
%if "&fmtlist" ne "0" %then %do;
|
||||||
|
select
|
||||||
|
%do i=1 %to %sysfunc(countw(&fmtlist));
|
||||||
|
%scan(&fmtlist,&i,%str( ))
|
||||||
|
%end;
|
||||||
|
;
|
||||||
|
%end;
|
||||||
|
run;
|
||||||
|
|
||||||
|
data &cntlout;
|
||||||
|
if 0 then set &ddlds;
|
||||||
|
set &cntlds;
|
||||||
|
if type="N" then do;
|
||||||
|
start=cats(start);
|
||||||
|
end=cats(end);
|
||||||
|
end;
|
||||||
|
run;
|
||||||
|
proc sort;
|
||||||
|
by fmtname start;
|
||||||
|
run;
|
||||||
|
|
||||||
|
proc sql;
|
||||||
|
drop table &ddlds,&cntlds;
|
||||||
|
|
||||||
|
%mend mp_cntlout;
|
||||||
|
/** @endcond */
|
||||||
@@ -54,6 +54,8 @@
|
|||||||
|
|
||||||
/* create folders and copy content */
|
/* create folders and copy content */
|
||||||
data _null_;
|
data _null_;
|
||||||
|
length msg $200;
|
||||||
|
call missing(msg);
|
||||||
set work.&tempds;
|
set work.&tempds;
|
||||||
if _n_ = 1 then dpos+sum(length(directory),2);
|
if _n_ = 1 then dpos+sum(length(directory),2);
|
||||||
filepath2="&target/"!!substr(filepath,dpos);
|
filepath2="&target/"!!substr(filepath,dpos);
|
||||||
@@ -63,9 +65,9 @@
|
|||||||
rc1=filename(fref1,filepath,'disk','recfm=n');
|
rc1=filename(fref1,filepath,'disk','recfm=n');
|
||||||
rc2=filename(fref2,filepath2,'disk','recfm=n');
|
rc2=filename(fref2,filepath2,'disk','recfm=n');
|
||||||
if fcopy(fref1,fref2) ne 0 then do;
|
if fcopy(fref1,fref2) ne 0 then do;
|
||||||
sysmsg=sysmsg();
|
msg=sysmsg();
|
||||||
putlog "%str(ERR)OR: Unable to copy " filepath " to " filepath2;
|
putlog "%str(ERR)OR: Unable to copy " filepath " to " filepath2;
|
||||||
putlog sysmg=;
|
putlog msg=;
|
||||||
end;
|
end;
|
||||||
end;
|
end;
|
||||||
rc=filename(fref1);
|
rc=filename(fref1);
|
||||||
|
|||||||
71
base/mp_coretable.sas
Normal file
71
base/mp_coretable.sas
Normal file
@@ -0,0 +1,71 @@
|
|||||||
|
/**
|
||||||
|
@file
|
||||||
|
@brief Create the permanent Core tables
|
||||||
|
@details Several macros in the [core](https://github.com/sasjs/core) library
|
||||||
|
make use of permanent tables. To avoid duplication in definitions, this
|
||||||
|
macro provides a central location for managing the corresponding DDL.
|
||||||
|
|
||||||
|
Note - this macro is likely to be deprecated in future in favour of a
|
||||||
|
dedicated "datamodel" folder (prefix mddl)
|
||||||
|
|
||||||
|
Any corresponding data would go in a seperate repo, to avoid this one
|
||||||
|
ballooning in size!
|
||||||
|
|
||||||
|
Example usage:
|
||||||
|
|
||||||
|
%mp_coretable(LOCKTABLE,libds=work.locktable)
|
||||||
|
|
||||||
|
@param [in] table_ref The type of table to create. Example values:
|
||||||
|
@li DIFFTABLE
|
||||||
|
@li FILTER_DETAIL
|
||||||
|
@li FILTER_SUMMARY
|
||||||
|
@li LOCKANYTABLE
|
||||||
|
@li MAXKEYTABLE
|
||||||
|
@param [in] libds= (0) The library.dataset reference used to create the table.
|
||||||
|
If not provided, then the DDL is simply printed to the log.
|
||||||
|
|
||||||
|
<h4> SAS Macros </h4>
|
||||||
|
@li mddl_dc_difftable.sas
|
||||||
|
@li mddl_dc_filterdetail.sas
|
||||||
|
@li mddl_dc_filtersummary.sas
|
||||||
|
@li mddl_dc_locktable.sas
|
||||||
|
@li mddl_dc_maxkeytable.sas
|
||||||
|
|
||||||
|
<h4> Related Macros </h4>
|
||||||
|
@li mp_filterstore.sas
|
||||||
|
@li mp_lockanytable.sas
|
||||||
|
@li mp_retainedkey.sas
|
||||||
|
@li mp_storediffs.sas
|
||||||
|
@li mp_stackdiffs.sas
|
||||||
|
|
||||||
|
@version 9.2
|
||||||
|
@author Allan Bowe
|
||||||
|
|
||||||
|
**/
|
||||||
|
|
||||||
|
%macro mp_coretable(table_ref,libds=0
|
||||||
|
)/*/STORE SOURCE*/;
|
||||||
|
%local outds ;
|
||||||
|
%let outds=%sysfunc(ifc(&libds=0,_data_,&libds));
|
||||||
|
proc sql;
|
||||||
|
%if &table_ref=DIFFTABLE %then %do;
|
||||||
|
%mddl_dc_difftable(libds=&outds)
|
||||||
|
%end;
|
||||||
|
%else %if &table_ref=LOCKTABLE %then %do;
|
||||||
|
%mddl_dc_locktable(libds=&outds)
|
||||||
|
%end;
|
||||||
|
%else %if &table_ref=FILTER_SUMMARY %then %do;
|
||||||
|
%mddl_dc_filtersummary(libds=&outds)
|
||||||
|
%end;
|
||||||
|
%else %if &table_ref=FILTER_DETAIL %then %do;
|
||||||
|
%mddl_dc_filterdetail(libds=&outds)
|
||||||
|
%end;
|
||||||
|
%else %if &table_ref=MAXKEYTABLE %then %do;
|
||||||
|
%mddl_dc_maxkeytable(libds=&outds)
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%if &libds=0 %then %do;
|
||||||
|
describe table &syslast;
|
||||||
|
drop table &syslast;
|
||||||
|
%end;
|
||||||
|
%mend mp_coretable;
|
||||||
@@ -24,20 +24,22 @@ Usage:
|
|||||||
%webout(OBJ,example2) * Object format, easier to work with ;
|
%webout(OBJ,example2) * Object format, easier to work with ;
|
||||||
%webout(CLOSE)
|
%webout(CLOSE)
|
||||||
;;;;
|
;;;;
|
||||||
%mp_createwebservice(path=/Public/app/common,name=appInit,code=ft15f001,replace=YES)
|
%mp_createwebservice(path=/Public/app/common,name=appInit,replace=YES)
|
||||||
|
|
||||||
<h4> SAS Macros </h4>
|
<h4> SAS Macros </h4>
|
||||||
@li mf_getplatform.sas
|
@li mf_getplatform.sas
|
||||||
@li mm_createwebservice.sas
|
@li mm_createwebservice.sas
|
||||||
@li mv_createwebservice.sas
|
@li mv_createwebservice.sas
|
||||||
|
|
||||||
@param path= The full folder path where the service will be created
|
@param [in,out] path= The full folder path where the service will be created
|
||||||
@param name= Service name. Avoid spaces.
|
@param [in,out] name= Service name. Avoid spaces.
|
||||||
@param desc= The description of the service (optional)
|
@param [in] desc= The description of the service (optional)
|
||||||
@param precode= Space separated list of filerefs, pointing to the code that
|
@param [in] precode= Space separated list of filerefs, pointing to the code
|
||||||
needs to be attached to the beginning of the service (optional)
|
that needs to be attached to the beginning of the service (optional)
|
||||||
@param code= Space seperated fileref(s) of the actual code to be added
|
@param [in] code= (ft15f001) Space seperated fileref(s) of the actual code to
|
||||||
@param replace= select YES to replace any existing service in that location
|
be added
|
||||||
|
@param [in] replace= (YES) Select YES to replace any existing service in that
|
||||||
|
location
|
||||||
|
|
||||||
|
|
||||||
@version 9.2
|
@version 9.2
|
||||||
|
|||||||
@@ -49,10 +49,6 @@
|
|||||||
,mac=&sysmacroname
|
,mac=&sysmacroname
|
||||||
,msg=%str(the BASEDS variable must be provided)
|
,msg=%str(the BASEDS variable must be provided)
|
||||||
)
|
)
|
||||||
%mp_abort(iftrue=( &baseds=0 )
|
|
||||||
,mac=&sysmacroname
|
|
||||||
,msg=%str(the BASEDS variable must be provided)
|
|
||||||
)
|
|
||||||
%mp_abort(iftrue=( %mf_existds(&baseds)=0 )
|
%mp_abort(iftrue=( %mf_existds(&baseds)=0 )
|
||||||
,mac=&sysmacroname
|
,mac=&sysmacroname
|
||||||
,msg=%str(the BASEDS dataset (&baseds) needs to be assigned, and to exist)
|
,msg=%str(the BASEDS dataset (&baseds) needs to be assigned, and to exist)
|
||||||
|
|||||||
@@ -51,6 +51,7 @@
|
|||||||
data _null_;
|
data _null_;
|
||||||
set work.&tempds end=last;
|
set work.&tempds end=last;
|
||||||
length fref $8;
|
length fref $8;
|
||||||
|
fref='';
|
||||||
rc=filename(fref,filepath);
|
rc=filename(fref,filepath);
|
||||||
rc=fdelete(fref);
|
rc=fdelete(fref);
|
||||||
if rc then do;
|
if rc then do;
|
||||||
|
|||||||
@@ -6,8 +6,7 @@
|
|||||||
Credit for the rename approach:
|
Credit for the rename approach:
|
||||||
https://communities.sas.com/t5/SAS-Programming/SAS-Function-to-convert-string-to-Legal-SAS-Name/m-p/27375/highlight/true#M5003
|
https://communities.sas.com/t5/SAS-Programming/SAS-Function-to-convert-string-to-Legal-SAS-Name/m-p/27375/highlight/true#M5003
|
||||||
|
|
||||||
|
Usage:
|
||||||
usage:
|
|
||||||
|
|
||||||
%mp_dirlist(path=/some/location, outds=myTable, maxdepth=MAX)
|
%mp_dirlist(path=/some/location, outds=myTable, maxdepth=MAX)
|
||||||
|
|
||||||
@@ -23,12 +22,12 @@
|
|||||||
X CMD) do please raise an issue!
|
X CMD) do please raise an issue!
|
||||||
|
|
||||||
|
|
||||||
@param [in] path= for which to return contents
|
@param [in] path= (%sysfunc(pathname(work))) Path for which to return contents
|
||||||
@param [in] fref= Provide a DISK engine fileref as an alternative to PATH
|
@param [in] fref= (0) Provide a DISK engine fileref as an alternative to PATH
|
||||||
@param [in] maxdepth= (0) Set to a positive integer to indicate the level of
|
@param [in] maxdepth= (0) Set to a positive integer to indicate the level of
|
||||||
subdirectory scan recursion - eg 3, to go `./3/levels/deep`. For unlimited
|
subdirectory scan recursion - eg 3, to go `./3/levels/deep`. For unlimited
|
||||||
recursion, set to MAX.
|
recursion, set to MAX.
|
||||||
@param [out] outds= the output dataset to create
|
@param [out] outds= (work.mp_dirlist) The output dataset to create
|
||||||
@param [out] getattrs= (NO) If getattrs=YES then the doptname / foptname
|
@param [out] getattrs= (NO) If getattrs=YES then the doptname / foptname
|
||||||
functions are used to scan all properties - any characters that are not
|
functions are used to scan all properties - any characters that are not
|
||||||
valid in a SAS name (v7) are simply stripped, and the table is transposed
|
valid in a SAS name (v7) are simply stripped, and the table is transposed
|
||||||
@@ -49,13 +48,15 @@
|
|||||||
- OS SPECIFIC variables, if <code>getattrs=</code> is used.
|
- OS SPECIFIC variables, if <code>getattrs=</code> is used.
|
||||||
|
|
||||||
<h4> SAS Macros </h4>
|
<h4> SAS Macros </h4>
|
||||||
|
@li mf_existds.sas
|
||||||
|
@li mf_getvarlist.sas
|
||||||
|
@li mf_wordsinstr1butnotstr2.sas
|
||||||
@li mp_dropmembers.sas
|
@li mp_dropmembers.sas
|
||||||
|
|
||||||
<h4> Related Macros </h4>
|
<h4> Related Macros </h4>
|
||||||
@li mp_dirlist.test.sas
|
@li mp_dirlist.test.sas
|
||||||
|
|
||||||
@version 9.2
|
@version 9.2
|
||||||
@author Allan Bowe
|
|
||||||
**/
|
**/
|
||||||
|
|
||||||
%macro mp_dirlist(path=%sysfunc(pathname(work))
|
%macro mp_dirlist(path=%sysfunc(pathname(work))
|
||||||
@@ -81,7 +82,8 @@ data &out_ds(compress=no
|
|||||||
keep=file_or_folder filepath filename ext msg directory level
|
keep=file_or_folder filepath filename ext msg directory level
|
||||||
);
|
);
|
||||||
length directory filepath $500 fref fref2 $8 file_or_folder $6 filename $80
|
length directory filepath $500 fref fref2 $8 file_or_folder $6 filename $80
|
||||||
ext $20 msg $200;
|
ext $20 msg $200 foption $16;
|
||||||
|
if _n_=1 then call missing(of _all_);
|
||||||
retain level &level;
|
retain level &level;
|
||||||
%if &fref=0 %then %do;
|
%if &fref=0 %then %do;
|
||||||
rc = filename(fref, "&path");
|
rc = filename(fref, "&path");
|
||||||
@@ -92,7 +94,13 @@ data &out_ds(compress=no
|
|||||||
%end;
|
%end;
|
||||||
if rc = 0 then do;
|
if rc = 0 then do;
|
||||||
did = dopen(fref);
|
did = dopen(fref);
|
||||||
directory=dinfo(did,'Directory');
|
/* attribute is OS-dependent - could be "Directory" or "Directory Name" */
|
||||||
|
numopts=doptnum(did);
|
||||||
|
do i=1 to numopts;
|
||||||
|
foption=doptname(did,i);
|
||||||
|
if foption=:'Directory' then i=numopts;
|
||||||
|
end;
|
||||||
|
directory=dinfo(did,foption);
|
||||||
if did=0 then do;
|
if did=0 then do;
|
||||||
putlog "NOTE: This directory is empty - " directory;
|
putlog "NOTE: This directory is empty - " directory;
|
||||||
msg=sysmsg();
|
msg=sysmsg();
|
||||||
@@ -193,9 +201,29 @@ data &out_ds;
|
|||||||
set &out_ds(where=(filepath ne ''));
|
set &out_ds(where=(filepath ne ''));
|
||||||
run;
|
run;
|
||||||
|
|
||||||
/* update main table */
|
/**
|
||||||
proc append base=&outds data=&out_ds;
|
* The above transpose can mean that some updates create additional columns.
|
||||||
run;
|
* This necessitates the occasional use of datastep over proc append.
|
||||||
|
*/
|
||||||
|
%if %mf_existds(&outds) %then %do;
|
||||||
|
%local basevars appvars newvars;
|
||||||
|
%let basevars=%mf_getvarlist(&outds);
|
||||||
|
%let appvars=%mf_getvarlist(&out_ds);
|
||||||
|
%let newvars=%length(%mf_wordsinstr1butnotstr2(Str1=&appvars,Str2=&basevars));
|
||||||
|
%if &newvars>0 %then %do;
|
||||||
|
data &outds;
|
||||||
|
set &outds &out_ds;
|
||||||
|
run;
|
||||||
|
%end;
|
||||||
|
%else %do;
|
||||||
|
proc append base=&outds data=&out_ds force nowarn;
|
||||||
|
run;
|
||||||
|
%end;
|
||||||
|
%end;
|
||||||
|
%else %do;
|
||||||
|
proc append base=&outds data=&out_ds;
|
||||||
|
run;
|
||||||
|
%end;
|
||||||
|
|
||||||
/* recursive call */
|
/* recursive call */
|
||||||
%if &maxdepth>&level or &maxdepth=MAX %then %do;
|
%if &maxdepth>&level or &maxdepth=MAX %then %do;
|
||||||
|
|||||||
@@ -139,8 +139,9 @@ create table datalines1 as
|
|||||||
/**
|
/**
|
||||||
Due to long decimals cannot use best. format
|
Due to long decimals cannot use best. format
|
||||||
So - use bestd. format and then use character functions to strip trailing
|
So - use bestd. format and then use character functions to strip trailing
|
||||||
zeros, if NOT an integer!!
|
zeros, if NOT an integer or missing!! Cannot use int() as it upsets
|
||||||
resolved code = ifc(int(VARIABLE)=VARIABLE
|
note2err when there are missings.
|
||||||
|
resolved code = ifc( mod(coalesce(VARIABLE,0),1)=0
|
||||||
,put(VARIABLE,best32.)
|
,put(VARIABLE,best32.)
|
||||||
,substrn(put(VARIABLE,bestd32.),1
|
,substrn(put(VARIABLE,bestd32.),1
|
||||||
,findc(put(VARIABLE,bestd32.),'0','TBK')));
|
,findc(put(VARIABLE,bestd32.),'0','TBK')));
|
||||||
@@ -151,7 +152,7 @@ data datalines_2;
|
|||||||
set datalines1 (where=(upcase(name) not in
|
set datalines1 (where=(upcase(name) not in
|
||||||
('PROCESSED_DTTM','VALID_FROM_DTTM','VALID_TO_DTTM')));
|
('PROCESSED_DTTM','VALID_FROM_DTTM','VALID_TO_DTTM')));
|
||||||
if type='num' then dataline=
|
if type='num' then dataline=
|
||||||
cats('ifc(int(',name,')=',name,'
|
cats('ifc(mod(coalesce(',name,',0),1)=0
|
||||||
,put(',name,',best32.-l)
|
,put(',name,',best32.-l)
|
||||||
,substrn(put(',name,',bestd32.-l),1
|
,substrn(put(',name,',bestd32.-l),1
|
||||||
,findc(put(',name,',bestd32.-l),"0","TBK")))');
|
,findc(put(',name,',bestd32.-l),"0","TBK")))');
|
||||||
|
|||||||
@@ -1,23 +1,82 @@
|
|||||||
/**
|
/**
|
||||||
@file
|
@file
|
||||||
@brief Export a dataset to a CSV file
|
@brief Export a dataset to a CSV file WITH leading blanks
|
||||||
@details Export to a file or a fileref
|
@details Export a dataset to a file or fileref, retaining leading blanks.
|
||||||
|
|
||||||
|
When using SASJS headerformat, the input statement is provided in the first
|
||||||
|
row of the CSV.
|
||||||
|
|
||||||
Usage:
|
Usage:
|
||||||
|
|
||||||
%mp_ds2csv(sashelp.class,outref="%sysfunc(pathname(work))/file.csv")
|
%mp_ds2csv(sashelp.class,outref="%sysfunc(pathname(work))/file.csv")
|
||||||
|
|
||||||
@param ds The dataset to be exported
|
filename example temp;
|
||||||
@param outfile= The output filename - should be quoted.
|
%mp_ds2csv(sashelp.air,outref=example,headerformat=SASJS)
|
||||||
@param outref= The output fileref (takes precedence if provided)
|
data; infile example; input;put _infile_; if _n_>5 then stop;run;
|
||||||
@param outencoding= The output encoding to use (unquoted)
|
|
||||||
|
data _null_;
|
||||||
|
infile example;
|
||||||
|
input;
|
||||||
|
call symputx('stmnt',_infile_);
|
||||||
|
stop;
|
||||||
|
run;
|
||||||
|
data work.want;
|
||||||
|
infile example dsd firstobs=2;
|
||||||
|
input &stmnt;
|
||||||
|
run;
|
||||||
|
|
||||||
|
Why use mp_ds2csv over, say, proc export?
|
||||||
|
|
||||||
|
1. Ability to retain leading blanks (this is a major one)
|
||||||
|
2. Control the header format
|
||||||
|
3. Simple one-liner
|
||||||
|
|
||||||
|
@param [in] ds The dataset to be exported
|
||||||
|
@param [in] dlm= (COMMA) The delimeter to apply. For SASJS, will always be
|
||||||
|
COMMA. Supported values:
|
||||||
|
@li COMMA
|
||||||
|
@li SEMICOLON
|
||||||
|
@param [in] headerformat= (LABEL) The format to use for the header section.
|
||||||
|
Valid values:
|
||||||
|
@li LABEL - Use the variable label (or name, if blank)
|
||||||
|
@li NAME - Use the variable name
|
||||||
|
@li SASJS - Used to create sasjs-formatted input CSVs, eg for use in
|
||||||
|
mp_testservice.sas. This format will supply an input statement in the
|
||||||
|
first row, making ingestion by datastep a breeze. Special misisng values
|
||||||
|
will be prefixed with a period (eg `.A`) to enable ingestion on both SAS 9
|
||||||
|
and Viya. Dates / Datetimes etc are identified by the format type (lookup
|
||||||
|
with mcf_getfmttype.sas) and converted to human readable formats (not
|
||||||
|
numbers).
|
||||||
|
@param [out] outfile= The output filename - should be quoted.
|
||||||
|
@param [out] outref= (0) The output fileref (takes precedence if provided)
|
||||||
|
@param [in] outencoding= (0) The (quoted) output encoding to use, eg `"UTF-8"`
|
||||||
|
@param [in] termstr= (CRLF) The line seperator to use. For SASJS, will
|
||||||
|
always be CRLF. Valid values:
|
||||||
|
@li CRLF
|
||||||
|
@li LF
|
||||||
|
|
||||||
|
<h4> SAS Macros </h4>
|
||||||
|
@li mcf_getfmttype.sas
|
||||||
|
@li mf_getuniquename.sas
|
||||||
|
@li mf_getvarformat.sas
|
||||||
|
@li mf_getvarlist.sas
|
||||||
|
@li mf_getvartype.sas
|
||||||
|
|
||||||
@version 9.2
|
@version 9.2
|
||||||
@author Allan Bowe (credit mjsq)
|
@author Allan Bowe (credit mjsq)
|
||||||
**/
|
**/
|
||||||
|
|
||||||
%macro mp_ds2csv(ds, outref=0, outfile=, outencoding=0
|
%macro mp_ds2csv(ds
|
||||||
|
,dlm=COMMA
|
||||||
|
,outref=0
|
||||||
|
,outfile=
|
||||||
|
,outencoding=0
|
||||||
|
,headerformat=LABEL
|
||||||
|
,termstr=CRLF
|
||||||
)/*/STORE SOURCE*/;
|
)/*/STORE SOURCE*/;
|
||||||
|
|
||||||
|
%local outloc delim i varlist var vcnt vat dsv vcom vmiss fmttype vfmt;
|
||||||
|
|
||||||
%if not %sysfunc(exist(&ds)) %then %do;
|
%if not %sysfunc(exist(&ds)) %then %do;
|
||||||
%put %str(WARN)ING: &ds does not exist;
|
%put %str(WARN)ING: &ds does not exist;
|
||||||
%return;
|
%return;
|
||||||
@@ -26,33 +85,128 @@
|
|||||||
%if %index(&ds,.)=0 %then %let ds=WORK.&ds;
|
%if %index(&ds,.)=0 %then %let ds=WORK.&ds;
|
||||||
|
|
||||||
%if &outencoding=0 %then %let outencoding=;
|
%if &outencoding=0 %then %let outencoding=;
|
||||||
%else %let outencoding=encoding="&outencoding";
|
%else %let outencoding=encoding=&outencoding;
|
||||||
|
|
||||||
%local outloc;
|
|
||||||
%if &outref=0 %then %let outloc=&outfile;
|
%if &outref=0 %then %let outloc=&outfile;
|
||||||
%else %let outloc=&outref;
|
%else %let outloc=&outref;
|
||||||
|
|
||||||
|
%if &headerformat=SASJS %then %do;
|
||||||
|
%let delim=",";
|
||||||
|
%let termstr=CRLF;
|
||||||
|
%mcf_getfmttype(wrap=YES)
|
||||||
|
%end;
|
||||||
|
%else %if &dlm=COMMA %then %let delim=",";
|
||||||
|
%else %let delim=";";
|
||||||
|
|
||||||
/* credit to mjsq - https://stackoverflow.com/a/55642267 */
|
/* credit to mjsq - https://stackoverflow.com/a/55642267 */
|
||||||
|
|
||||||
/* first get headers */
|
/* first get headers */
|
||||||
data _null_;
|
data _null_;
|
||||||
file &outloc dlm=',' dsd &outencoding lrecl=32767;
|
file &outloc &outencoding lrecl=32767 termstr=&termstr;
|
||||||
length header $ 2000;
|
length header $ 2000 varnm vfmt $32 dlm $1 fmttype $8;
|
||||||
|
call missing(of _all_);
|
||||||
dsid=open("&ds.","i");
|
dsid=open("&ds.","i");
|
||||||
num=attrn(dsid,"nvars");
|
num=attrn(dsid,"nvars");
|
||||||
|
dlm=&delim;
|
||||||
do i=1 to num;
|
do i=1 to num;
|
||||||
header = trim(left(coalescec(varlabel(dsid,i),varname(dsid,i))));
|
varnm=upcase(varname(dsid,i));
|
||||||
|
if i=num then dlm='';
|
||||||
|
%if &headerformat=NAME %then %do;
|
||||||
|
header=cats(varnm,dlm);
|
||||||
|
%end;
|
||||||
|
%else %if &headerformat=LABEL %then %do;
|
||||||
|
header = cats(coalescec(varlabel(dsid,i),varnm),dlm);
|
||||||
|
%end;
|
||||||
|
%else %if &headerformat=SASJS %then %do;
|
||||||
|
if vartype(dsid,i)='C' then header=cats(varnm,':$char',varlen(dsid,i),'.');
|
||||||
|
else do;
|
||||||
|
vfmt=coalescec(varfmt(dsid,i),'0');
|
||||||
|
fmttype=mcf_getfmttype(vfmt);
|
||||||
|
if fmttype='DATE' then header=cats(varnm,':date9.');
|
||||||
|
else if fmttype='DATETIME' then header=cats(varnm,':E8601DT26.6');
|
||||||
|
else if fmttype='TIME' then header=cats(varnm,':TIME12.');
|
||||||
|
else header=cats(varnm,':best.');
|
||||||
|
end;
|
||||||
|
%end;
|
||||||
|
%else %do;
|
||||||
|
%put &sysmacroname: Invalid headerformat value (&headerformat);
|
||||||
|
%return;
|
||||||
|
%end;
|
||||||
put header @;
|
put header @;
|
||||||
end;
|
end;
|
||||||
rc=close(dsid);
|
rc=close(dsid);
|
||||||
run;
|
run;
|
||||||
|
|
||||||
|
%let varlist=%mf_getvarlist(&ds);
|
||||||
|
%let vcnt=%sysfunc(countw(&varlist));
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The $quote modifier (without a width) will take the length from the variable
|
||||||
|
* and increase by two. However this will lead to truncation where the value
|
||||||
|
* contains double quotes (which are doubled up). To get around this, scan the
|
||||||
|
* data to see the max number of double quotes, so that the appropriate width
|
||||||
|
* can be applied in the subsequent step.
|
||||||
|
*/
|
||||||
|
data _null_;
|
||||||
|
set &ds end=last;
|
||||||
|
%do i=1 %to &vcnt;
|
||||||
|
%let var=%scan(&varlist,&i);
|
||||||
|
%if %mf_getvartype(&ds,&var)=C %then %do;
|
||||||
|
%let dsv1=%mf_getuniquename(prefix=csvcol1_);
|
||||||
|
%let dsv2=%mf_getuniquename(prefix=csvcol2_);
|
||||||
|
retain &dsv1 0;
|
||||||
|
&dsv2=length(&var)+countc(&var,'"');
|
||||||
|
if &dsv2>&dsv1 then &dsv1=&dsv2;
|
||||||
|
if last then call symputx(
|
||||||
|
"vlen&i"
|
||||||
|
/* should be no shorter than varlen, and no longer than 32767 */
|
||||||
|
,cats('$quote',min(&dsv1+2,32767),'.')
|
||||||
|
,'l'
|
||||||
|
);
|
||||||
|
%end;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%let vat=@;
|
||||||
|
%let vcom=&delim;
|
||||||
|
%let vmiss=%mf_getuniquename(prefix=csvcol3_);
|
||||||
/* next, export data */
|
/* next, export data */
|
||||||
data _null_;
|
data _null_;
|
||||||
set &ds.;
|
set &ds.;
|
||||||
file &outloc mod dlm=',' dsd &outencoding lrecl=32767;
|
file &outloc mod dlm=&delim dsd &outencoding lrecl=32767 termstr=&termstr;
|
||||||
put (_all_) (+0);
|
if _n_=1 then &vmiss=' ';
|
||||||
|
%do i=1 %to &vcnt;
|
||||||
|
%let var=%scan(&varlist,&i);
|
||||||
|
%if &i=&vcnt %then %do;
|
||||||
|
%let vat=;
|
||||||
|
%let vcom=;
|
||||||
|
%end;
|
||||||
|
%if %mf_getvartype(&ds,&var)=N %then %do;
|
||||||
|
%if &headerformat = SASJS %then %do;
|
||||||
|
%let vcom=&delim;
|
||||||
|
%let fmttype=%sysfunc(mcf_getfmttype(%mf_getvarformat(&ds,&var)0));
|
||||||
|
%if &fmttype=DATE %then %let vfmt=DATE9.;
|
||||||
|
%else %if &fmttype=DATETIME %then %let vfmt=E8601DT26.6;
|
||||||
|
%else %if &fmttype=TIME %then %let vfmt=TIME12.;
|
||||||
|
%else %do;
|
||||||
|
%let vfmt=;
|
||||||
|
%let vcom=;
|
||||||
|
%end;
|
||||||
|
%end;
|
||||||
|
%else %let vcom=;
|
||||||
|
|
||||||
|
/* must use period - in order to work in both 9.4 and Viya 3.5 */
|
||||||
|
if missing(&var) and &var ne %sysfunc(getoption(MISSING)) then do;
|
||||||
|
&vmiss=cats('.',&var);
|
||||||
|
put &vmiss &vat;
|
||||||
|
end;
|
||||||
|
else put &var &vfmt &vcom &vat;
|
||||||
|
|
||||||
|
%end;
|
||||||
|
%else %do;
|
||||||
|
%if &i ne &vcnt %then %let vcom=&delim;
|
||||||
|
put &var &&vlen&i &vcom &vat;
|
||||||
|
%end;
|
||||||
|
%end;
|
||||||
run;
|
run;
|
||||||
|
|
||||||
|
|
||||||
%mend mp_ds2csv;
|
%mend mp_ds2csv;
|
||||||
30
base/mp_ds2ddl.sas
Normal file
30
base/mp_ds2ddl.sas
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
/**
|
||||||
|
@file
|
||||||
|
@brief A wrapper for mp_getddl.sas
|
||||||
|
@details In the next release, this will be the main version.
|
||||||
|
|
||||||
|
|
||||||
|
<h4> SAS Macros </h4>
|
||||||
|
@li mp_getddl.sas
|
||||||
|
|
||||||
|
**/
|
||||||
|
|
||||||
|
%macro mp_ds2ddl(libds,fref=getddl,flavour=SAS,showlog=YES,schema=
|
||||||
|
,applydttm=NO
|
||||||
|
)/*/STORE SOURCE*/;
|
||||||
|
|
||||||
|
%local libref;
|
||||||
|
%let libds=%upcase(&libds);
|
||||||
|
%let libref=%scan(&libds,1,.);
|
||||||
|
%if &libref=&libds %then %let libds=WORK.&libds;
|
||||||
|
|
||||||
|
%mp_getddl(%scan(&libds,1,.)
|
||||||
|
,%scan(&libds,2,.)
|
||||||
|
,fref=&fref
|
||||||
|
,flavour=SAS
|
||||||
|
,showlog=&showlog
|
||||||
|
,schema=&schema
|
||||||
|
,applydttm=&applydttm
|
||||||
|
)
|
||||||
|
|
||||||
|
%mend mp_ds2ddl;
|
||||||
@@ -1,16 +1,22 @@
|
|||||||
/**
|
/**
|
||||||
@file
|
@file
|
||||||
@brief Converts every value in a dataset to it's formatted value
|
@brief Converts every value in a dataset to formatted value
|
||||||
@details Converts every value to it's formatted value. All variables will
|
@details Converts every value to it's formatted value. All variables will
|
||||||
become character, and will be in the same order.
|
become character, and will be in the same order as the original dataset.
|
||||||
|
|
||||||
|
Lengths will be adjusted according to the format lengths, where applicable.
|
||||||
|
|
||||||
Usage:
|
Usage:
|
||||||
|
|
||||||
%mp_ds2fmtds(sashelp.cars,work.cars)
|
%mp_ds2fmtds(sashelp.cars,work.cars)
|
||||||
|
%mp_ds2fmtds(sashelp.vallopt,vw_vallopt)
|
||||||
|
|
||||||
@param [in] libds The library.dataset to be converted
|
@param [in] libds The library.dataset to be converted
|
||||||
@param [out] outds The dataset to create.
|
@param [out] outds The dataset to create.
|
||||||
|
|
||||||
|
<h4> SAS Macros </h4>
|
||||||
|
@li mf_existds.sas
|
||||||
|
|
||||||
<h4> Related Macros <h4>
|
<h4> Related Macros <h4>
|
||||||
@li mp_jsonout.sas
|
@li mp_jsonout.sas
|
||||||
|
|
||||||
@@ -22,8 +28,9 @@
|
|||||||
)/*/STORE SOURCE*/;
|
)/*/STORE SOURCE*/;
|
||||||
|
|
||||||
/* validations */
|
/* validations */
|
||||||
%if not %sysfunc(exist(&libds)) %then %do;
|
|
||||||
%put %str(WARN)ING: &libds does not exist;
|
%if not %mf_existds(libds=&libds) %then %do;
|
||||||
|
%put %str(WARN)ING: &libds does not exist as either a VIEW or DATASET;
|
||||||
%return;
|
%return;
|
||||||
%end;
|
%end;
|
||||||
%if %index(&libds,.)=0 %then %let libds=WORK.&libds;
|
%if %index(&libds,.)=0 %then %let libds=WORK.&libds;
|
||||||
|
|||||||
@@ -116,6 +116,7 @@ data _null_;
|
|||||||
if _n_>&maxobs then stop;
|
if _n_>&maxobs then stop;
|
||||||
%end;
|
%end;
|
||||||
length _____str $32767;
|
length _____str $32767;
|
||||||
|
call missing(_____str);
|
||||||
format _numeric_ best.;
|
format _numeric_ best.;
|
||||||
format _character_ ;
|
format _character_ ;
|
||||||
%local i comma var vtype vfmt;
|
%local i comma var vtype vfmt;
|
||||||
|
|||||||
@@ -29,44 +29,44 @@
|
|||||||
|
|
||||||
Usage:
|
Usage:
|
||||||
|
|
||||||
%mp_mdtablewrite(libds=sashelp.class,showlog=YES)
|
%mp_ds2md(sashelp.class)
|
||||||
|
|
||||||
|
@param [in] libds the library / dataset to create or read from.
|
||||||
|
@param [out] outref= (mdtable) Fileref to contain the markdown
|
||||||
|
@param [out] showlog= (YES) Set to NO to avoid printing markdown to the log
|
||||||
|
|
||||||
<h4> SAS Macros </h4>
|
<h4> SAS Macros </h4>
|
||||||
@li mf_getvarlist.sas
|
@li mf_getvarlist.sas
|
||||||
@li mf_getvarformat.sas
|
@li mf_getvarformat.sas
|
||||||
|
|
||||||
@param [in] libds= the library / dataset to create or read from.
|
|
||||||
@param [out] fref= Fileref to contain the markdown. Default=mdtable.
|
|
||||||
@param [out] showlog= set to YES to show the markdown in the log. Default=NO.
|
|
||||||
|
|
||||||
@version 9.3
|
@version 9.3
|
||||||
@author Allan Bowe
|
@author Allan Bowe
|
||||||
**/
|
**/
|
||||||
|
|
||||||
%macro mp_mdtablewrite(
|
%macro mp_ds2md(
|
||||||
libds=,
|
libds,
|
||||||
fref=mdtable,
|
outref=mdtable,
|
||||||
showlog=NO
|
showlog=YES
|
||||||
)/*/STORE SOURCE*/;
|
)/*/STORE SOURCE*/;
|
||||||
|
|
||||||
/* check fileref is assigned */
|
/* check fileref is assigned */
|
||||||
%if %sysfunc(fileref(&fref)) > 0 %then %do;
|
%if %sysfunc(fileref(&outref)) > 0 %then %do;
|
||||||
filename &fref temp;
|
filename &outref temp;
|
||||||
%end;
|
%end;
|
||||||
|
|
||||||
%local vars;
|
%local vars;
|
||||||
%let vars=%mf_getvarlist(&libds);
|
%let vars=%upcase(%mf_getvarlist(&libds));
|
||||||
|
|
||||||
/* create the header row */
|
/* create the header row */
|
||||||
data _null_;
|
data _null_;
|
||||||
file &fref;
|
file &outref;
|
||||||
length line $32767;
|
length line $32767;
|
||||||
|
call missing(line);
|
||||||
put '|'
|
put '|'
|
||||||
%local i var fmt;
|
%local i var fmt;
|
||||||
%do i=1 %to %sysfunc(countw(&vars));
|
%do i=1 %to %sysfunc(countw(&vars));
|
||||||
%let var=%scan(&vars,&i);
|
%let var=%scan(&vars,&i);
|
||||||
%let fmt=%mf_getvarformat(&libds,&var,force=1);
|
%let fmt=%lowcase(%mf_getvarformat(&libds,&var,force=1));
|
||||||
"&var:&fmt|"
|
"&var:&fmt|"
|
||||||
%end;
|
%end;
|
||||||
;
|
;
|
||||||
@@ -79,20 +79,20 @@ run;
|
|||||||
|
|
||||||
/* write out the data */
|
/* write out the data */
|
||||||
data _null_;
|
data _null_;
|
||||||
file &fref mod dlm='|' lrecl=32767;
|
file &outref mod dlm='|' lrecl=32767;
|
||||||
set &libds ;
|
set &libds ;
|
||||||
length line $32767;
|
length line $32767;
|
||||||
line=cats('|',%mf_getvarlist(&libds,dlm=%str(,'|',)),'|');
|
line='|`'!!cats(%mf_getvarlist(&libds,dlm=%str(%)!!' `|`'!!cats%()))!!' `|';
|
||||||
put line;
|
put line;
|
||||||
run;
|
run;
|
||||||
|
|
||||||
%if %upcase(&showlog)=YES %then %do;
|
%if %upcase(&showlog)=YES %then %do;
|
||||||
options ps=max;
|
options ps=max;
|
||||||
data _null_;
|
data _null_;
|
||||||
infile &fref;
|
infile &outref;
|
||||||
input;
|
input;
|
||||||
putlog _infile_;
|
putlog _infile_;
|
||||||
run;
|
run;
|
||||||
%end;
|
%end;
|
||||||
|
|
||||||
%mend mp_mdtablewrite;
|
%mend mp_ds2md;
|
||||||
120
base/mp_ds2squeeze.sas
Normal file
120
base/mp_ds2squeeze.sas
Normal file
@@ -0,0 +1,120 @@
|
|||||||
|
/**
|
||||||
|
@file
|
||||||
|
@brief Create a smaller version of a dataset, without data loss
|
||||||
|
@details This macro will scan the input dataset and create a new one, that
|
||||||
|
has the minimum variable lengths needed to store the data without data loss.
|
||||||
|
|
||||||
|
Inspiration was taken from [How to Reduce the Disk Space Required by a
|
||||||
|
SAS® Data Set](https://www.lexjansen.com/nesug/nesug06/io/io18.pdf) by
|
||||||
|
Selvaratnam Sridharma. The end of the referenced paper presents a macro named
|
||||||
|
"squeeze", hence the nomenclature.
|
||||||
|
|
||||||
|
Usage:
|
||||||
|
|
||||||
|
data big;
|
||||||
|
length my big $32000;
|
||||||
|
do i=1 to 1e4;
|
||||||
|
my=repeat('oh my',100);
|
||||||
|
big='dawg';
|
||||||
|
special=._;
|
||||||
|
output;
|
||||||
|
end;
|
||||||
|
run;
|
||||||
|
|
||||||
|
%mp_ds2squeeze(work.big,outds=work.smaller)
|
||||||
|
|
||||||
|
The following will also be printed to the log (exact values may differ
|
||||||
|
depending on your OS and COMPRESS settings):
|
||||||
|
|
||||||
|
> MP_DS2SQUEEZE: work.big was 625MB
|
||||||
|
|
||||||
|
> MP_DS2SQUEEZE: work.smaller is 5MB
|
||||||
|
|
||||||
|
@param [in] libds The library.dataset to be squeezed
|
||||||
|
@param [out] outds= (work.mp_ds2squeeze) The squeezed dataset to create
|
||||||
|
@param [in] mdebug= (0) Set to 1 to enable DEBUG messages
|
||||||
|
|
||||||
|
<h4> SAS Macros </h4>
|
||||||
|
@li mf_getfilesize.sas
|
||||||
|
@li mf_getuniquefileref.sas
|
||||||
|
@li mf_getuniquename.sas
|
||||||
|
@li mp_getmaxvarlengths.sas
|
||||||
|
|
||||||
|
<h4> Related Programs </h4>
|
||||||
|
@li mp_ds2squeeze.test.sas
|
||||||
|
|
||||||
|
@version 9.3
|
||||||
|
@author Allan Bowe
|
||||||
|
**/
|
||||||
|
|
||||||
|
%macro mp_ds2squeeze(
|
||||||
|
libds,
|
||||||
|
outds=work.mp_ds2squeeze,
|
||||||
|
mdebug=0
|
||||||
|
)/*/STORE SOURCE*/;
|
||||||
|
%local dbg source;
|
||||||
|
%if &mdebug=1 %then %do;
|
||||||
|
%put &sysmacroname entry vars:;
|
||||||
|
%put _local_;
|
||||||
|
%end;
|
||||||
|
%else %do;
|
||||||
|
%let dbg=*;
|
||||||
|
%let source=/source2;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%local optval ds fref startsize;
|
||||||
|
%let ds=%mf_getuniquename();
|
||||||
|
%let fref=%mf_getuniquefileref();
|
||||||
|
%let startsize=%mf_getfilesize(libds=&libds,format=yes);
|
||||||
|
|
||||||
|
%mp_getmaxvarlengths(&libds,outds=&ds)
|
||||||
|
|
||||||
|
data _null_;
|
||||||
|
set &ds end=last;
|
||||||
|
file &fref;
|
||||||
|
/* grab the types */
|
||||||
|
retain dsid;
|
||||||
|
if _n_=1 then dsid=open("&libds",'is');
|
||||||
|
if dsid le 0 then do;
|
||||||
|
msg=sysmsg();
|
||||||
|
put msg=;
|
||||||
|
stop;
|
||||||
|
end;
|
||||||
|
type=vartype(dsid,varnum(dsid, name));
|
||||||
|
if last then rc=close(dsid);
|
||||||
|
/* write out the length statement */
|
||||||
|
if _n_=1 then put 'length ';
|
||||||
|
length len $6;
|
||||||
|
if type='C' then do;
|
||||||
|
if maxlen=0 then len='$1';
|
||||||
|
else len=cats('$',maxlen);
|
||||||
|
end;
|
||||||
|
else do;
|
||||||
|
if maxlen=0 then len='3';
|
||||||
|
else len=cats(maxlen);
|
||||||
|
end;
|
||||||
|
put ' ' name ' ' len;
|
||||||
|
if last then put ';';
|
||||||
|
run;
|
||||||
|
|
||||||
|
/* configure varlenchk - as we are explicitly shortening the variables */
|
||||||
|
%let optval=%sysfunc(getoption(varlenchk));
|
||||||
|
options varlenchk=NOWARN;
|
||||||
|
|
||||||
|
data &outds;
|
||||||
|
%inc &fref &source;
|
||||||
|
set &libds;
|
||||||
|
run;
|
||||||
|
|
||||||
|
options varlenchk=&optval;
|
||||||
|
|
||||||
|
%if &mdebug=0 %then %do;
|
||||||
|
proc sql;
|
||||||
|
drop table &ds;
|
||||||
|
filename &fref clear;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%put &sysmacroname: &libds was &startsize;
|
||||||
|
%put &sysmacroname: &outds is %mf_getfilesize(libds=&outds,format=yes);
|
||||||
|
|
||||||
|
%mend mp_ds2squeeze;
|
||||||
@@ -6,7 +6,8 @@
|
|||||||
SYSCC to 1008 if bad records are found, and call mp_abort.sas for a
|
SYSCC to 1008 if bad records are found, and call mp_abort.sas for a
|
||||||
graceful service exit (configurable).
|
graceful service exit (configurable).
|
||||||
|
|
||||||
Used for dynamic filtering in [Data Controller for SAS®](https://datacontroller.io).
|
Used for dynamic filtering in [Data Controller for SAS®](
|
||||||
|
https://datacontroller.io).
|
||||||
|
|
||||||
Usage:
|
Usage:
|
||||||
|
|
||||||
@@ -109,7 +110,7 @@ data &outds;
|
|||||||
output;
|
output;
|
||||||
end;
|
end;
|
||||||
if mod(SUBGROUP_ID,1) ne 0 then do;
|
if mod(SUBGROUP_ID,1) ne 0 then do;
|
||||||
REASON_CD='SUBGROUP_ID should be integer, not '!!left(subgroup_id);
|
REASON_CD='SUBGROUP_ID should be integer, not '!!cats(subgroup_id);
|
||||||
putlog REASON_CD= SUBGROUP_ID=;
|
putlog REASON_CD= SUBGROUP_ID=;
|
||||||
call symputx('reason_cd',reason_cd,'l');
|
call symputx('reason_cd',reason_cd,'l');
|
||||||
call symputx('nobs',_n_,'l');
|
call symputx('nobs',_n_,'l');
|
||||||
@@ -125,9 +126,9 @@ data &outds;
|
|||||||
output;
|
output;
|
||||||
end;
|
end;
|
||||||
if OPERATOR_NM not in
|
if OPERATOR_NM not in
|
||||||
('=','>','<','<=','>=','BETWEEN','IN','NOT IN','NE','CONTAINS')
|
('=','>','<','<=','>=','BETWEEN','IN','NOT IN','NE','CONTAINS','GE','LE')
|
||||||
then do;
|
then do;
|
||||||
REASON_CD='Invalid OPERATOR_NM: '!!left(OPERATOR_NM);
|
REASON_CD='Invalid OPERATOR_NM: '!!cats(OPERATOR_NM);
|
||||||
putlog REASON_CD= OPERATOR_NM=;
|
putlog REASON_CD= OPERATOR_NM=;
|
||||||
call symputx('reason_cd',reason_cd,'l');
|
call symputx('reason_cd',reason_cd,'l');
|
||||||
call symputx('nobs',_n_,'l');
|
call symputx('nobs',_n_,'l');
|
||||||
|
|||||||
254
base/mp_filterstore.sas
Normal file
254
base/mp_filterstore.sas
Normal file
@@ -0,0 +1,254 @@
|
|||||||
|
/**
|
||||||
|
@file
|
||||||
|
@brief Checks & Stores an input filter table and returns the Filter Key
|
||||||
|
@details Used to generate a FILTER_RK from an input query dataset. This
|
||||||
|
process requires several permanent tables (names are configurable). The
|
||||||
|
benefit of storing query values at backend is to enable stored 'views' of
|
||||||
|
filtered tables at frontend (ie, when building [SAS-Powered Apps](
|
||||||
|
https://sasapps.io)). This macro is also used in [Data Controller for SAS](
|
||||||
|
https://datacontroller.io).
|
||||||
|
|
||||||
|
A more recent feature of this macro is the ability to support filter queries
|
||||||
|
on Format Catalogs. This is achieved by adding a `-FC` suffix to the `libds`
|
||||||
|
parameter - where the "ds" in this case is the catalog name.
|
||||||
|
|
||||||
|
|
||||||
|
@param [in] libds= The target dataset to be filtered (lib should be assigned).
|
||||||
|
If filtering a format catalog, add the following suffix: `-FC`.
|
||||||
|
@param [in] queryds= (WORK.FILTERQUERY) The temporary input query dataset to
|
||||||
|
be validated. Has the following format:
|
||||||
|
|GROUP_LOGIC:$3|SUBGROUP_LOGIC:$3|SUBGROUP_ID:8.|VARIABLE_NM:$32|OPERATOR_NM:$10|RAW_VALUE:$32767|
|
||||||
|
|---|---|---|---|---|---|
|
||||||
|
|AND|AND|1|SOME_BESTNUM|>|1|
|
||||||
|
|AND|AND|1|SOME_TIME|=|77333|
|
||||||
|
@param [in] filter_summary= (PERM.FILTER_SUMMARY) Permanent table containing
|
||||||
|
summary filter values. The definition is available by running
|
||||||
|
mp_coretable.sas as follows: `mp_coretable(FILTER_SUMMARY)`. Example
|
||||||
|
values:
|
||||||
|
|FILTER_RK:best.|FILTER_HASH:$32.|FILTER_TABLE:$41.|PROCESSED_DTTM:datetime19.|
|
||||||
|
|---|---|---|---|
|
||||||
|
|`1 `|`540E96F566D194AB58DD4C413C99C9DB `|`VIYA6014.MPE_TABLES `|`1956084246 `|
|
||||||
|
|`2 `|`87737DB9EEE2650F5C89956CEAD0A14F `|`VIYA6014.MPE_X_TEST `|`1956084452.1`|
|
||||||
|
|`3 `|`8048BD908DBBD83D013560734E90D394 `|`VIYA6014.MPE_TABLES `|`1956093620.6`|
|
||||||
|
@param [in] filter_detail= (PERM.FILTER_DETAIL) Permanent table containing
|
||||||
|
detailed (raw) filter values. The definition is available by running
|
||||||
|
mp_coretable.sas as follows: `mp_coretable(FILTER_DETAIL)`. Example
|
||||||
|
values:
|
||||||
|
|FILTER_HASH:$32.|FILTER_LINE:best.|GROUP_LOGIC:$3.|SUBGROUP_LOGIC:$3.|SUBGROUP_ID:best.|VARIABLE_NM:$32.|OPERATOR_NM:$12.|RAW_VALUE:$4000.|PROCESSED_DTTM:datetime19.|
|
||||||
|
|---|---|---|---|---|---|---|---|---|
|
||||||
|
|`540E96F566D194AB58DD4C413C99C9DB `|`1 `|`AND `|`AND `|`1 `|`LIBREF `|`CONTAINS `|`DC`|`1956084245.8 `|
|
||||||
|
|`540E96F566D194AB58DD4C413C99C9DB `|`2 `|`AND `|`OR `|`2 `|`DSN `|`= `|` MPE_LOCK_ANYTABLE `|`1956084245.8 `|
|
||||||
|
|`87737DB9EEE2650F5C89956CEAD0A14F `|`1 `|`AND `|`AND `|`1 `|`PRIMARY_KEY_FIELD `|`IN `|`(1,2,3) `|`1956084451.9 `|
|
||||||
|
@param [in] lock_table= (PERM.LOCK_TABLE) Permanent locking table. Used to
|
||||||
|
manage concurrent access. The definition is available by running
|
||||||
|
mp_coretable.sas as follows: `mp_coretable(LOCKTABLE)`.
|
||||||
|
@param [in] maxkeytable= (0) Optional permanent reference table used for
|
||||||
|
retained key tracking. Described in mp_retainedkey.sas.
|
||||||
|
@param [in] mdebug= set to 1 to enable DEBUG messages
|
||||||
|
@param [out] outresult= The result table with the FILTER_RK
|
||||||
|
@param [out] outquery= The original query, taken as extract after table load
|
||||||
|
|
||||||
|
|
||||||
|
<h4> SAS Macros </h4>
|
||||||
|
@li mddl_sas_cntlout.sas
|
||||||
|
@li mf_getuniquename.sas
|
||||||
|
@li mf_getvalue.sas
|
||||||
|
@li mf_islibds.sas
|
||||||
|
@li mf_nobs.sas
|
||||||
|
@li mp_abort.sas
|
||||||
|
@li mp_filtercheck.sas
|
||||||
|
@li mp_hashdataset.sas
|
||||||
|
@li mp_retainedkey.sas
|
||||||
|
|
||||||
|
<h4> Related Macros </h4>
|
||||||
|
@li mp_filtercheck.sas
|
||||||
|
@li mp_filtergenerate.sas
|
||||||
|
@li mp_filtervalidate.sas
|
||||||
|
@li mp_filterstore.test.sas
|
||||||
|
|
||||||
|
@version 9.2
|
||||||
|
@author [Allan Bowe](https://www.linkedin.com/in/allanbowe)
|
||||||
|
|
||||||
|
**/
|
||||||
|
|
||||||
|
%macro mp_filterstore(libds=,
|
||||||
|
queryds=work.filterquery,
|
||||||
|
filter_summary=PERM.FILTER_SUMMARY,
|
||||||
|
filter_detail=PERM.FILTER_DETAIL,
|
||||||
|
lock_table=PERM.LOCK_TABLE,
|
||||||
|
maxkeytable=PERM.MAXKEYTABLE,
|
||||||
|
outresult=work.result,
|
||||||
|
outquery=work.query,
|
||||||
|
mdebug=1
|
||||||
|
);
|
||||||
|
%put &sysmacroname entry vars:;
|
||||||
|
%put _local_;
|
||||||
|
|
||||||
|
%local ds0 ds1 ds2 ds3 ds4 filter_hash orig_libds;
|
||||||
|
%let libds=%upcase(&libds);
|
||||||
|
%let orig_libds=&libds;
|
||||||
|
|
||||||
|
%mp_abort(iftrue= (&syscc ne 0)
|
||||||
|
,mac=mp_filterstore
|
||||||
|
,msg=%str(syscc=&syscc on macro entry)
|
||||||
|
)
|
||||||
|
%mp_abort(iftrue= (%mf_islibds(&filter_summary)=0)
|
||||||
|
,mac=mp_filterstore
|
||||||
|
,msg=%str(Invalid filter_summary value: &filter_summary)
|
||||||
|
)
|
||||||
|
%mp_abort(iftrue= (%mf_islibds(&filter_detail)=0)
|
||||||
|
,mac=mp_filterstore
|
||||||
|
,msg=%str(Invalid filter_detail value: &filter_detail)
|
||||||
|
)
|
||||||
|
%mp_abort(iftrue= (%mf_islibds(&lock_table)=0)
|
||||||
|
,mac=mp_filterstore
|
||||||
|
,msg=%str(Invalid lock_table value: &lock_table)
|
||||||
|
)
|
||||||
|
|
||||||
|
/**
|
||||||
|
* validate query
|
||||||
|
* use format catalog export, if a format
|
||||||
|
*/
|
||||||
|
%if "%substr(&libds,%length(&libds)-2,3)"="-FC" %then %do;
|
||||||
|
%let libds=%scan(&libds,1,-); /* chop off -FC extension */
|
||||||
|
%let ds0=%mf_getuniquename(prefix=fmtds_);
|
||||||
|
%let libds=&ds0;
|
||||||
|
/*
|
||||||
|
There is no need to export the entire format catalog here - the validations
|
||||||
|
are done against the data model, not the data values. So we can simply
|
||||||
|
hardcode the structure based on the cntlout dataset.
|
||||||
|
*/
|
||||||
|
%mddl_sas_cntlout(libds=&ds0)
|
||||||
|
|
||||||
|
%end;
|
||||||
|
%mp_filtercheck(&queryds,targetds=&libds,abort=YES)
|
||||||
|
|
||||||
|
/* hash the result */
|
||||||
|
%let ds1=%mf_getuniquename(prefix=hashds);
|
||||||
|
%mp_hashdataset(&queryds,outds=&ds1,salt=&orig_libds)
|
||||||
|
%let filter_hash=%upcase(%mf_getvalue(&ds1,hashkey));
|
||||||
|
%if &mdebug=1 %then %do;
|
||||||
|
data _null_;
|
||||||
|
putlog "filter_hash=&filter_hash";
|
||||||
|
set &ds1;
|
||||||
|
putlog (_all_)(=);
|
||||||
|
run;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
/* check if data already exists for this hash */
|
||||||
|
data &outresult;
|
||||||
|
set &filter_summary;
|
||||||
|
where filter_hash="&filter_hash";
|
||||||
|
run;
|
||||||
|
|
||||||
|
%mp_abort(iftrue= (&syscc ne 0)
|
||||||
|
,mac=mp_filterstore
|
||||||
|
,msg=%str(syscc=&syscc after hash check)
|
||||||
|
)
|
||||||
|
%mp_abort(iftrue= ("&filter_hash "=" ")
|
||||||
|
,mac=mp_filterstore
|
||||||
|
,msg=%str(problem with filter_hash generation)
|
||||||
|
)
|
||||||
|
|
||||||
|
%if %mf_nobs(&outresult)=0 %then %do;
|
||||||
|
|
||||||
|
/* first update summary table */
|
||||||
|
%let ds3=%mf_getuniquename(prefix=filtersum);
|
||||||
|
data work.&ds3;
|
||||||
|
if 0 then set &filter_summary;
|
||||||
|
filter_table="&orig_libds";
|
||||||
|
filter_hash="&filter_hash";
|
||||||
|
PROCESSED_DTTM=%sysfunc(datetime());
|
||||||
|
output;
|
||||||
|
stop;
|
||||||
|
run;
|
||||||
|
|
||||||
|
%mp_lockanytable(LOCK,
|
||||||
|
lib=%scan(&filter_summary,1,.)
|
||||||
|
,ds=%scan(&filter_summary,2,.)
|
||||||
|
,ref=MP_FILTERSTORE summary update - &filter_hash
|
||||||
|
,ctl_ds=&lock_table
|
||||||
|
)
|
||||||
|
|
||||||
|
%let ds4=%mf_getuniquename(prefix=filtersumappend);
|
||||||
|
%mp_retainedkey(
|
||||||
|
base_lib=%scan(&filter_summary,1,.)
|
||||||
|
,base_dsn=%scan(&filter_summary,2,.)
|
||||||
|
,append_lib=work
|
||||||
|
,append_dsn=&ds3
|
||||||
|
,retained_key=filter_rk
|
||||||
|
,business_key=filter_hash
|
||||||
|
,maxkeytable=&maxkeytable
|
||||||
|
,locktable=&lock_table
|
||||||
|
,outds=work.&ds4
|
||||||
|
)
|
||||||
|
proc append base=&filter_summary data=&ds4;
|
||||||
|
run;
|
||||||
|
|
||||||
|
%mp_lockanytable(UNLOCK,
|
||||||
|
lib=%scan(&filter_summary,1,.)
|
||||||
|
,ds=%scan(&filter_summary,2,.)
|
||||||
|
,ref=MP_FILTERSTORE summary update - &filter_hash
|
||||||
|
,ctl_ds=&lock_table
|
||||||
|
)
|
||||||
|
|
||||||
|
%if &syscc ne 0 %then %do;
|
||||||
|
data _null_;
|
||||||
|
set &ds4;
|
||||||
|
putlog (_all_)(=);
|
||||||
|
run;
|
||||||
|
%goto err;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
data &outresult;
|
||||||
|
set &filter_summary;
|
||||||
|
where filter_hash="&filter_hash";
|
||||||
|
run;
|
||||||
|
|
||||||
|
/* Next, update detail table */
|
||||||
|
%let ds2=%mf_getuniquename(prefix=filterdetail);
|
||||||
|
data &ds2;
|
||||||
|
if 0 then set &filter_detail;
|
||||||
|
set &queryds;
|
||||||
|
format filter_hash $hex32. filter_line 8.;
|
||||||
|
filter_hash="&filter_hash";
|
||||||
|
filter_line=_n_;
|
||||||
|
PROCESSED_DTTM=%sysfunc(datetime());
|
||||||
|
run;
|
||||||
|
%mp_lockanytable(LOCK,
|
||||||
|
lib=%scan(&filter_detail,1,.)
|
||||||
|
,ds=%scan(&filter_detail,2,.)
|
||||||
|
,ref=MP_FILTERSTORE update - &filter_hash
|
||||||
|
,ctl_ds=&lock_table
|
||||||
|
)
|
||||||
|
proc append base=&filter_detail data=&ds2;
|
||||||
|
run;
|
||||||
|
|
||||||
|
%mp_lockanytable(UNLOCK,
|
||||||
|
lib=%scan(&filter_detail,1,.)
|
||||||
|
,ds=%scan(&filter_detail,2,.)
|
||||||
|
,ref=MP_FILTERSTORE detail update &filter_hash
|
||||||
|
,ctl_ds=&lock_table
|
||||||
|
)
|
||||||
|
|
||||||
|
%if &syscc ne 0 %then %do;
|
||||||
|
data _null_;
|
||||||
|
set &ds2;
|
||||||
|
putlog (_all_)(=);
|
||||||
|
run;
|
||||||
|
%goto err;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%end;
|
||||||
|
|
||||||
|
proc sort data=&filter_detail(where=(filter_hash="&filter_hash")) out=&outquery;
|
||||||
|
by filter_line;
|
||||||
|
run;
|
||||||
|
|
||||||
|
%err:
|
||||||
|
%mp_abort(iftrue= (&syscc ne 0)
|
||||||
|
,mac=mp_filterstore
|
||||||
|
,msg=%str(syscc=&syscc on macro exit)
|
||||||
|
)
|
||||||
|
|
||||||
|
%mend mp_filterstore;
|
||||||
@@ -33,8 +33,8 @@
|
|||||||
@param [in] targetds The target dataset against which to verify the query
|
@param [in] targetds The target dataset against which to verify the query
|
||||||
@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= (work.mp_filtervalidate) Output dataset containing the
|
@param [out] outds= (work.mp_filtervalidate) Output dataset containing the
|
||||||
error / warning message, if one exists. If this table contains any rows,
|
err / warning message, if one exists. If this table contains any rows,
|
||||||
there are problems!
|
there are problems!
|
||||||
|
|
||||||
<h4> SAS Macros </h4>
|
<h4> SAS Macros </h4>
|
||||||
@li mf_getuniquefileref.sas
|
@li mf_getuniquefileref.sas
|
||||||
@@ -96,8 +96,7 @@ filename &fref1 clear;
|
|||||||
run;
|
run;
|
||||||
%mp_abort(
|
%mp_abort(
|
||||||
mac=&sysmacroname,
|
mac=&sysmacroname,
|
||||||
msg=%str(Filter validation issues. ERR=%superq(SYSERRORTEXT)
|
msg=%str(Filter validation issues.)
|
||||||
, WARN=%superq(SYSWARNINGTEXT) )
|
|
||||||
)
|
)
|
||||||
%end;
|
%end;
|
||||||
%let syscc=1008;
|
%let syscc=1008;
|
||||||
|
|||||||
@@ -14,11 +14,11 @@
|
|||||||
|
|
||||||
@param ds The dataset from which to obtain column metadata
|
@param ds The dataset from which to obtain column metadata
|
||||||
@param outds= (work.cols) The output dataset to create. Sample data:
|
@param outds= (work.cols) The output dataset to create. Sample data:
|
||||||
|NAME $|LENGTH 8|VARNUM 8|LABEL $|FORMAT $49|TYPE $1 |DDTYPE $|
|
|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.|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
|
||||||
@@ -30,26 +30,27 @@
|
|||||||
**/
|
**/
|
||||||
|
|
||||||
%macro mp_getcols(ds, outds=work.cols);
|
%macro mp_getcols(ds, outds=work.cols);
|
||||||
|
%local dropds;
|
||||||
proc contents noprint data=&ds
|
proc contents noprint data=&ds
|
||||||
out=_data_ (keep=name type length label varnum format:);
|
out=_data_ (keep=name type length label varnum format:);
|
||||||
run;
|
run;
|
||||||
data &outds(keep=name type length varnum format label ddtype);
|
%let dropds=&syslast;
|
||||||
set &syslast(rename=(format=format2 type=type2));
|
data &outds(keep=name type length varnum format label ddtype fmtname);
|
||||||
|
set &dropds(rename=(format=fmtname type=type2));
|
||||||
name=upcase(name);
|
name=upcase(name);
|
||||||
if type2=2 then do;
|
if type2=2 then do;
|
||||||
length format $49.;
|
length format $49.;
|
||||||
if format2='' then format=cats('$',length,'.');
|
if fmtname='' then format=cats('$',length,'.');
|
||||||
else if formatl=0 then format=cats(format2,'.');
|
else if formatl=0 then format=cats(fmtname,'.');
|
||||||
else format=cats(format2,formatl,'.');
|
else format=cats(fmtname,formatl,'.');
|
||||||
type='C';
|
type='C';
|
||||||
ddtype='CHARACTER';
|
ddtype='CHARACTER';
|
||||||
end;
|
end;
|
||||||
else do;
|
else do;
|
||||||
if format2='' then format=cats(length,'.');
|
if fmtname='' then format=cats(length,'.');
|
||||||
else if formatl=0 then format=cats(format2,'.');
|
else if formatl=0 then format=cats(fmtname,'.');
|
||||||
else if formatd=0 then format=cats(format2,formatl,'.');
|
else if formatd=0 then format=cats(fmtname,formatl,'.');
|
||||||
else format=cats(format2,formatl,'.',formatd);
|
else format=cats(fmtname,formatl,'.',formatd);
|
||||||
type='N';
|
type='N';
|
||||||
if format=:'DATETIME' or format=:'E8601DT' then ddtype='DATETIME';
|
if format=:'DATETIME' or format=:'E8601DT' then ddtype='DATETIME';
|
||||||
else if format=:'DATE' or format=:'DDMMYY' or format=:'MMDDYY'
|
else if format=:'DATE' or format=:'DDMMYY' or format=:'MMDDYY'
|
||||||
@@ -61,5 +62,6 @@ data &outds(keep=name type length varnum format label ddtype);
|
|||||||
end;
|
end;
|
||||||
if label='' then label=name;
|
if label='' then label=name;
|
||||||
run;
|
run;
|
||||||
|
proc sql;
|
||||||
|
drop table &dropds;
|
||||||
%mend mp_getcols;
|
%mend mp_getcols;
|
||||||
@@ -48,17 +48,17 @@
|
|||||||
%let vw=%mf_getuniquename(prefix=mp_getconstraints_vw_);
|
%let vw=%mf_getuniquename(prefix=mp_getconstraints_vw_);
|
||||||
data &vw /view=&vw;
|
data &vw /view=&vw;
|
||||||
set sashelp.vcncolu;
|
set sashelp.vcncolu;
|
||||||
where TABLE_CATALOG="&lib";
|
where table_catalog="&lib";
|
||||||
|
|
||||||
/* use retain approach to reset the constraint order with each constraint */
|
/* use retain approach to reset the constraint order with each constraint */
|
||||||
length tmp $1000;
|
length tmp $1000;
|
||||||
retain tmp;
|
retain tmp;
|
||||||
drop tmp;
|
drop tmp;
|
||||||
if tmp ne catx('|',libref,table_name,constraint_type,constraint_name) then do;
|
if tmp ne catx('|',table_catalog,table_name,constraint_name) then do;
|
||||||
constraint_order=1;
|
constraint_order=1;
|
||||||
end;
|
end;
|
||||||
else constraint_order+1;
|
else constraint_order+1;
|
||||||
tmp=catx('|',libref, table_name, constraint_type,constraint_name);
|
tmp=catx('|',table_catalog, table_name,constraint_name);
|
||||||
run;
|
run;
|
||||||
|
|
||||||
/* must use SQL as proc datasets does not support length changes */
|
/* must use SQL as proc datasets does not support length changes */
|
||||||
|
|||||||
@@ -139,7 +139,7 @@ run;
|
|||||||
%let curds=%scan(&dsnlist,&x);
|
%let curds=%scan(&dsnlist,&x);
|
||||||
data _null_;
|
data _null_;
|
||||||
file &fref mod;
|
file &fref mod;
|
||||||
length nm lab $1024 typ $20;
|
length lab $1024 typ $20;
|
||||||
set &colinfo (where=(upcase(memname)="&curds")) end=last;
|
set &colinfo (where=(upcase(memname)="&curds")) end=last;
|
||||||
|
|
||||||
if _n_=1 then do;
|
if _n_=1 then do;
|
||||||
@@ -158,7 +158,7 @@ run;
|
|||||||
lab=" label="!!cats("'",tranwrd(label,"'","''"),"'");
|
lab=" label="!!cats("'",tranwrd(label,"'","''"),"'");
|
||||||
if notnull='yes' then notnul=' not null';
|
if notnull='yes' then notnul=' not null';
|
||||||
if type='char' then typ=cats('char(',length,')');
|
if type='char' then typ=cats('char(',length,')');
|
||||||
else if length ne 8 then typ='num length='!!left(length);
|
else if length ne 8 then typ='num length='!!cats(length);
|
||||||
else typ='num';
|
else typ='num';
|
||||||
put name typ fmt notnul lab;
|
put name typ fmt notnul lab;
|
||||||
run;
|
run;
|
||||||
|
|||||||
124
base/mp_getformats.sas
Normal file
124
base/mp_getformats.sas
Normal file
@@ -0,0 +1,124 @@
|
|||||||
|
/**
|
||||||
|
@file
|
||||||
|
@brief Export format definitions
|
||||||
|
@details Formats are exported from the first (if any) catalog entry in the
|
||||||
|
FMTSEARCH path.
|
||||||
|
|
||||||
|
Formats are taken from the library / dataset reference and / or a static
|
||||||
|
format list.
|
||||||
|
|
||||||
|
Example usage:
|
||||||
|
|
||||||
|
%mp_getformats(lib=sashelp,ds=prdsale,outsummary=work.dictable)
|
||||||
|
|
||||||
|
@param [in] lib= (0) The libref for which to return formats.
|
||||||
|
@todo Enable exporting of formats for an entire library
|
||||||
|
@param [in] ds= (0) The dataset from which to obtain format definitions
|
||||||
|
@param [in] fmtlist= (0) A list of additional format names
|
||||||
|
@param [out] outsummary= (work.mp_getformats_summary) Output dataset
|
||||||
|
containing summary definitions - structure taken from dictionary.formats as
|
||||||
|
follows:
|
||||||
|
|
||||||
|
|libname:$8.|memname:$32.|path:$1024.|objname:$32.|fmtname:$32.|fmttype:$1.|source:$1.|minw:best.|mind:best.|maxw:best.|maxd:best.|defw:best.|defd:best.|
|
||||||
|
|---|---|---|---|---|---|---|---|---|---|---|---|---|
|
||||||
|
| | | | |$|F|B|1|0|32767|0|1|0|
|
||||||
|
| | | | |$|I|B|1|0|32767|0|1|0|
|
||||||
|
|` `|` `|/opt/sas/sas9/SASHome/SASFoundation/9.4/sasexe|UWIANYDT|$ANYDTIF|I|U|1|0|60|0|19|0|
|
||||||
|
| | |/opt/sas/sas9/SASHome/SASFoundation/9.4/sasexe|UWFASCII|$ASCII|F|U|1|0|32767|0|1|0|
|
||||||
|
| | |/opt/sas/sas9/SASHome/SASFoundation/9.4/sasexe|UWIASCII|$ASCII|I|U|1|0|32767|0|1|0|
|
||||||
|
| | |/opt/sas/sas9/SASHome/SASFoundation/9.4/sasexe|UWFBASE6|$BASE64X|F|U|1|0|32767|0|1|0|
|
||||||
|
|
||||||
|
|
||||||
|
@param [out] outdetail= (0) Provide an output dataset in which to export all
|
||||||
|
the custom format definitions (from proc format CNTLOUT). Definitions:
|
||||||
|
https://support.sas.com/documentation/cdl/en/proc/61895/HTML/default/viewer.htm#a002473477.htm
|
||||||
|
Sample data:
|
||||||
|
|
||||||
|
|FMTNAME:$32.|START:$16.|END:$16.|LABEL:$256.|MIN:best.|MAX:best.|DEFAULT:best.|LENGTH:best.|FUZZ:best.|PREFIX:$2.|MULT:best.|FILL:$1.|NOEDIT:best.|TYPE:$1.|SEXCL:$1.|EEXCL:$1.|HLO:$13.|DECSEP:$1.|DIG3SEP:$1.|DATATYPE:$8.|LANGUAGE:$8.|
|
||||||
|
|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
|
||||||
|
|`WHICHPATH `|`0 `|`0 `|`path1 `|`1 `|`40 `|`28 `|`28 `|`1E-12 `|` `|`0 `|` `|`0 `|`N `|`N `|`N `|` `|` `|` `|` `|` `|
|
||||||
|
|`WHICHPATH `|`**OTHER** `|`**OTHER** `|`big fat problem if not path1 `|`1 `|`40 `|`28 `|`28 `|`1E-12 `|` `|`0 `|` `|`0 `|`N `|`N `|`N `|`O `|` `|` `|` `|` `|
|
||||||
|
|
||||||
|
<h4> SAS Macros </h4>
|
||||||
|
@li mddl_sas_cntlout.sas
|
||||||
|
@li mf_dedup.sas
|
||||||
|
@li mf_getfmtlist.sas
|
||||||
|
@li mf_getfmtname.sas
|
||||||
|
@li mf_getquotedstr.sas
|
||||||
|
@li mf_getuniquename.sas
|
||||||
|
|
||||||
|
|
||||||
|
<h4> Related Macros </h4>
|
||||||
|
@li mp_applyformats.sas
|
||||||
|
@li mp_getformats.test.sas
|
||||||
|
|
||||||
|
@version 9.2
|
||||||
|
@author Allan Bowe
|
||||||
|
|
||||||
|
**/
|
||||||
|
|
||||||
|
%macro mp_getformats(lib=0
|
||||||
|
,ds=0
|
||||||
|
,fmtlist=0
|
||||||
|
,outsummary=work.mp_getformats_summary
|
||||||
|
,outdetail=0
|
||||||
|
);
|
||||||
|
|
||||||
|
%local i fmt allfmts tempds fmtcnt;
|
||||||
|
|
||||||
|
%if "&fmtlist" ne "0" %then %do i=1 %to %sysfunc(countw(&fmtlist,,%str( )));
|
||||||
|
/* ensure format list contains format _name_ only */
|
||||||
|
%let fmt=%scan(&fmtlist,&i,%str( ));
|
||||||
|
%let fmt=%mf_getfmtname(&fmt);
|
||||||
|
%let allfmts=&allfmts &fmt;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%if &ds=0 and &lib ne 0 %then %do;
|
||||||
|
/* grab formats from library */
|
||||||
|
/* to do */
|
||||||
|
%end;
|
||||||
|
%else %if &ds ne 0 and &lib ne 0 %then %do;
|
||||||
|
/* grab formats from dataset */
|
||||||
|
%let allfmts=%mf_getfmtlist(&lib..&ds) &allfmts;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
/* ensure list is unique */
|
||||||
|
%let allfmts=%mf_dedup(%upcase(&allfmts));
|
||||||
|
|
||||||
|
/* create summary table */
|
||||||
|
%if %index(&outsummary,.)=0 %then %let outsummary=WORK.&outsummary;
|
||||||
|
proc sql;
|
||||||
|
create table &outsummary as
|
||||||
|
select * from dictionary.formats
|
||||||
|
where fmtname in (%mf_getquotedstr(&allfmts,quote=D))
|
||||||
|
and fmttype='F';
|
||||||
|
|
||||||
|
%if "&outdetail" ne "0" %then %do;
|
||||||
|
/* ensure base table always exists */
|
||||||
|
%mddl_sas_cntlout(libds=&outdetail)
|
||||||
|
/* grab the location of each format */
|
||||||
|
%let fmtcnt=0;
|
||||||
|
data _null_;
|
||||||
|
set &outsummary;
|
||||||
|
if not missing(libname);
|
||||||
|
x+1;
|
||||||
|
call symputx(cats('fmtloc',x),cats(libname,'.',memname),'l');
|
||||||
|
call symputx(cats('fmtname',x),fmtname,'l');
|
||||||
|
call symputx('fmtcnt',x,'l');
|
||||||
|
run;
|
||||||
|
/* export each format and append to the output table */
|
||||||
|
%let tempds=%mf_getuniquename(prefix=mp_getformats);
|
||||||
|
%do i=1 %to &fmtcnt;
|
||||||
|
proc format library=&&fmtloc&i CNTLOUT=&tempds;
|
||||||
|
select &&fmtname&i;
|
||||||
|
run;
|
||||||
|
data &tempds;
|
||||||
|
if 0 then set &outdetail;
|
||||||
|
set &tempds;
|
||||||
|
run;
|
||||||
|
proc append base=&outdetail data=&tempds ;
|
||||||
|
run;
|
||||||
|
%end;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%mend mp_getformats;
|
||||||
@@ -1,28 +1,46 @@
|
|||||||
/**
|
/**
|
||||||
@file mp_getmaxvarlengths.sas
|
@file
|
||||||
@brief Scans a dataset to find the max length of the variable values
|
@brief Scans a dataset to find the max length of the variable values
|
||||||
@details
|
@details
|
||||||
This macro will scan a base dataset and produce an output dataset with two
|
This macro will scan a base dataset and produce an output dataset with two
|
||||||
columns:
|
columns:
|
||||||
|
|
||||||
- NAME Name of the base dataset column
|
- NAME Name of the base dataset column
|
||||||
- MAXLEN Maximum length of the data contained therein.
|
- MAXLEN Maximum length of the data contained therein.
|
||||||
|
|
||||||
Character fields may be allocated very large widths (eg 32000) of which the
|
Character fields are often allocated very large widths (eg 32000) of which the
|
||||||
maximum value is likely to be much narrower. This macro was designed to
|
maximum value is likely to be much narrower. Identifying such cases can be
|
||||||
enable a HTML table to be appropriately sized however this could be used as
|
helpful in the following scenarios:
|
||||||
part of a data audit to ensure we aren't over-sizing our tables in relation to
|
|
||||||
the data therein.
|
@li Enabling a HTML table to be appropriately sized (`num2char=YES`)
|
||||||
|
@li Reducing the size of a dataset to save on storage (mp_ds2squeeze.sas)
|
||||||
|
@li Identifying columns containing nothing but missing values (`MAXLEN=0` in
|
||||||
|
the output table)
|
||||||
|
|
||||||
|
If the entire column is made up of (non-special) missing values then a value
|
||||||
|
of 0 is returned.
|
||||||
|
|
||||||
Numeric fields are converted using the relevant format to determine the width.
|
|
||||||
Usage:
|
Usage:
|
||||||
|
|
||||||
%mp_getmaxvarlengths(sashelp.class,outds=work.myds)
|
%mp_getmaxvarlengths(sashelp.class,outds=work.myds)
|
||||||
|
|
||||||
@param libds Two part dataset (or view) reference.
|
@param [in] libds Two part dataset (or view) reference.
|
||||||
@param outds= The output dataset to create
|
@param [in] num2char= (NO) When set to NO, numeric fields are sized according
|
||||||
|
to the number of bytes used (or set to zero in the case of non-special
|
||||||
|
missings). When YES, the numeric field is converted to character (using the
|
||||||
|
format, if available), and that is sized instead, using `lengthn()`.
|
||||||
|
@param [out] outds= The output dataset to create, eg:
|
||||||
|
|NAME:$8.|MAXLEN:best.|
|
||||||
|
|---|---|
|
||||||
|
|`Name `|`7 `|
|
||||||
|
|`Sex `|`1 `|
|
||||||
|
|`Age `|`3 `|
|
||||||
|
|`Height `|`8 `|
|
||||||
|
|`Weight `|`3 `|
|
||||||
|
|
||||||
<h4> SAS Macros </h4>
|
<h4> SAS Macros </h4>
|
||||||
|
@li mcf_length.sas
|
||||||
|
@li mf_getuniquename.sas
|
||||||
@li mf_getvarlist.sas
|
@li mf_getvarlist.sas
|
||||||
@li mf_getvartype.sas
|
@li mf_getvartype.sas
|
||||||
@li mf_getvarformat.sas
|
@li mf_getvarformat.sas
|
||||||
@@ -30,20 +48,32 @@
|
|||||||
@version 9.2
|
@version 9.2
|
||||||
@author Allan Bowe
|
@author Allan Bowe
|
||||||
|
|
||||||
|
<h4> Related Macros </h4>
|
||||||
|
@li mp_ds2squeeze.sas
|
||||||
|
@li mp_getmaxvarlengths.test.sas
|
||||||
|
|
||||||
**/
|
**/
|
||||||
|
|
||||||
%macro mp_getmaxvarlengths(
|
%macro mp_getmaxvarlengths(
|
||||||
libds /* libref.dataset to analyse */
|
libds
|
||||||
,outds=work.mp_getmaxvarlengths /* name of output dataset to create */
|
,num2char=NO
|
||||||
|
,outds=work.mp_getmaxvarlengths
|
||||||
)/*/STORE SOURCE*/;
|
)/*/STORE SOURCE*/;
|
||||||
|
|
||||||
%local vars x var fmt;
|
%local vars prefix x var fmt;
|
||||||
%let vars=%mf_getvarlist(libds=&libds);
|
%let vars=%mf_getvarlist(libds=&libds);
|
||||||
|
%let prefix=%substr(%mf_getuniquename(),1,25);
|
||||||
|
%let num2char=%upcase(&num2char);
|
||||||
|
|
||||||
|
%if &num2char=NO %then %do;
|
||||||
|
/* compile length function for numeric fields */
|
||||||
|
%mcf_length(wrap=YES, insert_cmplib=YES)
|
||||||
|
%end;
|
||||||
|
|
||||||
proc sql;
|
proc sql;
|
||||||
create table &outds (rename=(
|
create table &outds (rename=(
|
||||||
%do x=1 %to %sysfunc(countw(&vars,%str( )));
|
%do x=1 %to %sysfunc(countw(&vars,%str( )));
|
||||||
________&x=%scan(&vars,&x)
|
&prefix.&x=%scan(&vars,&x)
|
||||||
%end;
|
%end;
|
||||||
))
|
))
|
||||||
as select
|
as select
|
||||||
@@ -51,18 +81,21 @@ create table &outds (rename=(
|
|||||||
%let var=%scan(&vars,&x);
|
%let var=%scan(&vars,&x);
|
||||||
%if &x>1 %then ,;
|
%if &x>1 %then ,;
|
||||||
%if %mf_getvartype(&libds,&var)=C %then %do;
|
%if %mf_getvartype(&libds,&var)=C %then %do;
|
||||||
max(length(&var)) as ________&x
|
max(lengthn(&var)) as &prefix.&x
|
||||||
%end;
|
%end;
|
||||||
%else %do;
|
%else %if &num2char=YES %then %do;
|
||||||
%let fmt=%mf_getvarformat(&libds,&var);
|
%let fmt=%mf_getvarformat(&libds,&var);
|
||||||
%put fmt=&fmt;
|
%put fmt=&fmt;
|
||||||
%if %str(&fmt)=%str() %then %do;
|
%if %str(&fmt)=%str() %then %do;
|
||||||
max(length(cats(&var))) as ________&x
|
max(lengthn(cats(&var))) as &prefix.&x
|
||||||
%end;
|
%end;
|
||||||
%else %do;
|
%else %do;
|
||||||
max(length(put(&var,&fmt))) as ________&x
|
max(lengthn(put(&var,&fmt))) as &prefix.&x
|
||||||
%end;
|
%end;
|
||||||
%end;
|
%end;
|
||||||
|
%else %do;
|
||||||
|
max(mcf_length(&var)) as &prefix.&x
|
||||||
|
%end;
|
||||||
%end;
|
%end;
|
||||||
from &libds;
|
from &libds;
|
||||||
|
|
||||||
|
|||||||
@@ -17,16 +17,21 @@
|
|||||||
%inc mc;
|
%inc mc;
|
||||||
%mp_guesspk(sashelp.class,outds=classpks)
|
%mp_guesspk(sashelp.class,outds=classpks)
|
||||||
|
|
||||||
@param baseds The dataset to analyse
|
@param [in] baseds The dataset to analyse
|
||||||
@param outds= The output dataset to contain the possible PKs
|
@param [out] outds= The output dataset to contain the possible PKs
|
||||||
@param max_guesses= (3) The total number of possible primary keys to generate.
|
@param [in] max_guesses= (3) The total number of possible primary keys to
|
||||||
A table may have multiple unlikely PKs, so no need to list them all.
|
generate. A table may have multiple (unlikely) PKs, so no need to list them
|
||||||
@param min_rows= (5) The minimum number of rows a table should have in order
|
all.
|
||||||
to try and guess the PK.
|
@param [in] min_rows= (5) The minimum number of rows a table should have in
|
||||||
|
order to try and guess the PK.
|
||||||
|
@param [in] ignore_cols (0) Space seperated list of columns which you are
|
||||||
|
sure are not part of the primary key (helps to avoid false positives)
|
||||||
|
@param [in] mdebug= Set to 1 to enable DEBUG messages and preserve outputs
|
||||||
|
|
||||||
<h4> SAS Macros </h4>
|
<h4> SAS Macros </h4>
|
||||||
@li mf_getvarlist.sas
|
@li mf_getvarlist.sas
|
||||||
@li mf_getuniquename.sas
|
@li mf_getuniquename.sas
|
||||||
|
@li mf_wordsInstr1butnotstr2.sas
|
||||||
@li mf_nobs.sas
|
@li mf_nobs.sas
|
||||||
|
|
||||||
<h4> Related Macros </h4>
|
<h4> Related Macros </h4>
|
||||||
@@ -38,179 +43,226 @@
|
|||||||
**/
|
**/
|
||||||
|
|
||||||
%macro mp_guesspk(baseds
|
%macro mp_guesspk(baseds
|
||||||
,outds=mp_guesspk
|
,outds=mp_guesspk
|
||||||
,max_guesses=3
|
,max_guesses=3
|
||||||
,min_rows=5
|
,min_rows=5
|
||||||
|
,ignore_cols=0
|
||||||
|
,mdebug=0
|
||||||
)/*/STORE SOURCE*/;
|
)/*/STORE SOURCE*/;
|
||||||
|
%local dbg;
|
||||||
|
%if &mdebug=1 %then %do;
|
||||||
|
%put &sysmacroname entry vars:;
|
||||||
|
%put _local_;
|
||||||
|
%end;
|
||||||
|
%else %let dbg=*;
|
||||||
|
|
||||||
/* declare local vars */
|
/* declare local vars */
|
||||||
%local var vars vcnt i j k l tmpvar tmpds rows posspks ppkcnt;
|
%local var vars vcnt i j k l tmpvar tmpds rows posspks ppkcnt;
|
||||||
%let vars=%mf_getvarlist(&baseds);
|
%let vars=%upcase(%mf_getvarlist(&baseds));
|
||||||
%let vcnt=%sysfunc(countw(&vars));
|
%let vars=%mf_wordsInStr1ButNotStr2(str1=&vars,str2=%upcase(&ignore_cols));
|
||||||
|
%let vcnt=%sysfunc(countw(&vars));
|
||||||
|
|
||||||
%if &vcnt=0 %then %do;
|
%if &vcnt=0 %then %do;
|
||||||
%put &sysmacroname: &baseds has no variables! Exiting.;
|
%put &sysmacroname: &baseds has no variables! Exiting.;
|
||||||
%return;
|
%return;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
/* get null count and row count */
|
||||||
|
%let tmpvar=%mf_getuniquename();
|
||||||
|
proc sql noprint;
|
||||||
|
create table _data_ as select
|
||||||
|
count(*) as &tmpvar
|
||||||
|
%do i=1 %to &vcnt;
|
||||||
|
%let var=%scan(&vars,&i);
|
||||||
|
,sum(case when &var is missing then 1 else 0 end) as &var
|
||||||
|
%end;
|
||||||
|
from &baseds;
|
||||||
|
|
||||||
|
/* transpose table and scan for not null cols */
|
||||||
|
proc transpose;
|
||||||
|
data _null_;
|
||||||
|
set &syslast end=last;
|
||||||
|
length vars $32767;
|
||||||
|
retain vars ;
|
||||||
|
if _name_="&tmpvar" then call symputx('rows',col1,'l');
|
||||||
|
else if col1=0 then vars=catx(' ',vars,_name_);
|
||||||
|
if last then call symputx('posspks',vars,'l');
|
||||||
|
run;
|
||||||
|
|
||||||
|
%let ppkcnt=%sysfunc(countw(&posspks));
|
||||||
|
%if &ppkcnt=0 %then %do;
|
||||||
|
%put &sysmacroname: &baseds has no non-missing variables! Exiting.;
|
||||||
|
%return;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
proc sort data=&baseds(keep=&posspks) out=_data_ noduprec;
|
||||||
|
by _all_;
|
||||||
|
run;
|
||||||
|
%local pkds; %let pkds=&syslast;
|
||||||
|
|
||||||
|
%if &rows > %mf_nobs(&pkds) %then %do;
|
||||||
|
%put &sysmacroname: &baseds has no combination of unique records! Exiting.;
|
||||||
|
%return;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
/* now check cardinality */
|
||||||
|
proc sql noprint;
|
||||||
|
create table _data_ as select
|
||||||
|
%do i=1 %to &ppkcnt;
|
||||||
|
%let var=%scan(&posspks,&i);
|
||||||
|
count(distinct &var) as &var
|
||||||
|
%if &i<&ppkcnt %then ,;
|
||||||
|
%end;
|
||||||
|
from &pkds;
|
||||||
|
|
||||||
|
/* transpose and sort by cardinality */
|
||||||
|
proc transpose;
|
||||||
|
proc sort; by descending col1;
|
||||||
|
run;
|
||||||
|
|
||||||
|
/* create initial PK list and re-order posspks list */
|
||||||
|
data &outds(keep=pkguesses);
|
||||||
|
length pkguesses $5000 vars $5000;
|
||||||
|
set &syslast end=last;
|
||||||
|
retain vars ;
|
||||||
|
vars=catx(' ',vars,_name_);
|
||||||
|
if col1=&rows then do;
|
||||||
|
pkguesses=_name_;
|
||||||
|
output;
|
||||||
|
end;
|
||||||
|
if last then call symputx('posspks',vars,'l');
|
||||||
|
run;
|
||||||
|
|
||||||
|
%if %mf_nobs(&outds) ge &max_guesses %then %do;
|
||||||
|
%put &sysmacroname: %mf_nobs(&outds) possible primary key values found;
|
||||||
|
%return;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%if &ppkcnt=1 %then %do;
|
||||||
|
%put &sysmacroname: No more PK guess possible;
|
||||||
|
%return;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
/* begin scanning for uniques on pairs of PKs */
|
||||||
|
%let tmpds=%mf_getuniquename();
|
||||||
|
%local lev1 lev2;
|
||||||
|
%do i=1 %to &ppkcnt;
|
||||||
|
%let lev1=%scan(&posspks,&i);
|
||||||
|
%do j=2 %to &ppkcnt;
|
||||||
|
%let lev2=%scan(&posspks,&j);
|
||||||
|
%if &lev1 ne &lev2 %then %do;
|
||||||
|
/* check for two level uniqueness */
|
||||||
|
proc sort data=&pkds(keep=&lev1 &lev2) out=&tmpds noduprec;
|
||||||
|
by _all_;
|
||||||
|
run;
|
||||||
|
%if %mf_nobs(&tmpds)=&rows %then %do;
|
||||||
|
proc sql;
|
||||||
|
insert into &outds values("&lev1 &lev2");
|
||||||
|
%if %mf_nobs(&outds) ge &max_guesses %then %do;
|
||||||
|
%put &sysmacroname: Max PKs reached at Level 2 for &baseds;
|
||||||
|
%goto exit;
|
||||||
|
%end;
|
||||||
|
%end;
|
||||||
|
%end;
|
||||||
%end;
|
%end;
|
||||||
|
%end;
|
||||||
|
|
||||||
/* get null count and row count */
|
%if &ppkcnt=2 %then %do;
|
||||||
%let tmpvar=%mf_getuniquename();
|
%put &sysmacroname: No more PK guess possible;
|
||||||
proc sql noprint;
|
%goto exit;
|
||||||
create table _data_ as select
|
%end;
|
||||||
count(*) as &tmpvar
|
|
||||||
%do i=1 %to &vcnt;
|
|
||||||
%let var=%scan(&vars,&i);
|
|
||||||
,sum(case when &var is missing then 1 else 0 end) as &var
|
|
||||||
%end;
|
|
||||||
from &baseds;
|
|
||||||
|
|
||||||
/* transpose table and scan for not null cols */
|
/* begin scanning for uniques on PK triplets */
|
||||||
proc transpose;
|
%local lev3;
|
||||||
data _null_;
|
%do i=1 %to &ppkcnt;
|
||||||
set &syslast end=last;
|
%let lev1=%scan(&posspks,&i);
|
||||||
length vars $32767;
|
%do j=2 %to &ppkcnt;
|
||||||
retain vars ;
|
%let lev2=%scan(&posspks,&j);
|
||||||
if _name_="&tmpvar" then call symputx('rows',col1,'l');
|
%if &lev1 ne &lev2 %then %do k=3 %to &ppkcnt;
|
||||||
else if col1=0 then vars=catx(' ',vars,_name_);
|
%let lev3=%scan(&posspks,&k);
|
||||||
if last then call symputx('posspks',vars,'l');
|
%if &lev1 ne &lev3 and &lev2 ne &lev3 %then %do;
|
||||||
run;
|
/* check for three level uniqueness */
|
||||||
|
proc sort data=&pkds(keep=&lev1 &lev2 &lev3) out=&tmpds noduprec;
|
||||||
%let ppkcnt=%sysfunc(countw(&posspks));
|
|
||||||
%if &ppkcnt=0 %then %do;
|
|
||||||
%put &sysmacroname: &baseds has no non-missing variables! Exiting.;
|
|
||||||
%return;
|
|
||||||
%end;
|
|
||||||
|
|
||||||
proc sort data=&baseds(keep=&posspks) out=_data_ noduprec;
|
|
||||||
by _all_;
|
|
||||||
run;
|
|
||||||
%local pkds; %let pkds=&syslast;
|
|
||||||
|
|
||||||
%if &rows > %mf_nobs(&pkds) %then %do;
|
|
||||||
%put &sysmacroname: &baseds has no combination of unique records! Exiting.;
|
|
||||||
%return;
|
|
||||||
%end;
|
|
||||||
|
|
||||||
/* now check cardinality */
|
|
||||||
proc sql noprint;
|
|
||||||
create table _data_ as select
|
|
||||||
%do i=1 %to &ppkcnt;
|
|
||||||
%let var=%scan(&posspks,&i);
|
|
||||||
count(distinct &var) as &var
|
|
||||||
%if &i<&ppkcnt %then ,;
|
|
||||||
%end;
|
|
||||||
from &pkds;
|
|
||||||
|
|
||||||
/* transpose and sort by cardinality */
|
|
||||||
proc transpose;
|
|
||||||
proc sort; by descending col1;
|
|
||||||
run;
|
|
||||||
|
|
||||||
/* create initial PK list and re-order posspks list */
|
|
||||||
data &outds(keep=pkguesses);
|
|
||||||
length pkguesses $5000 vars $5000;
|
|
||||||
set &syslast end=last;
|
|
||||||
retain vars ;
|
|
||||||
vars=catx(' ',vars,_name_);
|
|
||||||
if col1=&rows then do;
|
|
||||||
pkguesses=_name_;
|
|
||||||
output;
|
|
||||||
end;
|
|
||||||
if last then call symputx('posspks',vars,'l');
|
|
||||||
run;
|
|
||||||
|
|
||||||
%if %mf_nobs(&outds) ge &max_guesses %then %do;
|
|
||||||
%put &sysmacroname: %mf_nobs(&outds) possible primary key values found;
|
|
||||||
%return;
|
|
||||||
%end;
|
|
||||||
|
|
||||||
%if &ppkcnt=1 %then %do;
|
|
||||||
%put &sysmacroname: No more PK guess possible;
|
|
||||||
%return;
|
|
||||||
%end;
|
|
||||||
|
|
||||||
/* begin scanning for uniques on pairs of PKs */
|
|
||||||
%let tmpds=%mf_getuniquename();
|
|
||||||
%local lev1 lev2;
|
|
||||||
%do i=1 %to &ppkcnt;
|
|
||||||
%let lev1=%scan(&posspks,&i);
|
|
||||||
%do j=2 %to &ppkcnt;
|
|
||||||
%let lev2=%scan(&posspks,&j);
|
|
||||||
%if &lev1 ne &lev2 %then %do;
|
|
||||||
/* check for two level uniqueness */
|
|
||||||
proc sort data=&pkds(keep=&lev1 &lev2) out=&tmpds noduprec;
|
|
||||||
by _all_;
|
by _all_;
|
||||||
run;
|
run;
|
||||||
%if %mf_nobs(&tmpds)=&rows %then %do;
|
%if %mf_nobs(&tmpds)=&rows %then %do;
|
||||||
proc sql;
|
proc sql;
|
||||||
insert into &outds values("&lev1 &lev2");
|
insert into &outds values("&lev1 &lev2 &lev3");
|
||||||
%if %mf_nobs(&outds) ge &max_guesses %then %do;
|
%if %mf_nobs(&outds) ge &max_guesses %then %do;
|
||||||
%put &sysmacroname: Max PKs reached at Level 2 for &baseds;
|
%put &sysmacroname: Max PKs reached at Level 3 for &baseds;
|
||||||
%return;
|
%goto exit;
|
||||||
%end;
|
%end;
|
||||||
%end;
|
%end;
|
||||||
%end;
|
%end;
|
||||||
%end;
|
%end;
|
||||||
%end;
|
%end;
|
||||||
|
%end;
|
||||||
|
|
||||||
%if &ppkcnt=2 %then %do;
|
%if &ppkcnt=3 %then %do;
|
||||||
%put &sysmacroname: No more PK guess possible;
|
%put &sysmacroname: No more PK guess possible;
|
||||||
%return;
|
%goto exit;
|
||||||
%end;
|
%end;
|
||||||
|
|
||||||
/* begin scanning for uniques on PK triplets */
|
/* scan for uniques on up to 4 PK fields */
|
||||||
%local lev3;
|
%local lev4;
|
||||||
%do i=1 %to &ppkcnt;
|
%do i=1 %to &ppkcnt;
|
||||||
%let lev1=%scan(&posspks,&i);
|
%let lev1=%scan(&posspks,&i);
|
||||||
%do j=2 %to &ppkcnt;
|
%do j=2 %to &ppkcnt;
|
||||||
%let lev2=%scan(&posspks,&j);
|
%let lev2=%scan(&posspks,&j);
|
||||||
%if &lev1 ne &lev2 %then %do k=3 %to &ppkcnt;
|
%if &lev1 ne &lev2 %then %do k=3 %to &ppkcnt;
|
||||||
%let lev3=%scan(&posspks,&k);
|
%let lev3=%scan(&posspks,&k);
|
||||||
%if &lev1 ne &lev3 and &lev2 ne &lev3 %then %do;
|
%if &lev1 ne &lev3 and &lev2 ne &lev3 %then %do l=4 %to &ppkcnt;
|
||||||
/* check for three level uniqueness */
|
%let lev4=%scan(&posspks,&l);
|
||||||
proc sort data=&pkds(keep=&lev1 &lev2 &lev3) out=&tmpds noduprec;
|
%if &lev1 ne &lev4 and &lev2 ne &lev4 and &lev3 ne &lev4 %then %do;
|
||||||
|
/* check for four level uniqueness */
|
||||||
|
proc sort data=&pkds(keep=&lev1 &lev2 &lev3 &lev4)
|
||||||
|
out=&tmpds noduprec;
|
||||||
by _all_;
|
by _all_;
|
||||||
run;
|
run;
|
||||||
%if %mf_nobs(&tmpds)=&rows %then %do;
|
%if %mf_nobs(&tmpds)=&rows %then %do;
|
||||||
proc sql;
|
proc sql;
|
||||||
insert into &outds values("&lev1 &lev2 &lev3");
|
insert into &outds values("&lev1 &lev2 &lev3 &lev4");
|
||||||
%if %mf_nobs(&outds) ge &max_guesses %then %do;
|
%if %mf_nobs(&outds) ge &max_guesses %then %do;
|
||||||
%put &sysmacroname: Max PKs reached at Level 3 for &baseds;
|
%put &sysmacroname: Max PKs reached at Level 4 for &baseds;
|
||||||
%return;
|
%goto exit;
|
||||||
%end;
|
%end;
|
||||||
%end;
|
%end;
|
||||||
%end;
|
%end;
|
||||||
%end;
|
%end;
|
||||||
%end;
|
%end;
|
||||||
%end;
|
%end;
|
||||||
|
%end;
|
||||||
|
|
||||||
%if &ppkcnt=3 %then %do;
|
%if &ppkcnt=4 %then %do;
|
||||||
%put &sysmacroname: No more PK guess possible;
|
%put &sysmacroname: No more PK guess possible;
|
||||||
%return;
|
%goto exit;
|
||||||
%end;
|
%end;
|
||||||
|
|
||||||
/* scan for uniques on up to 4 PK fields */
|
/* scan for uniques on up to 4 PK fields */
|
||||||
%local lev4;
|
%local lev5 m;
|
||||||
%do i=1 %to &ppkcnt;
|
%do i=1 %to &ppkcnt;
|
||||||
%let lev1=%scan(&posspks,&i);
|
%let lev1=%scan(&posspks,&i);
|
||||||
%do j=2 %to &ppkcnt;
|
%do j=2 %to &ppkcnt;
|
||||||
%let lev2=%scan(&posspks,&j);
|
%let lev2=%scan(&posspks,&j);
|
||||||
%if &lev1 ne &lev2 %then %do k=3 %to &ppkcnt;
|
%if &lev1 ne &lev2 %then %do k=3 %to &ppkcnt;
|
||||||
%let lev3=%scan(&posspks,&k);
|
%let lev3=%scan(&posspks,&k);
|
||||||
%if &lev1 ne &lev3 and &lev2 ne &lev3 %then %do l=4 %to &ppkcnt;
|
%if &lev1 ne &lev3 and &lev2 ne &lev3 %then %do l=4 %to &ppkcnt;
|
||||||
%let lev4=%scan(&posspks,&l);
|
%let lev4=%scan(&posspks,&l);
|
||||||
%if &lev1 ne &lev4 and &lev2 ne &lev4 and &lev3 ne &lev4 %then %do;
|
%if &lev1 ne &lev4 and &lev2 ne &lev4 and &lev3 ne &lev4 %then
|
||||||
|
%do m=5 %to &ppkcnt;
|
||||||
|
%let lev5=%scan(&posspks,&m);
|
||||||
|
%if &lev1 ne &lev5 & &lev2 ne &lev5 & &lev3 ne &lev5 & &lev4 ne &lev5 %then %do;
|
||||||
/* check for four level uniqueness */
|
/* check for four level uniqueness */
|
||||||
proc sort data=&pkds(keep=&lev1 &lev2 &lev3 &lev4)
|
proc sort data=&pkds(keep=&lev1 &lev2 &lev3 &lev4 &lev5)
|
||||||
out=&tmpds noduprec;
|
out=&tmpds noduprec;
|
||||||
by _all_;
|
by _all_;
|
||||||
run;
|
run;
|
||||||
%if %mf_nobs(&tmpds)=&rows %then %do;
|
%if %mf_nobs(&tmpds)=&rows %then %do;
|
||||||
proc sql;
|
proc sql;
|
||||||
insert into &outds values("&lev1 &lev2 &lev3 &lev4");
|
insert into &outds values("&lev1 &lev2 &lev3 &lev4 &lev5");
|
||||||
%if %mf_nobs(&outds) ge &max_guesses %then %do;
|
%if %mf_nobs(&outds) ge &max_guesses %then %do;
|
||||||
%put &sysmacroname: Max PKs reached at Level 4 for &baseds;
|
%put &sysmacroname: Max PKs reached at Level 5 for &baseds;
|
||||||
%return;
|
%goto exit;
|
||||||
%end;
|
%end;
|
||||||
%end;
|
%end;
|
||||||
%end;
|
%end;
|
||||||
@@ -218,37 +270,44 @@
|
|||||||
%end;
|
%end;
|
||||||
%end;
|
%end;
|
||||||
%end;
|
%end;
|
||||||
|
%end;
|
||||||
|
|
||||||
%if &ppkcnt=4 %then %do;
|
%if &ppkcnt=5 %then %do;
|
||||||
%put &sysmacroname: No more PK guess possible;
|
%put &sysmacroname: No more PK guess possible;
|
||||||
%return;
|
%goto exit;
|
||||||
%end;
|
%end;
|
||||||
|
|
||||||
/* scan for uniques on up to 4 PK fields */
|
/* scan for uniques on up to 4 PK fields */
|
||||||
%local lev5 m;
|
%local lev6 n;
|
||||||
%do i=1 %to &ppkcnt;
|
%do i=1 %to &ppkcnt;
|
||||||
%let lev1=%scan(&posspks,&i);
|
%let lev1=%scan(&posspks,&i);
|
||||||
%do j=2 %to &ppkcnt;
|
%do j=2 %to &ppkcnt;
|
||||||
%let lev2=%scan(&posspks,&j);
|
%let lev2=%scan(&posspks,&j);
|
||||||
%if &lev1 ne &lev2 %then %do k=3 %to &ppkcnt;
|
%if &lev1 ne &lev2 %then %do k=3 %to &ppkcnt;
|
||||||
%let lev3=%scan(&posspks,&k);
|
%let lev3=%scan(&posspks,&k);
|
||||||
%if &lev1 ne &lev3 and &lev2 ne &lev3 %then %do l=4 %to &ppkcnt;
|
%if &lev1 ne &lev3 and &lev2 ne &lev3 %then %do l=4 %to &ppkcnt;
|
||||||
%let lev4=%scan(&posspks,&l);
|
%let lev4=%scan(&posspks,&l);
|
||||||
%if &lev1 ne &lev4 and &lev2 ne &lev4 and &lev3 ne &lev4 %then
|
%if &lev1 ne &lev4 and &lev2 ne &lev4 and &lev3 ne &lev4 %then
|
||||||
%do m=5 %to &ppkcnt;
|
%do m=5 %to &ppkcnt;
|
||||||
%let lev5=%scan(&posspks,&m);
|
%let lev5=%scan(&posspks,&m);
|
||||||
%if &lev1 ne &lev5 & &lev2 ne &lev5 & &lev3 ne &lev5 & &lev4 ne &lev5 %then %do;
|
%if &lev1 ne &lev5 & &lev2 ne &lev5 & &lev3 ne &lev5 & &lev4 ne &lev5
|
||||||
|
%then %do n=6 %to &ppkcnt;
|
||||||
|
%let lev6=%scan(&posspks,&n);
|
||||||
|
%if &lev1 ne &lev6 & &lev2 ne &lev6 & &lev3 ne &lev6
|
||||||
|
& &lev4 ne &lev6 & &lev5 ne &lev6 %then
|
||||||
|
%do;
|
||||||
/* check for four level uniqueness */
|
/* check for four level uniqueness */
|
||||||
proc sort data=&pkds(keep=&lev1 &lev2 &lev3 &lev4 &lev5)
|
proc sort data=&pkds(keep=&lev1 &lev2 &lev3 &lev4 &lev5 &lev6)
|
||||||
out=&tmpds noduprec;
|
out=&tmpds noduprec;
|
||||||
by _all_;
|
by _all_;
|
||||||
run;
|
run;
|
||||||
%if %mf_nobs(&tmpds)=&rows %then %do;
|
%if %mf_nobs(&tmpds)=&rows %then %do;
|
||||||
proc sql;
|
proc sql;
|
||||||
insert into &outds values("&lev1 &lev2 &lev3 &lev4 &lev5");
|
insert into &outds
|
||||||
|
values("&lev1 &lev2 &lev3 &lev4 &lev5 &lev6");
|
||||||
%if %mf_nobs(&outds) ge &max_guesses %then %do;
|
%if %mf_nobs(&outds) ge &max_guesses %then %do;
|
||||||
%put &sysmacroname: Max PKs reached at Level 5 for &baseds;
|
%put &sysmacroname: Max PKs reached at Level 6 for &baseds;
|
||||||
%return;
|
%goto exit;
|
||||||
%end;
|
%end;
|
||||||
%end;
|
%end;
|
||||||
%end;
|
%end;
|
||||||
@@ -257,56 +316,17 @@
|
|||||||
%end;
|
%end;
|
||||||
%end;
|
%end;
|
||||||
%end;
|
%end;
|
||||||
|
%end;
|
||||||
|
|
||||||
%if &ppkcnt=5 %then %do;
|
%if &ppkcnt=6 %then %do;
|
||||||
%put &sysmacroname: No more PK guess possible;
|
%put &sysmacroname: No more PK guess possible;
|
||||||
%return;
|
%goto exit;
|
||||||
%end;
|
%end;
|
||||||
|
|
||||||
/* scan for uniques on up to 4 PK fields */
|
%exit:
|
||||||
%local lev6 n;
|
%if &mdebug=0 %then %do;
|
||||||
%do i=1 %to &ppkcnt;
|
proc sql;
|
||||||
%let lev1=%scan(&posspks,&i);
|
drop table &tmpds;
|
||||||
%do j=2 %to &ppkcnt;
|
%end;
|
||||||
%let lev2=%scan(&posspks,&j);
|
|
||||||
%if &lev1 ne &lev2 %then %do k=3 %to &ppkcnt;
|
|
||||||
%let lev3=%scan(&posspks,&k);
|
|
||||||
%if &lev1 ne &lev3 and &lev2 ne &lev3 %then %do l=4 %to &ppkcnt;
|
|
||||||
%let lev4=%scan(&posspks,&l);
|
|
||||||
%if &lev1 ne &lev4 and &lev2 ne &lev4 and &lev3 ne &lev4 %then
|
|
||||||
%do m=5 %to &ppkcnt;
|
|
||||||
%let lev5=%scan(&posspks,&m);
|
|
||||||
%if &lev1 ne &lev5 & &lev2 ne &lev5 & &lev3 ne &lev5 & &lev4 ne &lev5 %then
|
|
||||||
%do n=6 %to &ppkcnt;
|
|
||||||
%let lev6=%scan(&posspks,&n);
|
|
||||||
%if &lev1 ne &lev6 & &lev2 ne &lev6 & &lev3 ne &lev6
|
|
||||||
& &lev4 ne &lev6 & &lev5 ne &lev6 %then
|
|
||||||
%do;
|
|
||||||
/* check for four level uniqueness */
|
|
||||||
proc sort data=&pkds(keep=&lev1 &lev2 &lev3 &lev4 &lev5 &lev6)
|
|
||||||
out=&tmpds noduprec;
|
|
||||||
by _all_;
|
|
||||||
run;
|
|
||||||
%if %mf_nobs(&tmpds)=&rows %then %do;
|
|
||||||
proc sql;
|
|
||||||
insert into &outds
|
|
||||||
values("&lev1 &lev2 &lev3 &lev4 &lev5 &lev6");
|
|
||||||
%if %mf_nobs(&outds) ge &max_guesses %then %do;
|
|
||||||
%put &sysmacroname: Max PKs reached at Level 6 for &baseds;
|
|
||||||
%return;
|
|
||||||
%end;
|
|
||||||
%end;
|
|
||||||
%end;
|
|
||||||
%end;
|
|
||||||
%end;
|
|
||||||
%end;
|
|
||||||
%end;
|
|
||||||
%end;
|
|
||||||
%end;
|
|
||||||
|
|
||||||
%if &ppkcnt=6 %then %do;
|
|
||||||
%put &sysmacroname: No more PK guess possible;
|
|
||||||
%return;
|
|
||||||
%end;
|
|
||||||
|
|
||||||
%mend mp_guesspk;
|
%mend mp_guesspk;
|
||||||
@@ -51,9 +51,10 @@ https://documentation.sas.com/doc/en/pgmsascdc/9.4_3.5/mcrolref/n1j5tcc0n2xczyn1
|
|||||||
this dataset.
|
this dataset.
|
||||||
It will then run an abort cancel FILE to stop the include running, and pass
|
It will then run an abort cancel FILE to stop the include running, and pass
|
||||||
the dataset back.
|
the dataset back.
|
||||||
NOTE - it is NOT possible to read this dataset as part of _this_ macro -
|
|
||||||
when running abort cancel FILE, ALL macros are closed, so instead it is
|
IMPORTANT NOTE - it is NOT possible to read this dataset as part of _this_
|
||||||
necessary to invoke "%mp_abort(mode=INCLUDE)" OUTSIDE of any macro wrappers.
|
macro! When running abort cancel FILE, ALL macros are closed, so instead it
|
||||||
|
is necessary to invoke "%mp_abort(mode=INCLUDE)" OUTSIDE of macro wrappers.
|
||||||
|
|
||||||
|
|
||||||
@version 9.4
|
@version 9.4
|
||||||
@@ -88,6 +89,7 @@ run;
|
|||||||
/* prepare the errds */
|
/* prepare the errds */
|
||||||
data &errds;
|
data &errds;
|
||||||
length msg mac $1000;
|
length msg mac $1000;
|
||||||
|
call missing(msg,mac);
|
||||||
iftrue='1=0';
|
iftrue='1=0';
|
||||||
run;
|
run;
|
||||||
|
|
||||||
|
|||||||
@@ -1,15 +1,25 @@
|
|||||||
/**
|
/**
|
||||||
@file
|
@file
|
||||||
@brief Initialise session with useful settings and variables
|
@brief Initialise session with useful settings and variables
|
||||||
@details Implements a set of recommended options for general SAS use. This
|
@details Implements a "strict" set of SAS options for use in defensive
|
||||||
macro is NOT used elsewhere within the core library (other than in tests),
|
programming. Highly recommended, if you want your code to run on some
|
||||||
but it is used by the SASjs team when building web services for
|
other machine.
|
||||||
SAS-Powered applications elsewhere.
|
|
||||||
|
|
||||||
If you have a good idea for an option, setting, or useful global variable -
|
This macro is recommended to be compiled and invoked in the `initProgram`
|
||||||
feel free to [raise an issue](https://github.com/sasjs/core/issues/new)!
|
for SASjs [Jobs](https://cli.sasjs.io/sasjsconfig.html#jobConfig_initProgram
|
||||||
|
), [Services](
|
||||||
|
https://cli.sasjs.io/sasjsconfig.html#serviceConfig_initProgram) and [Tests]
|
||||||
|
(https://cli.sasjs.io/sasjsconfig.html#testConfig_initProgram).
|
||||||
|
|
||||||
All global variables are prefixed with "SASJS_" (unless modfied with the
|
For non SASjs projects, you could invoke in the autoexec, or in your own
|
||||||
|
solution initialisation macro.
|
||||||
|
|
||||||
|
|
||||||
|
If you have a good idea for another useful option, setting, or global
|
||||||
|
variable - feel free to [raise an issue](
|
||||||
|
https://github.com/sasjs/core/issues/new)!
|
||||||
|
|
||||||
|
All global variables are prefixed with "SASJS" (unless modified with the
|
||||||
prefix parameter).
|
prefix parameter).
|
||||||
|
|
||||||
@param [in] prefix= (SASJS) The prefix to apply to the global macro variables
|
@param [in] prefix= (SASJS) The prefix to apply to the global macro variables
|
||||||
@@ -20,35 +30,47 @@
|
|||||||
|
|
||||||
**/
|
**/
|
||||||
|
|
||||||
%macro mp_init(prefix=
|
%macro mp_init(prefix=SASJS
|
||||||
)/*/STORE SOURCE*/;
|
)/*/STORE SOURCE*/;
|
||||||
|
|
||||||
%global
|
%if %symexist(SASJS_PREFIX) %then %return; /* only run once */
|
||||||
&prefix._INIT_NUM /* initialisation time as numeric */
|
|
||||||
&prefix._INIT_DTTM /* initialisation time in E8601DT26.6 format */
|
|
||||||
;
|
|
||||||
%if %eval(&&&prefix._INIT_NUM>0) %then %return; /* only run once */
|
|
||||||
|
|
||||||
data _null_;
|
%global
|
||||||
dttm=datetime();
|
SASJS_PREFIX /* the ONLY hard-coded global macro variable in SASjs */
|
||||||
call symputx("&prefix._init_num",dttm);
|
&prefix._FUNCTIONS /* used in mcf_init() to track core function compilation */
|
||||||
call symputx("&prefix._init_dttm",put(dttm,E8601DT26.6));
|
&prefix._INIT_NUM /* initialisation time as numeric */
|
||||||
run;
|
&prefix._INIT_DTTM /* initialisation time in E8601DT26.6 format */
|
||||||
|
&prefix.WORK /* avoid typing %sysfunc(pathname(work)) every time */
|
||||||
|
;
|
||||||
|
|
||||||
options
|
%let sasjs_prefix=&prefix;
|
||||||
autocorrect /* disallow mis-spelled procedure names */
|
|
||||||
compress=CHAR /* default is none so ensure we have something! */
|
data _null_;
|
||||||
datastmtchk=ALLKEYWORDS /* protection from overwriting input datasets */
|
dttm=datetime();
|
||||||
errorcheck=STRICT /* catch errors in libname/filename statements */
|
call symputx("&sasjs_prefix._init_num",dttm,'g');
|
||||||
fmterr /* ensure err when a format cannot be found */
|
call symputx("&sasjs_prefix._init_dttm",put(dttm,E8601DT26.6),'g');
|
||||||
mergenoby=%str(ERR)OR /* Throw err when a merge has no BY variables */
|
call symputx("&sasjs_prefix.work",pathname('WORK'),'g');
|
||||||
missing=. /* some sites change this which causes hard to detect errors */
|
run;
|
||||||
noquotelenmax /* avoid warnings for long strings */
|
|
||||||
noreplace /* avoid overwriting permanent datasets */
|
options
|
||||||
ps=max /* reduce log size slightly */
|
compress=CHAR /* default is none so ensure we have something! */
|
||||||
validmemname=COMPATIBLE /* avoid special characters etc in table names */
|
datastmtchk=ALLKEYWORDS /* protection from overwriting input datasets */
|
||||||
validvarname=V7 /* avoid special characters etc in variable names */
|
errorcheck=STRICT /* catch errs in libname/filename statements */
|
||||||
varlenchk=%str(ERR)OR /* fail hard if truncation (data loss) can result */
|
fmterr /* ensure err when a format cannot be found */
|
||||||
;
|
mergenoby=%str(ERR)OR /* throw err when a merge has no BY variables */
|
||||||
|
missing=. /* changing this can cause hard to detect errs */
|
||||||
|
noquotelenmax /* avoid warnings for long strings */
|
||||||
|
noreplace /* avoid overwriting permanent datasets */
|
||||||
|
ps=max /* reduce log size slightly */
|
||||||
|
ls=max /* reduce log even more and avoid word truncation */
|
||||||
|
validmemname=COMPATIBLE /* avoid special characters etc in table names */
|
||||||
|
validvarname=V7 /* avoid special characters etc in variable names */
|
||||||
|
varinitchk=%str(ERR)OR /* avoid data mistakes from variable name typos */
|
||||||
|
varlenchk=%str(ERR)OR /* fail hard if truncation (data loss) can result */
|
||||||
|
%if %substr(&sysver,1,1) ne 4 %then %do;
|
||||||
|
noautocorrect /* disallow misspelled procedure names */
|
||||||
|
dsoptions=note2err /* undocumented - convert bad NOTEs to ERRs */
|
||||||
|
%end;
|
||||||
|
;
|
||||||
|
|
||||||
%mend mp_init;
|
%mend mp_init;
|
||||||
@@ -19,11 +19,12 @@
|
|||||||
|
|
||||||
%mp_jsonout(OPEN,jref=tmp)
|
%mp_jsonout(OPEN,jref=tmp)
|
||||||
%mp_jsonout(OBJ,class,jref=tmp)
|
%mp_jsonout(OBJ,class,jref=tmp)
|
||||||
|
%mp_jsonout(OBJ,class,dslabel=class2,jref=tmp,showmeta=YES)
|
||||||
%mp_jsonout(CLOSE,jref=tmp)
|
%mp_jsonout(CLOSE,jref=tmp)
|
||||||
|
|
||||||
data _null_;
|
data _null_;
|
||||||
infile tmp;
|
infile tmp;
|
||||||
input;list;
|
input;putlog _infile_;
|
||||||
run;
|
run;
|
||||||
|
|
||||||
If you are building web apps with SAS then you are strongly encouraged to use
|
If you are building web apps with SAS then you are strongly encouraged to use
|
||||||
@@ -31,22 +32,24 @@
|
|||||||
[sasjs adapter](https://github.com/sasjs/adapter).
|
[sasjs adapter](https://github.com/sasjs/adapter).
|
||||||
For more information see https://sasjs.io
|
For more information see https://sasjs.io
|
||||||
|
|
||||||
@param action Valid values:
|
@param [in] action Valid values:
|
||||||
@li OPEN - opens the JSON
|
@li OPEN - opens the JSON
|
||||||
@li OBJ - sends a table with each row as an object
|
@li OBJ - sends a table with each row as an object
|
||||||
@li ARR - sends a table with each row in an array
|
@li ARR - sends a table with each row in an array
|
||||||
@li CLOSE - closes the JSON
|
@li CLOSE - closes the JSON
|
||||||
|
@param [in] ds The dataset to send. Must be a work table.
|
||||||
@param ds the dataset to send. Must be a work table.
|
@param [out] jref= (_webout) The fileref to which to send the JSON
|
||||||
@param jref= the fileref to which to send the JSON
|
@param [out] dslabel= The name to give the table in the exported JSON
|
||||||
@param dslabel= the name to give the table in the exported JSON
|
@param [in] fmt= (Y) Whether to keep (Y) or strip (N) formats from the table
|
||||||
@param fmt= Whether to keep or strip formats from the table
|
@param [in] engine= (DATASTEP) Which engine to use to send the JSON. Options:
|
||||||
@param engine= Which engine to use to send the JSON, valid options are:
|
|
||||||
@li PROCJSON (default)
|
@li PROCJSON (default)
|
||||||
@li DATASTEP (more reliable when data has non standard characters)
|
@li DATASTEP (more reliable when data has non standard characters)
|
||||||
|
@param [in] missing= (NULL) Special numeric missing values can be sent as NULL
|
||||||
@param dbg= DEPRECATED - was used to conditionally add PRETTY to
|
(eg `null`) or as STRING values (eg `".a"` or `".b"`)
|
||||||
proc json but this can cause line truncation in large files.
|
@param [in] showmeta= (NO) Set to YES to output metadata alongside each table,
|
||||||
|
such as the column formats and types. The metadata is contained inside an
|
||||||
|
object with the same name as the table but prefixed with a dollar sign - ie,
|
||||||
|
`,"$tablename":{"formats":{"col1":"$CHAR1"},"types":{"COL1":"C"}}`
|
||||||
|
|
||||||
<h4> Related Macros <h4>
|
<h4> Related Macros <h4>
|
||||||
@li mp_ds2fmtds.sas
|
@li mp_ds2fmtds.sas
|
||||||
@@ -57,129 +60,145 @@
|
|||||||
|
|
||||||
**/
|
**/
|
||||||
|
|
||||||
%macro mp_jsonout(action,ds,jref=_webout,dslabel=,fmt=Y,engine=DATASTEP,dbg=0
|
%macro mp_jsonout(action,ds,jref=_webout,dslabel=,fmt=Y
|
||||||
|
,engine=DATASTEP
|
||||||
|
,missing=NULL
|
||||||
|
,showmeta=NO
|
||||||
)/*/STORE SOURCE*/;
|
)/*/STORE SOURCE*/;
|
||||||
%put output location=&jref;
|
%local tempds colinfo fmtds i numcols;
|
||||||
|
%let numcols=0;
|
||||||
|
|
||||||
%if &action=OPEN %then %do;
|
%if &action=OPEN %then %do;
|
||||||
options nobomfile;
|
options nobomfile;
|
||||||
data _null_;file &jref encoding='utf-8';
|
data _null_;file &jref encoding='utf-8' ;
|
||||||
put '{"START_DTTM" : "' "%sysfunc(datetime(),datetime20.3)" '"';
|
put '{"PROCESSED_DTTM" : "' "%sysfunc(datetime(),E8601DT26.6)" '"';
|
||||||
run;
|
run;
|
||||||
%end;
|
%end;
|
||||||
%else %if (&action=ARR or &action=OBJ) %then %do;
|
%else %if (&action=ARR or &action=OBJ) %then %do;
|
||||||
options validvarname=upcase;
|
options validvarname=upcase;
|
||||||
data _null_;file &jref mod encoding='utf-8';
|
data _null_; file &jref encoding='utf-8' mod;
|
||||||
put ", ""%lowcase(%sysfunc(coalescec(&dslabel,&ds)))"":";
|
put ", ""%lowcase(%sysfunc(coalescec(&dslabel,&ds)))"":";
|
||||||
|
|
||||||
|
/* grab col defs */
|
||||||
|
proc contents noprint data=&ds
|
||||||
|
out=_data_(keep=name type length format formatl formatd varnum label);
|
||||||
|
run;
|
||||||
|
%let colinfo=%scan(&syslast,2,.);
|
||||||
|
proc sort data=&colinfo;
|
||||||
|
by varnum;
|
||||||
|
run;
|
||||||
|
/* move meta to mac vars */
|
||||||
|
data _null_;
|
||||||
|
if _n_=1 then call symputx('numcols',nobs,'l');
|
||||||
|
set &colinfo end=last nobs=nobs;
|
||||||
|
name=upcase(name);
|
||||||
|
/* fix formats */
|
||||||
|
if type=2 or type=6 then do;
|
||||||
|
typelong='char';
|
||||||
|
length fmt $49.;
|
||||||
|
if format='' then fmt=cats('$',length,'.');
|
||||||
|
else if formatl=0 then fmt=cats(format,'.');
|
||||||
|
else fmt=cats(format,formatl,'.');
|
||||||
|
newlen=max(formatl,length);
|
||||||
|
end;
|
||||||
|
else do;
|
||||||
|
typelong='num';
|
||||||
|
if format='' then fmt='best.';
|
||||||
|
else if formatl=0 then fmt=cats(format,'.');
|
||||||
|
else if formatd=0 then fmt=cats(format,formatl,'.');
|
||||||
|
else fmt=cats(format,formatl,'.',formatd);
|
||||||
|
/* needs to be wide, for datetimes etc */
|
||||||
|
newlen=max(length,formatl,24);
|
||||||
|
end;
|
||||||
|
/* 32 char unique name */
|
||||||
|
newname='sasjs'!!substr(cats(put(md5(name),$hex32.)),1,27);
|
||||||
|
|
||||||
|
call symputx(cats('name',_n_),name,'l');
|
||||||
|
call symputx(cats('newname',_n_),newname,'l');
|
||||||
|
call symputx(cats('len',_n_),newlen,'l');
|
||||||
|
call symputx(cats('length',_n_),length,'l');
|
||||||
|
call symputx(cats('fmt',_n_),fmt,'l');
|
||||||
|
call symputx(cats('type',_n_),type,'l');
|
||||||
|
call symputx(cats('typelong',_n_),typelong,'l');
|
||||||
|
call symputx(cats('label',_n_),coalescec(label,name),'l');
|
||||||
|
run;
|
||||||
|
|
||||||
|
%let tempds=%substr(_%sysfunc(compress(%sysfunc(uuidgen()),-)),1,32);
|
||||||
|
|
||||||
%if &engine=PROCJSON %then %do;
|
%if &engine=PROCJSON %then %do;
|
||||||
data;run;%let tempds=&syslast;
|
%if &missing=STRING %then %do;
|
||||||
proc sql;drop table &tempds;
|
%put &sysmacroname: Special Missings not supported in proc json.;
|
||||||
data &tempds /view=&tempds;set &ds;
|
%put &sysmacroname: Switching to DATASTEP engine;
|
||||||
|
%goto datastep;
|
||||||
|
%end;
|
||||||
|
data &tempds;set &ds;
|
||||||
%if &fmt=N %then format _numeric_ best32.;;
|
%if &fmt=N %then format _numeric_ best32.;;
|
||||||
|
/* PRETTY is necessary to avoid line truncation in large files */
|
||||||
proc json out=&jref pretty
|
proc json out=&jref pretty
|
||||||
%if &action=ARR %then nokeys ;
|
%if &action=ARR %then nokeys ;
|
||||||
;export &tempds / nosastags fmtnumeric;
|
;export &tempds / nosastags fmtnumeric;
|
||||||
run;
|
run;
|
||||||
proc sql;drop view &tempds;
|
|
||||||
%end;
|
%end;
|
||||||
%else %if &engine=DATASTEP %then %do;
|
%else %if &engine=DATASTEP %then %do;
|
||||||
%local cols i tempds;
|
%datastep:
|
||||||
%let cols=0;
|
%if %sysfunc(exist(&ds)) ne 1 & %sysfunc(exist(&ds,VIEW)) ne 1
|
||||||
%if %sysfunc(exist(&ds)) ne 1 & %sysfunc(exist(&ds,VIEW)) ne 1 %then %do;
|
%then %do;
|
||||||
%put &sysmacroname: &ds NOT FOUND!!!;
|
%put &sysmacroname: &ds NOT FOUND!!!;
|
||||||
%return;
|
%return;
|
||||||
%end;
|
%end;
|
||||||
%if &fmt=Y %then %do;
|
|
||||||
%put converting every variable to a formatted variable;
|
|
||||||
/* see mp_ds2fmtds.sas for source */
|
|
||||||
proc contents noprint data=&ds
|
|
||||||
out=_data_(keep=name type length format formatl formatd varnum);
|
|
||||||
run;
|
|
||||||
proc sort;
|
|
||||||
by varnum;
|
|
||||||
run;
|
|
||||||
%local fmtds;
|
|
||||||
%let fmtds=%scan(&syslast,2,.);
|
|
||||||
/* prepare formats and varnames */
|
|
||||||
data _null_;
|
|
||||||
if _n_=1 then call symputx('nobs',nobs,'l');
|
|
||||||
set &fmtds end=last nobs=nobs;
|
|
||||||
name=upcase(name);
|
|
||||||
/* fix formats */
|
|
||||||
if type=2 or type=6 then do;
|
|
||||||
length fmt $49.;
|
|
||||||
if format='' then fmt=cats('$',length,'.');
|
|
||||||
else if formatl=0 then fmt=cats(format,'.');
|
|
||||||
else fmt=cats(format,formatl,'.');
|
|
||||||
newlen=max(formatl,length);
|
|
||||||
end;
|
|
||||||
else do;
|
|
||||||
if format='' then fmt='best.';
|
|
||||||
else if formatl=0 then fmt=cats(format,'.');
|
|
||||||
else if formatd=0 then fmt=cats(format,formatl,'.');
|
|
||||||
else fmt=cats(format,formatl,'.',formatd);
|
|
||||||
/* needs to be wide, for datetimes etc */
|
|
||||||
newlen=max(length,formatl,24);
|
|
||||||
end;
|
|
||||||
/* 32 char unique name */
|
|
||||||
newname='sasjs'!!substr(cats(put(md5(name),$hex32.)),1,27);
|
|
||||||
|
|
||||||
call symputx(cats('name',_n_),name,'l');
|
%if &fmt=Y %then %do;
|
||||||
call symputx(cats('newname',_n_),newname,'l');
|
data _data_;
|
||||||
call symputx(cats('len',_n_),newlen,'l');
|
|
||||||
call symputx(cats('fmt',_n_),fmt,'l');
|
|
||||||
call symputx(cats('type',_n_),type,'l');
|
|
||||||
run;
|
|
||||||
data &fmtds;
|
|
||||||
/* rename on entry */
|
/* rename on entry */
|
||||||
set &ds(rename=(
|
set &ds(rename=(
|
||||||
%local i;
|
%do i=1 %to &numcols;
|
||||||
%do i=1 %to &nobs;
|
|
||||||
&&name&i=&&newname&i
|
&&name&i=&&newname&i
|
||||||
%end;
|
%end;
|
||||||
));
|
));
|
||||||
%do i=1 %to &nobs;
|
%do i=1 %to &numcols;
|
||||||
length &&name&i $&&len&i;
|
length &&name&i $&&len&i;
|
||||||
&&name&i=left(put(&&newname&i,&&fmt&i));
|
%if &&typelong&i=num %then %do;
|
||||||
|
&&name&i=left(put(&&newname&i,&&fmt&i));
|
||||||
|
%end;
|
||||||
|
%else %do;
|
||||||
|
&&name&i=put(&&newname&i,&&fmt&i);
|
||||||
|
%end;
|
||||||
drop &&newname&i;
|
drop &&newname&i;
|
||||||
%end;
|
%end;
|
||||||
if _error_ then call symputx('syscc',1012);
|
if _error_ then call symputx('syscc',1012);
|
||||||
run;
|
run;
|
||||||
%let ds=&fmtds;
|
%let fmtds=&syslast;
|
||||||
%end; /* &fmt=Y */
|
%end;
|
||||||
data _null_;file &jref mod encoding='utf-8';
|
|
||||||
put "["; call symputx('cols',0,'l');
|
|
||||||
proc sort
|
|
||||||
data=sashelp.vcolumn(where=(libname='WORK' & memname="%upcase(&ds)"))
|
|
||||||
out=_data_;
|
|
||||||
by varnum;
|
|
||||||
|
|
||||||
data _null_;
|
|
||||||
set _last_ end=last;
|
|
||||||
call symputx(cats('name',_n_),name,'l');
|
|
||||||
call symputx(cats('type',_n_),type,'l');
|
|
||||||
call symputx(cats('len',_n_),length,'l');
|
|
||||||
if last then call symputx('cols',_n_,'l');
|
|
||||||
run;
|
|
||||||
|
|
||||||
proc format; /* credit yabwon for special null removal */
|
proc format; /* credit yabwon for special null removal */
|
||||||
value bart ._ - .z = null
|
value bart (default=40)
|
||||||
|
%if &missing=NULL %then %do;
|
||||||
|
._ - .z = null
|
||||||
|
%end;
|
||||||
|
%else %do;
|
||||||
|
._ = [quote()]
|
||||||
|
. = null
|
||||||
|
.a - .z = [quote()]
|
||||||
|
%end;
|
||||||
other = [best.];
|
other = [best.];
|
||||||
|
|
||||||
data;run; %let tempds=&syslast; /* temp table for spesh char management */
|
data &tempds;
|
||||||
proc sql; drop table &tempds;
|
|
||||||
data &tempds/view=&tempds;
|
|
||||||
attrib _all_ label='';
|
attrib _all_ label='';
|
||||||
%do i=1 %to &cols;
|
%do i=1 %to &numcols;
|
||||||
%if &&type&i=char %then %do;
|
%if &&typelong&i=char or &fmt=Y %then %do;
|
||||||
length &&name&i $32767;
|
length &&name&i $32767;
|
||||||
format &&name&i $32767.;
|
format &&name&i $32767.;
|
||||||
%end;
|
%end;
|
||||||
%end;
|
%end;
|
||||||
set &ds;
|
%if &fmt=Y %then %do;
|
||||||
|
set &fmtds;
|
||||||
|
%end;
|
||||||
|
%else %do;
|
||||||
|
set &ds;
|
||||||
|
%end;
|
||||||
format _numeric_ bart.;
|
format _numeric_ bart.;
|
||||||
%do i=1 %to &cols;
|
%do i=1 %to &numcols;
|
||||||
%if &&type&i=char %then %do;
|
%if &&typelong&i=char or &fmt=Y %then %do;
|
||||||
&&name&i='"'!!trim(prxchange('s/"/\"/',-1,
|
&&name&i='"'!!trim(prxchange('s/"/\"/',-1,
|
||||||
prxchange('s/'!!'0A'x!!'/\n/',-1,
|
prxchange('s/'!!'0A'x!!'/\n/',-1,
|
||||||
prxchange('s/'!!'0D'x!!'/\r/',-1,
|
prxchange('s/'!!'0D'x!!'/\r/',-1,
|
||||||
@@ -189,44 +208,64 @@
|
|||||||
%end;
|
%end;
|
||||||
%end;
|
%end;
|
||||||
run;
|
run;
|
||||||
|
|
||||||
/* write to temp loc to avoid _webout truncation
|
/* write to temp loc to avoid _webout truncation
|
||||||
- https://support.sas.com/kb/49/325.html */
|
- https://support.sas.com/kb/49/325.html */
|
||||||
filename _sjs temp lrecl=131068 encoding='utf-8';
|
filename _sjs temp lrecl=131068 encoding='utf-8';
|
||||||
data _null_; file _sjs lrecl=131068 encoding='utf-8' mod;
|
data _null_; file _sjs lrecl=131068 encoding='utf-8' mod ;
|
||||||
|
if _n_=1 then put "[";
|
||||||
set &tempds;
|
set &tempds;
|
||||||
if _n_>1 then put "," @; put
|
if _n_>1 then put "," @; put
|
||||||
%if &action=ARR %then "[" ; %else "{" ;
|
%if &action=ARR %then "[" ; %else "{" ;
|
||||||
%do i=1 %to &cols;
|
%do i=1 %to &numcols;
|
||||||
%if &i>1 %then "," ;
|
%if &i>1 %then "," ;
|
||||||
%if &action=OBJ %then """&&name&i"":" ;
|
%if &action=OBJ %then """&&name&i"":" ;
|
||||||
&&name&i
|
&&name&i
|
||||||
%end;
|
%end;
|
||||||
%if &action=ARR %then "]" ; %else "}" ; ;
|
%if &action=ARR %then "]" ; %else "}" ; ;
|
||||||
proc sql;
|
|
||||||
drop view &tempds;
|
|
||||||
/* now write the long strings to _webout 1 byte at a time */
|
/* now write the long strings to _webout 1 byte at a time */
|
||||||
data _null_;
|
data _null_;
|
||||||
length filein 8 fileid 8;
|
length filein 8 fileid 8;
|
||||||
filein = fopen("_sjs",'I',1,'B');
|
filein=fopen("_sjs",'I',1,'B');
|
||||||
fileid = fopen("&jref",'A',1,'B');
|
fileid=fopen("&jref",'A',1,'B');
|
||||||
rec = '20'x;
|
rec='20'x;
|
||||||
do while(fread(filein)=0);
|
do while(fread(filein)=0);
|
||||||
rc = fget(filein,rec,1);
|
rc=fget(filein,rec,1);
|
||||||
rc = fput(fileid, rec);
|
rc=fput(fileid, rec);
|
||||||
rc =fwrite(fileid);
|
rc=fwrite(fileid);
|
||||||
end;
|
end;
|
||||||
rc = fclose(filein);
|
/* close out the table */
|
||||||
rc = fclose(fileid);
|
rc=fput(fileid, "]");
|
||||||
|
rc=fwrite(fileid);
|
||||||
|
rc=fclose(filein);
|
||||||
|
rc=fclose(fileid);
|
||||||
run;
|
run;
|
||||||
filename _sjs clear;
|
filename _sjs clear;
|
||||||
data _null_; file &jref mod encoding='utf-8';
|
%end;
|
||||||
put "]";
|
|
||||||
|
proc sql;
|
||||||
|
drop table &colinfo, &tempds;
|
||||||
|
|
||||||
|
%if &showmeta=YES %then %do;
|
||||||
|
data _null_; file &jref encoding='utf-8' mod;
|
||||||
|
put ", ""$%lowcase(%sysfunc(coalescec(&dslabel,&ds)))"":{""vars"":{";
|
||||||
|
do i=1 to &numcols;
|
||||||
|
name=quote(trim(symget(cats('name',i))));
|
||||||
|
format=quote(trim(symget(cats('fmt',i))));
|
||||||
|
label=quote(trim(symget(cats('label',i))));
|
||||||
|
length=quote(trim(symget(cats('length',i))));
|
||||||
|
type=quote(trim(symget(cats('typelong',i))));
|
||||||
|
if i>1 then put "," @@;
|
||||||
|
put name ':{"format":' format ',"label":' label
|
||||||
|
',"length":' length ',"type":' type '}';
|
||||||
|
end;
|
||||||
|
put '}}';
|
||||||
run;
|
run;
|
||||||
%end;
|
%end;
|
||||||
%end;
|
%end;
|
||||||
|
|
||||||
%else %if &action=CLOSE %then %do;
|
%else %if &action=CLOSE %then %do;
|
||||||
data _null_;file &jref encoding='utf-8' mod;
|
data _null_; file &jref encoding='utf-8' mod ;
|
||||||
put "}";
|
put "}";
|
||||||
run;
|
run;
|
||||||
%end;
|
%end;
|
||||||
|
|||||||
298
base/mp_loadformat.sas
Normal file
298
base/mp_loadformat.sas
Normal file
@@ -0,0 +1,298 @@
|
|||||||
|
/**
|
||||||
|
@file
|
||||||
|
@brief Loads a format catalog from a staging dataset
|
||||||
|
@details When loading staged data, it is common to receive only the records
|
||||||
|
that have actually changed. However, when loading a format catalog, if
|
||||||
|
records are missing they are presumed to be no longer required.
|
||||||
|
|
||||||
|
This macro will augment a staging dataset with other records from the same
|
||||||
|
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=`).
|
||||||
|
|
||||||
|
This macro can also be used to identify which records would be (or were)
|
||||||
|
considered new, modified or deleted (`loadtarget=`) by creating the following
|
||||||
|
tables:
|
||||||
|
|
||||||
|
@li work.outds_add
|
||||||
|
@li work.outds_del
|
||||||
|
@li work.outds_mod
|
||||||
|
|
||||||
|
For example usage, see mp_loadformat.test.sas
|
||||||
|
|
||||||
|
@param [in] libcat The format catalog to be loaded
|
||||||
|
@param [in] libds The staging table to load
|
||||||
|
@param [in] loadtarget= (NO) Set to YES to actually load the target catalog
|
||||||
|
@param [in] delete_col= (_____DELETE__THIS__RECORD_____) The column used to
|
||||||
|
mark a record for deletion. Values should be "Yes" or "No".
|
||||||
|
@param [out] auditlibds= (0) For change tracking, set to the libds of an audit
|
||||||
|
table as defined in mddl_dc_difftable.sas
|
||||||
|
@param [in] locklibds= (0) For multi-user (parallel) situations, set to the
|
||||||
|
libds of the DC lock table as defined in the mddl_dc_locktable.sas macro.
|
||||||
|
@param [out] outds_add= (0) Set a libds here to see the new records added
|
||||||
|
@param [out] outds_del= (0) Set a libds here to see the records deleted
|
||||||
|
@param [out] outds_mod= (0) Set a libds here to see the modified records
|
||||||
|
@param [in] mdebug= (0) Set to 1 to enable DEBUG messages and preserve outputs
|
||||||
|
|
||||||
|
<h4> SAS Macros </h4>
|
||||||
|
@li mddl_sas_cntlout.sas
|
||||||
|
@li mf_getuniquename.sas
|
||||||
|
@li mf_nobs.sas
|
||||||
|
@li mp_abort.sas
|
||||||
|
@li mp_cntlout.sas
|
||||||
|
@li mp_lockanytable.sas
|
||||||
|
|
||||||
|
<h4> Related Macros </h4>
|
||||||
|
@li mddl_dc_difftable.sas
|
||||||
|
@li mddl_dc_locktable.sas
|
||||||
|
@li mp_loadformat.test.sas
|
||||||
|
@li mp_lockanytable.sas
|
||||||
|
@li mp_storediffs.sas
|
||||||
|
@li mp_stackdiffs.sas
|
||||||
|
|
||||||
|
|
||||||
|
@version 9.2
|
||||||
|
@author Allan Bowe
|
||||||
|
|
||||||
|
**/
|
||||||
|
|
||||||
|
%macro mp_loadformat(libcat,libds
|
||||||
|
,loadtarget=NO
|
||||||
|
,auditlibds=0
|
||||||
|
,locklibds=0
|
||||||
|
,delete_col=_____DELETE__THIS__RECORD_____
|
||||||
|
,outds_add=0
|
||||||
|
,outds_del=0
|
||||||
|
,outds_mod=0
|
||||||
|
,mdebug=0
|
||||||
|
);
|
||||||
|
/* set up local macro variables and temporary tables (with a prefix) */
|
||||||
|
%local err msg prefix dslist i var fmtlist ibufsize;
|
||||||
|
%let dslist=base_fmts template inlibds ds1 stagedata storediffs;
|
||||||
|
%if &outds_add=0 %then %let dslist=&dslist outds_add;
|
||||||
|
%if &outds_del=0 %then %let dslist=&dslist outds_del;
|
||||||
|
%if &outds_mod=0 %then %let dslist=&dslist outds_mod;
|
||||||
|
%let prefix=%substr(%mf_getuniquename(),1,21);
|
||||||
|
%do i=1 %to %sysfunc(countw(&dslist));
|
||||||
|
%let var=%scan(&dslist,&i);
|
||||||
|
%local &var;
|
||||||
|
%let &var=%upcase(&prefix._&var);
|
||||||
|
%end;
|
||||||
|
|
||||||
|
/*
|
||||||
|
format values can be up to 32767 wide. SQL joins on such a wide column can
|
||||||
|
cause buffer issues. Update ibufsize and reset at the end.
|
||||||
|
*/
|
||||||
|
%let ibufsize=%sysfunc(getoption(ibufsize));
|
||||||
|
options ibufsize=32767 ;
|
||||||
|
|
||||||
|
/* in DC, format catalogs maybe specified in the libds with a -FC extension */
|
||||||
|
%let libcat=%scan(&libcat,1,-);
|
||||||
|
|
||||||
|
/* perform input validations */
|
||||||
|
%let err=0;
|
||||||
|
%let msg=0;
|
||||||
|
data _null_;
|
||||||
|
if _n_=1 then putlog "&sysmacroname entry vars:";
|
||||||
|
set sashelp.vmacro;
|
||||||
|
where scope="&sysmacroname";
|
||||||
|
value=upcase(value);
|
||||||
|
if &mdebug=0 then put name '=' value;
|
||||||
|
if name=:'LOAD' and value not in ('YES','NO') then do;
|
||||||
|
call symputx('msg',"invalid value for "!!name!!":"!!value);
|
||||||
|
call symputx('err',1);
|
||||||
|
stop;
|
||||||
|
end;
|
||||||
|
else if name='LIBCAT' then do;
|
||||||
|
if exist(value,'CATALOG') le 0 then do;
|
||||||
|
call symputx('msg',"Unable to open catalog: "!!value);
|
||||||
|
call symputx('err',1);
|
||||||
|
stop;
|
||||||
|
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'))
|
||||||
|
and missing(value) then do;
|
||||||
|
call symputx('msg',"missing value in var: "!!name);
|
||||||
|
call symputx('err',1);
|
||||||
|
stop;
|
||||||
|
end;
|
||||||
|
run;
|
||||||
|
|
||||||
|
%mp_abort(
|
||||||
|
iftrue=(&err ne 0)
|
||||||
|
,mac=&sysmacroname
|
||||||
|
,msg=%str(&msg)
|
||||||
|
)
|
||||||
|
|
||||||
|
/**
|
||||||
|
* First, extract only relevant formats from the catalog
|
||||||
|
*/
|
||||||
|
proc sql noprint;
|
||||||
|
select distinct fmtname into: fmtlist separated by ' ' from &libds;
|
||||||
|
|
||||||
|
%mp_cntlout(libcat=&libcat,fmtlist=&fmtlist,cntlout=&base_fmts)
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Ensure input table and base_formats have consistent lengths and types
|
||||||
|
*/
|
||||||
|
%mddl_sas_cntlout(libds=&template)
|
||||||
|
data &inlibds;
|
||||||
|
if 0 then set &template;
|
||||||
|
set &libds;
|
||||||
|
if missing(type) then do;
|
||||||
|
if substr(fmtname,1,1)='$' then type='C';
|
||||||
|
else type='N';
|
||||||
|
end;
|
||||||
|
if type='N' then do;
|
||||||
|
start=cats(start);
|
||||||
|
end=cats(end);
|
||||||
|
end;
|
||||||
|
run;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Identify new records
|
||||||
|
*/
|
||||||
|
proc sql;
|
||||||
|
create table &outds_add(drop=&delete_col) as
|
||||||
|
select a.*
|
||||||
|
from &inlibds a
|
||||||
|
left join &base_fmts b
|
||||||
|
on a.fmtname=b.fmtname
|
||||||
|
and a.start=b.start
|
||||||
|
where b.fmtname is null
|
||||||
|
and upcase(a.&delete_col) ne "YES"
|
||||||
|
order by fmtname, start;;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Identify deleted records
|
||||||
|
*/
|
||||||
|
create table &outds_del(drop=&delete_col) as
|
||||||
|
select a.*
|
||||||
|
from &inlibds a
|
||||||
|
inner join &base_fmts b
|
||||||
|
on a.fmtname=b.fmtname
|
||||||
|
and a.start=b.start
|
||||||
|
where upcase(a.&delete_col)="YES"
|
||||||
|
order by fmtname, start;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Identify modified records
|
||||||
|
*/
|
||||||
|
create table &outds_mod (drop=&delete_col) as
|
||||||
|
select a.*
|
||||||
|
from &inlibds a
|
||||||
|
inner join &base_fmts b
|
||||||
|
on a.fmtname=b.fmtname
|
||||||
|
and a.start=b.start
|
||||||
|
where upcase(a.&delete_col) ne "YES"
|
||||||
|
order by fmtname, start;
|
||||||
|
|
||||||
|
options ibufsize=&ibufsize;
|
||||||
|
|
||||||
|
%mp_abort(
|
||||||
|
iftrue=(&syscc ne 0)
|
||||||
|
,mac=&sysmacroname
|
||||||
|
,msg=%str(SYSCC=&syscc prior to load prep)
|
||||||
|
)
|
||||||
|
|
||||||
|
%if &loadtarget=YES %then %do;
|
||||||
|
data &ds1;
|
||||||
|
merge &base_fmts(in=base)
|
||||||
|
&outds_mod(in=mod)
|
||||||
|
&outds_add(in=add)
|
||||||
|
&outds_del(in=del);
|
||||||
|
if not del and not mod;
|
||||||
|
by fmtname start;
|
||||||
|
run;
|
||||||
|
data &stagedata;
|
||||||
|
set &ds1 &outds_mod;
|
||||||
|
run;
|
||||||
|
proc sort;
|
||||||
|
by fmtname start;
|
||||||
|
run;
|
||||||
|
%end;
|
||||||
|
/* mp abort needs to run outside of conditional blocks */
|
||||||
|
%mp_abort(
|
||||||
|
iftrue=(&syscc ne 0)
|
||||||
|
,mac=&sysmacroname
|
||||||
|
,msg=%str(SYSCC=&syscc prior to actual load)
|
||||||
|
)
|
||||||
|
%if &loadtarget=YES %then %do;
|
||||||
|
%if %mf_nobs(&stagedata)=0 %then %do;
|
||||||
|
%put There are no changes to load in &libcat!;
|
||||||
|
%return;
|
||||||
|
%end;
|
||||||
|
%if &locklibds ne 0 %then %do;
|
||||||
|
/* prevent parallel updates */
|
||||||
|
%mp_lockanytable(LOCK
|
||||||
|
,lib=%scan(&libcat,1,.)
|
||||||
|
,ds=%scan(&libcat,2,.)-FC
|
||||||
|
,ref=MP_LOADFORMAT commencing format load
|
||||||
|
,ctl_ds=&locklibds
|
||||||
|
)
|
||||||
|
%end;
|
||||||
|
/* do the actual load */
|
||||||
|
proc format lib=&libcat cntlin=&stagedata;
|
||||||
|
run;
|
||||||
|
%if &locklibds ne 0 %then %do;
|
||||||
|
/* unlock the table */
|
||||||
|
%mp_lockanytable(UNLOCK
|
||||||
|
,lib=%scan(&libcat,1,.)
|
||||||
|
,ds=%scan(&libcat,2,.)-FC
|
||||||
|
,ref=MP_LOADFORMAT completed format load
|
||||||
|
,ctl_ds=&locklibds
|
||||||
|
)
|
||||||
|
%end;
|
||||||
|
/* track the changes */
|
||||||
|
%if &auditlibds ne 0 %then %do;
|
||||||
|
%if &locklibds ne 0 %then %do;
|
||||||
|
%mp_lockanytable(LOCK
|
||||||
|
,lib=%scan(&auditlibds,1,.)
|
||||||
|
,ds=%scan(&auditlibds,2,.)
|
||||||
|
,ref=MP_LOADFORMAT commencing audit table load
|
||||||
|
,ctl_ds=&locklibds
|
||||||
|
)
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%mp_storediffs(&libcat-FC
|
||||||
|
,&inlibds
|
||||||
|
,FMTNAME START
|
||||||
|
,delds=&outds_del
|
||||||
|
,modds=&outds_mod
|
||||||
|
,appds=&outds_add
|
||||||
|
,outds=&storediffs
|
||||||
|
,mdebug=&mdebug
|
||||||
|
)
|
||||||
|
|
||||||
|
%if &locklibds ne 0 %then %do;
|
||||||
|
%mp_lockanytable(UNLOCK
|
||||||
|
,lib=%scan(&auditlibds,1,.)
|
||||||
|
,ds=%scan(&auditlibds,2,.)
|
||||||
|
,ref=MP_LOADFORMAT commencing audit table load
|
||||||
|
,ctl_ds=&locklibds
|
||||||
|
)
|
||||||
|
%end;
|
||||||
|
%end;
|
||||||
|
%end;
|
||||||
|
%mp_abort(
|
||||||
|
iftrue=(&syscc ne 0)
|
||||||
|
,mac=&sysmacroname
|
||||||
|
,msg=%str(SYSCC=&syscc after load)
|
||||||
|
)
|
||||||
|
|
||||||
|
%if &mdebug=0 %then %do;
|
||||||
|
proc datasets lib=work;
|
||||||
|
delete &prefix:;
|
||||||
|
run;
|
||||||
|
%put &sysmacroname exit vars:;
|
||||||
|
%put _local_;
|
||||||
|
%end;
|
||||||
|
%mend mp_loadformat;
|
||||||
@@ -1,23 +1,23 @@
|
|||||||
/**
|
/**
|
||||||
@file
|
@file
|
||||||
@brief Mechanism for locking tables to prevent parallel modifications
|
@brief Mechanism for locking tables to prevent parallel modifications
|
||||||
@details Uses a control table to enable ANY table to be locked for updates.
|
@details Uses a control table to enable ANY table to be locked for updates
|
||||||
|
(not just SAS datasets).
|
||||||
Only useful if every update uses the macro! Used heavily within
|
Only useful if every update uses the macro! Used heavily within
|
||||||
[Data Controller for SAS](https://datacontroller.io).
|
[Data Controller for SAS](https://datacontroller.io).
|
||||||
|
|
||||||
The underlying table is structured as per the MAKETABLE action.
|
|
||||||
|
|
||||||
@param [in] action The action to be performed. Valid values:
|
@param [in] action The action to be performed. Valid values:
|
||||||
@li LOCK - Sets the lock flag, also confirms if a SAS lock is available
|
@li LOCK - Sets the lock flag, also confirms if a SAS lock is available
|
||||||
@li UNLOCK - Unlocks the table
|
@li UNLOCK - Unlocks the table
|
||||||
@li MAKETABLE - creates the control table (ctl_ds)
|
|
||||||
@param [in] lib= (WORK) The libref of the table to lock. Should already be
|
@param [in] lib= (WORK) The libref of the table to lock. Should already be
|
||||||
assigned.
|
assigned.
|
||||||
@param [in] ds= The dataset to lock
|
@param [in] ds= The dataset to lock
|
||||||
@param [in] ref= A meaningful reference to enable the lock to be traced. Max
|
@param [in] ref= A meaningful reference to enable the lock to be traced. Max
|
||||||
length is 200 characters.
|
length is 200 characters.
|
||||||
@param [out] ctl_ds= (0) The control table which controls the actual locking.
|
@param [out] ctl_ds= (0) The control table which controls the actual locking.
|
||||||
Should already be assigned and available.
|
Should already be assigned and available. The definition is available by
|
||||||
|
running the mddl_dc_locktable.sas macro.
|
||||||
|
|
||||||
@param [in] loops= (25) Number of times to check for a lock.
|
@param [in] loops= (25) Number of times to check for a lock.
|
||||||
@param [in] loop_secs= (1) Seconds to wait between each lock attempt
|
@param [in] loop_secs= (1) Seconds to wait between each lock attempt
|
||||||
|
|
||||||
@@ -49,7 +49,7 @@ data _null_;
|
|||||||
put name '=' value;
|
put name '=' value;
|
||||||
run;
|
run;
|
||||||
|
|
||||||
%mp_abort(iftrue= (&ds=0 and &action ne MAKETABLE)
|
%mp_abort(iftrue= ("&ds"="0" and &action ne MAKETABLE)
|
||||||
,mac=&sysmacroname
|
,mac=&sysmacroname
|
||||||
,msg=%str(dataset was not provided)
|
,msg=%str(dataset was not provided)
|
||||||
)
|
)
|
||||||
@@ -86,7 +86,7 @@ run;
|
|||||||
/* do not proceed if no observations can be processed */
|
/* do not proceed if no observations can be processed */
|
||||||
%mp_abort(iftrue= (%sysfunc(getoption(OBS))=0)
|
%mp_abort(iftrue= (%sysfunc(getoption(OBS))=0)
|
||||||
,mac=&sysmacroname
|
,mac=&sysmacroname
|
||||||
,msg=%str(options obs = 0. syserrortext=&syserrortext)
|
,msg=%str(cannot continue when options obs = 0)
|
||||||
)
|
)
|
||||||
|
|
||||||
%if &ACTION=LOCK %then %do;
|
%if &ACTION=LOCK %then %do;
|
||||||
@@ -180,7 +180,7 @@ run;
|
|||||||
%else %do;
|
%else %do;
|
||||||
data _null_;
|
data _null_;
|
||||||
putlog 'NOTE-' / 'NOTE-';
|
putlog 'NOTE-' / 'NOTE-';
|
||||||
putlog "NOTE- &sysmacroname: Table &lib..&ds locked at "@
|
putlog "NOTE- &sysmacroname: Table &lib..&ds locked at "@;
|
||||||
putlog " %sysfunc(datetime(),datetime19.) (iteration &x)"@;
|
putlog " %sysfunc(datetime(),datetime19.) (iteration &x)"@;
|
||||||
putlog 'NOTE-' / 'NOTE-';
|
putlog 'NOTE-' / 'NOTE-';
|
||||||
run;
|
run;
|
||||||
@@ -221,25 +221,12 @@ run;
|
|||||||
%let abortme=1;
|
%let abortme=1;
|
||||||
%end;
|
%end;
|
||||||
%end;
|
%end;
|
||||||
%else %if &action=MAKETABLE %then %do;
|
|
||||||
proc sql;
|
|
||||||
create table &ctl_ds(
|
|
||||||
lock_lib char(8),
|
|
||||||
lock_ds char(32),
|
|
||||||
lock_status_cd char(10) not null,
|
|
||||||
lock_user_nm char(100) not null ,
|
|
||||||
lock_ref char(200),
|
|
||||||
lock_pid char(10),
|
|
||||||
lock_start_dttm num format=E8601DT26.6,
|
|
||||||
lock_end_dttm num format=E8601DT26.6,
|
|
||||||
constraint pk_mp_lockanytable primary key(lock_lib,lock_ds));
|
|
||||||
%end;
|
|
||||||
%else %do;
|
%else %do;
|
||||||
%let msg=lock_anytable given unsupported action (&action);
|
%let msg=lock_anytable given unsupported action (&action);
|
||||||
%let abortme=1;
|
%let abortme=1;
|
||||||
%end;
|
%end;
|
||||||
|
|
||||||
/* catch errors - mp_abort must be called outside of a logic block */
|
/* catch errs - mp_abort must be called outside of a logic block */
|
||||||
%mp_abort(iftrue=(&abortme=1),
|
%mp_abort(iftrue=(&abortme=1),
|
||||||
msg=%superq(msg),
|
msg=%superq(msg),
|
||||||
mac=&sysmacroname
|
mac=&sysmacroname
|
||||||
|
|||||||
@@ -37,7 +37,7 @@ run;
|
|||||||
,mac=checklock.sas
|
,mac=checklock.sas
|
||||||
,msg=Aborting with syscc=&syscc on entry.
|
,msg=Aborting with syscc=&syscc on entry.
|
||||||
)
|
)
|
||||||
%mp_abort(iftrue= (&libds=0)
|
%mp_abort(iftrue= ("&libds"="0")
|
||||||
,mac=&sysmacroname
|
,mac=&sysmacroname
|
||||||
,msg=%str(libds not provided)
|
,msg=%str(libds not provided)
|
||||||
)
|
)
|
||||||
@@ -46,6 +46,12 @@ run;
|
|||||||
%let lib=%upcase(%scan(&libds,1,.));
|
%let lib=%upcase(%scan(&libds,1,.));
|
||||||
%let ds=%upcase(%scan(&libds,2,.));
|
%let ds=%upcase(%scan(&libds,2,.));
|
||||||
|
|
||||||
|
/* in DC, format catalogs are passed with a -FC suffix. No saslock here! */
|
||||||
|
%if %scan(&libds,2,-)=FC %then %do;
|
||||||
|
%put &sysmacroname: Format Catalog detected, no lockfile applied to &libds;
|
||||||
|
%return;
|
||||||
|
%end;
|
||||||
|
|
||||||
/* do not proceed if no observations can be processed */
|
/* do not proceed if no observations can be processed */
|
||||||
%let msg=options obs = 0. syserrortext=%superq(syserrortext);
|
%let msg=options obs = 0. syserrortext=%superq(syserrortext);
|
||||||
%mp_abort(iftrue= (%sysfunc(getoption(OBS))=0)
|
%mp_abort(iftrue= (%sysfunc(getoption(OBS))=0)
|
||||||
|
|||||||
@@ -10,8 +10,6 @@
|
|||||||
according to the variable types and formats.
|
according to the variable types and formats.
|
||||||
|
|
||||||
TODO:
|
TODO:
|
||||||
@li Respect PKs
|
|
||||||
@li Respect NOT NULLs
|
|
||||||
@li Consider dates, datetimes, times, integers etc
|
@li Consider dates, datetimes, times, integers etc
|
||||||
|
|
||||||
Usage:
|
Usage:
|
||||||
@@ -27,16 +25,23 @@
|
|||||||
);
|
);
|
||||||
%mp_makedata(work.example)
|
%mp_makedata(work.example)
|
||||||
|
|
||||||
@param [in] libds The empty table in which to create data
|
@param [in] libds The empty table (libref.dataset) in which to create data
|
||||||
@param [out] obs= (500) The number of records to create.
|
@param [out] obs= (500) The maximum number of records to create. The table
|
||||||
|
is sorted with nodup on the primary key, so the actual number of records may
|
||||||
|
be lower than this.
|
||||||
|
|
||||||
<h4> SAS Macros </h4>
|
<h4> SAS Macros </h4>
|
||||||
@li mf_getuniquename.sas
|
@li mf_getuniquename.sas
|
||||||
@li mf_getvarlen.sas
|
@li mf_getvarlen.sas
|
||||||
|
@li mf_getvarlist.sas
|
||||||
|
@li mf_islibds.sas
|
||||||
@li mf_nobs.sas
|
@li mf_nobs.sas
|
||||||
@li mp_getcols.sas
|
@li mp_getcols.sas
|
||||||
@li mp_getpk.sas
|
@li mp_getpk.sas
|
||||||
|
|
||||||
|
<h4> Related Macros </h4>
|
||||||
|
@li mp_makedata.test.sas
|
||||||
|
|
||||||
@version 9.2
|
@version 9.2
|
||||||
@author Allan Bowe
|
@author Allan Bowe
|
||||||
|
|
||||||
@@ -44,44 +49,59 @@
|
|||||||
|
|
||||||
%macro mp_makedata(libds
|
%macro mp_makedata(libds
|
||||||
,obs=500
|
,obs=500
|
||||||
|
,seed=1
|
||||||
)/*/STORE SOURCE*/;
|
)/*/STORE SOURCE*/;
|
||||||
|
|
||||||
%local ds1 c1 n1 i col charvars numvars;
|
%local ds1 ds2 lib ds pk_fields i col charvars numvars ispk;
|
||||||
|
|
||||||
%if %mf_nobs(&libds)>0 %then %do;
|
%if %mf_islibds(&libds)=0 %then %do;
|
||||||
|
%put &sysmacroname: Invalid libds (&libds) - should be library.dataset format;
|
||||||
|
%return;
|
||||||
|
%end;
|
||||||
|
%else %if %mf_nobs(&libds)>0 %then %do;
|
||||||
%put &sysmacroname: &libds has data, it will not be recreated;
|
%put &sysmacroname: &libds has data, it will not be recreated;
|
||||||
%return;
|
%return;
|
||||||
%end;
|
%end;
|
||||||
|
|
||||||
%local ds1 c1 n1;
|
/* set up temporary vars */
|
||||||
%let ds1=%mf_getuniquename(prefix=mp_makedata);
|
%let ds1=%mf_getuniquename(prefix=mp_makedatads1);
|
||||||
%let c1=%mf_getuniquename(prefix=mp_makedatacol);
|
%let ds2=%mf_getuniquename(prefix=mp_makedatads2);
|
||||||
%let n1=%mf_getuniquename(prefix=mp_makedatacol);
|
%let lib=%scan(&libds,1,.);
|
||||||
data &ds1;
|
%let ds=%scan(&libds,2,.);
|
||||||
|
|
||||||
|
/* grab the primary key vars */
|
||||||
|
%mp_getpk(&lib,ds=&ds,outds=&ds1)
|
||||||
|
|
||||||
|
proc sql noprint;
|
||||||
|
select coalescec(pk_fields,'_all_') into: pk_fields from &ds1;
|
||||||
|
|
||||||
|
data &ds2;
|
||||||
if 0 then set &libds;
|
if 0 then set &libds;
|
||||||
do _n_=1 to &obs;
|
do _n_=1 to &obs;
|
||||||
&c1=repeat(uuidgen(),10);
|
|
||||||
&n1=ranuni(1)*5000000;
|
|
||||||
drop &c1 &n1;
|
|
||||||
%let charvars=%mf_getvarlist(&libds,typefilter=C);
|
%let charvars=%mf_getvarlist(&libds,typefilter=C);
|
||||||
%do i=1 %to %sysfunc(countw(&charvars));
|
%if &charvars ^= %then %do i=1 %to %sysfunc(countw(&charvars));
|
||||||
%let col=%scan(&charvars,&i);
|
%let col=%scan(&charvars,&i);
|
||||||
&col=subpad(&c1,1,%mf_getvarlen(&libds,&col));
|
/* create random value based on observation number and colum length */
|
||||||
|
&col=repeat(put(md5(cats(_n_)),$hex32.),%mf_getvarlen(&libds,&col)/32);
|
||||||
%end;
|
%end;
|
||||||
|
|
||||||
%let numvars=%mf_getvarlist(&libds,typefilter=N);
|
%let numvars=%mf_getvarlist(&libds,typefilter=N);
|
||||||
%do i=1 %to %sysfunc(countw(&numvars));
|
%if &numvars ^= %then %do i=1 %to %sysfunc(countw(&numvars));
|
||||||
%let col=%scan(&numvars,&i);
|
%let col=%scan(&numvars,&i);
|
||||||
&col=&n1;
|
&col=_n_;
|
||||||
%end;
|
%end;
|
||||||
output;
|
output;
|
||||||
end;
|
end;
|
||||||
|
stop;
|
||||||
|
run;
|
||||||
|
proc sort data=&ds2 nodupkey;
|
||||||
|
by &pk_fields;
|
||||||
run;
|
run;
|
||||||
|
|
||||||
proc append base=&libds data=&ds1;
|
proc append base=&libds data=&ds2;
|
||||||
run;
|
run;
|
||||||
|
|
||||||
proc sql;
|
proc sql;
|
||||||
drop table &ds1;
|
drop table &ds1, &ds2;
|
||||||
|
|
||||||
%mend mp_makedata;
|
%mend mp_makedata;
|
||||||
58
base/mp_md5.sas
Normal file
58
base/mp_md5.sas
Normal file
@@ -0,0 +1,58 @@
|
|||||||
|
/**
|
||||||
|
@file
|
||||||
|
@brief Generates an md5 expression for hashing a set of variables
|
||||||
|
@details This is the same algorithm used to hash records in
|
||||||
|
[Data Controller for SAS](https://datacontroller.io) (free for up
|
||||||
|
to 5 users).
|
||||||
|
|
||||||
|
It is not designed to be efficient - it is designed to be effective,
|
||||||
|
given the range of edge cases (large floating points, special missing
|
||||||
|
numerics, thousands of columns, very wide columns).
|
||||||
|
|
||||||
|
It can be used only in data step, eg as follows:
|
||||||
|
|
||||||
|
data _null_;
|
||||||
|
set sashelp.class;
|
||||||
|
hashvar=%mp_md5(cvars=name sex, nvars=age height weight);
|
||||||
|
put hashvar=;
|
||||||
|
run;
|
||||||
|
|
||||||
|
Unfortunately it will not run in SQL - it fails with the following message:
|
||||||
|
|
||||||
|
> The width value for HEX is out of bounds. It should be between 1 and 16
|
||||||
|
|
||||||
|
The macro will also cause errors if the data contains (non-special) missings
|
||||||
|
and the (undocumented) `options dsoptions=nonote2err;` is in effect.
|
||||||
|
|
||||||
|
This can be avoided in two ways:
|
||||||
|
|
||||||
|
@li Global option: `options dsoptions=nonote2err;`
|
||||||
|
@li Data step option: `data YOURLIB.YOURDATASET /nonote2err;`
|
||||||
|
|
||||||
|
@param cvars= Space seperated list of character variables
|
||||||
|
@param nvars= Space seperated list of numeric variables
|
||||||
|
|
||||||
|
<h4> Related Programs </h4>
|
||||||
|
@li mp_init.sas
|
||||||
|
|
||||||
|
@version 9.2
|
||||||
|
@author Allan Bowe
|
||||||
|
**/
|
||||||
|
|
||||||
|
%macro mp_md5(cvars=,nvars=);
|
||||||
|
%local i var sep;
|
||||||
|
put(md5(
|
||||||
|
%do i=1 %to %sysfunc(countw(&cvars));
|
||||||
|
%let var=%scan(&cvars,&i,%str( ));
|
||||||
|
&sep put(md5(trim(&var)),$hex32.)
|
||||||
|
%let sep=!!;
|
||||||
|
%end;
|
||||||
|
%do i=1 %to %sysfunc(countw(&nvars));
|
||||||
|
%let var=%scan(&nvars,&i,%str( ));
|
||||||
|
/* multiply by 1 to strip precision errors (eg 0 != 0) */
|
||||||
|
/* but ONLY if not missing, else will lose any special missing values */
|
||||||
|
&sep put(md5(trim(put(ifn(missing(&var),&var,&var*1),binary64.))),$hex32.)
|
||||||
|
%let sep=!!;
|
||||||
|
%end;
|
||||||
|
),hex32.)
|
||||||
|
%mend mp_md5;
|
||||||
149
base/mp_replace.sas
Normal file
149
base/mp_replace.sas
Normal file
@@ -0,0 +1,149 @@
|
|||||||
|
/**
|
||||||
|
@file
|
||||||
|
@brief Performs a text substitution on a file
|
||||||
|
@details Performs a find and replace on a file, either in place or to a new
|
||||||
|
file. Can be used on files where lines are longer than 32767.
|
||||||
|
|
||||||
|
Works by reading in the file byte by byte, then marking the beginning and end
|
||||||
|
of each matched string, before finally doing the replace.
|
||||||
|
|
||||||
|
Full credit for this highly efficient and syntactically satisfying SAS logic
|
||||||
|
goes to [Bartosz Jabłoński](https://www.linkedin.com/in/yabwon), founder of
|
||||||
|
the [SAS Packages](https://github.com/yabwon/SAS_PACKAGES) framework.
|
||||||
|
|
||||||
|
Usage:
|
||||||
|
|
||||||
|
%let file="%sysfunc(pathname(work))/file.txt";
|
||||||
|
%let str=replace/me;
|
||||||
|
%let rep=with/this;
|
||||||
|
data _null_;
|
||||||
|
file &file;
|
||||||
|
put 'blahblah';
|
||||||
|
put "blahblah&str.blah";
|
||||||
|
put 'blahblahblah';
|
||||||
|
run;
|
||||||
|
%mp_replace(&file, findvar=str, replacevar=rep)
|
||||||
|
data _null_;
|
||||||
|
infile &file;
|
||||||
|
input;
|
||||||
|
list;
|
||||||
|
run;
|
||||||
|
|
||||||
|
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
|
||||||
|
|
||||||
|
@param infile The QUOTED path to the file on which to perform the substitution
|
||||||
|
@param findvar= Macro variable NAME containing the string to search for
|
||||||
|
@param replacevar= Macro variable NAME containing the replacement string
|
||||||
|
@param outfile= (0) Optional QUOTED path to an the adjusted output file (to
|
||||||
|
avoid overwriting the first file).
|
||||||
|
|
||||||
|
<h4> SAS Macros </h4>
|
||||||
|
@li mf_getuniquefileref.sas
|
||||||
|
@li mf_getuniquename.sas
|
||||||
|
|
||||||
|
<h4> Related Macros </h4>
|
||||||
|
@li mp_gsubfile.test.sas
|
||||||
|
|
||||||
|
@version 9.4
|
||||||
|
@author Bartosz Jabłoński
|
||||||
|
@author Allan Bowe
|
||||||
|
**/
|
||||||
|
|
||||||
|
%macro mp_replace(infile,
|
||||||
|
findvar=,
|
||||||
|
replacevar=,
|
||||||
|
outfile=0
|
||||||
|
)/*/STORE SOURCE*/;
|
||||||
|
|
||||||
|
%local inref dttm ds1;
|
||||||
|
%let inref=%mf_getuniquefileref();
|
||||||
|
%let outref=%mf_getuniquefileref();
|
||||||
|
%if &outfile=0 %then %let outfile=&infile;
|
||||||
|
%let ds1=%mf_getuniquename(prefix=allchars);
|
||||||
|
%let ds2=%mf_getuniquename(prefix=startmark);
|
||||||
|
|
||||||
|
/* START */
|
||||||
|
%let dttm=%sysfunc(datetime());
|
||||||
|
|
||||||
|
filename &inref &infile lrecl=1 recfm=n;
|
||||||
|
|
||||||
|
data &ds1;
|
||||||
|
infile &inref;
|
||||||
|
input sourcechar $ 1. @@;
|
||||||
|
format sourcechar hex2.;
|
||||||
|
run;
|
||||||
|
|
||||||
|
data &ds2;
|
||||||
|
/* set find string to length in bytes to cover trailing spaces */
|
||||||
|
length string $ %length(%superq(&findvar));
|
||||||
|
string =symget("&findvar");
|
||||||
|
drop string;
|
||||||
|
|
||||||
|
firstchar=char(string,1);
|
||||||
|
findlen=lengthm(string); /* <- for trailing bytes */
|
||||||
|
|
||||||
|
do _N_=1 to nobs;
|
||||||
|
set &ds1 nobs=nobs point=_N_;
|
||||||
|
if sourcechar=firstchar then do;
|
||||||
|
pos=1;
|
||||||
|
s=0;
|
||||||
|
do point=_N_ to min(_N_ + findlen -1,nobs);
|
||||||
|
set &ds1 point=point;
|
||||||
|
if sourcechar=char(string, pos) then s + 1;
|
||||||
|
else goto _leave_;
|
||||||
|
pos+1;
|
||||||
|
end;
|
||||||
|
_leave_:
|
||||||
|
if s=findlen then do;
|
||||||
|
START =_N_;
|
||||||
|
_N_ =_N_+ s - 1;
|
||||||
|
STOP =_N_;
|
||||||
|
output;
|
||||||
|
end;
|
||||||
|
end;
|
||||||
|
end;
|
||||||
|
stop;
|
||||||
|
keep START STOP;
|
||||||
|
run;
|
||||||
|
|
||||||
|
data &ds1;
|
||||||
|
declare hash HS(dataset:"&ds2(keep=start)");
|
||||||
|
HS.defineKey("start");
|
||||||
|
HS.defineDone();
|
||||||
|
declare hash HE(dataset:"&ds2(keep=stop)");
|
||||||
|
HE.defineKey("stop");
|
||||||
|
HE.defineDone();
|
||||||
|
do until(eof);
|
||||||
|
set &ds1 end=eof curobs =n;
|
||||||
|
start = ^HS.check(key:n);
|
||||||
|
stop = ^HE.check(key:n);
|
||||||
|
length strt $ 1;
|
||||||
|
strt =put(start,best. -L);
|
||||||
|
retain out 1;
|
||||||
|
if out then output;
|
||||||
|
if start then out=0;
|
||||||
|
if stop then out=1;
|
||||||
|
end;
|
||||||
|
stop;
|
||||||
|
keep sourcechar strt;
|
||||||
|
run;
|
||||||
|
|
||||||
|
filename &outref &outfile recfm=n;
|
||||||
|
|
||||||
|
data _null_;
|
||||||
|
length replace $ %length(%superq(&replacevar));
|
||||||
|
replace=symget("&replacevar");
|
||||||
|
file &outref;
|
||||||
|
do until(eof);
|
||||||
|
set &ds1 end=eof;
|
||||||
|
if strt ="1" then put replace char.;
|
||||||
|
else put sourcechar char1.;
|
||||||
|
end;
|
||||||
|
stop;
|
||||||
|
run;
|
||||||
|
|
||||||
|
/* END */
|
||||||
|
%put &sysmacroname took %sysevalf(%sysfunc(datetime())-&dttm) seconds to run;
|
||||||
|
|
||||||
|
%mend mp_replace;
|
||||||
@@ -3,13 +3,15 @@
|
|||||||
@brief Reset an option to original value
|
@brief Reset an option to original value
|
||||||
@details Inspired by the SAS Jedi -
|
@details Inspired by the SAS Jedi -
|
||||||
https://blogs.sas.com/content/sastraining/2012/08/14/jedi-sas-tricks-reset-sas-system-options
|
https://blogs.sas.com/content/sastraining/2012/08/14/jedi-sas-tricks-reset-sas-system-options
|
||||||
Called as follows:
|
|
||||||
|
|
||||||
options obs=30;
|
Called as follows:
|
||||||
%mp_resetoption(OBS)
|
|
||||||
|
options obs=30 ps=max;
|
||||||
|
%mp_resetoption(OBS)
|
||||||
|
%mp_resetoption(PS)
|
||||||
|
|
||||||
|
|
||||||
@param option the option to reset
|
@param [in] option the option to reset
|
||||||
|
|
||||||
@version 9.2
|
@version 9.2
|
||||||
@author Allan Bowe
|
@author Allan Bowe
|
||||||
|
|||||||
246
base/mp_retainedkey.sas
Normal file
246
base/mp_retainedkey.sas
Normal file
@@ -0,0 +1,246 @@
|
|||||||
|
/**
|
||||||
|
@file
|
||||||
|
@brief Generate and apply retained key values to a staging table
|
||||||
|
@details This macro will populate a staging table with a Retained Key based on
|
||||||
|
a business key and a base (target) table.
|
||||||
|
|
||||||
|
Definition of retained key ([source](
|
||||||
|
http://bukhantsov.org/2012/04/what-is-data-vault/)):
|
||||||
|
|
||||||
|
> The retained key is a key which is mapped to business key one-to-one. In
|
||||||
|
> comparison, the surrogate key includes time and there can be many surrogate
|
||||||
|
> keys corresponding to one business key. This explains the name of the key,
|
||||||
|
> it is retained with insertion of a new version of a row while surrogate key
|
||||||
|
> is increasing.
|
||||||
|
|
||||||
|
This macro is designed to be used as part of a wider load / ETL process (such
|
||||||
|
as the one in [Data Controller for SAS](https://datacontroller.io)).
|
||||||
|
|
||||||
|
Specifically, the macro assumes that the base table has already been 'locked'
|
||||||
|
(eg with the mp_lockanytable.sas macro) prior to invocation. Also, several
|
||||||
|
tables are assumed to exist (names are configurable):
|
||||||
|
|
||||||
|
@li work.staging_table - the staged data, minus the retained key element
|
||||||
|
@li permlib.base_table - the target table to be loaded (**not** loaded by this
|
||||||
|
macro)
|
||||||
|
@li permlib.maxkeytable - optional, used to store load metaadata.
|
||||||
|
The definition is available by running mp_coretable.sas as follows:
|
||||||
|
`mp_coretable(MAXKEYTABLE)`.
|
||||||
|
@li permlib.locktable - Necessary if maxkeytable is being populated. The
|
||||||
|
definition is available by running mp_coretable.sas as follows:
|
||||||
|
`mp_coretable(LOCKTABLE)`.
|
||||||
|
|
||||||
|
|
||||||
|
@param [in] base_lib= (WORK) Libref of the base (target) table.
|
||||||
|
@param [in] base_dsn= (BASETABLE) Name of the base (target) table.
|
||||||
|
@param [in] append_lib= (WORK) Libref of the staging table
|
||||||
|
@param [in] append_dsn= (APPENDTABLE) Name of the staging table
|
||||||
|
@param [in] retained_key= (DEFAULT_RK) Name of RK to generate (should exist on
|
||||||
|
base table)
|
||||||
|
@param [in] business_key= (PK1 PK2) Business key against which to generate
|
||||||
|
RK values. Should be unique and not null on the staging table.
|
||||||
|
@param [in] check_uniqueness=(NO) Set to yes to perform a uniqueness check.
|
||||||
|
Recommended if there is a chance that the staging data is not unique on the
|
||||||
|
business key.
|
||||||
|
@param [in] maxkeytable= (0) Provide a maxkeytable libds reference here, to
|
||||||
|
store load metadata (maxkey val, load time). Set to zero if metadata is not
|
||||||
|
required, eg, when preparing a 'dummy' load. Structure is described above.
|
||||||
|
See below for sample data.
|
||||||
|
|KEYTABLE:$32.|KEYCOLUMN:$32.|MAX_KEY:best.|PROCESSED_DTTM:E8601DT26.6|
|
||||||
|
|---|---|---|---|
|
||||||
|
|`DC487173.MPE_SELECTBOX `|`SELECTBOX_RK `|`55 `|`1950427787.8 `|
|
||||||
|
|`DC487173.MPE_FILTERANYTABLE `|`filter_rk `|`14 `|`1951053886.8 `|
|
||||||
|
@param [in] locktable= (0) If updating the maxkeytable, provide the libds
|
||||||
|
reference to the lock table (per mp_lockanytable.sas macro)
|
||||||
|
@param [in] filter_str= Apply a filter - useful for SCD2 or BITEMPORAL loads.
|
||||||
|
Example: `filter_str=%str( (where=( &now < &tech_to)) )`
|
||||||
|
@param [out] outds= (WORK.APPEND) Output table (staging table + retained key)
|
||||||
|
|
||||||
|
<h4> SAS Macros </h4>
|
||||||
|
@li mf_existvar.sas
|
||||||
|
@li mf_getquotedstr.sas
|
||||||
|
@li mf_getuniquename.sas
|
||||||
|
@li mf_nobs.sas
|
||||||
|
@li mp_abort.sas
|
||||||
|
@li mp_lockanytable.sas
|
||||||
|
|
||||||
|
<h4> Related Macros </h4>
|
||||||
|
@li mp_filterstore.sas
|
||||||
|
@li mp_retainedkey.test.sas
|
||||||
|
|
||||||
|
@version 9.2
|
||||||
|
|
||||||
|
**/
|
||||||
|
|
||||||
|
%macro mp_retainedkey(
|
||||||
|
base_lib=WORK
|
||||||
|
,base_dsn=BASETABLE
|
||||||
|
,append_lib=WORK
|
||||||
|
,append_dsn=APPENDTABLE
|
||||||
|
,retained_key=DEFAULT_RK
|
||||||
|
,business_key= PK1 PK2
|
||||||
|
,check_uniqueness=NO
|
||||||
|
,maxkeytable=0
|
||||||
|
,locktable=0
|
||||||
|
,outds=WORK.APPEND
|
||||||
|
,filter_str=
|
||||||
|
);
|
||||||
|
%put &sysmacroname entry vars:;
|
||||||
|
%put _local_;
|
||||||
|
|
||||||
|
%local base_libds app_libds key_field check maxkey idx_pk newkey_cnt iserr
|
||||||
|
msg x tempds1 tempds2 comma_pk appnobs checknobs dropvar tempvar idx_val;
|
||||||
|
%let base_libds=%upcase(&base_lib..&base_dsn);
|
||||||
|
%let app_libds=%upcase(&append_lib..&append_dsn);
|
||||||
|
%let tempds1=%mf_getuniquename();
|
||||||
|
%let tempds2=%mf_getuniquename();
|
||||||
|
%let comma_pk=%mf_getquotedstr(in_str=%str(&business_key),dlm=%str(,),quote=);
|
||||||
|
%let outds=%sysfunc(ifc(%index(&outds,.)=0,work.&outds,&outds));
|
||||||
|
/* validation checks */
|
||||||
|
%let iserr=0;
|
||||||
|
%if &syscc>0 %then %do;
|
||||||
|
%let iserr=1;
|
||||||
|
%let msg=%str(SYSCC=&syscc on macro entry);
|
||||||
|
%end;
|
||||||
|
%else %if %sysfunc(exist(&base_libds))=0 %then %do;
|
||||||
|
%let iserr=1;
|
||||||
|
%let msg=%str(Base LIBDS (&base_libds) expected but NOT FOUND);
|
||||||
|
%end;
|
||||||
|
%else %if %sysfunc(exist(&app_libds))=0 %then %do;
|
||||||
|
%let iserr=1;
|
||||||
|
%let msg=%str(Append LIBDS (&app_libds) expected but NOT FOUND);
|
||||||
|
%end;
|
||||||
|
%else %if &maxkeytable ne 0 and %sysfunc(exist(&maxkeytable))=0 %then %do;
|
||||||
|
%let iserr=1;
|
||||||
|
%let msg=%str(Maxkeytable (&maxkeytable) expected but NOT FOUND);
|
||||||
|
%end;
|
||||||
|
%else %if &maxkeytable ne 0 and %sysfunc(exist(&locktable))=0 %then %do;
|
||||||
|
%let iserr=1;
|
||||||
|
%let msg=%str(Locktable (&locktable) expected but NOT FOUND);
|
||||||
|
%end;
|
||||||
|
%else %if %length(&business_key)=0 %then %do;
|
||||||
|
%let iserr=1;
|
||||||
|
%let msg=%str(Business key (&business_key) expected but NOT FOUND);
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%do x=1 %to %sysfunc(countw(&business_key));
|
||||||
|
/* check business key values exist */
|
||||||
|
%let key_field=%scan(&business_key,&x,%str( ));
|
||||||
|
%if not %mf_existvar(&app_libds,&key_field) %then %do;
|
||||||
|
%let iserr=1;
|
||||||
|
%let msg=Business key (&key_field) not found on &app_libds!;
|
||||||
|
%goto err;
|
||||||
|
%end;
|
||||||
|
%else %if not %mf_existvar(&base_libds,&key_field) %then %do;
|
||||||
|
%let iserr=1;
|
||||||
|
%let msg=Business key (&key_field) not found on &base_libds!;
|
||||||
|
%goto err;
|
||||||
|
%end;
|
||||||
|
%end;
|
||||||
|
%err:
|
||||||
|
%if &iserr=1 %then %do;
|
||||||
|
/* err case so first perform an unlock of the base table before exiting */
|
||||||
|
%mp_lockanytable(
|
||||||
|
UNLOCK,lib=&base_lib,ds=&base_dsn,ref=%superq(msg),ctl_ds=&locktable
|
||||||
|
)
|
||||||
|
%end;
|
||||||
|
%mp_abort(iftrue=(&iserr=1),mac=mp_retainedkey,msg=%superq(msg))
|
||||||
|
|
||||||
|
proc sql noprint;
|
||||||
|
select sum(max(&retained_key),0) into: maxkey from &base_libds;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* get base table RK and bus field values for lookup
|
||||||
|
*/
|
||||||
|
proc sql noprint;
|
||||||
|
create table &tempds1 as
|
||||||
|
select distinct &comma_pk,&retained_key
|
||||||
|
from &base_libds &filter_str
|
||||||
|
order by &comma_pk,&retained_key;
|
||||||
|
|
||||||
|
%if &check_uniqueness=YES %then %do;
|
||||||
|
select count(*) into:checknobs
|
||||||
|
from (select distinct &comma_pk from &app_libds);
|
||||||
|
select count(*) into: appnobs from &app_libds; /* might be view */
|
||||||
|
%if &checknobs ne &appnobs %then %do;
|
||||||
|
%let msg=Source table &app_libds is not unique on (&business_key);
|
||||||
|
%let iserr=1;
|
||||||
|
%end;
|
||||||
|
%end;
|
||||||
|
%if &iserr=1 %then %do;
|
||||||
|
/* err case so first perform an unlock of the base table before exiting */
|
||||||
|
%mp_lockanytable(
|
||||||
|
UNLOCK,lib=&base_lib,ds=&base_dsn,ref=%superq(msg),ctl_ds=&locktable
|
||||||
|
)
|
||||||
|
%end;
|
||||||
|
%mp_abort(iftrue= (&iserr=1),mac=mp_retainedkey,msg=%superq(msg))
|
||||||
|
|
||||||
|
%if %mf_existvar(&app_libds,&retained_key)
|
||||||
|
%then %let dropvar=(drop=&retained_key);
|
||||||
|
|
||||||
|
/* prepare interim table with retained key populated for matching keys */
|
||||||
|
proc sql noprint;
|
||||||
|
create table &tempds2 as
|
||||||
|
select b.&retained_key, a.*
|
||||||
|
from &app_libds &dropvar a
|
||||||
|
left join &tempds1 b
|
||||||
|
on 1
|
||||||
|
%do idx_pk=1 %to %sysfunc(countw(&business_key));
|
||||||
|
%let idx_val=%scan(&business_key,&idx_pk);
|
||||||
|
and a.&idx_val=b.&idx_val
|
||||||
|
%end;
|
||||||
|
order by &retained_key;
|
||||||
|
|
||||||
|
/* identify the number of entries without retained keys (new records) */
|
||||||
|
select count(*) into: newkey_cnt
|
||||||
|
from &tempds2
|
||||||
|
where missing(&retained_key);
|
||||||
|
quit;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Update maxkey table if link provided
|
||||||
|
*/
|
||||||
|
%if &maxkeytable ne 0 %then %do;
|
||||||
|
proc sql noprint;
|
||||||
|
select count(*) into: check from &maxkeytable
|
||||||
|
where upcase(keytable)="&base_libds";
|
||||||
|
|
||||||
|
%mp_lockanytable(LOCK
|
||||||
|
,lib=%scan(&maxkeytable,1,.)
|
||||||
|
,ds=%scan(&maxkeytable,2,.)
|
||||||
|
,ref=Updating maxkeyvalues with mp_retainedkey
|
||||||
|
,ctl_ds=&locktable
|
||||||
|
)
|
||||||
|
proc sql;
|
||||||
|
%if &check=0 %then %do;
|
||||||
|
insert into &maxkeytable
|
||||||
|
set keytable="&base_libds"
|
||||||
|
,keycolumn="&retained_key"
|
||||||
|
,max_key=%eval(&maxkey+&newkey_cnt)
|
||||||
|
,processed_dttm="%sysfunc(datetime(),E8601DT26.6)"dt;
|
||||||
|
%end;
|
||||||
|
%else %do;
|
||||||
|
update &maxkeytable
|
||||||
|
set max_key=%eval(&maxkey+&newkey_cnt)
|
||||||
|
,processed_dttm="%sysfunc(datetime(),E8601DT26.6)"dt
|
||||||
|
where keytable="&base_libds";
|
||||||
|
%end;
|
||||||
|
%mp_lockanytable(UNLOCK
|
||||||
|
,lib=%scan(&maxkeytable,1,.)
|
||||||
|
,ds=%scan(&maxkeytable,2,.)
|
||||||
|
,ref=Updating maxkeyvalues with maxkey=%eval(&maxkey+&newkey_cnt)
|
||||||
|
,ctl_ds=&locktable
|
||||||
|
)
|
||||||
|
%end;
|
||||||
|
|
||||||
|
/* fill in the missing retained key values */
|
||||||
|
%let tempvar=%mf_getuniquename();
|
||||||
|
data &outds(drop=&tempvar);
|
||||||
|
retain &tempvar %eval(&maxkey+1);
|
||||||
|
set &tempds2;
|
||||||
|
if &retained_key =. then &retained_key=&tempvar;
|
||||||
|
&tempvar=&tempvar+1;
|
||||||
|
run;
|
||||||
|
|
||||||
|
%mend mp_retainedkey;
|
||||||
|
|
||||||
@@ -11,22 +11,28 @@
|
|||||||
Usage:
|
Usage:
|
||||||
|
|
||||||
%mp_searchdata(lib=sashelp, string=Jan)
|
%mp_searchdata(lib=sashelp, string=Jan)
|
||||||
%mp_searchdata(lib=sashelp, numval=1)
|
%mp_searchdata(lib=sashelp, ds=bird, numval=1)
|
||||||
|
%mp_searchdata(lib=sashelp, ds=class, string=l,outobs=5)
|
||||||
|
|
||||||
|
|
||||||
Outputs zero or more tables to an MPSEARCH library with specific records.
|
Outputs zero or more tables to an MPSEARCH library with specific records.
|
||||||
|
|
||||||
@param lib= the libref to search (should be already assigned)
|
@param [in] lib= The libref to search (should be already assigned)
|
||||||
@param ds= the dataset to search (leave blank to search entire library)
|
@param [in] ds= The dataset to search (leave blank to search entire library)
|
||||||
@param string= the string value to search
|
@param [in] string= String value to search (case sensitive, can be partial)
|
||||||
@param numval= the numeric value to search (must be exact)
|
@param [in] numval= Numeric value to search (must be exact)
|
||||||
@param outloc= the directory in which to create the output datasets with
|
@param [out] outloc= (0) Optionally specify the directory in which to
|
||||||
matching rows. Will default to a subfolder in the WORK library.
|
create the the output datasets with matching rows. By default it will
|
||||||
@param outobs= set to a positive integer to restrict the number of
|
write them to a temporary subdirectory within the WORK folder.
|
||||||
|
@param [out] outlib= (MPSEARCH) Assign a different libref to the output
|
||||||
|
library containing the matching datasets / records
|
||||||
|
@param [in] outobs= set to a positive integer to restrict the number of
|
||||||
observations
|
observations
|
||||||
@param filter_text= add a (valid) filter clause to further filter the results
|
@param [in] filter_text= (1=1) Add a (valid) filter clause to further filter
|
||||||
|
the results.
|
||||||
|
|
||||||
<h4> SAS Macros </h4>
|
<h4> SAS Macros </h4>
|
||||||
|
@li mf_getuniquename.sas
|
||||||
@li mf_getvarlist.sas
|
@li mf_getvarlist.sas
|
||||||
@li mf_getvartype.sas
|
@li mf_getvartype.sas
|
||||||
@li mf_mkdir.sas
|
@li mf_mkdir.sas
|
||||||
@@ -36,11 +42,12 @@
|
|||||||
@author Allan Bowe
|
@author Allan Bowe
|
||||||
**/
|
**/
|
||||||
|
|
||||||
%macro mp_searchdata(lib=sashelp
|
%macro mp_searchdata(lib=
|
||||||
,ds=
|
,ds=
|
||||||
,string= /* the query will use a contains (?) operator */
|
,string= /* the query will use a contains (?) operator */
|
||||||
,numval= /* numeric must match exactly */
|
,numval= /* numeric must match exactly */
|
||||||
,outloc=%sysfunc(pathname(work))/mpsearch
|
,outloc=0
|
||||||
|
,outlib=MPSEARCH
|
||||||
,outobs=-1
|
,outobs=-1
|
||||||
,filter_text=%str(1=1)
|
,filter_text=%str(1=1)
|
||||||
)/*/STORE SOURCE*/;
|
)/*/STORE SOURCE*/;
|
||||||
@@ -57,8 +64,12 @@
|
|||||||
%if &string = %then %let type=N;
|
%if &string = %then %let type=N;
|
||||||
%else %let type=C;
|
%else %let type=C;
|
||||||
|
|
||||||
|
%if "&outloc"="0" %then %do;
|
||||||
|
%let outloc=%sysfunc(pathname(work))/%mf_getuniquename();
|
||||||
|
%end;
|
||||||
|
|
||||||
%mf_mkdir(&outloc)
|
%mf_mkdir(&outloc)
|
||||||
libname mpsearch "&outloc";
|
libname &outlib "&outloc";
|
||||||
|
|
||||||
/* get the list of tables in the library */
|
/* get the list of tables in the library */
|
||||||
proc sql noprint;
|
proc sql noprint;
|
||||||
@@ -70,11 +81,6 @@ select distinct memname into: table_list separated by ' '
|
|||||||
%end;
|
%end;
|
||||||
;
|
;
|
||||||
/* check that we have something to check */
|
/* check that we have something to check */
|
||||||
proc sql
|
|
||||||
%if &outobs>-1 %then %do;
|
|
||||||
outobs=&outobs
|
|
||||||
%end;
|
|
||||||
;
|
|
||||||
%if %length(&table_list)=0 %then %put library &lib contains no tables!;
|
%if %length(&table_list)=0 %then %put library &lib contains no tables!;
|
||||||
/* loop through each table */
|
/* loop through each table */
|
||||||
%else %do table_num=1 %to %sysfunc(countw(&table_list,%str( )));
|
%else %do table_num=1 %to %sysfunc(countw(&table_list,%str( )));
|
||||||
@@ -85,10 +91,10 @@ proc sql
|
|||||||
%end;
|
%end;
|
||||||
%else %do;
|
%else %do;
|
||||||
%let check_tm=%sysfunc(datetime());
|
%let check_tm=%sysfunc(datetime());
|
||||||
/* build sql statement */
|
/* prep input */
|
||||||
create table mpsearch.&table as select * from &lib..&table
|
data &outlib..&table;
|
||||||
where %unquote(&filter_text) and
|
set &lib..&table;
|
||||||
(0
|
where %unquote(&filter_text) and ( 0
|
||||||
/* loop through columns */
|
/* loop through columns */
|
||||||
%do colnum=1 %to %sysfunc(countw(&vars,%str( )));
|
%do colnum=1 %to %sysfunc(countw(&vars,%str( )));
|
||||||
%let col=%scan(&vars,&colnum,%str( ));
|
%let col=%scan(&vars,&colnum,%str( ));
|
||||||
@@ -102,15 +108,20 @@ proc sql
|
|||||||
or ("&col"n = &numval)
|
or ("&col"n = &numval)
|
||||||
%end;
|
%end;
|
||||||
%end;
|
%end;
|
||||||
);
|
);
|
||||||
|
%if &outobs>-1 %then %do;
|
||||||
|
if _n_ > &outobs then stop;
|
||||||
|
%end;
|
||||||
|
run;
|
||||||
%put Search query for &table took
|
%put Search query for &table took
|
||||||
%sysevalf(%sysfunc(datetime())-&check_tm) seconds;
|
%sysevalf(%sysfunc(datetime())-&check_tm) seconds;
|
||||||
%if &sqlrc ne 0 %then %do;
|
%if &syscc ne 0 %then %do;
|
||||||
%put %str(WAR)NING: SQLRC=&sqlrc when processing &table;
|
%put %str(ERR)ROR: SYSCC=&syscc when processing &lib..&table;
|
||||||
%return;
|
%return;
|
||||||
%end;
|
%end;
|
||||||
%if %mf_nobs(mpsearch.&table)=0 %then %do;
|
%if %mf_nobs(&outlib..&table)=0 %then %do;
|
||||||
drop table mpsearch.&table;
|
proc sql;
|
||||||
|
drop table &outlib..&table;
|
||||||
%end;
|
%end;
|
||||||
%end;
|
%end;
|
||||||
%end;
|
%end;
|
||||||
|
|||||||
@@ -5,9 +5,9 @@
|
|||||||
sort it before performing operations such as merges / joins etc.
|
sort it before performing operations such as merges / joins etc.
|
||||||
That said, there are a few edge cases where it can be desirable:
|
That said, there are a few edge cases where it can be desirable:
|
||||||
|
|
||||||
@li To improve performance for particular scenarios
|
|
||||||
@li To allow adjacent records to be viewed directly in the dataset
|
@li To allow adjacent records to be viewed directly in the dataset
|
||||||
@li To reduce dataset size (eg when there are deleted records)
|
@li To apply compression, or to remove deleted records
|
||||||
|
@li To improve performance for specific queries
|
||||||
|
|
||||||
This macro will only work for BASE (V9) engine libraries. It works by
|
This macro will only work for BASE (V9) engine libraries. It works by
|
||||||
creating a copy of the dataset (without data, WITH constraints) in the same
|
creating a copy of the dataset (without data, WITH constraints) in the same
|
||||||
@@ -29,6 +29,7 @@
|
|||||||
@li mf_getengine.sas
|
@li mf_getengine.sas
|
||||||
@li mf_getquotedstr.sas
|
@li mf_getquotedstr.sas
|
||||||
@li mf_getuniquename.sas
|
@li mf_getuniquename.sas
|
||||||
|
@li mf_getvarlist.sas
|
||||||
@li mf_nobs.sas
|
@li mf_nobs.sas
|
||||||
@li mp_abort.sas
|
@li mp_abort.sas
|
||||||
@li mp_getpk.sas
|
@li mp_getpk.sas
|
||||||
@@ -38,7 +39,6 @@
|
|||||||
|
|
||||||
@version 9.2
|
@version 9.2
|
||||||
@author Allan Bowe
|
@author Allan Bowe
|
||||||
@source https://github.com/sasjs/core
|
|
||||||
|
|
||||||
**/
|
**/
|
||||||
|
|
||||||
@@ -74,9 +74,13 @@
|
|||||||
%return;
|
%return;
|
||||||
%end;
|
%end;
|
||||||
|
|
||||||
|
/* fallback sortkey is all fields */
|
||||||
|
%let sortkey=%mf_getvarlist(&libds);
|
||||||
|
|
||||||
|
/* overlay actual sort key if it exists */
|
||||||
data _null_;
|
data _null_;
|
||||||
set work.&tempds1;
|
set work.&tempds1;
|
||||||
call symputx('sortkey',pk_fields);
|
call symputx('sortkey',coalescec(pk_fields,symget('sortkey')));
|
||||||
run;
|
run;
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
592
base/mp_stackdiffs.sas
Normal file
592
base/mp_stackdiffs.sas
Normal file
@@ -0,0 +1,592 @@
|
|||||||
|
/**
|
||||||
|
@file
|
||||||
|
@brief Prepares an audit table for stacking (re-applying) the changes.
|
||||||
|
@details When the underlying data from a Base Table is refreshed, it can be
|
||||||
|
helpful to have any previously-applied changes, re-applied.
|
||||||
|
|
||||||
|
Such situation might arise if you are applying those changes using a tool
|
||||||
|
like [Data Controller for SAS®](https://datacontroller.io) - which records
|
||||||
|
all such changes in an audit table.
|
||||||
|
It may also apply if you are preparing a series of specific cell-level
|
||||||
|
transactions, that you would like to apply to multiple sets of (similarly
|
||||||
|
structured) Base Tables.
|
||||||
|
|
||||||
|
In both cases, it is necessary that the transactions are stored using
|
||||||
|
the mp_storediffs.sas macro, or at least that the underlying table is
|
||||||
|
structured as per the definition in mp_coretable.sas (DIFFTABLE entry)
|
||||||
|
|
||||||
|
<b>This</b> macro is used to convert the stored changes (tall format) into
|
||||||
|
staged changes (wide format), with base table values incorporated (in the
|
||||||
|
case of modified rows), ready for the subsequent load process.
|
||||||
|
|
||||||
|
Essentially then, what this macro does, is turn a table like this:
|
||||||
|
|
||||||
|
|KEY_HASH:$32.|MOVE_TYPE:$1.|TGTVAR_NM:$32.|IS_PK:best.|IS_DIFF:best.|TGTVAR_TYPE:$1.|OLDVAL_NUM:best32.|NEWVAL_NUM:best32.|OLDVAL_CHAR:$32765.|NEWVAL_CHAR:$32765.|
|
||||||
|
|---|---|---|---|---|---|---|---|---|---|
|
||||||
|
|`27AA6F7581052E7FF48E1BCA901313FB `|`A `|`NAME `|`1 `|`-1 `|`C `|`. `|`. `|` `|`Newbie `|
|
||||||
|
|`27AA6F7581052E7FF48E1BCA901313FB `|`A `|`AGE `|`0 `|`-1 `|`N `|`. `|`13 `|` `|` `|
|
||||||
|
|`27AA6F7581052E7FF48E1BCA901313FB `|`A `|`HEIGHT `|`0 `|`-1 `|`N `|`. `|`65.3 `|` `|` `|
|
||||||
|
|`27AA6F7581052E7FF48E1BCA901313FB `|`A `|`SEX `|`0 `|`-1 `|`C `|`. `|`. `|` `|`F `|
|
||||||
|
|`27AA6F7581052E7FF48E1BCA901313FB `|`A `|`WEIGHT `|`0 `|`-1 `|`N `|`. `|`98 `|` `|` `|
|
||||||
|
|`86703FDE9E87DD5C0F8E1072545D0128 `|`D `|`NAME `|`1 `|`-1 `|`C `|`. `|`. `|`Alfred `|` `|
|
||||||
|
|`86703FDE9E87DD5C0F8E1072545D0128 `|`D `|`AGE `|`0 `|`-1 `|`N `|`14 `|`. `|` `|` `|
|
||||||
|
|`86703FDE9E87DD5C0F8E1072545D0128 `|`D `|`HEIGHT `|`0 `|`-1 `|`N `|`69 `|`. `|` `|` `|
|
||||||
|
|`86703FDE9E87DD5C0F8E1072545D0128 `|`D `|`SEX `|`0 `|`-1 `|`C `|`. `|`. `|`M `|` `|
|
||||||
|
|`86703FDE9E87DD5C0F8E1072545D0128 `|`D `|`WEIGHT `|`0 `|`-1 `|`N `|`112.5 `|`. `|` `|` `|
|
||||||
|
|`64489C85DC2FE0787B85CD87214B3810 `|`M `|`NAME `|`1 `|`0 `|`C `|`. `|`. `|`Alice `|`Alice `|
|
||||||
|
|`64489C85DC2FE0787B85CD87214B3810 `|`M `|`AGE `|`0 `|`1 `|`N `|`13 `|`99 `|` `|` `|
|
||||||
|
|`64489C85DC2FE0787B85CD87214B3810 `|`M `|`HEIGHT `|`0 `|`0 `|`N `|`56.5 `|`56.5 `|` `|` `|
|
||||||
|
|`64489C85DC2FE0787B85CD87214B3810 `|`M `|`SEX `|`0 `|`0 `|`C `|`. `|`. `|`F `|`F `|
|
||||||
|
|`64489C85DC2FE0787B85CD87214B3810 `|`M `|`WEIGHT `|`0 `|`0 `|`N `|`84 `|`84 `|` `|` `|
|
||||||
|
|
||||||
|
Into three tables like this:
|
||||||
|
|
||||||
|
<b> `work.outmod`: </b>
|
||||||
|
|NAME:$8.|SEX:$1.|AGE:best.|HEIGHT:best.|WEIGHT:best.|
|
||||||
|
|---|---|---|---|---|
|
||||||
|
|`Alice `|`F `|`99 `|`56.5 `|`84 `|
|
||||||
|
|
||||||
|
<b> `work.outadd`: </b>
|
||||||
|
|NAME:$8.|SEX:$1.|AGE:best.|HEIGHT:best.|WEIGHT:best.|
|
||||||
|
|---|---|---|---|---|
|
||||||
|
|`Newbie `|`F `|`13 `|`65.3 `|`98 `|
|
||||||
|
|
||||||
|
<b> `work.outdel`: </b>
|
||||||
|
|NAME:$8.|SEX:$1.|AGE:best.|HEIGHT:best.|WEIGHT:best.|
|
||||||
|
|---|---|---|---|---|
|
||||||
|
|`Alfred `|`M `|`14 `|`69 `|`112.5 `|
|
||||||
|
|
||||||
|
As you might expect, there are a bunch of extra features and checks.
|
||||||
|
|
||||||
|
The macro supports both SCD2 (TXTEMPORAL) and UPDATE loadtypes. If the
|
||||||
|
base table contains a PROCESSED_DTTM column (or similar), this can be
|
||||||
|
ignored by declaring it in the `processed_dttm_var` parameter.
|
||||||
|
|
||||||
|
The macro is also flexible where columns have been added or removed from
|
||||||
|
the base table UNLESS there is a change to the primary key.
|
||||||
|
|
||||||
|
Changes to the primary key fields are NOT supported, and are likely to cause
|
||||||
|
unexpected results.
|
||||||
|
|
||||||
|
The following pre-flight checks are made:
|
||||||
|
|
||||||
|
@li All primary key columns exist on the base table
|
||||||
|
@li There is no change in variable TYPE for any of the columns
|
||||||
|
@li There is no reduction in variable LENGTH below the max-length of the
|
||||||
|
supplied values
|
||||||
|
|
||||||
|
Rules for stacking changes are as follows:
|
||||||
|
|
||||||
|
<table>
|
||||||
|
<tr>
|
||||||
|
<th>Transaction Type</th><th>Key Behaviour</th><th>Column Behaviour</th>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td>Deletes</td>
|
||||||
|
<td>
|
||||||
|
The row is added to `&outDEL.` UNLESS it no longer exists
|
||||||
|
in the base table, in which case it is added to `&errDS.` instead.
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
Deletes are unaffected by the addition or removal of non Primary-Key
|
||||||
|
columns.
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td>Inserts</td>
|
||||||
|
<td>
|
||||||
|
Previously newly added rows are added to the `outADD` table UNLESS they
|
||||||
|
are present in the Base table.<br>In this case they are added to the
|
||||||
|
`&errDS.` table instead.
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
Inserts are unaffected by the addition of columns in the Base Table
|
||||||
|
(they are padded with blanks). Deleted columns are only a problem if
|
||||||
|
they appear on the previous insert - in which case the record is added
|
||||||
|
to `&errDS.`.
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td>Updates</td>
|
||||||
|
<td>
|
||||||
|
Previously modified rows are merged with base table values such that
|
||||||
|
only the individual cells that were _previously_ changed are re-applied.
|
||||||
|
Where the row contains cells that were not marked as having changed in
|
||||||
|
the prior transaction, the 'blanks' are filled with base table values in
|
||||||
|
the `outMOD` table.<br>
|
||||||
|
If the row no longer exists on the base table, then the row is added to
|
||||||
|
the `errDS` table instead.
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
Updates are unaffected by the addition of columns in the Base Table -
|
||||||
|
the new cells are simply populated with Base Table values. Deleted
|
||||||
|
columns are only a problem if they relate to a modified cell
|
||||||
|
(`is_diff=1`) - in which case the record is added to `&errDS.`.
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
</table>
|
||||||
|
|
||||||
|
To illustrate the above with a diagram:
|
||||||
|
|
||||||
|
@dot
|
||||||
|
digraph {
|
||||||
|
rankdir="TB"
|
||||||
|
start[label="Transaction Type?" shape=Mdiamond]
|
||||||
|
del[label="Does Base Row exist?" shape=rectangle]
|
||||||
|
add [label="Does Base Row exist?" shape=rectangle]
|
||||||
|
mod [label="Does Base Row exist?" shape=rectangle]
|
||||||
|
chkmod [label="Do all modified\n(is_diff=1) cells exist?" shape=rectangle]
|
||||||
|
chkadd [label="Do all inserted cells exist?" shape=rectangle]
|
||||||
|
outmod [label="outMOD\nTable" shape=Msquare style=filled]
|
||||||
|
outadd [label="outADD\nTable" shape=Msquare style=filled]
|
||||||
|
outdel [label="outDEL\nTable" shape=Msquare style=filled]
|
||||||
|
outerr [label="ErrDS Table" shape=Msquare fillcolor=Orange style=filled]
|
||||||
|
start -> del [label="Delete"]
|
||||||
|
start -> add [label="Insert"]
|
||||||
|
start -> mod [label="Update"]
|
||||||
|
|
||||||
|
del -> outdel [label="Yes"]
|
||||||
|
del -> outerr [label="No" color="Red" fontcolor="Red"]
|
||||||
|
add -> chkadd [label="No"]
|
||||||
|
add -> outerr [label="Yes" color="Red" fontcolor="Red"]
|
||||||
|
mod -> outerr [label="No" color="Red" fontcolor="Red"]
|
||||||
|
mod -> chkmod [label="Yes"]
|
||||||
|
chkmod -> outerr [label="No" color="Red" fontcolor="Red"]
|
||||||
|
chkmod -> outmod [label="Yes"]
|
||||||
|
chkadd -> outerr [label="No" color="Red" fontcolor="Red"]
|
||||||
|
chkadd -> outadd [label="Yes"]
|
||||||
|
|
||||||
|
}
|
||||||
|
@enddot
|
||||||
|
|
||||||
|
For examples of usage, check out the mp_stackdiffs.test.sas program.
|
||||||
|
|
||||||
|
|
||||||
|
@param [in] baselibds Base Table against which the changes will be applied,
|
||||||
|
in libref.dataset format.
|
||||||
|
@param [in] auditlibds Dataset with previously applied transactions, to be
|
||||||
|
re-applied. Use libref.dataset format.
|
||||||
|
DDL as follows: %mp_coretable(DIFFTABLE)
|
||||||
|
@param [in] key Space seperated list of key variables
|
||||||
|
@param [in] mdebug= Set to 1 to enable DEBUG messages and preserve outputs
|
||||||
|
@param [in] processed_dttm_var= (0) If a variable is being used to mark
|
||||||
|
the processed datetime, put the name of the variable here. It will NOT
|
||||||
|
be included in the staged dataset (the load process is expected to
|
||||||
|
provide this)
|
||||||
|
@param [out] errds= (work.errds) Output table containing problematic records.
|
||||||
|
The columns of this table are:
|
||||||
|
@li PK_VARS - Space separated list of primary key variable names
|
||||||
|
@li PK_VALS - Slash separted list of PK variable values
|
||||||
|
@li ERR_MSG - Explanation of why this record is problematic
|
||||||
|
@param [out] outmod= (work.outmod) Output table containing modified records
|
||||||
|
@param [out] outadd= (work.outadd) Output table containing additional records
|
||||||
|
@param [out] outdel= (work.outdel) Output table containing deleted records
|
||||||
|
|
||||||
|
<h4> SAS Macros </h4>
|
||||||
|
@li mf_existvarlist.sas
|
||||||
|
@li mf_getquotedstr.sas
|
||||||
|
@li mf_getuniquefileref.sas
|
||||||
|
@li mf_getuniquename.sas
|
||||||
|
@li mf_islibds.sas
|
||||||
|
@li mf_nobs.sas
|
||||||
|
@li mf_wordsinstr1butnotstr2.sas
|
||||||
|
@li mp_abort.sas
|
||||||
|
@li mp_ds2squeeze.sas
|
||||||
|
|
||||||
|
<h4> Related Macros </h4>
|
||||||
|
@li mp_coretable.sas
|
||||||
|
@li mp_stackdiffs.test.sas
|
||||||
|
@li mp_storediffs.sas
|
||||||
|
|
||||||
|
@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
|
||||||
|
mf_getuniquename.sas) when such a use case arises.
|
||||||
|
|
||||||
|
@version 9.2
|
||||||
|
@author Allan Bowe
|
||||||
|
**/
|
||||||
|
/** @cond */
|
||||||
|
|
||||||
|
%macro mp_stackdiffs(baselibds
|
||||||
|
,auditlibds
|
||||||
|
,key
|
||||||
|
,mdebug=0
|
||||||
|
,processed_dttm_var=0
|
||||||
|
,errds=work.errds
|
||||||
|
,outmod=work.outmod
|
||||||
|
,outadd=work.outadd
|
||||||
|
,outdel=work.outdel
|
||||||
|
)/*/STORE SOURCE*/;
|
||||||
|
%local dbg;
|
||||||
|
%if &mdebug=1 %then %do;
|
||||||
|
%put &sysmacroname entry vars:;
|
||||||
|
%put _local_;
|
||||||
|
%end;
|
||||||
|
%else %let dbg=*;
|
||||||
|
|
||||||
|
/* input parameter validations */
|
||||||
|
%mp_abort(iftrue= (%mf_islibds(&baselibds) ne 1)
|
||||||
|
,mac=&sysmacroname
|
||||||
|
,msg=%str(Invalid baselibds: &baselibds)
|
||||||
|
)
|
||||||
|
%mp_abort(iftrue= (%mf_islibds(&auditlibds) ne 1)
|
||||||
|
,mac=&sysmacroname
|
||||||
|
,msg=%str(Invalid auditlibds: &auditlibds)
|
||||||
|
)
|
||||||
|
%mp_abort(iftrue= (%length(&key)=0)
|
||||||
|
,mac=&sysmacroname
|
||||||
|
,msg=%str(Missing key variables!)
|
||||||
|
)
|
||||||
|
%mp_abort(iftrue= (
|
||||||
|
%mf_existVarList(&auditlibds,LIBREF DSN MOVE_TYPE KEY_HASH TGTVAR_NM IS_PK
|
||||||
|
IS_DIFF TGTVAR_TYPE OLDVAL_NUM NEWVAL_NUM OLDVAL_CHAR NEWVAL_CHAR)=0
|
||||||
|
)
|
||||||
|
,mac=&sysmacroname
|
||||||
|
,msg=%str(Input &auditlibds is missing required columns!)
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
/* set up macro vars */
|
||||||
|
%local prefix dslist x var keyjoin commakey keepvars missvars fref;
|
||||||
|
%let prefix=%substr(%mf_getuniquename(),1,25);
|
||||||
|
%let dslist=ds1d ds2d ds3d ds1a ds2a ds3a ds1m ds2m ds3m pks dups base
|
||||||
|
delrec delerr addrec adderr modrec moderr;
|
||||||
|
%do x=1 %to %sysfunc(countw(&dslist));
|
||||||
|
%let var=%scan(&dslist,&x);
|
||||||
|
%local &var;
|
||||||
|
%let &var=%upcase(&prefix._&var);
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%let key=%upcase(&key);
|
||||||
|
%let commakey=%mf_getquotedstr(&key,quote=N);
|
||||||
|
|
||||||
|
%let keyjoin=1=1;
|
||||||
|
%do x=1 %to %sysfunc(countw(&key));
|
||||||
|
%let var=%scan(&key,&x);
|
||||||
|
%let keyjoin=&keyjoin and a.&var=b.&var;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
data &errds;
|
||||||
|
length pk_vars $256 pk_vals $4098 err_msg $512;
|
||||||
|
call missing (of _all_);
|
||||||
|
stop;
|
||||||
|
run;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Prepare raw DELETE table
|
||||||
|
* Records are in the OLDVAL_xxx columns
|
||||||
|
*/
|
||||||
|
%let keepvars=MOVE_TYPE KEY_HASH TGTVAR_NM TGTVAR_TYPE IS_PK
|
||||||
|
OLDVAL_NUM OLDVAL_CHAR
|
||||||
|
NEWVAL_NUM NEWVAL_CHAR;
|
||||||
|
proc sort data=&auditlibds(where=(move_type='D') keep=&keepvars)
|
||||||
|
out=&ds1d(drop=move_type);
|
||||||
|
by KEY_HASH TGTVAR_NM;
|
||||||
|
run;
|
||||||
|
proc transpose data=&ds1d(where=(tgtvar_type='N'))
|
||||||
|
out=&ds2d(drop=_name_);
|
||||||
|
by KEY_HASH;
|
||||||
|
id TGTVAR_NM;
|
||||||
|
var OLDVAL_NUM;
|
||||||
|
run;
|
||||||
|
proc transpose data=&ds1d(where=(tgtvar_type='C'))
|
||||||
|
out=&ds3d(drop=_name_);
|
||||||
|
by KEY_HASH;
|
||||||
|
id TGTVAR_NM;
|
||||||
|
var OLDVAL_CHAR;
|
||||||
|
run;
|
||||||
|
%mp_ds2squeeze(&ds2d,outds=&ds2d)
|
||||||
|
%mp_ds2squeeze(&ds3d,outds=&ds3d)
|
||||||
|
data &outdel;
|
||||||
|
if 0 then set &baselibds;
|
||||||
|
set &ds2d;
|
||||||
|
set &ds3d;
|
||||||
|
drop key_hash;
|
||||||
|
if not missing(%scan(&key,1));
|
||||||
|
run;
|
||||||
|
proc sort;
|
||||||
|
by &key;
|
||||||
|
run;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Prepare raw APPEND table
|
||||||
|
* Records are in the NEWVAL_xxx columns
|
||||||
|
*/
|
||||||
|
proc sort data=&auditlibds(where=(move_type='A') keep=&keepvars)
|
||||||
|
out=&ds1a(drop=move_type);
|
||||||
|
by KEY_HASH TGTVAR_NM;
|
||||||
|
run;
|
||||||
|
proc transpose data=&ds1a(where=(tgtvar_type='N'))
|
||||||
|
out=&ds2a(drop=_name_);
|
||||||
|
by KEY_HASH;
|
||||||
|
id TGTVAR_NM;
|
||||||
|
var NEWVAL_NUM;
|
||||||
|
run;
|
||||||
|
proc transpose data=&ds1a(where=(tgtvar_type='C'))
|
||||||
|
out=&ds3a(drop=_name_);
|
||||||
|
by KEY_HASH;
|
||||||
|
id TGTVAR_NM;
|
||||||
|
var NEWVAL_CHAR;
|
||||||
|
run;
|
||||||
|
%mp_ds2squeeze(&ds2a,outds=&ds2a)
|
||||||
|
%mp_ds2squeeze(&ds3a,outds=&ds3a)
|
||||||
|
data &outadd;
|
||||||
|
if 0 then set &baselibds;
|
||||||
|
set &ds2a;
|
||||||
|
set &ds3a;
|
||||||
|
drop key_hash;
|
||||||
|
if not missing(%scan(&key,1));
|
||||||
|
run;
|
||||||
|
proc sort;
|
||||||
|
by &key;
|
||||||
|
run;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Prepare raw MODIFY table
|
||||||
|
* Keep only primary key - will add modified values later
|
||||||
|
*/
|
||||||
|
proc sort data=&auditlibds(
|
||||||
|
where=(move_type='M' and is_pk=1) keep=&keepvars
|
||||||
|
) out=&ds1m(drop=move_type);
|
||||||
|
by KEY_HASH TGTVAR_NM;
|
||||||
|
run;
|
||||||
|
proc transpose data=&ds1m(where=(tgtvar_type='N'))
|
||||||
|
out=&ds2m(drop=_name_);
|
||||||
|
by KEY_HASH ;
|
||||||
|
id TGTVAR_NM;
|
||||||
|
var NEWVAL_NUM;
|
||||||
|
run;
|
||||||
|
proc transpose data=&ds1m(where=(tgtvar_type='C'))
|
||||||
|
out=&ds3m(drop=_name_);
|
||||||
|
by KEY_HASH;
|
||||||
|
id TGTVAR_NM;
|
||||||
|
var NEWVAL_CHAR;
|
||||||
|
run;
|
||||||
|
%mp_ds2squeeze(&ds2m,outds=&ds2m)
|
||||||
|
%mp_ds2squeeze(&ds3m,outds=&ds3m)
|
||||||
|
data &outmod;
|
||||||
|
if 0 then set &baselibds;
|
||||||
|
set &ds2m;
|
||||||
|
set &ds3m;
|
||||||
|
if not missing(%scan(&key,1));
|
||||||
|
run;
|
||||||
|
proc sort;
|
||||||
|
by &key;
|
||||||
|
run;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Extract matching records from the base table
|
||||||
|
* Do this in one join for efficiency.
|
||||||
|
* At a later date, this should be optimised for large database tables by using
|
||||||
|
* passthrough and a temporary table.
|
||||||
|
*/
|
||||||
|
data &pks;
|
||||||
|
if 0 then set &baselibds;
|
||||||
|
set &outadd &outmod &outdel;
|
||||||
|
keep &key;
|
||||||
|
run;
|
||||||
|
|
||||||
|
proc sort noduprec dupout=&dups;
|
||||||
|
by &key;
|
||||||
|
run;
|
||||||
|
data _null_;
|
||||||
|
set &dups;
|
||||||
|
putlog (_all_)(=);
|
||||||
|
run;
|
||||||
|
%mp_abort(iftrue= (%mf_nobs(&dups) ne 0)
|
||||||
|
,mac=&sysmacroname
|
||||||
|
,msg=%str(duplicates (%mf_nobs(&dups)) found on &auditlibds!)
|
||||||
|
)
|
||||||
|
|
||||||
|
proc sql;
|
||||||
|
create table &base as
|
||||||
|
select a.*
|
||||||
|
from &baselibds a, &pks b
|
||||||
|
where &keyjoin;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* delete check
|
||||||
|
* This is straightforward as it relates to records only
|
||||||
|
*/
|
||||||
|
proc sql;
|
||||||
|
create table &delrec as
|
||||||
|
select a.*
|
||||||
|
from &outdel a
|
||||||
|
left join &base b
|
||||||
|
on &keyjoin
|
||||||
|
where b.%scan(&key,1) is null
|
||||||
|
order by &commakey;
|
||||||
|
|
||||||
|
data &delerr;
|
||||||
|
if 0 then set &errds;
|
||||||
|
set &delrec;
|
||||||
|
PK_VARS="&key";
|
||||||
|
PK_VALS=catx('/',&commakey);
|
||||||
|
ERR_MSG="Rows cannot be deleted as they do not exist on the Base dataset";
|
||||||
|
keep PK_VARS PK_VALS ERR_MSG;
|
||||||
|
run;
|
||||||
|
proc append base=&errds data=&delerr;
|
||||||
|
run;
|
||||||
|
|
||||||
|
data &outdel;
|
||||||
|
merge &outdel (in=a) &delrec (in=b);
|
||||||
|
by &key;
|
||||||
|
if not b;
|
||||||
|
run;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* add check
|
||||||
|
* Problems - where record already exists, or base table has columns missing
|
||||||
|
*/
|
||||||
|
%let missvars=%mf_wordsinstr1butnotstr2(
|
||||||
|
Str1=%upcase(%mf_getvarlist(&outadd)),
|
||||||
|
Str2=%upcase(%mf_getvarlist(&baselibds))
|
||||||
|
);
|
||||||
|
%if %length(&missvars)>0 %then %do;
|
||||||
|
/* add them to the err table */
|
||||||
|
data &adderr;
|
||||||
|
if 0 then set &errds;
|
||||||
|
set &outadd;
|
||||||
|
PK_VARS="&key";
|
||||||
|
PK_VALS=catx('/',&commakey);
|
||||||
|
ERR_MSG="Rows cannot be added due to missing base vars: &missvars";
|
||||||
|
keep PK_VARS PK_VALS ERR_MSG;
|
||||||
|
run;
|
||||||
|
proc append base=&errds data=&adderr;
|
||||||
|
run;
|
||||||
|
proc sql;
|
||||||
|
delete * from &outadd;
|
||||||
|
%end;
|
||||||
|
%else %do;
|
||||||
|
proc sql;
|
||||||
|
/* find records that already exist on base table */
|
||||||
|
create table &addrec as
|
||||||
|
select a.*
|
||||||
|
from &outadd a
|
||||||
|
inner join &base b
|
||||||
|
on &keyjoin
|
||||||
|
order by &commakey;
|
||||||
|
|
||||||
|
/* add them to the err table */
|
||||||
|
data &adderr;
|
||||||
|
if 0 then set &errds;
|
||||||
|
set &addrec;
|
||||||
|
PK_VARS="&key";
|
||||||
|
PK_VALS=catx('/',&commakey);
|
||||||
|
ERR_MSG="Rows cannot be added as they already exist on the Base dataset";
|
||||||
|
keep PK_VARS PK_VALS ERR_MSG;
|
||||||
|
run;
|
||||||
|
proc append base=&errds data=&adderr;
|
||||||
|
run;
|
||||||
|
|
||||||
|
/* remove invalid rows from the outadd table */
|
||||||
|
data &outadd;
|
||||||
|
merge &outadd (in=a) &addrec (in=b);
|
||||||
|
by &key;
|
||||||
|
if not b;
|
||||||
|
run;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* mod check
|
||||||
|
* Problems - where record does not exist or baseds has modified cols missing
|
||||||
|
*/
|
||||||
|
proc sql noprint;
|
||||||
|
select distinct tgtvar_nm into: missvars separated by ' '
|
||||||
|
from &auditlibds
|
||||||
|
where move_type='M' and is_diff=1;
|
||||||
|
%let missvars=%mf_wordsinstr1butnotstr2(
|
||||||
|
Str1=&missvars,
|
||||||
|
Str2=%upcase(%mf_getvarlist(&baselibds))
|
||||||
|
);
|
||||||
|
%if %length(&missvars)>0 %then %do;
|
||||||
|
/* add them to the err table */
|
||||||
|
data &moderr;
|
||||||
|
if 0 then set &errds;
|
||||||
|
set &outmod;
|
||||||
|
PK_VARS="&key";
|
||||||
|
PK_VALS=catx('/',&commakey);
|
||||||
|
ERR_MSG="Rows cannot be modified due to missing base vars: &missvars";
|
||||||
|
keep PK_VARS PK_VALS ERR_MSG;
|
||||||
|
run;
|
||||||
|
proc append base=&errds data=&moderr;
|
||||||
|
run;
|
||||||
|
proc sql;
|
||||||
|
delete * from &outmod;
|
||||||
|
%end;
|
||||||
|
%else %do;
|
||||||
|
/* now check for records that do not exist (therefore cannot be modified) */
|
||||||
|
proc sql;
|
||||||
|
create table &modrec as
|
||||||
|
select a.*
|
||||||
|
from &outmod a
|
||||||
|
left join &base b
|
||||||
|
on &keyjoin
|
||||||
|
where b.%scan(&key,1) is null
|
||||||
|
order by &commakey;
|
||||||
|
data &moderr;
|
||||||
|
if 0 then set &errds;
|
||||||
|
set &modrec;
|
||||||
|
PK_VARS="&key";
|
||||||
|
PK_VALS=catx('/',&commakey);
|
||||||
|
ERR_MSG="Rows cannot be modified as they do not exist on the Base dataset";
|
||||||
|
keep PK_VARS PK_VALS ERR_MSG;
|
||||||
|
run;
|
||||||
|
proc append base=&errds data=&moderr;
|
||||||
|
run;
|
||||||
|
/* delete the above records from the outmod table */
|
||||||
|
data &outmod;
|
||||||
|
merge &outmod (in=a) &modrec (in=b);
|
||||||
|
by &key;
|
||||||
|
if not b;
|
||||||
|
run;
|
||||||
|
/* now - we can prepare the final MOD table (which is currently PK only) */
|
||||||
|
proc sql undo_policy=none;
|
||||||
|
create table &outmod as
|
||||||
|
select a.key_hash
|
||||||
|
,b.*
|
||||||
|
from &outmod a
|
||||||
|
inner join &base b
|
||||||
|
on &keyjoin
|
||||||
|
order by &commakey;
|
||||||
|
/* now - to update outmod with modified (is_diff=1) values */
|
||||||
|
%let fref=%mf_getuniquefileref();
|
||||||
|
data _null_;
|
||||||
|
file &fref;
|
||||||
|
set &auditlibds(where=(move_type='M')) end=lastobs;
|
||||||
|
by key_hash;
|
||||||
|
retain comma 'N';
|
||||||
|
if _n_=1 then put 'proc sql;';
|
||||||
|
if first.key_hash then do;
|
||||||
|
comma='N';
|
||||||
|
put "update &outmod set " @@;
|
||||||
|
end;
|
||||||
|
if is_diff=1 then do;
|
||||||
|
if comma='N' then do;
|
||||||
|
put ' '@@;
|
||||||
|
comma='Y';
|
||||||
|
end;
|
||||||
|
else put ' ,'@@;
|
||||||
|
if tgtvar_type='C' then do;
|
||||||
|
length qstr $32767;
|
||||||
|
qstr=quote(trim(NEWVAL_CHAR));
|
||||||
|
put tgtvar_nm '=' qstr;
|
||||||
|
end;
|
||||||
|
else put tgtvar_nm '=' newval_num;
|
||||||
|
if comma=' ' then comma=' ,';
|
||||||
|
end;
|
||||||
|
if last.key_hash then put ' where key_hash=trim("' key_hash '");';
|
||||||
|
if lastobs then put "alter table &outmod drop key_hash;";
|
||||||
|
run;
|
||||||
|
%inc &fref/source2;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%if &mdebug=0 %then %do;
|
||||||
|
proc datasets lib=work;
|
||||||
|
delete &prefix:;
|
||||||
|
run;
|
||||||
|
%put &sysmacroname exit vars:;
|
||||||
|
%put _local_;
|
||||||
|
%end;
|
||||||
|
%mend mp_stackdiffs;
|
||||||
|
/** @endcond */
|
||||||
228
base/mp_storediffs.sas
Normal file
228
base/mp_storediffs.sas
Normal file
@@ -0,0 +1,228 @@
|
|||||||
|
/**
|
||||||
|
@file
|
||||||
|
@brief Converts deletes/changes/appends into a single audit table.
|
||||||
|
@details When tracking changes to data over time, it can be helpful to have
|
||||||
|
a single base table to track ALL modifications - enabling audit trail,
|
||||||
|
data recovery, and change re-application. This macro is one of many
|
||||||
|
data management utilities used in [Data Controller for SAS](
|
||||||
|
https:datacontroller.io) - a comprehensive data ingestion solution, which
|
||||||
|
works on any SAS platform (Viya, SAS 9, Foundation) and is free for up to 5
|
||||||
|
users.
|
||||||
|
|
||||||
|
NOTE - this macro does not validate the inputs. It is assumed that the
|
||||||
|
datasets containing the new / changed / deleted rows are CORRECT, contain
|
||||||
|
no additional (or missing columns), and that the originals dataset contains
|
||||||
|
all relevant base records (and no additionals).
|
||||||
|
|
||||||
|
Usage:
|
||||||
|
|
||||||
|
data work.orig work.deleted work.changed work.appended;
|
||||||
|
set sashelp.class;
|
||||||
|
if _n_=1 then do;
|
||||||
|
output work.orig work.deleted;
|
||||||
|
end;
|
||||||
|
else if _n_=2 then do;
|
||||||
|
output work.orig;
|
||||||
|
age=99;
|
||||||
|
output work.changed;
|
||||||
|
end;
|
||||||
|
else do;
|
||||||
|
name='Newbie';
|
||||||
|
output work.appended;
|
||||||
|
stop;
|
||||||
|
end;
|
||||||
|
run;
|
||||||
|
|
||||||
|
%mp_storediffs(sashelp.class,work.orig,NAME
|
||||||
|
,delds=work.deleted
|
||||||
|
,modds=work.changed
|
||||||
|
,appds=work.appended
|
||||||
|
,outds=work.final
|
||||||
|
,mdebug=1
|
||||||
|
)
|
||||||
|
|
||||||
|
@param [in] libds Target table against which the changes were applied
|
||||||
|
@param [in] origds Dataset with original (unchanged) records. Can be empty if
|
||||||
|
only appending.
|
||||||
|
@param [in] key Space seperated list of key variables
|
||||||
|
@param [in] delds= (0) Dataset with deleted records
|
||||||
|
@param [in] appds= (0) Dataset with appended records
|
||||||
|
@param [in] modds= (0) Dataset with modified records
|
||||||
|
@param [out] outds= (work.mp_storediffs) Output table containing stored data.
|
||||||
|
DDL as follows: %mp_coretable(DIFFTABLE)
|
||||||
|
|
||||||
|
@param [in] processed_dttm= (0) Provide a datetime constant in relation to
|
||||||
|
the actual load time. If not provided, current timestamp is used.
|
||||||
|
@param [in] mdebug= set to 1 to enable DEBUG messages and preserve outputs
|
||||||
|
@param [out] loadref= (0) Provide a unique key to reference the load,
|
||||||
|
otherwise a UUID will be generated.
|
||||||
|
|
||||||
|
<h4> SAS Macros </h4>
|
||||||
|
@li mf_getquotedstr.sas
|
||||||
|
@li mf_getuniquename.sas
|
||||||
|
@li mf_getvarlist.sas
|
||||||
|
|
||||||
|
<h4> Related Macros </h4>
|
||||||
|
@li mp_stackdiffs.sas
|
||||||
|
@li mp_storediffs.test.sas
|
||||||
|
|
||||||
|
@version 9.2
|
||||||
|
@author Allan Bowe
|
||||||
|
**/
|
||||||
|
/** @cond */
|
||||||
|
|
||||||
|
%macro mp_storediffs(libds
|
||||||
|
,origds
|
||||||
|
,key
|
||||||
|
,delds=0
|
||||||
|
,appds=0
|
||||||
|
,modds=0
|
||||||
|
,outds=work.mp_storediffs
|
||||||
|
,loadref=0
|
||||||
|
,processed_dttm=0
|
||||||
|
,mdebug=0
|
||||||
|
)/*/STORE SOURCE*/;
|
||||||
|
%local dbg;
|
||||||
|
%if &mdebug=1 %then %do;
|
||||||
|
%put &sysmacroname entry vars:;
|
||||||
|
%put _local_;
|
||||||
|
%end;
|
||||||
|
%else %let dbg=*;
|
||||||
|
|
||||||
|
/* set up unique and temporary vars */
|
||||||
|
%local ds1 ds2 ds3 ds4 hashkey inds_auto inds_keep dslist vlist;
|
||||||
|
%let ds1=%upcase(work.%mf_getuniquename(prefix=mpsd_ds1));
|
||||||
|
%let ds2=%upcase(work.%mf_getuniquename(prefix=mpsd_ds2));
|
||||||
|
%let ds3=%upcase(work.%mf_getuniquename(prefix=mpsd_ds3));
|
||||||
|
%let ds4=%upcase(work.%mf_getuniquename(prefix=mpsd_ds4));
|
||||||
|
%let hashkey=%upcase(%mf_getuniquename(prefix=mpsd_hashkey));
|
||||||
|
%let inds_auto=%upcase(%mf_getuniquename(prefix=mpsd_inds_auto));
|
||||||
|
%let inds_keep=%upcase(%mf_getuniquename(prefix=mpsd_inds_keep));
|
||||||
|
|
||||||
|
%let dslist=&origds;
|
||||||
|
%if &delds ne 0 %then %do;
|
||||||
|
%let delds=%upcase(&delds);
|
||||||
|
%if %scan(&delds,-1,.)=&delds %then %let delds=WORK.&delds;
|
||||||
|
%let dslist=&dslist &delds;
|
||||||
|
%end;
|
||||||
|
%if &appds ne 0 %then %do;
|
||||||
|
%let appds=%upcase(&appds);
|
||||||
|
%if %scan(&appds,-1,.)=&appds %then %let appds=WORK.&appds;
|
||||||
|
%let dslist=&dslist &appds;
|
||||||
|
%end;
|
||||||
|
%if &modds ne 0 %then %do;
|
||||||
|
%let modds=%upcase(&modds);
|
||||||
|
%if %scan(&modds,-1,.)=&modds %then %let modds=WORK.&modds;
|
||||||
|
%let dslist=&dslist &modds;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%let origds=%upcase(&origds);
|
||||||
|
%if %scan(&origds,-1,.)=&origds %then %let origds=WORK.&origds;
|
||||||
|
|
||||||
|
%let key=%upcase(&key);
|
||||||
|
|
||||||
|
/* hash the key and append all the tables (marking the source) */
|
||||||
|
data &ds1;
|
||||||
|
set &dslist indsname=&inds_auto;
|
||||||
|
&hashkey=put(md5(catx('|',%mf_getquotedstr(&key,quote=N))),$hex32.);
|
||||||
|
&inds_keep=&inds_auto;
|
||||||
|
proc sort;
|
||||||
|
by &inds_keep &hashkey;
|
||||||
|
run;
|
||||||
|
|
||||||
|
/* transpose numeric & char vars */
|
||||||
|
proc transpose data=&ds1
|
||||||
|
out=&ds2(rename=(&hashkey=key_hash _name_=tgtvar_nm col1=newval_num));
|
||||||
|
by &inds_keep &hashkey;
|
||||||
|
var _numeric_;
|
||||||
|
run;
|
||||||
|
proc transpose data=&ds1
|
||||||
|
out=&ds3(
|
||||||
|
rename=(&hashkey=key_hash _name_=tgtvar_nm col1=newval_char)
|
||||||
|
where=(tgtvar_nm not in ("&hashkey","&inds_keep"))
|
||||||
|
);
|
||||||
|
by &inds_keep &hashkey;
|
||||||
|
var _character_;
|
||||||
|
run;
|
||||||
|
|
||||||
|
%if %index(&libds,-)>0 and %scan(&libds,2,-)=FC %then %do;
|
||||||
|
/* this is a format catalog - cannot query cols directly */
|
||||||
|
%let vlist="FMTNAME","START","END","LABEL","MIN","MAX","DEFAULT","LENGTH"
|
||||||
|
,"FUZZ","PREFIX","MULT","FILL","NOEDIT","TYPE","SEXCL","EEXCL","HLO"
|
||||||
|
,"DECSEP","DIG3SEP","DATATYPE","LANGUAGE";
|
||||||
|
%end;
|
||||||
|
%else %let vlist=%mf_getvarlist(&libds,dlm=%str(,),quote=DOUBLE);
|
||||||
|
|
||||||
|
data &ds4;
|
||||||
|
length &inds_keep $41 tgtvar_nm $32;
|
||||||
|
set &ds2 &ds3 indsname=&inds_auto;
|
||||||
|
|
||||||
|
tgtvar_nm=upcase(tgtvar_nm);
|
||||||
|
if tgtvar_nm in (%upcase(&vlist));
|
||||||
|
|
||||||
|
if &inds_auto="&ds2" then tgtvar_type='N';
|
||||||
|
else if &inds_auto="&ds3" then tgtvar_type='C';
|
||||||
|
else do;
|
||||||
|
putlog "%str(ERR)OR: unidentified vartype input!" &inds_auto;
|
||||||
|
call symputx('syscc',98);
|
||||||
|
end;
|
||||||
|
|
||||||
|
if &inds_keep="&appds" then move_type='A';
|
||||||
|
else if &inds_keep="&delds" then move_type='D';
|
||||||
|
else if &inds_keep="&modds" then move_type='M';
|
||||||
|
else if &inds_keep="&origds" then move_type='O';
|
||||||
|
else do;
|
||||||
|
putlog "%str(ERR)OR: unidentified movetype input!" &inds_keep;
|
||||||
|
call symputx('syscc',99);
|
||||||
|
end;
|
||||||
|
tgtvar_nm=upcase(tgtvar_nm);
|
||||||
|
if tgtvar_nm in (%mf_getquotedstr(&key)) then is_pk=1;
|
||||||
|
else is_pk=0;
|
||||||
|
drop &inds_keep;
|
||||||
|
run;
|
||||||
|
|
||||||
|
%if "&loadref"="0" %then %let loadref=%sysfunc(uuidgen());
|
||||||
|
%if &processed_dttm=0 %then %let processed_dttm=%sysfunc(datetime());
|
||||||
|
%let libds=%upcase(&libds);
|
||||||
|
|
||||||
|
/* join orig vals for modified & deleted */
|
||||||
|
proc sql;
|
||||||
|
create table &outds as
|
||||||
|
select "&loadref" as load_ref length=36
|
||||||
|
,&processed_dttm as processed_dttm format=E8601DT26.6
|
||||||
|
,"%scan(&libds,1,.)" as libref length=8
|
||||||
|
,"%scan(&libds,2,.)" as dsn length=32
|
||||||
|
,b.key_hash length=32
|
||||||
|
,b.move_type length=1
|
||||||
|
,b.tgtvar_nm length=32
|
||||||
|
,b.is_pk
|
||||||
|
,case when b.move_type ne 'M' then -1
|
||||||
|
when a.newval_num=b.newval_num and a.newval_char=b.newval_char then 0
|
||||||
|
else 1
|
||||||
|
end as is_diff
|
||||||
|
,b.tgtvar_type length=1
|
||||||
|
,case when b.move_type='D' then b.newval_num
|
||||||
|
else a.newval_num
|
||||||
|
end as oldval_num format=best32.
|
||||||
|
,case when b.move_type='D' then .
|
||||||
|
else b.newval_num
|
||||||
|
end as newval_num format=best32.
|
||||||
|
,case when b.move_type='D' then b.newval_char
|
||||||
|
else a.newval_char
|
||||||
|
end as oldval_char length=32765
|
||||||
|
,case when b.move_type='D' then ''
|
||||||
|
else b.newval_char
|
||||||
|
end as newval_char length=32765
|
||||||
|
from &ds4(where=(move_type='O')) as a
|
||||||
|
right join &ds4(where=(move_type ne 'O')) as b
|
||||||
|
on a.tgtvar_nm=b.tgtvar_nm
|
||||||
|
and a.key_hash=b.key_hash
|
||||||
|
order by move_type, key_hash,is_pk desc, tgtvar_nm;
|
||||||
|
|
||||||
|
%if &mdebug=0 %then %do;
|
||||||
|
proc sql;
|
||||||
|
drop table &ds1, &ds2, &ds3, &ds4;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%mend mp_storediffs;
|
||||||
|
/** @endcond */
|
||||||
@@ -12,17 +12,20 @@
|
|||||||
|
|
||||||
%mp_streamfile(contenttype=csv,inloc=/some/where.txt,outname=myfile.txt)
|
%mp_streamfile(contenttype=csv,inloc=/some/where.txt,outname=myfile.txt)
|
||||||
|
|
||||||
|
@param [in] contenttype= (TEXT) Either TEXT, ZIP, CSV, EXCEL
|
||||||
|
@param [in] inloc= /path/to/file.ext to be sent
|
||||||
|
@param [in] inref= fileref of file to be sent (if provided, overrides `inloc`)
|
||||||
|
@param [in] iftrue= (1=1) Provide a condition under which to execute.
|
||||||
|
@param [out] outname= the name of the file, as downloaded by the browser
|
||||||
|
@param [out] outref= (_webout) The destination where the file should be
|
||||||
|
streamed.
|
||||||
|
|
||||||
<h4> SAS Macros </h4>
|
<h4> SAS Macros </h4>
|
||||||
@li mf_getplatform.sas
|
@li mf_getplatform.sas
|
||||||
|
@li mfs_httpheader.sas
|
||||||
@li mp_binarycopy.sas
|
@li mp_binarycopy.sas
|
||||||
|
|
||||||
@param contenttype= Either TEXT, ZIP, CSV, EXCEL (default TEXT)
|
|
||||||
@param inloc= /path/to/file.ext to be sent
|
|
||||||
@param inref= fileref of file to be sent (if provided, overrides `inloc`)
|
|
||||||
@param outname= the name of the file, as downloaded by the browser
|
|
||||||
|
|
||||||
@author Allan Bowe
|
@author Allan Bowe
|
||||||
@source https://github.com/sasjs/core
|
|
||||||
|
|
||||||
**/
|
**/
|
||||||
|
|
||||||
@@ -30,12 +33,16 @@
|
|||||||
contenttype=TEXT
|
contenttype=TEXT
|
||||||
,inloc=
|
,inloc=
|
||||||
,inref=0
|
,inref=0
|
||||||
|
,iftrue=%str(1=1)
|
||||||
,outname=
|
,outname=
|
||||||
|
,outref=_webout
|
||||||
)/*/STORE SOURCE*/;
|
)/*/STORE SOURCE*/;
|
||||||
|
|
||||||
%let contentype=%upcase(&contenttype);
|
%if not(%eval(%unquote(&iftrue))) %then %return;
|
||||||
%local platform; %let platform=%mf_getplatform();
|
|
||||||
|
|
||||||
|
%let contentype=%upcase(&contenttype);
|
||||||
|
%let outref=%upcase(&outref);
|
||||||
|
%local platform; %let platform=%mf_getplatform();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* check engine type to avoid the below err message:
|
* check engine type to avoid the below err message:
|
||||||
@@ -44,39 +51,101 @@
|
|||||||
%local streamweb;
|
%local streamweb;
|
||||||
%let streamweb=0;
|
%let streamweb=0;
|
||||||
data _null_;
|
data _null_;
|
||||||
set sashelp.vextfl(where=(upcase(fileref)="_WEBOUT"));
|
set sashelp.vextfl(where=(upcase(fileref)="&outref"));
|
||||||
if xengine='STREAM' then call symputx('streamweb',1,'l');
|
if xengine='STREAM' then call symputx('streamweb',1,'l');
|
||||||
run;
|
run;
|
||||||
|
|
||||||
%if &contentype=ZIP %then %do;
|
%if &contentype=CSV %then %do;
|
||||||
%if &platform=SASMETA and &streamweb=1 %then %do;
|
%if (&platform=SASMETA and &streamweb=1) %then %do;
|
||||||
data _null_;
|
data _null_;
|
||||||
rc=stpsrv_header('Content-type','application/zip');
|
rc=stpsrv_header('Content-type','application/csv');
|
||||||
rc=stpsrv_header('Content-disposition',"attachment; filename=&outname");
|
rc=stpsrv_header('Content-disposition',"attachment; filename=&outname");
|
||||||
run;
|
run;
|
||||||
%end;
|
%end;
|
||||||
%else %if &platform=SASVIYA %then %do;
|
%else %if &platform=SASVIYA %then %do;
|
||||||
filename _webout filesrvc parenturi="&SYS_JES_JOB_URI" name='_webout.zip'
|
filename &outref filesrvc parenturi="&SYS_JES_JOB_URI" name='_webout.txt'
|
||||||
contenttype='application/zip'
|
contenttype='application/csv'
|
||||||
contentdisp="attachment; filename=&outname";
|
contentdisp="attachment; filename=&outname";
|
||||||
%end;
|
%end;
|
||||||
|
%else %if &platform=SASJS %then %do;
|
||||||
|
%mfs_httpheader(Content-type,application/csv)
|
||||||
|
%mfs_httpheader(Content-disposition,%str(attachment; filename=&outname))
|
||||||
|
%end;
|
||||||
%end;
|
%end;
|
||||||
%else %if &contentype=EXCEL %then %do;
|
%else %if &contentype=EXCEL %then %do;
|
||||||
/* suitable for XLS format */
|
/* suitable for XLS format */
|
||||||
%if &platform=SASMETA and &streamweb=1 %then %do;
|
%if (&platform=SASMETA and &streamweb=1) %then %do;
|
||||||
data _null_;
|
data _null_;
|
||||||
rc=stpsrv_header('Content-type','application/vnd.ms-excel');
|
rc=stpsrv_header('Content-type','application/vnd.ms-excel');
|
||||||
rc=stpsrv_header('Content-disposition',"attachment; filename=&outname");
|
rc=stpsrv_header('Content-disposition',"attachment; filename=&outname");
|
||||||
run;
|
run;
|
||||||
%end;
|
%end;
|
||||||
%else %if &platform=SASVIYA %then %do;
|
%else %if &platform=SASVIYA %then %do;
|
||||||
filename _webout filesrvc parenturi="&SYS_JES_JOB_URI" name='_webout.xls'
|
filename &outref filesrvc parenturi="&SYS_JES_JOB_URI" name='_webout.xls'
|
||||||
contenttype='application/vnd.ms-excel'
|
contenttype='application/vnd.ms-excel'
|
||||||
contentdisp="attachment; filename=&outname";
|
contentdisp="attachment; filename=&outname";
|
||||||
%end;
|
%end;
|
||||||
|
%else %if &platform=SASJS %then %do;
|
||||||
|
%mfs_httpheader(Content-type,application/vnd.ms-excel)
|
||||||
|
%mfs_httpheader(Content-disposition,%str(attachment; filename=&outname))
|
||||||
|
%end;
|
||||||
|
%end;
|
||||||
|
%else %if &contentype=GIF or &contentype=JPEG or &contentype=PNG %then %do;
|
||||||
|
%if (&platform=SASMETA and &streamweb=1) %then %do;
|
||||||
|
data _null_;
|
||||||
|
rc=stpsrv_header('Content-type',"image/%lowcase(&contenttype)");
|
||||||
|
run;
|
||||||
|
%end;
|
||||||
|
%else %if &platform=SASVIYA %then %do;
|
||||||
|
filename &outref filesrvc parenturi="&SYS_JES_JOB_URI"
|
||||||
|
contenttype="image/%lowcase(&contenttype)";
|
||||||
|
%end;
|
||||||
|
%else %if &platform=SASJS %then %do;
|
||||||
|
%mfs_httpheader(Content-type,image/%lowcase(&contenttype))
|
||||||
|
%end;
|
||||||
|
%end;
|
||||||
|
%else %if &contentype=HTML %then %do;
|
||||||
|
%if &platform=SASVIYA %then %do;
|
||||||
|
filename &outref filesrvc parenturi="&SYS_JES_JOB_URI" name="_webout.json"
|
||||||
|
contenttype="text/html";
|
||||||
|
%end;
|
||||||
|
%else %if &platform=SASJS %then %do;
|
||||||
|
%mfs_httpheader(Content-type,text/html)
|
||||||
|
%end;
|
||||||
|
%end;
|
||||||
|
%else %if &contentype=TEXT %then %do;
|
||||||
|
%if (&platform=SASMETA and &streamweb=1) %then %do;
|
||||||
|
data _null_;
|
||||||
|
rc=stpsrv_header('Content-type','application/text');
|
||||||
|
rc=stpsrv_header('Content-disposition',"attachment; filename=&outname");
|
||||||
|
run;
|
||||||
|
%end;
|
||||||
|
%else %if &platform=SASVIYA %then %do;
|
||||||
|
filename &outref filesrvc parenturi="&SYS_JES_JOB_URI" name='_webout.txt'
|
||||||
|
contenttype='application/text'
|
||||||
|
contentdisp="attachment; filename=&outname";
|
||||||
|
%end;
|
||||||
|
%else %if &platform=SASJS %then %do;
|
||||||
|
%mfs_httpheader(Content-type,application/text)
|
||||||
|
%mfs_httpheader(Content-disposition,%str(attachment; filename=&outname))
|
||||||
|
%end;
|
||||||
|
%end;
|
||||||
|
%else %if &contentype=WOFF or &contentype=WOFF2 or &contentype=TTF %then %do;
|
||||||
|
%if (&platform=SASMETA and &streamweb=1) %then %do;
|
||||||
|
data _null_;
|
||||||
|
rc=stpsrv_header('Content-type',"font/%lowcase(&contenttype)");
|
||||||
|
run;
|
||||||
|
%end;
|
||||||
|
%else %if &platform=SASVIYA %then %do;
|
||||||
|
filename &outref filesrvc parenturi="&SYS_JES_JOB_URI"
|
||||||
|
contenttype="font/%lowcase(&contenttype)";
|
||||||
|
%end;
|
||||||
|
%else %if &platform=SASJS %then %do;
|
||||||
|
%mfs_httpheader(Content-type,font/%lowcase(&contenttype))
|
||||||
|
%end;
|
||||||
%end;
|
%end;
|
||||||
%else %if &contentype=XLSX %then %do;
|
%else %if &contentype=XLSX %then %do;
|
||||||
%if &platform=SASMETA and &streamweb=1 %then %do;
|
%if (&platform=SASMETA and &streamweb=1) %then %do;
|
||||||
data _null_;
|
data _null_;
|
||||||
rc=stpsrv_header('Content-type',
|
rc=stpsrv_header('Content-type',
|
||||||
'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet');
|
'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet');
|
||||||
@@ -84,54 +153,44 @@ run;
|
|||||||
run;
|
run;
|
||||||
%end;
|
%end;
|
||||||
%else %if &platform=SASVIYA %then %do;
|
%else %if &platform=SASVIYA %then %do;
|
||||||
filename _webout filesrvc parenturi="&SYS_JES_JOB_URI" name='_webout.xls'
|
filename &outref filesrvc parenturi="&SYS_JES_JOB_URI" name='_webout.xls'
|
||||||
contenttype=
|
contenttype=
|
||||||
'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet'
|
'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet'
|
||||||
contentdisp="attachment; filename=&outname";
|
contentdisp="attachment; filename=&outname";
|
||||||
%end;
|
%end;
|
||||||
|
%else %if &platform=SASJS %then %do;
|
||||||
|
%mfs_httpheader(Content-type
|
||||||
|
,application/vnd.openxmlformats-officedocument.spreadsheetml.sheet
|
||||||
|
)
|
||||||
|
%mfs_httpheader(Content-disposition,%str(attachment; filename=&outname))
|
||||||
|
%end;
|
||||||
%end;
|
%end;
|
||||||
%else %if &contentype=TEXT %then %do;
|
%else %if &contentype=ZIP %then %do;
|
||||||
%if &platform=SASMETA and &streamweb=1 %then %do;
|
%if (&platform=SASMETA and &streamweb=1) %then %do;
|
||||||
data _null_;
|
data _null_;
|
||||||
rc=stpsrv_header('Content-type','application/text');
|
rc=stpsrv_header('Content-type','application/zip');
|
||||||
rc=stpsrv_header('Content-disposition',"attachment; filename=&outname");
|
rc=stpsrv_header('Content-disposition',"attachment; filename=&outname");
|
||||||
run;
|
run;
|
||||||
%end;
|
%end;
|
||||||
%else %if &platform=SASVIYA %then %do;
|
%else %if &platform=SASVIYA %then %do;
|
||||||
filename _webout filesrvc parenturi="&SYS_JES_JOB_URI" name='_webout.txt'
|
filename &outref filesrvc parenturi="&SYS_JES_JOB_URI" name='_webout.zip'
|
||||||
contenttype='application/text'
|
contenttype='application/zip'
|
||||||
contentdisp="attachment; filename=&outname";
|
contentdisp="attachment; filename=&outname";
|
||||||
%end;
|
%end;
|
||||||
%end;
|
%else %if &platform=SASJS %then %do;
|
||||||
%else %if &contentype=CSV %then %do;
|
%mfs_httpheader(Content-type,application/zip)
|
||||||
%if &platform=SASMETA and &streamweb=1 %then %do;
|
%mfs_httpheader(Content-disposition,%str(attachment; filename=&outname))
|
||||||
data _null_;
|
|
||||||
rc=stpsrv_header('Content-type','application/csv');
|
|
||||||
rc=stpsrv_header('Content-disposition',"attachment; filename=&outname");
|
|
||||||
run;
|
|
||||||
%end;
|
|
||||||
%else %if &platform=SASVIYA %then %do;
|
|
||||||
filename _webout filesrvc parenturi="&SYS_JES_JOB_URI" name='_webout.txt'
|
|
||||||
contenttype='application/csv'
|
|
||||||
contentdisp="attachment; filename=&outname";
|
|
||||||
%end;
|
|
||||||
%end;
|
|
||||||
%else %if &contentype=HTML %then %do;
|
|
||||||
%if &platform=SASVIYA %then %do;
|
|
||||||
filename _webout filesrvc parenturi="&SYS_JES_JOB_URI" name="_webout.json"
|
|
||||||
contenttype="text/html";
|
|
||||||
%end;
|
%end;
|
||||||
%end;
|
%end;
|
||||||
%else %do;
|
%else %do;
|
||||||
%put %str(ERR)OR: Content Type &contenttype NOT SUPPORTED by &sysmacroname!;
|
%put %str(ERR)OR: Content Type &contenttype NOT SUPPORTED by &sysmacroname!;
|
||||||
%return;
|
|
||||||
%end;
|
%end;
|
||||||
|
|
||||||
%if &inref ne 0 %then %do;
|
%if &inref ne 0 %then %do;
|
||||||
%mp_binarycopy(inref=&inref,outref=_webout)
|
%mp_binarycopy(inref=&inref,outref=&outref)
|
||||||
%end;
|
%end;
|
||||||
%else %do;
|
%else %do;
|
||||||
%mp_binarycopy(inloc="&inloc",outref=_webout)
|
%mp_binarycopy(inloc="&inloc",outref=&outref)
|
||||||
%end;
|
%end;
|
||||||
|
|
||||||
%mend mp_streamfile;
|
%mend mp_streamfile;
|
||||||
|
|||||||
@@ -1,12 +1,10 @@
|
|||||||
/**
|
/**
|
||||||
@file mp_testservice.sas
|
@file
|
||||||
@brief Will execute a test against a SASjs web service on SAS 9 or Viya
|
@brief Will execute a SASjs web service on SAS 9 or Viya
|
||||||
@details Prepares the input files and retrieves the resulting datasets from
|
@details Prepares the input files and retrieves the resulting datasets from
|
||||||
the response JSON.
|
the response JSON.
|
||||||
|
|
||||||
%mp_testjob(
|
|
||||||
duration=60*5
|
|
||||||
)
|
|
||||||
|
|
||||||
Note - the _webout fileref should NOT be assigned prior to running this macro.
|
Note - the _webout fileref should NOT be assigned prior to running this macro.
|
||||||
|
|
||||||
@@ -14,6 +12,10 @@
|
|||||||
@param [in] inputfiles=(0) A list of space seperated fileref:filename pairs as
|
@param [in] inputfiles=(0) A list of space seperated fileref:filename pairs as
|
||||||
follows:
|
follows:
|
||||||
inputfiles=inref:filename inref2:filename2
|
inputfiles=inref:filename inref2:filename2
|
||||||
|
@param [in] inputdatasets= (0) All datasets in this space seperated list are
|
||||||
|
converted into SASJS-formatted CSVs (see mp_ds2csv.sas) files and added to
|
||||||
|
the list of `inputfiles` for ingestion. The dataset will be sent with the
|
||||||
|
same name (no need for a colon modifier).
|
||||||
@param [in] inputparams=(0) A dataset containing name/value pairs in the
|
@param [in] inputparams=(0) A dataset containing name/value pairs in the
|
||||||
following format:
|
following format:
|
||||||
|name:$32|value:$1000|
|
|name:$32|value:$1000|
|
||||||
@@ -38,9 +40,13 @@
|
|||||||
@li mf_getuniquename.sas
|
@li mf_getuniquename.sas
|
||||||
@li mp_abort.sas
|
@li mp_abort.sas
|
||||||
@li mp_binarycopy.sas
|
@li mp_binarycopy.sas
|
||||||
|
@li mp_ds2csv.sas
|
||||||
@li mv_getjobresult.sas
|
@li mv_getjobresult.sas
|
||||||
@li mv_jobflow.sas
|
@li mv_jobflow.sas
|
||||||
|
|
||||||
|
<h4> Related Programs </h4>
|
||||||
|
@li mp_testservice.test.sas
|
||||||
|
|
||||||
@version 9.4
|
@version 9.4
|
||||||
@author Allan Bowe
|
@author Allan Bowe
|
||||||
|
|
||||||
@@ -48,6 +54,7 @@
|
|||||||
|
|
||||||
%macro mp_testservice(program,
|
%macro mp_testservice(program,
|
||||||
inputfiles=0,
|
inputfiles=0,
|
||||||
|
inputdatasets=0,
|
||||||
inputparams=0,
|
inputparams=0,
|
||||||
debug=log,
|
debug=log,
|
||||||
mdebug=0,
|
mdebug=0,
|
||||||
@@ -56,7 +63,7 @@
|
|||||||
viyaresult=WEBOUT_JSON,
|
viyaresult=WEBOUT_JSON,
|
||||||
viyacontext=SAS Job Execution compute context
|
viyacontext=SAS Job Execution compute context
|
||||||
)/*/STORE SOURCE*/;
|
)/*/STORE SOURCE*/;
|
||||||
%local dbg;
|
%local dbg pcnt fref1 webref i webcount var platform;
|
||||||
%if &mdebug=1 %then %do;
|
%if &mdebug=1 %then %do;
|
||||||
%put &sysmacroname entry vars:;
|
%put &sysmacroname entry vars:;
|
||||||
%put _local_;
|
%put _local_;
|
||||||
@@ -64,7 +71,6 @@
|
|||||||
%else %let dbg=*;
|
%else %let dbg=*;
|
||||||
|
|
||||||
/* sanitise inputparams */
|
/* sanitise inputparams */
|
||||||
%local pcnt;
|
|
||||||
%let pcnt=0;
|
%let pcnt=0;
|
||||||
%if &inputparams ne 0 %then %do;
|
%if &inputparams ne 0 %then %do;
|
||||||
data _null_;
|
data _null_;
|
||||||
@@ -76,7 +82,7 @@
|
|||||||
else do;
|
else do;
|
||||||
x+1;
|
x+1;
|
||||||
call symputx(name,quote(cats(value)),'l');
|
call symputx(name,quote(cats(value)),'l');
|
||||||
call symputx('pval'!!left(x),name,'l');
|
call symputx(cats('pval',x),name,'l');
|
||||||
call symputx('pcnt',x,'l');
|
call symputx('pcnt',x,'l');
|
||||||
end;
|
end;
|
||||||
run;
|
run;
|
||||||
@@ -86,17 +92,25 @@
|
|||||||
)
|
)
|
||||||
%end;
|
%end;
|
||||||
|
|
||||||
|
/* convert inputdatasets to filerefs */
|
||||||
|
%if "&inputdatasets" ne "0" %then %do;
|
||||||
|
%if %quote(&inputfiles)=0 %then %let inputfiles=;
|
||||||
|
%do i=1 %to %sysfunc(countw(&inputdatasets,%str( )));
|
||||||
|
%let var=%scan(&inputdatasets,&i,%str( ));
|
||||||
|
%local dsref&i;
|
||||||
|
%let dsref&i=%mf_getuniquefileref();
|
||||||
|
%mp_ds2csv(&var,outref=&&dsref&i,headerformat=SASJS)
|
||||||
|
%let inputfiles=&inputfiles &&dsref&i:%scan(&var,-1,.);
|
||||||
|
%end;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
|
||||||
%local fref1 webref;
|
|
||||||
%let fref1=%mf_getuniquefileref();
|
%let fref1=%mf_getuniquefileref();
|
||||||
%let webref=%mf_getuniquefileref();
|
%let webref=%mf_getuniquefileref();
|
||||||
|
|
||||||
%local platform;
|
|
||||||
%let platform=%mf_getplatform();
|
%let platform=%mf_getplatform();
|
||||||
%if &platform=SASMETA %then %do;
|
%if &platform=SASMETA %then %do;
|
||||||
|
|
||||||
/* parse the input files */
|
/* parse the input files */
|
||||||
%local webcount i var;
|
|
||||||
%if %quote(&inputfiles) ne 0 %then %do;
|
%if %quote(&inputfiles) ne 0 %then %do;
|
||||||
%let webcount=%sysfunc(countw(&inputfiles));
|
%let webcount=%sysfunc(countw(&inputfiles));
|
||||||
%put &=webcount;
|
%put &=webcount;
|
||||||
|
|||||||
@@ -20,15 +20,24 @@
|
|||||||
;;;;
|
;;;;
|
||||||
run;
|
run;
|
||||||
|
|
||||||
|
For more examples, see mp_validatecol.test.sas
|
||||||
|
|
||||||
|
Tip - when contributing, use https://regex101.com to test the regex validity!
|
||||||
|
|
||||||
@param [in] incol The column to be validated
|
@param [in] incol The column to be validated
|
||||||
@param [in] rule The rule to apply. Current rules:
|
@param [in] rule The rule to apply. Current rules:
|
||||||
|
@li ISINT - checks if the variable is an integer
|
||||||
@li ISNUM - checks if the variable is numeric
|
@li ISNUM - checks if the variable is numeric
|
||||||
@li LIBDS - matches LIBREF.DATASET format
|
@li LIBDS - matches LIBREF.DATASET format
|
||||||
|
@li FORMAT - checks if the provided format is syntactically valid
|
||||||
@param [out] outcol The variable to create, with the results of the match
|
@param [out] outcol The variable to create, with the results of the match
|
||||||
|
|
||||||
<h4> SAS Macros </h4>
|
<h4> SAS Macros </h4>
|
||||||
@li mf_getuniquename.sas
|
@li mf_getuniquename.sas
|
||||||
|
|
||||||
|
<h4> Related Macros </h4>
|
||||||
|
@li mp_validatecol.test.sas
|
||||||
|
|
||||||
@version 9.3
|
@version 9.3
|
||||||
**/
|
**/
|
||||||
|
|
||||||
@@ -38,7 +47,15 @@
|
|||||||
%local tempcol;
|
%local tempcol;
|
||||||
%let tempcol=%mf_getuniquename();
|
%let tempcol=%mf_getuniquename();
|
||||||
|
|
||||||
%if &rule=ISNUM %then %do;
|
%if &rule=ISINT %then %do;
|
||||||
|
&outcol=0;
|
||||||
|
if not missing(&incol) then do;
|
||||||
|
&tempcol=input(&incol,?? best32.);
|
||||||
|
if not missing(&tempcol) then if mod(&tempcol,1)=0 then &outcol=1;
|
||||||
|
end;
|
||||||
|
drop &tempcol;
|
||||||
|
%end;
|
||||||
|
%else %if &rule=ISNUM %then %do;
|
||||||
/*
|
/*
|
||||||
credit SØREN LASSEN
|
credit SØREN LASSEN
|
||||||
https://sasmacro.blogspot.com/2009/06/welcome-isnum-macro.html
|
https://sasmacro.blogspot.com/2009/06/welcome-isnum-macro.html
|
||||||
@@ -62,5 +79,19 @@
|
|||||||
if prxmatch(&tempcol, trim(&incol)) then &outcol=1;
|
if prxmatch(&tempcol, trim(&incol)) then &outcol=1;
|
||||||
else &outcol=0;
|
else &outcol=0;
|
||||||
%end;
|
%end;
|
||||||
|
%else %if &rule=FORMAT %then %do;
|
||||||
|
/* match valid format - regex could probably be improved */
|
||||||
|
if _n_=1 then do;
|
||||||
|
retain &tempcol;
|
||||||
|
&tempcol=prxparse('/^[_a-z\$]\w{0,31}\.[0-9]*$/i');
|
||||||
|
if missing(&tempcol) then do;
|
||||||
|
putlog "%str(ERR)OR: Invalid expression for FORMAT";
|
||||||
|
stop;
|
||||||
|
end;
|
||||||
|
drop &tempcol;
|
||||||
|
end;
|
||||||
|
if prxmatch(&tempcol, trim(&incol)) then &outcol=1;
|
||||||
|
else &outcol=0;
|
||||||
|
%end;
|
||||||
|
|
||||||
%mend mp_validatecol;
|
%mend mp_validatecol;
|
||||||
|
|||||||
@@ -4,9 +4,7 @@
|
|||||||
@details Loops with a `sleep()` command until a file arrives or the max wait
|
@details Loops with a `sleep()` command until a file arrives or the max wait
|
||||||
period expires.
|
period expires.
|
||||||
|
|
||||||
@example
|
Example: Wait 3 minutes OR for /tmp/flag.txt to appear
|
||||||
|
|
||||||
Wait 3 minutes OR for /tmp/flag.txt to appear
|
|
||||||
|
|
||||||
%mp_wait4file(/tmp/flag.txt , maxwait=60*3)
|
%mp_wait4file(/tmp/flag.txt , maxwait=60*3)
|
||||||
|
|
||||||
|
|||||||
2
build.py
2
build.py
@@ -84,7 +84,7 @@ options noquotelenmax;
|
|||||||
"""
|
"""
|
||||||
f = open('all.sas', "w") # r / r+ / rb / rb+ / w / wb
|
f = open('all.sas', "w") # r / r+ / rb / rb+ / w / wb
|
||||||
f.write(header)
|
f.write(header)
|
||||||
folders=['base','meta','metax','server','viya','lua','fcmp']
|
folders=['base','ddl','meta','metax','server','viya','lua','fcmp']
|
||||||
for folder in folders:
|
for folder in folders:
|
||||||
filenames = [fn for fn in Path('./' + folder).iterdir() if fn.match("*.sas")]
|
filenames = [fn for fn in Path('./' + folder).iterdir() if fn.match("*.sas")]
|
||||||
filenames.sort()
|
filenames.sort()
|
||||||
|
|||||||
34
ddl/mddl_dc_difftable.sas
Normal file
34
ddl/mddl_dc_difftable.sas
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
/**
|
||||||
|
@file
|
||||||
|
@brief Difftable DDL
|
||||||
|
@details Used to store changes to tables. Used by mp_storediffs.sas
|
||||||
|
and mp_stackdiffs.sas
|
||||||
|
|
||||||
|
**/
|
||||||
|
|
||||||
|
|
||||||
|
%macro mddl_dc_difftable(libds=WORK.DIFFTABLE);
|
||||||
|
|
||||||
|
proc sql;
|
||||||
|
create table &libds(
|
||||||
|
load_ref char(36) label='unique load reference',
|
||||||
|
processed_dttm num format=E8601DT26.6 label='Processed at timestamp',
|
||||||
|
libref char(8) label='Library Reference (8 chars)',
|
||||||
|
dsn char(32) label='Dataset Name (32 chars)',
|
||||||
|
key_hash char(32) label=
|
||||||
|
'MD5 Hash of primary key values (pipe seperated)',
|
||||||
|
move_type char(1) label='Either (A)ppended, (D)eleted or (M)odified',
|
||||||
|
is_pk num label='Is Primary Key Field? (1/0)',
|
||||||
|
is_diff num label=
|
||||||
|
'Did value change? (1/0/-1). Always -1 for appends and deletes.',
|
||||||
|
tgtvar_type char(1) label='Either (C)haracter or (N)umeric',
|
||||||
|
tgtvar_nm char(32) label='Target variable name (32 chars)',
|
||||||
|
oldval_num num format=best32. label='Old (numeric) value',
|
||||||
|
newval_num num format=best32. label='New (numeric) value',
|
||||||
|
oldval_char char(32765) label='Old (character) value',
|
||||||
|
newval_char char(32765) label='New (character) value',
|
||||||
|
constraint pk_mpe_audit
|
||||||
|
primary key(load_ref,libref,dsn,key_hash,tgtvar_nm)
|
||||||
|
);
|
||||||
|
|
||||||
|
%mend mddl_dc_difftable;
|
||||||
27
ddl/mddl_dc_filterdetail.sas
Normal file
27
ddl/mddl_dc_filterdetail.sas
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
/**
|
||||||
|
@file
|
||||||
|
@brief Filtertable DDL
|
||||||
|
@details For storing detailed filter values. Used by
|
||||||
|
mp_filterstore.sas.
|
||||||
|
|
||||||
|
**/
|
||||||
|
|
||||||
|
|
||||||
|
%macro mddl_dc_filterdetail(libds=WORK.FILTER_DETAIL);
|
||||||
|
|
||||||
|
proc sql;
|
||||||
|
create table &libds(
|
||||||
|
filter_hash char(32) not null,
|
||||||
|
filter_line num not null,
|
||||||
|
group_logic char(3) not null,
|
||||||
|
subgroup_logic char(3) not null,
|
||||||
|
subgroup_id num not null,
|
||||||
|
variable_nm varchar(32) not null,
|
||||||
|
operator_nm varchar(12) not null,
|
||||||
|
raw_value varchar(4000) not null,
|
||||||
|
processed_dttm num not null format=E8601DT26.6,
|
||||||
|
constraint pk_mpe_filteranytable
|
||||||
|
primary key(filter_hash,filter_line)
|
||||||
|
);
|
||||||
|
|
||||||
|
%mend mddl_dc_filterdetail;
|
||||||
22
ddl/mddl_dc_filtersummary.sas
Normal file
22
ddl/mddl_dc_filtersummary.sas
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
/**
|
||||||
|
@file
|
||||||
|
@brief Filtersummary DDL
|
||||||
|
@details For storing summary filter values. Used by
|
||||||
|
mp_filterstore.sas.
|
||||||
|
|
||||||
|
**/
|
||||||
|
|
||||||
|
|
||||||
|
%macro mddl_dc_filtersummary(libds=WORK.FILTER_SUMMARY);
|
||||||
|
|
||||||
|
proc sql;
|
||||||
|
create table &libds(
|
||||||
|
filter_rk num not null,
|
||||||
|
filter_hash char(32) not null,
|
||||||
|
filter_table char(41) not null,
|
||||||
|
processed_dttm num not null format=E8601DT26.6,
|
||||||
|
constraint pk_mpe_filteranytable
|
||||||
|
primary key(filter_rk)
|
||||||
|
);
|
||||||
|
|
||||||
|
%mend mddl_dc_filtersummary;
|
||||||
25
ddl/mddl_dc_locktable.sas
Normal file
25
ddl/mddl_dc_locktable.sas
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
/**
|
||||||
|
@file
|
||||||
|
@brief Locktable DDL
|
||||||
|
@details For "locking" tables prior to multipass loads. Used by
|
||||||
|
mp_lockanytable.sas
|
||||||
|
|
||||||
|
**/
|
||||||
|
|
||||||
|
|
||||||
|
%macro mddl_dc_locktable(libds=WORK.LOCKTABLE);
|
||||||
|
|
||||||
|
proc sql;
|
||||||
|
create table &libds(
|
||||||
|
lock_lib char(8),
|
||||||
|
lock_ds char(32),
|
||||||
|
lock_status_cd char(10) not null,
|
||||||
|
lock_user_nm char(100) not null ,
|
||||||
|
lock_ref char(200),
|
||||||
|
lock_pid char(10),
|
||||||
|
lock_start_dttm num format=E8601DT26.6,
|
||||||
|
lock_end_dttm num format=E8601DT26.6,
|
||||||
|
constraint pk_mp_lockanytable primary key(lock_lib,lock_ds)
|
||||||
|
);
|
||||||
|
|
||||||
|
%mend mddl_dc_locktable;
|
||||||
24
ddl/mddl_dc_maxkeytable.sas
Normal file
24
ddl/mddl_dc_maxkeytable.sas
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
/**
|
||||||
|
@file
|
||||||
|
@brief Maxkeytable DDL
|
||||||
|
@details For storing the maximum retained key information. Used
|
||||||
|
by mp_retainedkey.sas
|
||||||
|
|
||||||
|
**/
|
||||||
|
|
||||||
|
|
||||||
|
%macro mddl_dc_maxkeytable(libds=WORK.MAXKEYTABLE);
|
||||||
|
|
||||||
|
proc sql;
|
||||||
|
create table &libds(
|
||||||
|
keytable varchar(41) label='Base table in libref.dataset format',
|
||||||
|
keycolumn char(32) format=$32.
|
||||||
|
label='The Retained key field containing the key values.',
|
||||||
|
max_key num label=
|
||||||
|
'Integer representing current max RK or SK value in the KEYTABLE',
|
||||||
|
processed_dttm num format=E8601DT26.6
|
||||||
|
label='Datetime this value was last updated',
|
||||||
|
constraint pk_mpe_maxkeyvalues
|
||||||
|
primary key(keytable));
|
||||||
|
|
||||||
|
%mend mddl_dc_maxkeytable;
|
||||||
43
ddl/mddl_sas_cntlout.sas
Normal file
43
ddl/mddl_sas_cntlout.sas
Normal file
@@ -0,0 +1,43 @@
|
|||||||
|
/**
|
||||||
|
@file
|
||||||
|
@brief The CNTLOUT table generated by proc format
|
||||||
|
@details This table will actually change format depending on the data values,
|
||||||
|
therefore the max possible lengths are described here to enable consistency
|
||||||
|
when dealing with format data.
|
||||||
|
|
||||||
|
**/
|
||||||
|
|
||||||
|
|
||||||
|
%macro mddl_sas_cntlout(libds=WORK.CNTLOUT);
|
||||||
|
|
||||||
|
proc sql;
|
||||||
|
create table &libds(
|
||||||
|
FMTNAME char(32) label='Format name'
|
||||||
|
/*
|
||||||
|
to accomodate larger START values, mp_loadformat.sas will need the
|
||||||
|
SQL dependency removed (proc sql needs to accomodate 3 index values in
|
||||||
|
a 32767 ibufsize limit)
|
||||||
|
*/
|
||||||
|
,START char(10000) label='Starting value for format'
|
||||||
|
,END char(32767) label='Ending value for format'
|
||||||
|
,LABEL char(32767) label='Format value label'
|
||||||
|
,MIN num length=3 label='Minimum length'
|
||||||
|
,MAX num length=3 label='Maximum length'
|
||||||
|
,DEFAULT num length=3 label='Default length'
|
||||||
|
,LENGTH num length=3 label='Format length'
|
||||||
|
,FUZZ num label='Fuzz value'
|
||||||
|
,PREFIX char(2) label='Prefix characters'
|
||||||
|
,MULT num label='Multiplier'
|
||||||
|
,FILL char(1) label='Fill character'
|
||||||
|
,NOEDIT num length=3 label='Is picture string noedit?'
|
||||||
|
,TYPE char(1) label='Type of format'
|
||||||
|
,SEXCL char(1) label='Start exclusion'
|
||||||
|
,EEXCL char(1) label='End exclusion'
|
||||||
|
,HLO char(13) label='Additional information'
|
||||||
|
,DECSEP char(1) label='Decimal separator'
|
||||||
|
,DIG3SEP char(1) label='Three-digit separator'
|
||||||
|
,DATATYPE char(8) label='Date/time/datetime?'
|
||||||
|
,LANGUAGE char(8) label='Language for date strings'
|
||||||
|
);
|
||||||
|
|
||||||
|
%mend mddl_sas_cntlout;
|
||||||
118
fcmp/mcf_getfmttype.sas
Normal file
118
fcmp/mcf_getfmttype.sas
Normal file
@@ -0,0 +1,118 @@
|
|||||||
|
/**
|
||||||
|
@file
|
||||||
|
@brief Returns the type of the format
|
||||||
|
@details
|
||||||
|
Returns the type, eg DATE / DATETIME / TIME (based on hard-coded lookup)
|
||||||
|
else CHAR / NUM.
|
||||||
|
|
||||||
|
This macro may be extended in the future to support custom formats - this
|
||||||
|
would necessitate a call to `dosubl()` for running a proc format with cntlout.
|
||||||
|
|
||||||
|
The function itself takes the following (positional) parameters:
|
||||||
|
|
||||||
|
| PARAMETER | DESCRIPTION |
|
||||||
|
|---|---|
|
||||||
|
|fmtnm| Format name to be tested. Can be with or without the w.d extension.|
|
||||||
|
|
||||||
|
Usage:
|
||||||
|
|
||||||
|
%mcf_getfmttype(wrap=YES, insert_cmplib=YES)
|
||||||
|
|
||||||
|
data _null_;
|
||||||
|
fmt1=mcf_getfmttype('DATE9.');
|
||||||
|
fmt2=mcf_getfmttype('DATETIME');
|
||||||
|
put (fmt:)(=);
|
||||||
|
run;
|
||||||
|
%put fmt3=%sysfunc(mcf_getfmttype(TIME9.));
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
|
||||||
|
> fmt1=DATE fmt2=DATETIME
|
||||||
|
> fmt3=TIME
|
||||||
|
|
||||||
|
@param [out] wrap= (NO) Choose YES to add the proc fcmp wrapper.
|
||||||
|
@param [out] lib= (work) The output library in which to create the catalog.
|
||||||
|
@param [out] cat= (sasjs) The output catalog in which to create the package.
|
||||||
|
@param [out] pkg= (utils) The output package in which to create the function.
|
||||||
|
Uses a 3 part format: libref.catalog.package
|
||||||
|
@param [out] insert_cmplib= DEPRECATED - The CMPLIB option is checked and
|
||||||
|
values inserted only if needed.
|
||||||
|
|
||||||
|
<h4> SAS Macros </h4>
|
||||||
|
@li mcf_init.sas
|
||||||
|
|
||||||
|
<h4> Related Programs </h4>
|
||||||
|
@li mcf_getfmttype.test.sas
|
||||||
|
@li mp_init.sas
|
||||||
|
|
||||||
|
@todo "Custom Format Lookups" To enable site-specific formats, make
|
||||||
|
use of a set of SASJS_FMTLIST_(DATATYPE) global variables.
|
||||||
|
|
||||||
|
**/
|
||||||
|
|
||||||
|
%macro mcf_getfmttype(wrap=NO
|
||||||
|
,insert_cmplib=DEPRECATED
|
||||||
|
,lib=WORK
|
||||||
|
,cat=SASJS
|
||||||
|
,pkg=UTILS
|
||||||
|
)/*/STORE SOURCE*/;
|
||||||
|
%local i var cmpval found;
|
||||||
|
|
||||||
|
%if %mcf_init(mcf_getfmttype)=1 %then %return;
|
||||||
|
|
||||||
|
%if &wrap=YES %then %do;
|
||||||
|
proc fcmp outlib=&lib..&cat..&pkg;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
function mcf_getfmttype(fmtnm $) $8;
|
||||||
|
if substr(fmtnm,1,1)='$' then return('CHAR');
|
||||||
|
else do;
|
||||||
|
/* extract NAME */
|
||||||
|
length fmt $32;
|
||||||
|
fmt=scan(fmtnm,1,'.');
|
||||||
|
do while (
|
||||||
|
substr(fmt,length(fmt),1) in ('1','2','3','4','5','6','7','8','9','0')
|
||||||
|
);
|
||||||
|
if length(fmt)=1 then fmt='W';
|
||||||
|
else fmt=substr(fmt,1,length(fmt)-1);
|
||||||
|
end;
|
||||||
|
|
||||||
|
/* apply lookups */
|
||||||
|
if cats(fmt) in ('DATETIME','B8601DN','B8601DN','B8601DT','B8601DT'
|
||||||
|
,'B8601DZ','B8601DZ','DATEAMPM','DTDATE','DTMONYY','DTWKDATX','DTYEAR'
|
||||||
|
,'DTYYQC','E8601DN','E8601DN','E8601DT','E8601DT','E8601DZ','E8601DZ')
|
||||||
|
then return('DATETIME');
|
||||||
|
else if fmt in ('DATE','YYMMDD','B8601DA','B8601DA','DAY','DDMMYY'
|
||||||
|
,'DDMMYYB','DDMMYYC','DDMMYYD','DDMMYYN','DDMMYYP','DDMMYYS','DDMMYYx'
|
||||||
|
,'DOWNAME','E8601DA','E8601DA','JULDAY','JULIAN','MMDDYY','MMDDYYB'
|
||||||
|
,'MMDDYYC','MMDDYYD','MMDDYYN','MMDDYYP','MMDDYYS','MMDDYYx','MMYY'
|
||||||
|
,'MMYYC','MMYYD','MMYYN','MMYYP','MMYYS','MMYYx','MONNAME','MONTH'
|
||||||
|
,'MONYY','PDJULG','PDJULI','QTR','QTRR','WEEKDATE','WEEKDATX','WEEKDAY'
|
||||||
|
,'WEEKU','WEEKV','WEEKW','WORDDATE','WORDDATX','YEAR','YYMM','YYMMC'
|
||||||
|
,'YYMMD','YYMMDDB','YYMMDDC','YYMMDDD','YYMMDDN','YYMMDDP','YYMMDDS'
|
||||||
|
,'YYMMDDx','YYMMN','YYMMP','YYMMS','YYMMx','YYMON','YYQ','YYQC','YYQD'
|
||||||
|
,'YYQN','YYQP','YYQR','YYQRC','YYQRD','YYQRN','YYQRP','YYQRS','YYQRx'
|
||||||
|
,'YYQS','YYQx','YYQZ') then return('DATE');
|
||||||
|
else if fmt in ('TIME','B8601LZ','B8601LZ','B8601TM','B8601TM','B8601TZ'
|
||||||
|
,'B8601TZ','E8601LZ','E8601LZ','E8601TM','E8601TM','E8601TZ','E8601TZ'
|
||||||
|
,'HHMM','HOUR','MMSS','TIMEAMPM','TOD') then return('TIME');
|
||||||
|
else return('NUM');
|
||||||
|
end;
|
||||||
|
endsub;
|
||||||
|
|
||||||
|
%if &wrap=YES %then %do;
|
||||||
|
quit;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
/* insert the CMPLIB if not already there */
|
||||||
|
%let cmpval=%sysfunc(getoption(cmplib));
|
||||||
|
%let found=0;
|
||||||
|
%do i=1 %to %sysfunc(countw(&cmpval,%str( %(%))));
|
||||||
|
%let var=%scan(&cmpval,&i,%str( %(%)));
|
||||||
|
%if &var=&lib..&cat %then %let found=1;
|
||||||
|
%end;
|
||||||
|
%if &found=0 %then %do;
|
||||||
|
options insert=(CMPLIB=(&lib..&cat));
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%mend mcf_getfmttype;
|
||||||
44
fcmp/mcf_init.sas
Normal file
44
fcmp/mcf_init.sas
Normal file
@@ -0,0 +1,44 @@
|
|||||||
|
/**
|
||||||
|
@file
|
||||||
|
@brief Sets up the mcf_xx functions
|
||||||
|
@details
|
||||||
|
There is no (efficient) way to determine if an mcf_xx macro has already been
|
||||||
|
invoked. So, we make use of a global macro variable list to keep track.
|
||||||
|
|
||||||
|
Usage:
|
||||||
|
|
||||||
|
%mcf_init(MCF_LENGTH)
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
|
||||||
|
> 1 (if already initialised) else 0
|
||||||
|
|
||||||
|
@param [in] func The function to be initialised
|
||||||
|
|
||||||
|
<h4> Related Macros </h4>
|
||||||
|
@li mcf_init.test.sas
|
||||||
|
|
||||||
|
**/
|
||||||
|
|
||||||
|
%macro mcf_init(func
|
||||||
|
)/*/STORE SOURCE*/;
|
||||||
|
|
||||||
|
%if not (%symexist(SASJS_PREFIX)) %then %do;
|
||||||
|
%global SASJS_PREFIX;
|
||||||
|
%let SASJS_PREFIX=SASJS;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%let func=%upcase(&func);
|
||||||
|
|
||||||
|
/* the / character is just a seperator */
|
||||||
|
%global &sasjs_prefix._FUNCTIONS;
|
||||||
|
%if %index(&&&sasjs_prefix._FUNCTIONS,&func/)>0 %then %do;
|
||||||
|
1
|
||||||
|
%return;
|
||||||
|
%end;
|
||||||
|
%else %do;
|
||||||
|
%let &sasjs_prefix._FUNCTIONS=&&&sasjs_prefix._FUNCTIONS &func/;
|
||||||
|
0
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%mend mcf_init;
|
||||||
90
fcmp/mcf_length.sas
Normal file
90
fcmp/mcf_length.sas
Normal file
@@ -0,0 +1,90 @@
|
|||||||
|
/**
|
||||||
|
@file
|
||||||
|
@brief Returns the length of a numeric value
|
||||||
|
@details
|
||||||
|
Returns the length, in bytes, of a numeric value. If the value is
|
||||||
|
missing, then 0 is returned.
|
||||||
|
|
||||||
|
The function itself takes the following (positional) parameters:
|
||||||
|
|
||||||
|
| PARAMETER | DESCRIPTION |
|
||||||
|
|---|---|
|
||||||
|
| var | variable (or value) to be tested|
|
||||||
|
|
||||||
|
Usage:
|
||||||
|
|
||||||
|
%mcf_length(wrap=YES, insert_cmplib=YES)
|
||||||
|
|
||||||
|
data _null_;
|
||||||
|
ina=1;
|
||||||
|
inb=10000000;
|
||||||
|
inc=12345678;
|
||||||
|
ind=.;
|
||||||
|
outa=mcf_length(ina);
|
||||||
|
outb=mcf_length(inb);
|
||||||
|
outc=mcf_length(inc);
|
||||||
|
outd=mcf_length(ind);
|
||||||
|
put (out:)(=);
|
||||||
|
run;
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
|
||||||
|
> outa=3 outb=4 outc=5 outd=0
|
||||||
|
|
||||||
|
@param [out] wrap= (NO) Choose YES to add the proc fcmp wrapper.
|
||||||
|
@param [out] lib= (work) The output library in which to create the catalog.
|
||||||
|
@param [out] cat= (sasjs) The output catalog in which to create the package.
|
||||||
|
@param [out] pkg= (utils) The output package in which to create the function.
|
||||||
|
Uses a 3 part format: libref.catalog.package
|
||||||
|
@param [out] insert_cmplib= DEPRECATED - The CMPLIB option is checked and
|
||||||
|
values inserted only if needed.
|
||||||
|
|
||||||
|
<h4> SAS Macros </h4>
|
||||||
|
@li mcf_init.sas
|
||||||
|
|
||||||
|
<h4> Related Programs </h4>
|
||||||
|
@li mcf_length.test.sas
|
||||||
|
@li mp_init.sas
|
||||||
|
|
||||||
|
**/
|
||||||
|
|
||||||
|
%macro mcf_length(wrap=NO
|
||||||
|
,insert_cmplib=DEPRECATED
|
||||||
|
,lib=WORK
|
||||||
|
,cat=SASJS
|
||||||
|
,pkg=UTILS
|
||||||
|
)/*/STORE SOURCE*/;
|
||||||
|
%local i var cmpval found;
|
||||||
|
%if %mcf_init(mcf_length)=1 %then %return;
|
||||||
|
|
||||||
|
%if &wrap=YES %then %do;
|
||||||
|
proc fcmp outlib=&lib..&cat..&pkg;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
function mcf_length(var);
|
||||||
|
if var=. then len=0;
|
||||||
|
else if missing(var) or trunc(var,3)=var then len=3;
|
||||||
|
else if trunc(var,4)=var then len=4;
|
||||||
|
else if trunc(var,5)=var then len=5;
|
||||||
|
else if trunc(var,6)=var then len=6;
|
||||||
|
else if trunc(var,7)=var then len=7;
|
||||||
|
else len=8;
|
||||||
|
return(len);
|
||||||
|
endsub;
|
||||||
|
|
||||||
|
%if &wrap=YES %then %do;
|
||||||
|
quit;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
/* insert the CMPLIB if not already there */
|
||||||
|
%let cmpval=%sysfunc(getoption(cmplib));
|
||||||
|
%let found=0;
|
||||||
|
%do i=1 %to %sysfunc(countw(&cmpval,%str( %(%))));
|
||||||
|
%let var=%scan(&cmpval,&i,%str( %(%)));
|
||||||
|
%if &var=&lib..&cat %then %let found=1;
|
||||||
|
%end;
|
||||||
|
%if &found=0 %then %do;
|
||||||
|
options insert=(CMPLIB=(&lib..&cat));
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%mend mcf_length;
|
||||||
@@ -47,29 +47,33 @@
|
|||||||
|
|
||||||
|
|
||||||
@param [out] wrap= (NO) Choose YES to add the proc fcmp wrapper.
|
@param [out] wrap= (NO) Choose YES to add the proc fcmp wrapper.
|
||||||
@param [out] insert_cmplib= (NO) Choose YES to insert the package into the
|
|
||||||
CMPLIB reference.
|
|
||||||
@param [out] lib= (work) The output library in which to create the catalog.
|
@param [out] lib= (work) The output library in which to create the catalog.
|
||||||
@param [out] cat= (sasjs) The output catalog in which to create the package.
|
@param [out] cat= (sasjs) The output catalog in which to create the package.
|
||||||
@param [out] pkg= (utils) The output package in which to create the function.
|
@param [out] pkg= (utils) The output package in which to create the function.
|
||||||
Uses a 3 part format: libref.catalog.package
|
Uses a 3 part format: libref.catalog.package
|
||||||
|
@param [out] insert_cmplib= DEPRECATED - The CMPLIB option is checked and
|
||||||
|
values inserted only if needed.
|
||||||
|
|
||||||
<h4> SAS Macros </h4>
|
<h4> SAS Macros </h4>
|
||||||
@li mf_existfunction.sas
|
@li mcf_init.sas
|
||||||
|
|
||||||
|
<h4> Related Programs </h4>
|
||||||
|
@li mcf_stpsrv_header.test.sas
|
||||||
|
@li mp_init.sas
|
||||||
|
|
||||||
**/
|
**/
|
||||||
|
|
||||||
%macro mcf_stpsrv_header(wrap=NO
|
%macro mcf_stpsrv_header(wrap=NO
|
||||||
,insert_cmplib=NO
|
,insert_cmplib=DEPRECATED
|
||||||
,lib=WORK
|
,lib=WORK
|
||||||
,cat=SASJS
|
,cat=SASJS
|
||||||
,pkg=UTILS
|
,pkg=UTILS
|
||||||
)/*/STORE SOURCE*/;
|
)/*/STORE SOURCE*/;
|
||||||
|
%local i var cmpval found;
|
||||||
%if %mf_existfunction(stpsrv_header)=1 %then %return;
|
%if %mcf_init(stpsrv_header)=1 %then %return;
|
||||||
|
|
||||||
%if &wrap=YES %then %do;
|
%if &wrap=YES %then %do;
|
||||||
proc fcmp outcat=&lib..&cat..&pkg;
|
proc fcmp outlib=&lib..&cat..&pkg;
|
||||||
%end;
|
%end;
|
||||||
|
|
||||||
function stpsrv_header(name $, value $);
|
function stpsrv_header(name $, value $);
|
||||||
@@ -92,7 +96,14 @@ endsub;
|
|||||||
quit;
|
quit;
|
||||||
%end;
|
%end;
|
||||||
|
|
||||||
%if &insert_cmplib=YES %then %do;
|
/* insert the CMPLIB if not already there */
|
||||||
|
%let cmpval=%sysfunc(getoption(cmplib));
|
||||||
|
%let found=0;
|
||||||
|
%do i=1 %to %sysfunc(countw(&cmpval,%str( %(%))));
|
||||||
|
%let var=%scan(&cmpval,&i,%str( %(%)));
|
||||||
|
%if &var=&lib..&cat %then %let found=1;
|
||||||
|
%end;
|
||||||
|
%if &found=0 %then %do;
|
||||||
options insert=(CMPLIB=(&lib..&cat));
|
options insert=(CMPLIB=(&lib..&cat));
|
||||||
%end;
|
%end;
|
||||||
|
|
||||||
|
|||||||
@@ -32,24 +32,33 @@
|
|||||||
run;
|
run;
|
||||||
|
|
||||||
@param [out] wrap= (NO) Choose YES to add the proc fcmp wrapper.
|
@param [out] wrap= (NO) Choose YES to add the proc fcmp wrapper.
|
||||||
@param [out] insert_cmplib= (NO) Choose YES to insert the package into the
|
|
||||||
CMPLIB reference.
|
|
||||||
@param [out] lib= (work) The output library in which to create the catalog.
|
@param [out] lib= (work) The output library in which to create the catalog.
|
||||||
@param [out] cat= (sasjs) The output catalog in which to create the package.
|
@param [out] cat= (sasjs) The output catalog in which to create the package.
|
||||||
@param [out] pkg= (utils) The output package in which to create the function.
|
@param [out] pkg= (utils) The output package in which to create the function.
|
||||||
Uses a 3 part format: libref.catalog.package
|
Uses a 3 part format: libref.catalog.package
|
||||||
|
@param [out] insert_cmplib= DEPRECATED - The CMPLIB option is checked and
|
||||||
|
values inserted only if needed.
|
||||||
|
|
||||||
|
<h4> SAS Macros </h4>
|
||||||
|
@li mcf_init.sas
|
||||||
|
|
||||||
|
<h4> Related Programs </h4>
|
||||||
|
@li mcf_stpsrv_header.test.sas
|
||||||
|
@li mp_init.sas
|
||||||
|
|
||||||
**/
|
**/
|
||||||
|
|
||||||
%macro mcf_string2file(wrap=NO
|
%macro mcf_string2file(wrap=NO
|
||||||
,insert_cmplib=NO
|
,insert_cmplib=DEPRECATED
|
||||||
,lib=WORK
|
,lib=WORK
|
||||||
,cat=SASJS
|
,cat=SASJS
|
||||||
,pkg=UTILS
|
,pkg=UTILS
|
||||||
)/*/STORE SOURCE*/;
|
)/*/STORE SOURCE*/;
|
||||||
|
%local i var cmpval found;
|
||||||
|
%if %mcf_init(mcf_string2file)=1 %then %return;
|
||||||
|
|
||||||
%if &wrap=YES %then %do;
|
%if &wrap=YES %then %do;
|
||||||
proc fcmp outcat=&lib..&cat..&pkg;
|
proc fcmp outlib=&lib..&cat..&pkg;
|
||||||
%end;
|
%end;
|
||||||
|
|
||||||
function mcf_string2file(filepath $, string $, mode $);
|
function mcf_string2file(filepath $, string $, mode $);
|
||||||
@@ -72,7 +81,14 @@ endsub;
|
|||||||
quit;
|
quit;
|
||||||
%end;
|
%end;
|
||||||
|
|
||||||
%if &insert_cmplib=YES %then %do;
|
/* insert the CMPLIB if not already there */
|
||||||
|
%let cmpval=%sysfunc(getoption(cmplib));
|
||||||
|
%let found=0;
|
||||||
|
%do i=1 %to %sysfunc(countw(&cmpval,%str( %(%))));
|
||||||
|
%let var=%scan(&cmpval,&i,%str( %(%)));
|
||||||
|
%if &var=&lib..&cat %then %let found=1;
|
||||||
|
%end;
|
||||||
|
%if &found=0 %then %do;
|
||||||
options insert=(CMPLIB=(&lib..&cat));
|
options insert=(CMPLIB=(&lib..&cat));
|
||||||
%end;
|
%end;
|
||||||
|
|
||||||
|
|||||||
@@ -295,7 +295,7 @@ run;
|
|||||||
prop='Connection.DBMS.Property.SERVER.Name.xmlKey.txt';
|
prop='Connection.DBMS.Property.SERVER.Name.xmlKey.txt';
|
||||||
rc=metadata_getprop(uri,prop,server,"");
|
rc=metadata_getprop(uri,prop,server,"");
|
||||||
end;
|
end;
|
||||||
if server^='' then server='server='!!server;
|
if server^='' then server='server='!!quote(cats(server));
|
||||||
call symputx('server',server,'l');
|
call symputx('server',server,'l');
|
||||||
|
|
||||||
/* get SCHEMA value */
|
/* get SCHEMA value */
|
||||||
@@ -441,11 +441,11 @@ run;
|
|||||||
run;
|
run;
|
||||||
|
|
||||||
%put NOTE: Executing the following:/; %put NOTE-;
|
%put NOTE: Executing the following:/; %put NOTE-;
|
||||||
%put NOTE- libname &libref TERADATA server=&path schema=&schema ;
|
%put NOTE- libname &libref TERADATA server="&path" schema=&schema ;
|
||||||
%put NOTe- authdomain=&authdomain;
|
%put NOTe- authdomain=&authdomain;
|
||||||
%put NOTE-;
|
%put NOTE-;
|
||||||
|
|
||||||
libname &libref TERADATA server=&path schema=&schema authdomain=&authdomain;
|
libname &libref TERADATA server="&path" schema=&schema authdomain=&authdomain;
|
||||||
%end;
|
%end;
|
||||||
%else %if &engine= %then %do;
|
%else %if &engine= %then %do;
|
||||||
%put NOTE: Libref &libref is not registered in metadata;
|
%put NOTE: Libref &libref is not registered in metadata;
|
||||||
|
|||||||
@@ -40,6 +40,12 @@
|
|||||||
/* now try and assign it */
|
/* now try and assign it */
|
||||||
if libname("&libref",,'meta',cats('liburi="',liburi,'";')) ne 0 then do;
|
if libname("&libref",,'meta',cats('liburi="',liburi,'";')) ne 0 then do;
|
||||||
putlog "&libref could not be assigned";
|
putlog "&libref could not be assigned";
|
||||||
|
putlog liburi=;
|
||||||
|
/**
|
||||||
|
* Fetch the system message for display in the abort modal. This is
|
||||||
|
* not always helpful though. One example, previously received:
|
||||||
|
* NOTE: Libref XX refers to the same library metadata as libref XX.
|
||||||
|
*/
|
||||||
call symputx('msg',sysmsg(),'l');
|
call symputx('msg',sysmsg(),'l');
|
||||||
if "&mabort"='HARD' then call symputx('mp_abort',1,'l');
|
if "&mabort"='HARD' then call symputx('mp_abort',1,'l');
|
||||||
end;
|
end;
|
||||||
@@ -61,7 +67,7 @@
|
|||||||
|
|
||||||
%if &mp_abort=1 %then %do;
|
%if &mp_abort=1 %then %do;
|
||||||
%mp_abort(iftrue= (&mp_abort=1)
|
%mp_abort(iftrue= (&mp_abort=1)
|
||||||
,mac=&sysmacroname
|
,mac=mm_assignlib.sas
|
||||||
,msg=&msg
|
,msg=&msg
|
||||||
)
|
)
|
||||||
%return;
|
%return;
|
||||||
|
|||||||
@@ -59,7 +59,10 @@
|
|||||||
%&mD.put Executing &sysmacroname..sas;
|
%&mD.put Executing &sysmacroname..sas;
|
||||||
%&mD.put _local_;
|
%&mD.put _local_;
|
||||||
|
|
||||||
%mf_verifymacvars(tree name)
|
%mp_abort(iftrue= (%mf_verifymacvars(tree name)=0)
|
||||||
|
,mac=&sysmacroname
|
||||||
|
,msg=%str(Empty inputs: tree name)
|
||||||
|
)
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* check tree exists
|
* check tree exists
|
||||||
|
|||||||
@@ -47,7 +47,10 @@
|
|||||||
%&mD.put Executing &sysmacroname..sas;
|
%&mD.put Executing &sysmacroname..sas;
|
||||||
%&mD.put _local_;
|
%&mD.put _local_;
|
||||||
|
|
||||||
%mf_verifymacvars(tree name)
|
%mp_abort(iftrue= (%mf_verifymacvars(tree name)=0)
|
||||||
|
,mac=&sysmacroname
|
||||||
|
,msg=%str(Empty inputs: tree name)
|
||||||
|
)
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* check tree exists
|
* check tree exists
|
||||||
|
|||||||
@@ -12,13 +12,14 @@
|
|||||||
The macro is idempotent - if you run it twice, it will only create a folder
|
The macro is idempotent - if you run it twice, it will only create a folder
|
||||||
once.
|
once.
|
||||||
|
|
||||||
usage:
|
Usage:
|
||||||
|
|
||||||
%mm_createfolder(path=/some/meta/folder)
|
%mm_createfolder(path=/some/meta/folder)
|
||||||
|
|
||||||
@param [in] path= Name of the folder to create.
|
@param [in] path= Name of the folder to create.
|
||||||
@param [in] mdebug= set DBG to 1 to disable DEBUG messages
|
@param [in] mdebug= set DBG to 1 to disable DEBUG messages
|
||||||
|
|
||||||
|
|
||||||
@version 9.4
|
@version 9.4
|
||||||
@author Allan Bowe
|
@author Allan Bowe
|
||||||
|
|
||||||
|
|||||||
@@ -133,12 +133,14 @@ run;
|
|||||||
filename &frefin temp;
|
filename &frefin temp;
|
||||||
filename &frefout temp;
|
filename &frefout temp;
|
||||||
|
|
||||||
|
%mp_abort(iftrue= (
|
||||||
|
&engine=BASE & %mf_verifymacvars(libname libref engine servercontext tree)=0
|
||||||
|
)
|
||||||
|
,mac=&sysmacroname
|
||||||
|
,msg=%str(Empty inputs: libname libref engine servercontext tree)
|
||||||
|
)
|
||||||
|
|
||||||
%if &engine=BASE %then %do;
|
%if &engine=BASE %then %do;
|
||||||
|
|
||||||
%mf_verifymacvars(libname libref engine servercontext tree)
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Check that the ServerContext exists
|
* Check that the ServerContext exists
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -12,7 +12,7 @@
|
|||||||
This macro is idempotent - if you run it twice, it will only create an STP
|
This macro is idempotent - if you run it twice, it will only create an STP
|
||||||
once.
|
once.
|
||||||
|
|
||||||
usage (type 1 STP):
|
Usage (type 1 STP):
|
||||||
|
|
||||||
%mm_createstp(stpname=MyNewSTP
|
%mm_createstp(stpname=MyNewSTP
|
||||||
,filename=mySpecialProgram.sas
|
,filename=mySpecialProgram.sas
|
||||||
@@ -31,7 +31,8 @@
|
|||||||
putlog (_all_)(=);
|
putlog (_all_)(=);
|
||||||
run;
|
run;
|
||||||
|
|
||||||
usage (type 2 STP):
|
Usage (type 2 STP):
|
||||||
|
|
||||||
%mm_createstp(stpname=MyNewType2STP
|
%mm_createstp(stpname=MyNewType2STP
|
||||||
,filename=mySpecialProgram.sas
|
,filename=mySpecialProgram.sas
|
||||||
,directory=SASEnvironment/SASCode/STPs
|
,directory=SASEnvironment/SASCode/STPs
|
||||||
@@ -39,14 +40,6 @@
|
|||||||
,Server=SASApp
|
,Server=SASApp
|
||||||
,stptype=2)
|
,stptype=2)
|
||||||
|
|
||||||
<h4> SAS Macros </h4>
|
|
||||||
@li mf_nobs.sas
|
|
||||||
@li mf_verifymacvars.sas
|
|
||||||
@li mm_getdirectories.sas
|
|
||||||
@li mm_updatestpsourcecode.sas
|
|
||||||
@li mp_dropmembers.sas
|
|
||||||
@li mm_getservercontexts.sas
|
|
||||||
|
|
||||||
@param stpname= Stored Process name. Avoid spaces - testing has shown that
|
@param stpname= 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.
|
||||||
@@ -77,6 +70,18 @@
|
|||||||
- fileuri
|
- fileuri
|
||||||
- texturi
|
- texturi
|
||||||
|
|
||||||
|
<h4> SAS Macros </h4>
|
||||||
|
@li mf_nobs.sas
|
||||||
|
@li mf_verifymacvars.sas
|
||||||
|
@li mm_getdirectories.sas
|
||||||
|
@li mm_updatestpsourcecode.sas
|
||||||
|
@li mm_getservercontexts.sas
|
||||||
|
@li mp_abort.sas
|
||||||
|
@li mp_dropmembers.sas
|
||||||
|
|
||||||
|
<h4> Related Macros </h4>
|
||||||
|
@li mm_createwebservice.sas
|
||||||
|
|
||||||
@version 9.2
|
@version 9.2
|
||||||
@author Allan Bowe
|
@author Allan Bowe
|
||||||
|
|
||||||
@@ -105,7 +110,12 @@
|
|||||||
%&mD.put Executing mm_CreateSTP.sas;
|
%&mD.put Executing mm_CreateSTP.sas;
|
||||||
%&mD.put _local_;
|
%&mD.put _local_;
|
||||||
|
|
||||||
%mf_verifymacvars(stpname filename directory tree)
|
%mp_abort(
|
||||||
|
iftrue=(%mf_verifymacvars(stpname filename directory tree)=0)
|
||||||
|
,mac=&sysmacroname
|
||||||
|
,msg=%str(Empty inputs: stpname filename directory tree)
|
||||||
|
)
|
||||||
|
|
||||||
%mp_dropmembers(%scan(&outds,2,.))
|
%mp_dropmembers(%scan(&outds,2,.))
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -174,6 +184,7 @@ run;
|
|||||||
|
|
||||||
data &outds (keep=stpuri prompturi fileuri texturi);
|
data &outds (keep=stpuri prompturi fileuri texturi);
|
||||||
length stpuri prompturi fileuri texturi serveruri $256 ;
|
length stpuri prompturi fileuri texturi serveruri $256 ;
|
||||||
|
if _n_=1 then call missing (of _all_);
|
||||||
set &outds;
|
set &outds;
|
||||||
|
|
||||||
/* final checks on uris */
|
/* final checks on uris */
|
||||||
|
|||||||
@@ -12,6 +12,7 @@ Usage:
|
|||||||
%* parmcards lets us write to a text file from open code ;
|
%* parmcards lets us write to a text file from open code ;
|
||||||
filename ft15f001 temp;
|
filename ft15f001 temp;
|
||||||
parmcards4;
|
parmcards4;
|
||||||
|
%webout(FETCH)
|
||||||
%* do some sas, any inputs are now already WORK tables;
|
%* do some sas, any inputs are now already WORK tables;
|
||||||
data example1 example2;
|
data example1 example2;
|
||||||
set sashelp.class;
|
set sashelp.class;
|
||||||
@@ -24,11 +25,8 @@ Usage:
|
|||||||
;;;;
|
;;;;
|
||||||
%mm_createwebservice(path=/Public/app/common,name=appInit,code=ft15f001)
|
%mm_createwebservice(path=/Public/app/common,name=appInit,code=ft15f001)
|
||||||
|
|
||||||
<h4> SAS Macros </h4>
|
For more examples of using these web services with the SASjs Adapter, see:
|
||||||
@li mm_createstp.sas
|
https://github.com/sasjs/adapter#readme
|
||||||
@li mf_getuser.sas
|
|
||||||
@li mm_createfolder.sas
|
|
||||||
@li mm_deletestp.sas
|
|
||||||
|
|
||||||
@param path= The full path (in SAS Metadata) where the service will be created
|
@param path= The full path (in SAS Metadata) where the service will be created
|
||||||
@param name= Stored Process name. Avoid spaces - testing has shown that
|
@param name= Stored Process name. Avoid spaces - testing has shown that
|
||||||
@@ -37,16 +35,22 @@ Usage:
|
|||||||
@param desc= The description of the service (optional)
|
@param desc= The description of the service (optional)
|
||||||
@param precode= Space separated list of filerefs, pointing to the code that
|
@param precode= Space separated list of filerefs, pointing to the code that
|
||||||
needs to be attached to the beginning of the service (optional)
|
needs to be attached to the beginning of the service (optional)
|
||||||
@param code=(ft15f001) Space seperated fileref(s) of the actual code to be
|
@param code= (ft15f001) Space seperated fileref(s) of the actual code to be
|
||||||
added
|
added
|
||||||
@param server=(SASApp) The server which will run the STP. Server name or uri
|
@param server= (SASApp) The server which will run the STP. Server name or uri
|
||||||
is fine.
|
is fine.
|
||||||
@param mDebug=(0) set to 1 to show debug messages in the log
|
@param mDebug= (0) set to 1 to show debug messages in the log
|
||||||
@param replace=(YES) select NO to avoid replacing an existing service in that
|
@param replace= (YES) select NO to avoid replacing an existing service in that
|
||||||
location
|
location
|
||||||
@param adapter=(sasjs) the macro uses the sasjs adapter by default. To use
|
@param adapter= (sasjs) the macro uses the sasjs adapter by default. To use
|
||||||
another adapter, add a (different) fileref here.
|
another adapter, add a (different) fileref here.
|
||||||
|
|
||||||
|
<h4> SAS Macros </h4>
|
||||||
|
@li mm_createstp.sas
|
||||||
|
@li mf_getuser.sas
|
||||||
|
@li mm_createfolder.sas
|
||||||
|
@li mm_deletestp.sas
|
||||||
|
|
||||||
@version 9.2
|
@version 9.2
|
||||||
@author Allan Bowe
|
@author Allan Bowe
|
||||||
|
|
||||||
@@ -89,129 +93,145 @@ data _null_;
|
|||||||
put "/* Created on %sysfunc(datetime(),datetime19.) by %mf_getuser() */";
|
put "/* Created on %sysfunc(datetime(),datetime19.) by %mf_getuser() */";
|
||||||
/* WEBOUT BEGIN */
|
/* WEBOUT BEGIN */
|
||||||
put ' ';
|
put ' ';
|
||||||
put '%macro mp_jsonout(action,ds,jref=_webout,dslabel=,fmt=Y,engine=DATASTEP,dbg=0 ';
|
put '%macro mp_jsonout(action,ds,jref=_webout,dslabel=,fmt=Y ';
|
||||||
|
put ' ,engine=DATASTEP ';
|
||||||
|
put ' ,missing=NULL ';
|
||||||
|
put ' ,showmeta=NO ';
|
||||||
put ')/*/STORE SOURCE*/; ';
|
put ')/*/STORE SOURCE*/; ';
|
||||||
put '%put output location=&jref; ';
|
put '%local tempds colinfo fmtds i numcols; ';
|
||||||
|
put '%let numcols=0; ';
|
||||||
|
put ' ';
|
||||||
put '%if &action=OPEN %then %do; ';
|
put '%if &action=OPEN %then %do; ';
|
||||||
put ' options nobomfile; ';
|
put ' options nobomfile; ';
|
||||||
put ' data _null_;file &jref encoding=''utf-8''; ';
|
put ' data _null_;file &jref encoding=''utf-8'' ; ';
|
||||||
put ' put ''{"START_DTTM" : "'' "%sysfunc(datetime(),datetime20.3)" ''"''; ';
|
put ' put ''{"PROCESSED_DTTM" : "'' "%sysfunc(datetime(),E8601DT26.6)" ''"''; ';
|
||||||
put ' run; ';
|
put ' run; ';
|
||||||
put '%end; ';
|
put '%end; ';
|
||||||
put '%else %if (&action=ARR or &action=OBJ) %then %do; ';
|
put '%else %if (&action=ARR or &action=OBJ) %then %do; ';
|
||||||
put ' options validvarname=upcase; ';
|
put ' options validvarname=upcase; ';
|
||||||
put ' data _null_;file &jref mod encoding=''utf-8''; ';
|
put ' data _null_; file &jref encoding=''utf-8'' mod; ';
|
||||||
put ' put ", ""%lowcase(%sysfunc(coalescec(&dslabel,&ds)))"":"; ';
|
put ' put ", ""%lowcase(%sysfunc(coalescec(&dslabel,&ds)))"":"; ';
|
||||||
put ' ';
|
put ' ';
|
||||||
|
put ' /* grab col defs */ ';
|
||||||
|
put ' proc contents noprint data=&ds ';
|
||||||
|
put ' out=_data_(keep=name type length format formatl formatd varnum label); ';
|
||||||
|
put ' run; ';
|
||||||
|
put ' %let colinfo=%scan(&syslast,2,.); ';
|
||||||
|
put ' proc sort data=&colinfo; ';
|
||||||
|
put ' by varnum; ';
|
||||||
|
put ' run; ';
|
||||||
|
put ' /* move meta to mac vars */ ';
|
||||||
|
put ' data _null_; ';
|
||||||
|
put ' if _n_=1 then call symputx(''numcols'',nobs,''l''); ';
|
||||||
|
put ' set &colinfo end=last nobs=nobs; ';
|
||||||
|
put ' name=upcase(name); ';
|
||||||
|
put ' /* fix formats */ ';
|
||||||
|
put ' if type=2 or type=6 then do; ';
|
||||||
|
put ' typelong=''char''; ';
|
||||||
|
put ' length fmt $49.; ';
|
||||||
|
put ' if format='''' then fmt=cats(''$'',length,''.''); ';
|
||||||
|
put ' else if formatl=0 then fmt=cats(format,''.''); ';
|
||||||
|
put ' else fmt=cats(format,formatl,''.''); ';
|
||||||
|
put ' newlen=max(formatl,length); ';
|
||||||
|
put ' end; ';
|
||||||
|
put ' else do; ';
|
||||||
|
put ' typelong=''num''; ';
|
||||||
|
put ' if format='''' then fmt=''best.''; ';
|
||||||
|
put ' else if formatl=0 then fmt=cats(format,''.''); ';
|
||||||
|
put ' else if formatd=0 then fmt=cats(format,formatl,''.''); ';
|
||||||
|
put ' else fmt=cats(format,formatl,''.'',formatd); ';
|
||||||
|
put ' /* needs to be wide, for datetimes etc */ ';
|
||||||
|
put ' newlen=max(length,formatl,24); ';
|
||||||
|
put ' end; ';
|
||||||
|
put ' /* 32 char unique name */ ';
|
||||||
|
put ' newname=''sasjs''!!substr(cats(put(md5(name),$hex32.)),1,27); ';
|
||||||
|
put ' ';
|
||||||
|
put ' call symputx(cats(''name'',_n_),name,''l''); ';
|
||||||
|
put ' call symputx(cats(''newname'',_n_),newname,''l''); ';
|
||||||
|
put ' call symputx(cats(''len'',_n_),newlen,''l''); ';
|
||||||
|
put ' call symputx(cats(''length'',_n_),length,''l''); ';
|
||||||
|
put ' call symputx(cats(''fmt'',_n_),fmt,''l''); ';
|
||||||
|
put ' call symputx(cats(''type'',_n_),type,''l''); ';
|
||||||
|
put ' call symputx(cats(''typelong'',_n_),typelong,''l''); ';
|
||||||
|
put ' call symputx(cats(''label'',_n_),coalescec(label,name),''l''); ';
|
||||||
|
put ' run; ';
|
||||||
|
put ' ';
|
||||||
|
put ' %let tempds=%substr(_%sysfunc(compress(%sysfunc(uuidgen()),-)),1,32); ';
|
||||||
|
put ' ';
|
||||||
put ' %if &engine=PROCJSON %then %do; ';
|
put ' %if &engine=PROCJSON %then %do; ';
|
||||||
put ' data;run;%let tempds=&syslast; ';
|
put ' %if &missing=STRING %then %do; ';
|
||||||
put ' proc sql;drop table &tempds; ';
|
put ' %put &sysmacroname: Special Missings not supported in proc json.; ';
|
||||||
put ' data &tempds /view=&tempds;set &ds; ';
|
put ' %put &sysmacroname: Switching to DATASTEP engine; ';
|
||||||
|
put ' %goto datastep; ';
|
||||||
|
put ' %end; ';
|
||||||
|
put ' data &tempds;set &ds; ';
|
||||||
put ' %if &fmt=N %then format _numeric_ best32.;; ';
|
put ' %if &fmt=N %then format _numeric_ best32.;; ';
|
||||||
|
put ' /* PRETTY is necessary to avoid line truncation in large files */ ';
|
||||||
put ' proc json out=&jref pretty ';
|
put ' proc json out=&jref pretty ';
|
||||||
put ' %if &action=ARR %then nokeys ; ';
|
put ' %if &action=ARR %then nokeys ; ';
|
||||||
put ' ;export &tempds / nosastags fmtnumeric; ';
|
put ' ;export &tempds / nosastags fmtnumeric; ';
|
||||||
put ' run; ';
|
put ' run; ';
|
||||||
put ' proc sql;drop view &tempds; ';
|
|
||||||
put ' %end; ';
|
put ' %end; ';
|
||||||
put ' %else %if &engine=DATASTEP %then %do; ';
|
put ' %else %if &engine=DATASTEP %then %do; ';
|
||||||
put ' %local cols i tempds; ';
|
put ' %datastep: ';
|
||||||
put ' %let cols=0; ';
|
put ' %if %sysfunc(exist(&ds)) ne 1 & %sysfunc(exist(&ds,VIEW)) ne 1 ';
|
||||||
put ' %if %sysfunc(exist(&ds)) ne 1 & %sysfunc(exist(&ds,VIEW)) ne 1 %then %do; ';
|
put ' %then %do; ';
|
||||||
put ' %put &sysmacroname: &ds NOT FOUND!!!; ';
|
put ' %put &sysmacroname: &ds NOT FOUND!!!; ';
|
||||||
put ' %return; ';
|
put ' %return; ';
|
||||||
put ' %end; ';
|
put ' %end; ';
|
||||||
put ' %if &fmt=Y %then %do; ';
|
|
||||||
put ' %put converting every variable to a formatted variable; ';
|
|
||||||
put ' /* see mp_ds2fmtds.sas for source */ ';
|
|
||||||
put ' proc contents noprint data=&ds ';
|
|
||||||
put ' out=_data_(keep=name type length format formatl formatd varnum); ';
|
|
||||||
put ' run; ';
|
|
||||||
put ' proc sort; ';
|
|
||||||
put ' by varnum; ';
|
|
||||||
put ' run; ';
|
|
||||||
put ' %local fmtds; ';
|
|
||||||
put ' %let fmtds=%scan(&syslast,2,.); ';
|
|
||||||
put ' /* prepare formats and varnames */ ';
|
|
||||||
put ' data _null_; ';
|
|
||||||
put ' if _n_=1 then call symputx(''nobs'',nobs,''l''); ';
|
|
||||||
put ' set &fmtds end=last nobs=nobs; ';
|
|
||||||
put ' name=upcase(name); ';
|
|
||||||
put ' /* fix formats */ ';
|
|
||||||
put ' if type=2 or type=6 then do; ';
|
|
||||||
put ' length fmt $49.; ';
|
|
||||||
put ' if format='''' then fmt=cats(''$'',length,''.''); ';
|
|
||||||
put ' else if formatl=0 then fmt=cats(format,''.''); ';
|
|
||||||
put ' else fmt=cats(format,formatl,''.''); ';
|
|
||||||
put ' newlen=max(formatl,length); ';
|
|
||||||
put ' end; ';
|
|
||||||
put ' else do; ';
|
|
||||||
put ' if format='''' then fmt=''best.''; ';
|
|
||||||
put ' else if formatl=0 then fmt=cats(format,''.''); ';
|
|
||||||
put ' else if formatd=0 then fmt=cats(format,formatl,''.''); ';
|
|
||||||
put ' else fmt=cats(format,formatl,''.'',formatd); ';
|
|
||||||
put ' /* needs to be wide, for datetimes etc */ ';
|
|
||||||
put ' newlen=max(length,formatl,24); ';
|
|
||||||
put ' end; ';
|
|
||||||
put ' /* 32 char unique name */ ';
|
|
||||||
put ' newname=''sasjs''!!substr(cats(put(md5(name),$hex32.)),1,27); ';
|
|
||||||
put ' ';
|
put ' ';
|
||||||
put ' call symputx(cats(''name'',_n_),name,''l''); ';
|
put ' %if &fmt=Y %then %do; ';
|
||||||
put ' call symputx(cats(''newname'',_n_),newname,''l''); ';
|
put ' data _data_; ';
|
||||||
put ' call symputx(cats(''len'',_n_),newlen,''l''); ';
|
|
||||||
put ' call symputx(cats(''fmt'',_n_),fmt,''l''); ';
|
|
||||||
put ' call symputx(cats(''type'',_n_),type,''l''); ';
|
|
||||||
put ' run; ';
|
|
||||||
put ' data &fmtds; ';
|
|
||||||
put ' /* rename on entry */ ';
|
put ' /* rename on entry */ ';
|
||||||
put ' set &ds(rename=( ';
|
put ' set &ds(rename=( ';
|
||||||
put ' %local i; ';
|
put ' %do i=1 %to &numcols; ';
|
||||||
put ' %do i=1 %to &nobs; ';
|
|
||||||
put ' &&name&i=&&newname&i ';
|
put ' &&name&i=&&newname&i ';
|
||||||
put ' %end; ';
|
put ' %end; ';
|
||||||
put ' )); ';
|
put ' )); ';
|
||||||
put ' %do i=1 %to &nobs; ';
|
put ' %do i=1 %to &numcols; ';
|
||||||
put ' length &&name&i $&&len&i; ';
|
put ' length &&name&i $&&len&i; ';
|
||||||
put ' &&name&i=left(put(&&newname&i,&&fmt&i)); ';
|
put ' %if &&typelong&i=num %then %do; ';
|
||||||
|
put ' &&name&i=left(put(&&newname&i,&&fmt&i)); ';
|
||||||
|
put ' %end; ';
|
||||||
|
put ' %else %do; ';
|
||||||
|
put ' &&name&i=put(&&newname&i,&&fmt&i); ';
|
||||||
|
put ' %end; ';
|
||||||
put ' drop &&newname&i; ';
|
put ' drop &&newname&i; ';
|
||||||
put ' %end; ';
|
put ' %end; ';
|
||||||
put ' if _error_ then call symputx(''syscc'',1012); ';
|
put ' if _error_ then call symputx(''syscc'',1012); ';
|
||||||
put ' run; ';
|
put ' run; ';
|
||||||
put ' %let ds=&fmtds; ';
|
put ' %let fmtds=&syslast; ';
|
||||||
put ' %end; /* &fmt=Y */ ';
|
put ' %end; ';
|
||||||
put ' data _null_;file &jref mod encoding=''utf-8''; ';
|
|
||||||
put ' put "["; call symputx(''cols'',0,''l''); ';
|
|
||||||
put ' proc sort ';
|
|
||||||
put ' data=sashelp.vcolumn(where=(libname=''WORK'' & memname="%upcase(&ds)")) ';
|
|
||||||
put ' out=_data_; ';
|
|
||||||
put ' by varnum; ';
|
|
||||||
put ' ';
|
|
||||||
put ' data _null_; ';
|
|
||||||
put ' set _last_ end=last; ';
|
|
||||||
put ' call symputx(cats(''name'',_n_),name,''l''); ';
|
|
||||||
put ' call symputx(cats(''type'',_n_),type,''l''); ';
|
|
||||||
put ' call symputx(cats(''len'',_n_),length,''l''); ';
|
|
||||||
put ' if last then call symputx(''cols'',_n_,''l''); ';
|
|
||||||
put ' run; ';
|
|
||||||
put ' ';
|
put ' ';
|
||||||
put ' proc format; /* credit yabwon for special null removal */ ';
|
put ' proc format; /* credit yabwon for special null removal */ ';
|
||||||
put ' value bart ._ - .z = null ';
|
put ' value bart (default=40) ';
|
||||||
|
put ' %if &missing=NULL %then %do; ';
|
||||||
|
put ' ._ - .z = null ';
|
||||||
|
put ' %end; ';
|
||||||
|
put ' %else %do; ';
|
||||||
|
put ' ._ = [quote()] ';
|
||||||
|
put ' . = null ';
|
||||||
|
put ' .a - .z = [quote()] ';
|
||||||
|
put ' %end; ';
|
||||||
put ' other = [best.]; ';
|
put ' other = [best.]; ';
|
||||||
put ' ';
|
put ' ';
|
||||||
put ' data;run; %let tempds=&syslast; /* temp table for spesh char management */ ';
|
put ' data &tempds; ';
|
||||||
put ' proc sql; drop table &tempds; ';
|
|
||||||
put ' data &tempds/view=&tempds; ';
|
|
||||||
put ' attrib _all_ label=''''; ';
|
put ' attrib _all_ label=''''; ';
|
||||||
put ' %do i=1 %to &cols; ';
|
put ' %do i=1 %to &numcols; ';
|
||||||
put ' %if &&type&i=char %then %do; ';
|
put ' %if &&typelong&i=char or &fmt=Y %then %do; ';
|
||||||
put ' length &&name&i $32767; ';
|
put ' length &&name&i $32767; ';
|
||||||
put ' format &&name&i $32767.; ';
|
put ' format &&name&i $32767.; ';
|
||||||
put ' %end; ';
|
put ' %end; ';
|
||||||
put ' %end; ';
|
put ' %end; ';
|
||||||
put ' set &ds; ';
|
put ' %if &fmt=Y %then %do; ';
|
||||||
|
put ' set &fmtds; ';
|
||||||
|
put ' %end; ';
|
||||||
|
put ' %else %do; ';
|
||||||
|
put ' set &ds; ';
|
||||||
|
put ' %end; ';
|
||||||
put ' format _numeric_ bart.; ';
|
put ' format _numeric_ bart.; ';
|
||||||
put ' %do i=1 %to &cols; ';
|
put ' %do i=1 %to &numcols; ';
|
||||||
put ' %if &&type&i=char %then %do; ';
|
put ' %if &&typelong&i=char or &fmt=Y %then %do; ';
|
||||||
put ' &&name&i=''"''!!trim(prxchange(''s/"/\"/'',-1, ';
|
put ' &&name&i=''"''!!trim(prxchange(''s/"/\"/'',-1, ';
|
||||||
put ' prxchange(''s/''!!''0A''x!!''/\n/'',-1, ';
|
put ' prxchange(''s/''!!''0A''x!!''/\n/'',-1, ';
|
||||||
put ' prxchange(''s/''!!''0D''x!!''/\r/'',-1, ';
|
put ' prxchange(''s/''!!''0D''x!!''/\r/'',-1, ';
|
||||||
@@ -221,49 +241,71 @@ data _null_;
|
|||||||
put ' %end; ';
|
put ' %end; ';
|
||||||
put ' %end; ';
|
put ' %end; ';
|
||||||
put ' run; ';
|
put ' run; ';
|
||||||
|
put ' ';
|
||||||
put ' /* write to temp loc to avoid _webout truncation ';
|
put ' /* write to temp loc to avoid _webout truncation ';
|
||||||
put ' - https://support.sas.com/kb/49/325.html */ ';
|
put ' - https://support.sas.com/kb/49/325.html */ ';
|
||||||
put ' filename _sjs temp lrecl=131068 encoding=''utf-8''; ';
|
put ' filename _sjs temp lrecl=131068 encoding=''utf-8''; ';
|
||||||
put ' data _null_; file _sjs lrecl=131068 encoding=''utf-8'' mod; ';
|
put ' data _null_; file _sjs lrecl=131068 encoding=''utf-8'' mod ; ';
|
||||||
|
put ' if _n_=1 then put "["; ';
|
||||||
put ' set &tempds; ';
|
put ' set &tempds; ';
|
||||||
put ' if _n_>1 then put "," @; put ';
|
put ' if _n_>1 then put "," @; put ';
|
||||||
put ' %if &action=ARR %then "[" ; %else "{" ; ';
|
put ' %if &action=ARR %then "[" ; %else "{" ; ';
|
||||||
put ' %do i=1 %to &cols; ';
|
put ' %do i=1 %to &numcols; ';
|
||||||
put ' %if &i>1 %then "," ; ';
|
put ' %if &i>1 %then "," ; ';
|
||||||
put ' %if &action=OBJ %then """&&name&i"":" ; ';
|
put ' %if &action=OBJ %then """&&name&i"":" ; ';
|
||||||
put ' &&name&i ';
|
put ' &&name&i ';
|
||||||
put ' %end; ';
|
put ' %end; ';
|
||||||
put ' %if &action=ARR %then "]" ; %else "}" ; ; ';
|
put ' %if &action=ARR %then "]" ; %else "}" ; ; ';
|
||||||
put ' proc sql; ';
|
|
||||||
put ' drop view &tempds; ';
|
|
||||||
put ' /* now write the long strings to _webout 1 byte at a time */ ';
|
put ' /* now write the long strings to _webout 1 byte at a time */ ';
|
||||||
put ' data _null_; ';
|
put ' data _null_; ';
|
||||||
put ' length filein 8 fileid 8; ';
|
put ' length filein 8 fileid 8; ';
|
||||||
put ' filein = fopen("_sjs",''I'',1,''B''); ';
|
put ' filein=fopen("_sjs",''I'',1,''B''); ';
|
||||||
put ' fileid = fopen("&jref",''A'',1,''B''); ';
|
put ' fileid=fopen("&jref",''A'',1,''B''); ';
|
||||||
put ' rec = ''20''x; ';
|
put ' rec=''20''x; ';
|
||||||
put ' do while(fread(filein)=0); ';
|
put ' do while(fread(filein)=0); ';
|
||||||
put ' rc = fget(filein,rec,1); ';
|
put ' rc=fget(filein,rec,1); ';
|
||||||
put ' rc = fput(fileid, rec); ';
|
put ' rc=fput(fileid, rec); ';
|
||||||
put ' rc =fwrite(fileid); ';
|
put ' rc=fwrite(fileid); ';
|
||||||
put ' end; ';
|
put ' end; ';
|
||||||
put ' rc = fclose(filein); ';
|
put ' /* close out the table */ ';
|
||||||
put ' rc = fclose(fileid); ';
|
put ' rc=fput(fileid, "]"); ';
|
||||||
|
put ' rc=fwrite(fileid); ';
|
||||||
|
put ' rc=fclose(filein); ';
|
||||||
|
put ' rc=fclose(fileid); ';
|
||||||
put ' run; ';
|
put ' run; ';
|
||||||
put ' filename _sjs clear; ';
|
put ' filename _sjs clear; ';
|
||||||
put ' data _null_; file &jref mod encoding=''utf-8''; ';
|
put ' %end; ';
|
||||||
put ' put "]"; ';
|
put ' ';
|
||||||
|
put ' proc sql; ';
|
||||||
|
put ' drop table &colinfo, &tempds; ';
|
||||||
|
put ' ';
|
||||||
|
put ' %if &showmeta=YES %then %do; ';
|
||||||
|
put ' data _null_; file &jref encoding=''utf-8'' mod; ';
|
||||||
|
put ' put ", ""$%lowcase(%sysfunc(coalescec(&dslabel,&ds)))"":{""vars"":{"; ';
|
||||||
|
put ' do i=1 to &numcols; ';
|
||||||
|
put ' name=quote(trim(symget(cats(''name'',i)))); ';
|
||||||
|
put ' format=quote(trim(symget(cats(''fmt'',i)))); ';
|
||||||
|
put ' label=quote(trim(symget(cats(''label'',i)))); ';
|
||||||
|
put ' length=quote(trim(symget(cats(''length'',i)))); ';
|
||||||
|
put ' type=quote(trim(symget(cats(''typelong'',i)))); ';
|
||||||
|
put ' if i>1 then put "," @@; ';
|
||||||
|
put ' put name '':{"format":'' format '',"label":'' label ';
|
||||||
|
put ' '',"length":'' length '',"type":'' type ''}''; ';
|
||||||
|
put ' end; ';
|
||||||
|
put ' put ''}}''; ';
|
||||||
put ' run; ';
|
put ' run; ';
|
||||||
put ' %end; ';
|
put ' %end; ';
|
||||||
put '%end; ';
|
put '%end; ';
|
||||||
put ' ';
|
put ' ';
|
||||||
put '%else %if &action=CLOSE %then %do; ';
|
put '%else %if &action=CLOSE %then %do; ';
|
||||||
put ' data _null_;file &jref encoding=''utf-8'' mod; ';
|
put ' data _null_; file &jref encoding=''utf-8'' mod ; ';
|
||||||
put ' put "}"; ';
|
put ' put "}"; ';
|
||||||
put ' run; ';
|
put ' run; ';
|
||||||
put '%end; ';
|
put '%end; ';
|
||||||
put '%mend mp_jsonout; ';
|
put '%mend mp_jsonout; ';
|
||||||
put '%macro mm_webout(action,ds,dslabel=,fref=_webout,fmt=Y); ';
|
put '%macro mm_webout(action,ds,dslabel=,fref=_webout,fmt=Y,missing=NULL ';
|
||||||
|
put ' ,showmeta=NO ';
|
||||||
|
put '); ';
|
||||||
put '%global _webin_file_count _webin_fileref1 _webin_name1 _program _debug ';
|
put '%global _webin_file_count _webin_fileref1 _webin_name1 _program _debug ';
|
||||||
put ' sasjs_tables; ';
|
put ' sasjs_tables; ';
|
||||||
put '%local i tempds jsonengine; ';
|
put '%local i tempds jsonengine; ';
|
||||||
@@ -321,14 +363,15 @@ data _null_;
|
|||||||
put ' %if %str(&_debug) ge 131 %then %do; ';
|
put ' %if %str(&_debug) ge 131 %then %do; ';
|
||||||
put ' put ''>>weboutBEGIN<<''; ';
|
put ' put ''>>weboutBEGIN<<''; ';
|
||||||
put ' %end; ';
|
put ' %end; ';
|
||||||
put ' put ''{"START_DTTM" : "'' "%sysfunc(datetime(),datetime20.3)" ''"''; ';
|
put ' put ''{"SYSDATE" : "'' "&SYSDATE" ''"''; ';
|
||||||
|
put ' put '',"SYSTIME" : "'' "&SYSTIME" ''"''; ';
|
||||||
put ' run; ';
|
put ' run; ';
|
||||||
put ' ';
|
put ' ';
|
||||||
put '%end; ';
|
put '%end; ';
|
||||||
put ' ';
|
put ' ';
|
||||||
put '%else %if &action=ARR or &action=OBJ %then %do; ';
|
put '%else %if &action=ARR or &action=OBJ %then %do; ';
|
||||||
put ' %mp_jsonout(&action,&ds,dslabel=&dslabel,fmt=&fmt,jref=&fref ';
|
put ' %mp_jsonout(&action,&ds,dslabel=&dslabel,fmt=&fmt,jref=&fref ';
|
||||||
put ' ,engine=&jsonengine,dbg=%str(&_debug) ';
|
put ' ,engine=&jsonengine,missing=&missing,showmeta=&showmeta ';
|
||||||
put ' ) ';
|
put ' ) ';
|
||||||
put '%end; ';
|
put '%end; ';
|
||||||
put '%else %if &action=CLOSE %then %do; ';
|
put '%else %if &action=CLOSE %then %do; ';
|
||||||
@@ -343,15 +386,12 @@ data _null_;
|
|||||||
put ' set &tempds; ';
|
put ' set &tempds; ';
|
||||||
put ' if not (upcase(name) =:"DATA"); /* ignore temp datasets */ ';
|
put ' if not (upcase(name) =:"DATA"); /* ignore temp datasets */ ';
|
||||||
put ' i+1; ';
|
put ' i+1; ';
|
||||||
put ' call symputx(''wt''!!left(i),name,''l''); ';
|
put ' call symputx(cats(''wt'',i),name,''l''); ';
|
||||||
put ' call symputx(''wtcnt'',i,''l''); ';
|
put ' call symputx(''wtcnt'',i,''l''); ';
|
||||||
put ' data _null_; file &fref mod encoding=''utf-8''; ';
|
put ' data _null_; file &fref mod encoding=''utf-8''; ';
|
||||||
put ' put ",""WORK"":{"; ';
|
put ' put ",""WORK"":{"; ';
|
||||||
put ' %do i=1 %to &wtcnt; ';
|
put ' %do i=1 %to &wtcnt; ';
|
||||||
put ' %let wt=&&wt&i; ';
|
put ' %let wt=&&wt&i; ';
|
||||||
put ' proc contents noprint data=&wt ';
|
|
||||||
put ' out=_data_ (keep=name type length format:); ';
|
|
||||||
put ' run;%let tempds=%scan(&syslast,2,.); ';
|
|
||||||
put ' data _null_; file &fref mod encoding=''utf-8''; ';
|
put ' data _null_; file &fref mod encoding=''utf-8''; ';
|
||||||
put ' dsid=open("WORK.&wt",''is''); ';
|
put ' dsid=open("WORK.&wt",''is''); ';
|
||||||
put ' nlobs=attrn(dsid,''NLOBS''); ';
|
put ' nlobs=attrn(dsid,''NLOBS''); ';
|
||||||
@@ -361,8 +401,7 @@ data _null_;
|
|||||||
put ' put " ""&wt"" : {"; ';
|
put ' put " ""&wt"" : {"; ';
|
||||||
put ' put ''"nlobs":'' nlobs; ';
|
put ' put ''"nlobs":'' nlobs; ';
|
||||||
put ' put '',"nvars":'' nvars; ';
|
put ' put '',"nvars":'' nvars; ';
|
||||||
put ' %mp_jsonout(OBJ,&tempds,jref=&fref,dslabel=colattrs,engine=&jsonengine) ';
|
put ' %mp_jsonout(OBJ,&wt,jref=&fref,dslabel=first10rows,showmeta=YES) ';
|
||||||
put ' %mp_jsonout(OBJ,&wt,jref=&fref,dslabel=first10rows,engine=&jsonengine) ';
|
|
||||||
put ' data _null_; file &fref mod encoding=''utf-8''; ';
|
put ' data _null_; file &fref mod encoding=''utf-8''; ';
|
||||||
put ' put "}"; ';
|
put ' put "}"; ';
|
||||||
put ' %end; ';
|
put ' %end; ';
|
||||||
@@ -390,7 +429,11 @@ data _null_;
|
|||||||
put ' sysvlong=quote(trim(symget(''sysvlong''))); ';
|
put ' sysvlong=quote(trim(symget(''sysvlong''))); ';
|
||||||
put ' put '',"SYSVLONG" : '' sysvlong; ';
|
put ' put '',"SYSVLONG" : '' sysvlong; ';
|
||||||
put ' put ",""SYSWARNINGTEXT"" : ""&syswarningtext"" "; ';
|
put ' put ",""SYSWARNINGTEXT"" : ""&syswarningtext"" "; ';
|
||||||
put ' put '',"END_DTTM" : "'' "%sysfunc(datetime(),datetime20.3)" ''" ''; ';
|
put ' put '',"END_DTTM" : "'' "%sysfunc(datetime(),E8601DT26.6)" ''" ''; ';
|
||||||
|
put ' length memsize $32; ';
|
||||||
|
put ' memsize="%sysfunc(INPUTN(%sysfunc(getoption(memsize)), best.),sizekmg.)"; ';
|
||||||
|
put ' memsize=quote(cats(memsize)); ';
|
||||||
|
put ' put '',"MEMSIZE" : '' memsize; ';
|
||||||
put ' put "}" @; ';
|
put ' put "}" @; ';
|
||||||
put ' %if %str(&_debug) ge 131 %then %do; ';
|
put ' %if %str(&_debug) ge 131 %then %do; ';
|
||||||
put ' put ''>>weboutEND<<''; ';
|
put ' put ''>>weboutEND<<''; ';
|
||||||
@@ -418,8 +461,10 @@ data _null_;
|
|||||||
put ' ';
|
put ' ';
|
||||||
put '%mend mf_getuser; ';
|
put '%mend mf_getuser; ';
|
||||||
/* WEBOUT END */
|
/* WEBOUT END */
|
||||||
put '%macro webout(action,ds,dslabel=,fmt=);';
|
put '%macro webout(action,ds,dslabel=,fmt=,missing=NULL,showmeta=NO);';
|
||||||
put ' %mm_webout(&action,ds=&ds,dslabel=&dslabel,fmt=&fmt)';
|
put ' %mm_webout(&action,ds=&ds,dslabel=&dslabel,fmt=&fmt,missing=&missing';
|
||||||
|
put ' ,showmeta=&showmeta';
|
||||||
|
put ' )';
|
||||||
put '%mend;';
|
put '%mend;';
|
||||||
run;
|
run;
|
||||||
|
|
||||||
|
|||||||
@@ -1,16 +1,21 @@
|
|||||||
/**
|
/**
|
||||||
@file mm_getauthinfo.sas
|
@file mm_getauthinfo.sas
|
||||||
@brief extracts authentication info
|
@brief Extracts authentication info for each user in metadata
|
||||||
@details usage:
|
@details
|
||||||
|
Usage:
|
||||||
|
|
||||||
%mm_getauthinfo(outds=auths)
|
%mm_getauthinfo(outds=auths)
|
||||||
|
|
||||||
@param outds= the ONE LEVEL work dataset to create
|
|
||||||
|
@param [in] mdebug= (0) Set to 1 to enable DEBUG messages and preserve outputs
|
||||||
|
@param [out] outds= (mm_getauthinfo) The output dataset to create
|
||||||
|
|
||||||
<h4> SAS Macros </h4>
|
<h4> SAS Macros </h4>
|
||||||
@li mm_getobjects.sas
|
|
||||||
@li mf_getuniquefileref.sas
|
@li mf_getuniquefileref.sas
|
||||||
|
@li mf_getuniquename.sas
|
||||||
@li mm_getdetails.sas
|
@li mm_getdetails.sas
|
||||||
|
@li mm_getobjects.sas
|
||||||
|
|
||||||
|
|
||||||
@version 9.4
|
@version 9.4
|
||||||
@author Allan Bowe
|
@author Allan Bowe
|
||||||
@@ -18,67 +23,69 @@
|
|||||||
**/
|
**/
|
||||||
|
|
||||||
%macro mm_getauthinfo(outds=mm_getauthinfo
|
%macro mm_getauthinfo(outds=mm_getauthinfo
|
||||||
|
,mdebug=0
|
||||||
)/*/STORE SOURCE*/;
|
)/*/STORE SOURCE*/;
|
||||||
|
%local prefix fileref;
|
||||||
|
%let prefix=%substr(%mf_getuniquename(),1,25);
|
||||||
|
|
||||||
%if %length(&outds)>30 %then %do;
|
%mm_getobjects(type=Login,outds=&prefix.0)
|
||||||
%put %str(ERR)OR: Temp tables are created with the &outds prefix, which
|
|
||||||
therefore needs to be 30 characters or less;
|
|
||||||
%return;
|
|
||||||
%end;
|
|
||||||
%if %index(&outds,'.')>0 %then %do;
|
|
||||||
%put %str(ERR)OR: Table &outds should be ONE LEVEL (no library);
|
|
||||||
%return;
|
|
||||||
%end;
|
|
||||||
|
|
||||||
%mm_getobjects(type=Login,outds=&outds.0)
|
|
||||||
|
|
||||||
%local fileref;
|
%local fileref;
|
||||||
%let fileref=%mf_getuniquefileref();
|
%let fileref=%mf_getuniquefileref();
|
||||||
|
|
||||||
data _null_;
|
data _null_;
|
||||||
file &fileref;
|
file &fileref;
|
||||||
set &outds.0 end=last;
|
set &prefix.0 end=last;
|
||||||
/* run macro */
|
/* run macro */
|
||||||
str=cats('%mm_getdetails(uri=',id,",outattrs=&outds.d",_n_
|
str=cats('%mm_getdetails(uri=',id,",outattrs=&prefix.d",_n_
|
||||||
,",outassocs=&outds.a",_n_,")");
|
,",outassocs=&prefix.a",_n_,")");
|
||||||
put str;
|
put str;
|
||||||
/* transpose attributes */
|
/* transpose attributes */
|
||||||
str=cats("proc transpose data=&outds.d",_n_,"(drop=type) out=&outds.da"
|
str=cats("proc transpose data=&prefix.d",_n_,"(drop=type) out=&prefix.da"
|
||||||
,_n_,"(drop=_name_);var value;id name;run;");
|
,_n_,"(drop=_name_);var value;id name;run;");
|
||||||
put str;
|
put str;
|
||||||
/* add extra info to attributes */
|
/* add extra info to attributes */
|
||||||
str=cats("data &outds.da",_n_,";length login_id login_name $256; login_id="
|
str=cats("data &prefix.da",_n_,";length login_id login_name $256; login_id="
|
||||||
,quote(trim(id)),";set &outds.da",_n_
|
,quote(trim(id)),";set &prefix.da",_n_
|
||||||
,";login_name=trim(subpad(name,1,256));drop name;run;");
|
,";login_name=trim(subpad(name,1,256));drop name;run;");
|
||||||
put str;
|
put str;
|
||||||
/* add extra info to associations */
|
/* add extra info to associations */
|
||||||
str=cats("data &outds.a",_n_,";length login_id login_name $256; login_id="
|
str=cats("data &prefix.a",_n_,";length login_id login_name $256; login_id="
|
||||||
,quote(trim(id)),";login_name=",quote(trim(name))
|
,quote(trim(id)),";login_name=",quote(trim(name))
|
||||||
,";set &outds.a",_n_,";run;");
|
,";set &prefix.a",_n_,";run;");
|
||||||
put str;
|
put str;
|
||||||
if last then do;
|
if last then do;
|
||||||
/* collate attributes */
|
/* collate attributes */
|
||||||
str=cats("data &outds._logat; set &outds.da1-&outds.da",_n_,";run;");
|
str=cats("data &prefix._logat; set &prefix.da1-&prefix.da",_n_,";run;");
|
||||||
put str;
|
put str;
|
||||||
/* collate associations */
|
/* collate associations */
|
||||||
str=cats("data &outds._logas; set &outds.a1-&outds.a",_n_,";run;");
|
str=cats("data &prefix._logas; set &prefix.a1-&prefix.a",_n_,";run;");
|
||||||
put str;
|
put str;
|
||||||
/* tidy up */
|
/* tidy up */
|
||||||
str=cats("proc delete data=&outds.da1-&outds.da",_n_,";run;");
|
str=cats("proc delete data=&prefix.da1-&prefix.da",_n_,";run;");
|
||||||
put str;
|
put str;
|
||||||
str=cats("proc delete data=&outds.d1-&outds.d",_n_,";run;");
|
str=cats("proc delete data=&prefix.d1-&prefix.d",_n_,";run;");
|
||||||
put str;
|
put str;
|
||||||
str=cats("proc delete data=&outds.a1-&outds.a",_n_,";run;");
|
str=cats("proc delete data=&prefix.a1-&prefix.a",_n_,";run;");
|
||||||
put str;
|
put str;
|
||||||
end;
|
end;
|
||||||
run;
|
run;
|
||||||
|
|
||||||
|
%if &mdebug=1 %then %do;
|
||||||
|
data _null_;
|
||||||
|
infile &fileref;
|
||||||
|
if _n_=1 then putlog // "Now executing the following code:" //;
|
||||||
|
input; putlog _infile_;
|
||||||
|
run;
|
||||||
|
%end;
|
||||||
%inc &fileref;
|
%inc &fileref;
|
||||||
|
filename &fileref clear;
|
||||||
|
|
||||||
/* get libraries */
|
/* get libraries */
|
||||||
proc sort data=&outds._logas(where=(assoc='Libraries')) out=&outds._temp;
|
proc sort data=&prefix._logas(where=(assoc='Libraries')) out=&prefix._temp;
|
||||||
by login_id;
|
by login_id;
|
||||||
data &outds._temp;
|
data &prefix._temp;
|
||||||
set &outds._temp;
|
set &prefix._temp;
|
||||||
by login_id;
|
by login_id;
|
||||||
length library_list $32767;
|
length library_list $32767;
|
||||||
retain library_list;
|
retain library_list;
|
||||||
@@ -86,31 +93,27 @@ data &outds._temp;
|
|||||||
else library_list=catx(' !! ',library_list,name);
|
else library_list=catx(' !! ',library_list,name);
|
||||||
proc sql;
|
proc sql;
|
||||||
/* get auth domain */
|
/* get auth domain */
|
||||||
create table &outds._dom as
|
create table &prefix._dom as
|
||||||
select login_id,name as domain
|
select login_id,name as domain
|
||||||
from &outds._logas
|
from &prefix._logas
|
||||||
where assoc='Domain';
|
where assoc='Domain';
|
||||||
create unique index login_id on &outds._dom(login_id);
|
create unique index login_id on &prefix._dom(login_id);
|
||||||
/* join it all together */
|
/* join it all together */
|
||||||
create table &outds._logins as
|
create table &outds as
|
||||||
select a.*
|
select a.*
|
||||||
,c.domain
|
,c.domain
|
||||||
,b.library_list
|
,b.library_list
|
||||||
from &outds._logat (drop=ishidden lockedby usageversion publictype) a
|
from &prefix._logat (drop=ishidden lockedby usageversion publictype) a
|
||||||
left join &outds._temp b
|
left join &prefix._temp b
|
||||||
on a.login_id=b.login_id
|
on a.login_id=b.login_id
|
||||||
left join &outds._dom c
|
left join &prefix._dom c
|
||||||
on a.login_id=c.login_id;
|
on a.login_id=c.login_id;
|
||||||
drop table &outds._temp;
|
|
||||||
drop table &outds._logat;
|
|
||||||
drop table &outds._logas;
|
|
||||||
|
|
||||||
data _null_;
|
%if &mdebug=0 %then %do;
|
||||||
infile &fileref;
|
proc datasets lib=work;
|
||||||
if _n_=1 then putlog // "Now executing the following code:" //;
|
delete &prefix:;
|
||||||
input; putlog _infile_;
|
run;
|
||||||
run;
|
%end;
|
||||||
|
|
||||||
filename &fileref clear;
|
|
||||||
|
|
||||||
%mend mm_getauthinfo;
|
%mend mm_getauthinfo;
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user