mirror of
https://github.com/sasjs/core.git
synced 2026-04-09 18:13:14 +00:00
Compare commits
89 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
2b74caede4 | ||
|
|
b5a76600d6 | ||
|
|
13113cacaf | ||
|
|
ae7f93aa4e | ||
|
|
e3b8ee69a9 | ||
|
|
78287ed5d3 | ||
|
|
5944619488 | ||
|
|
df0c9899cf | ||
|
|
737eb65251 | ||
|
|
c50555a6e2 | ||
|
|
c69639a228 | ||
|
|
9930b84785 | ||
|
|
7686b7fb99 | ||
|
|
def0514731 | ||
|
|
502fafa53d | ||
|
|
6721e73ecd | ||
|
|
9e36e82ff2 | ||
|
|
87ce565321 | ||
|
|
3bb902b74e | ||
|
|
dd5e4edc80 | ||
|
|
835369381c | ||
|
|
c32819df9f | ||
|
|
c1f1fcdebf | ||
|
|
641966eed8 | ||
|
|
16922c525c | ||
|
|
f315f803db | ||
|
|
bae5431d24 | ||
|
|
76728cbc6c | ||
|
|
a221a706b4 | ||
|
|
f3b712ecee | ||
|
|
db15c66e68 | ||
|
|
62796ab6e6 | ||
|
|
7eca3b5e07 | ||
|
|
66ceb738c8 | ||
|
|
9d37856fc2 | ||
|
|
14987e3914 | ||
|
|
10857b2153 | ||
|
|
ac2a054c84 | ||
|
|
f60b298bbb | ||
|
|
bf6beded5f | ||
|
|
f98d401bcd | ||
|
|
b808c69e93 | ||
|
|
7c2b7dca1f | ||
|
|
cb94a94a21 | ||
|
|
c23262198b | ||
|
|
f2b0988b42 | ||
|
|
b3178a87ee | ||
|
|
cc57907c0c | ||
|
|
7b24faaa21 | ||
|
|
a3c0ba92cc | ||
|
|
c15e7db1c6 | ||
|
|
3faf4cf325 | ||
|
|
b49ac96766 | ||
|
|
b3298143c7 | ||
|
|
366b6e7fa4 | ||
|
|
9bf2870357 | ||
|
|
f71681c352 | ||
|
|
6008db999c | ||
|
|
b6f020e897 | ||
|
|
9d0533fe3b | ||
|
|
7dd2597041 | ||
|
|
426c0bf9f2 | ||
|
|
1cd8ba03c5 | ||
|
|
569533b218 | ||
|
|
14aeb585ae | ||
|
|
7dd219e9f1 | ||
|
|
cdd2b88b09 | ||
|
|
7e4fb4a640 | ||
|
|
a428b4f66c | ||
|
|
e2f0577e78 | ||
|
|
d53eff7771 | ||
|
|
5b56c85455 | ||
|
|
ff519c7f39 | ||
|
|
7d7778fd36 | ||
|
|
b47f31cfe6 | ||
|
|
542039b425 | ||
|
|
cc908a82bc | ||
|
|
71c31046f4 | ||
|
|
33a487b2b4 | ||
|
|
7240cf08d6 | ||
|
|
1cb702149c | ||
|
|
a12ea6a7cb | ||
|
|
a6b52b5d9e | ||
|
|
0faba3581b | ||
|
|
749309b749 | ||
|
|
e54de44d4b | ||
|
|
40436be14f | ||
|
|
909fef7143 | ||
|
|
bcb93e62d4 |
4
.github/workflows/main.yml
vendored
4
.github/workflows/main.yml
vendored
@@ -13,7 +13,7 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v6
|
||||||
|
|
||||||
- name: Install dependencies
|
- name: Install dependencies
|
||||||
run: |
|
run: |
|
||||||
@@ -54,7 +54,7 @@ jobs:
|
|||||||
echo "REFRESH_TOKEN=${{secrets.SAS9_4GL_IO_REFRESH_TOKEN}}" >> .env.server
|
echo "REFRESH_TOKEN=${{secrets.SAS9_4GL_IO_REFRESH_TOKEN}}" >> .env.server
|
||||||
|
|
||||||
- name: Semantic Release
|
- name: Semantic Release
|
||||||
uses: cycjimmy/semantic-release-action@v4
|
uses: cycjimmy/semantic-release-action@v6
|
||||||
env:
|
env:
|
||||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
NPM_TOKEN: ${{ secrets.NPM_TOKEN }}
|
NPM_TOKEN: ${{ secrets.NPM_TOKEN }}
|
||||||
|
|||||||
2
.github/workflows/notmain.yml
vendored
2
.github/workflows/notmain.yml
vendored
@@ -13,7 +13,7 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v6
|
||||||
|
|
||||||
- name: Install dependencies
|
- name: Install dependencies
|
||||||
run: |
|
run: |
|
||||||
|
|||||||
7
.github/workflows/run-tests.yml
vendored
7
.github/workflows/run-tests.yml
vendored
@@ -12,10 +12,10 @@ jobs:
|
|||||||
|
|
||||||
strategy:
|
strategy:
|
||||||
matrix:
|
matrix:
|
||||||
node-version: [lts/hydrogen]
|
node-version: [lts/iron]
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v2
|
- uses: actions/checkout@v6
|
||||||
- name: Use Node.js ${{ matrix.node-version }}
|
- name: Use Node.js ${{ matrix.node-version }}
|
||||||
uses: actions/setup-node@v2
|
uses: actions/setup-node@v2
|
||||||
with:
|
with:
|
||||||
@@ -73,6 +73,8 @@ jobs:
|
|||||||
|
|
||||||
- name: Build & Deploy Project to SAS server
|
- name: Build & Deploy Project to SAS server
|
||||||
run: npx @sasjs/cli cbd -t server
|
run: npx @sasjs/cli cbd -t server
|
||||||
|
env:
|
||||||
|
macroCorePath: .
|
||||||
|
|
||||||
- name: Run all tests
|
- name: Run all tests
|
||||||
run: npx @sasjs/cli test -t server
|
run: npx @sasjs/cli test -t server
|
||||||
@@ -84,3 +86,4 @@ jobs:
|
|||||||
SAS_PASSWORD: ${{secrets.SAS_PASSWORD}}
|
SAS_PASSWORD: ${{secrets.SAS_PASSWORD}}
|
||||||
ACCESS_TOKEN: ${{secrets.ACCESS_TOKEN}}
|
ACCESS_TOKEN: ${{secrets.ACCESS_TOKEN}}
|
||||||
REFRESH_TOKEN: ${{secrets.REFRESH_TOKEN}}
|
REFRESH_TOKEN: ${{secrets.REFRESH_TOKEN}}
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +0,0 @@
|
|||||||
FROM gitpod/workspace-full
|
|
||||||
|
|
||||||
RUN sudo apt-get update \
|
|
||||||
&& sudo apt-get install -y doxygen \
|
|
||||||
&& sudo apt-get install -y graphviz \
|
|
||||||
&& sudo rm -rf /var/lib/apt/lists/*
|
|
||||||
27
.gitpod.yml
27
.gitpod.yml
@@ -1,27 +0,0 @@
|
|||||||
tasks:
|
|
||||||
- init: npm install -g npm
|
|
||||||
- command: npm i
|
|
||||||
- command: npm i -g @sasjs/cli
|
|
||||||
|
|
||||||
image:
|
|
||||||
file: .gitpod.dockerfile
|
|
||||||
vscode:
|
|
||||||
extensions:
|
|
||||||
- sasjs.sasjs-for-vscode
|
|
||||||
|
|
||||||
github:
|
|
||||||
prebuilds:
|
|
||||||
# enable for the master/default branch (defaults to true)
|
|
||||||
master: true
|
|
||||||
# enable for all branches in this repo (defaults to false)
|
|
||||||
branches: false
|
|
||||||
# enable for pull requests coming from this repo (defaults to true)
|
|
||||||
pullRequests: true
|
|
||||||
# enable for pull requests coming from forks (defaults to false)
|
|
||||||
pullRequestsFromForks: true
|
|
||||||
# add a "Review in Gitpod" button as a comment to pull requests (defaults to true)
|
|
||||||
addComment: true
|
|
||||||
# add a "Review in Gitpod" button to pull requests (defaults to false)
|
|
||||||
addBadge: false
|
|
||||||
# add a label once the prebuild is ready to pull requests (defaults to false)
|
|
||||||
addLabel: prebuilt-in-gitpod
|
|
||||||
@@ -1,6 +1,5 @@
|
|||||||
all.sas
|
all.sas
|
||||||
build.py
|
build.py
|
||||||
.gitpod*
|
|
||||||
tests/
|
tests/
|
||||||
sasjs/
|
sasjs/
|
||||||
.github/
|
.github/
|
||||||
|
|||||||
22
README.md
22
README.md
@@ -1,19 +1,12 @@
|
|||||||
# Macro Core
|
# Macro Core
|
||||||
[![npm package][npm-image]][npm-url]
|
|
||||||
[![Github Workflow][githubworkflow-image]][githubworkflow-url]
|
[](http://npmjs.org/package/@sasjs/core)
|
||||||
|
[](https://github.com/sasjs/core/blob/main/.github/workflows/main.yml)
|
||||||

|

|
||||||

|

|
||||||
[](https://github.com/sasjs/core/issues?q=is%3Aissue+is%3Aclosed)
|
[](https://github.com/sasjs/core/issues?q=is%3Aissue+is%3Aclosed)
|
||||||
[](https://github.com/sasjs/core/issues)
|
[](https://github.com/sasjs/core/issues)
|
||||||

|

|
||||||
[](https://gitpod.io/#https://github.com/sasjs/core)
|
|
||||||
|
|
||||||
|
|
||||||
[npm-image]:https://img.shields.io/npm/v/@sasjs/core.svg
|
|
||||||
[npm-url]:http://npmjs.org/package/@sasjs/core
|
|
||||||
[githubworkflow-image]:https://github.com/sasjs/core/actions/workflows/main.yml/badge.svg
|
|
||||||
[githubworkflow-url]:https://github.com/sasjs/core/blob/main/.github/workflows/main.yml
|
|
||||||
[dependency-url]:https://github.com/sasjs/core/blob/main/package.json
|
|
||||||
|
|
||||||
|
|
||||||
Much quality. Many standards. The **Macro Core** library exists to save time and development effort! Herein ye shall find a veritable host of MIT-licenced, production quality SAS macros. These are a mix of tools, utilities, functions and code generators that are useful in the context of [Application Development](https://sasapps.io) on the SAS platform (eg https://datacontroller.io). [Contributions](https://github.com/sasjs/core/blob/main/.github/CONTRIBUTING.md) are welcome.
|
Much quality. Many standards. The **Macro Core** library exists to save time and development effort! Herein ye shall find a veritable host of MIT-licenced, production quality SAS macros. These are a mix of tools, utilities, functions and code generators that are useful in the context of [Application Development](https://sasapps.io) on the SAS platform (eg https://datacontroller.io). [Contributions](https://github.com/sasjs/core/blob/main/.github/CONTRIBUTING.md) are welcome.
|
||||||
@@ -220,12 +213,13 @@ When contributing to this library, it is therefore important to ensure that all
|
|||||||
|
|
||||||
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:
|
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)
|
* mcf_xxx macros to have `insert_cmplib` option deprecated (the option is now checked automatically with value inserted only if 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.
|
* 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.
|
* mf_getuniquelibref.sas to have the deprecated maxtries parameter removed (no longer needed)
|
||||||
|
* mp_abort.sas will have the redundant type= parameter removed.
|
||||||
* mp_coretable.sas will be replaced by the standalone macros in the `ddl` folder (which are already available)
|
* mp_coretable.sas will be replaced by the standalone macros in the `ddl` folder (which are already available)
|
||||||
|
* 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_testservice.sas to be renamed as mp_execute.sas (as it doesn't actually test anything)
|
||||||
|
|
||||||
## Star Gazing
|
## Star Gazing
|
||||||
|
|
||||||
|
|||||||
622
all.sas
622
all.sas
@@ -1018,8 +1018,9 @@ or %index(&pgm,/tests/testteardown)
|
|||||||
%local dsid vnum rc schema;
|
%local dsid vnum rc schema;
|
||||||
/* in case the parameter is a libref.tablename, pull off just the libref */
|
/* in case the parameter is a libref.tablename, pull off just the libref */
|
||||||
%let libref = %upcase(%scan(&libref, 1, %str(.)));
|
%let libref = %upcase(%scan(&libref, 1, %str(.)));
|
||||||
|
/* sysname can be 'Schema/Owner' or just 'Schema' (eg snowflake) */
|
||||||
%let dsid=%sysfunc(open(sashelp.vlibnam(where=(
|
%let dsid=%sysfunc(open(sashelp.vlibnam(where=(
|
||||||
libname="%upcase(&libref)" and sysname='Schema/Owner'
|
libname="%upcase(&libref)" and sysname=:'Schema'
|
||||||
)),i));
|
)),i));
|
||||||
%if (&dsid ^= 0) %then %do;
|
%if (&dsid ^= 0) %then %do;
|
||||||
%let vnum=%sysfunc(varnum(&dsid,SYSVALUE));
|
%let vnum=%sysfunc(varnum(&dsid,SYSVALUE));
|
||||||
@@ -1121,7 +1122,7 @@ or %index(&pgm,/tests/testteardown)
|
|||||||
@author Allan Bowe
|
@author Allan Bowe
|
||||||
**/
|
**/
|
||||||
|
|
||||||
%macro mf_getuniquelibref(prefix=mclib,maxtries=1000);
|
%macro mf_getuniquelibref(prefix=mc,maxtries=1000);
|
||||||
%local x;
|
%local x;
|
||||||
|
|
||||||
%if ( %length(&prefix) gt 7 ) %then %do;
|
%if ( %length(&prefix) gt 7 ) %then %do;
|
||||||
@@ -1214,8 +1215,9 @@ or %index(&pgm,/tests/testteardown)
|
|||||||
%mend mf_getuser;
|
%mend mf_getuser;
|
||||||
/**
|
/**
|
||||||
@file
|
@file
|
||||||
@brief Retrieves a value from a dataset. If no filter supplied, then first
|
@brief Retrieves a value from a dataset. Returned value is fetched from the
|
||||||
record is used.
|
'fetchobs=' record (row 1 by default), after applying the optional filter.
|
||||||
|
|
||||||
@details Be sure to <code>%quote()</code> your where clause. Example usage:
|
@details Be sure to <code>%quote()</code> your where clause. Example usage:
|
||||||
|
|
||||||
%put %mf_getvalue(sashelp.class,name,filter=%quote(age=15));
|
%put %mf_getvalue(sashelp.class,name,filter=%quote(age=15));
|
||||||
@@ -1230,24 +1232,43 @@ or %index(&pgm,/tests/testteardown)
|
|||||||
@param [in] libds dataset to query
|
@param [in] libds dataset to query
|
||||||
@param [in] variable the variable which contains the value to return.
|
@param [in] variable the variable which contains the value to return.
|
||||||
@param [in] filter= (1) contents of where clause
|
@param [in] filter= (1) contents of where clause
|
||||||
|
@param [in] fetchobs= (1) observation to fetch. NB: Filter applies first.
|
||||||
|
|
||||||
@version 9.2
|
@version 9.2
|
||||||
@author Allan Bowe
|
@author Allan Bowe
|
||||||
**/
|
**/
|
||||||
|
|
||||||
%macro mf_getvalue(libds,variable,filter=1
|
%macro mf_getvalue(libds,variable,filter=1,fetchobs=1
|
||||||
)/*/STORE SOURCE*/;
|
)/*/STORE SOURCE*/;
|
||||||
%if %mf_getattrn(&libds,NLOBS)>0 %then %do;
|
%local dsid;
|
||||||
%local dsid rc &variable;
|
|
||||||
%let dsid=%sysfunc(open(&libds(where=(&filter))));
|
%let dsid=%sysfunc(open(&libds(where=(&filter))));
|
||||||
|
%if (&dsid) %then %do;
|
||||||
|
%local rc &variable;
|
||||||
%syscall set(dsid);
|
%syscall set(dsid);
|
||||||
%let rc = %sysfunc(fetch(&dsid));
|
%let rc = %sysfunc(fetchobs(&dsid,&fetchobs));
|
||||||
|
%if (&rc ne 0) %then %do;
|
||||||
|
%put NOTE: Problem reading obs &fetchobs from &libds..;
|
||||||
|
%put %sysfunc(sysmsg());
|
||||||
|
/* Coerce an rc value of -1 (read past end of data) to a 4
|
||||||
|
that, in SAS condition code terms, represents the sysmsg
|
||||||
|
w@rning it generates. */
|
||||||
|
%if &rc eq -1 %then %let rc = 4;
|
||||||
|
/* And update SYSCC if the &rc value is higher */
|
||||||
|
%let syscc = %sysfunc(max(&syscc,&rc));
|
||||||
|
%end;
|
||||||
%let rc = %sysfunc(close(&dsid));
|
%let rc = %sysfunc(close(&dsid));
|
||||||
|
|
||||||
%trim(&&&variable)
|
%trim(&&&variable)
|
||||||
|
|
||||||
%end;
|
%end;
|
||||||
%mend mf_getvalue;/**
|
%else %do;
|
||||||
|
%put %sysfunc(sysmsg());
|
||||||
|
%let syscc = %sysfunc(max(&syscc,%sysfunc(sysrc())));
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%mend mf_getvalue;
|
||||||
|
/**
|
||||||
@file
|
@file
|
||||||
@brief Returns number of variables in a dataset
|
@brief Returns number of variables in a dataset
|
||||||
@details Useful to identify those renagade datasets that have no columns!
|
@details Useful to identify those renagade datasets that have no columns!
|
||||||
@@ -4002,10 +4023,6 @@ run;
|
|||||||
ignorelist=,
|
ignorelist=,
|
||||||
outds=work.test_results
|
outds=work.test_results
|
||||||
)/*/STORE SOURCE*/;
|
)/*/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
|
* this sets up the global vars, it will also enter STRICT mode. If this
|
||||||
* behaviour is not desired, simply initiate the following global macro
|
* behaviour is not desired, simply initiate the following global macro
|
||||||
@@ -4013,6 +4030,10 @@ run;
|
|||||||
*/
|
*/
|
||||||
%mp_init()
|
%mp_init()
|
||||||
|
|
||||||
|
%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);
|
||||||
|
|
||||||
/* get current variables */
|
/* get current variables */
|
||||||
%if &action=SNAPSHOT %then %do;
|
%if &action=SNAPSHOT %then %do;
|
||||||
proc sql;
|
proc sql;
|
||||||
@@ -11943,7 +11964,7 @@ data _null_;
|
|||||||
run;
|
run;
|
||||||
|
|
||||||
/* END */
|
/* END */
|
||||||
%put &sysmacroname took %sysevalf(%sysfunc(datetime())-&dttm) seconds to run;
|
/* %put &sysmacroname took %sysevalf(%sysfunc(datetime())-&dttm) secs to run; */
|
||||||
|
|
||||||
%mend mp_replace;
|
%mend mp_replace;
|
||||||
/**
|
/**
|
||||||
@@ -13859,6 +13880,7 @@ run;
|
|||||||
@li mf_islibds.sas
|
@li mf_islibds.sas
|
||||||
@li mf_wordsinstr1butnotstr2.sas
|
@li mf_wordsinstr1butnotstr2.sas
|
||||||
@li mp_abort.sas
|
@li mp_abort.sas
|
||||||
|
@li mp_ds2squeeze.sas
|
||||||
|
|
||||||
<h4> Related Macros </h4>
|
<h4> Related Macros </h4>
|
||||||
@li mddl_dc_difftable.sas
|
@li mddl_dc_difftable.sas
|
||||||
@@ -15371,7 +15393,8 @@ run;
|
|||||||
%else %if &engine=ODBC %then %do;
|
%else %if &engine=ODBC %then %do;
|
||||||
%&mD.put NOTE: Retrieving ODBC connection details;
|
%&mD.put NOTE: Retrieving ODBC connection details;
|
||||||
data _null_;
|
data _null_;
|
||||||
length connx_uri conprop_uri value datasource up_uri schema domprop_uri authdomain $256.;
|
length connx_uri conprop_uri value datasource up_uri schema domprop_uri
|
||||||
|
authdomain $256.;
|
||||||
call missing (of _all_);
|
call missing (of _all_);
|
||||||
/* get source connection ID */
|
/* get source connection ID */
|
||||||
rc=metadata_getnasn("&liburi",'LibraryConnection',1,connx_uri);
|
rc=metadata_getnasn("&liburi",'LibraryConnection',1,connx_uri);
|
||||||
@@ -15571,6 +15594,55 @@ run;
|
|||||||
|
|
||||||
libname &libref SQLSVR datasrc=&path schema=&schema user="&user" pass="&pass";
|
libname &libref SQLSVR datasrc=&path schema=&schema user="&user" pass="&pass";
|
||||||
%end;
|
%end;
|
||||||
|
%else %if &engine=SASIOSNF or &engine=SNOW %then %do;
|
||||||
|
%&mD.put NOTE: Retrieving SNOW connection details;
|
||||||
|
data _null_;
|
||||||
|
length connx_uri conprop_uri value server up_uri schema domprop_uri
|
||||||
|
authdomain database $256.;
|
||||||
|
call missing (of _all_);
|
||||||
|
/* get source connection ID */
|
||||||
|
rc=metadata_getnasn("&liburi",'LibraryConnection',1,connx_uri);
|
||||||
|
/* get connection properties */
|
||||||
|
i=0;
|
||||||
|
do until (rc2<0);
|
||||||
|
i+1;
|
||||||
|
rc2=metadata_getnasn(connx_uri,'Properties',i,conprop_uri);
|
||||||
|
rc3=metadata_getattr(conprop_uri,'Name',value);
|
||||||
|
if value='Connection.DBMS.Property.SERVER.Name.xmlKey.txt' then do;
|
||||||
|
rc4=metadata_getattr(conprop_uri,'DefaultValue',server);
|
||||||
|
rc2=-1;
|
||||||
|
end;
|
||||||
|
end;
|
||||||
|
|
||||||
|
/* get auth domain */
|
||||||
|
autrc=metadata_getnasn(connx_uri,"Domain",1,domprop_uri);
|
||||||
|
arc=metadata_getattr(domprop_uri,"Name",authdomain);
|
||||||
|
if not missing(authdomain) then authdomain=cats('AUTHDOMAIN=',authdomain);
|
||||||
|
call symputx('authdomain',authdomain,'l');
|
||||||
|
|
||||||
|
/* get SCHEMA */
|
||||||
|
rc6=metadata_getnasn("&liburi",'UsingPackages',1,up_uri);
|
||||||
|
rc7=metadata_getattr(up_uri,'SchemaName',schema);
|
||||||
|
&mD.put rc= connx_uri= rc2= conprop_uri= rc3= value= rc4= server=
|
||||||
|
rc6= up_uri= rc7= schema=;
|
||||||
|
|
||||||
|
/* get database value */
|
||||||
|
prop='Connection.DBMS.Property.DB.Name.xmlKey.txt';
|
||||||
|
rc=metadata_getprop("&liburi",prop,database,"");
|
||||||
|
if database^='' then database='database='!!quote(trim(database));
|
||||||
|
call symputx('database',database,'l');
|
||||||
|
|
||||||
|
call symputx('snow_schema',schema,'l');
|
||||||
|
call symputx('snow_server',server,'l');
|
||||||
|
run;
|
||||||
|
|
||||||
|
libname &libref SNOW SERVER="&snow_server" SCHEMA=&snow_schema &authdomain
|
||||||
|
&database;
|
||||||
|
%if %length(&open_passthrough)>0 %then %do;
|
||||||
|
proc sql;
|
||||||
|
connect using &libref as &open_passthrough;
|
||||||
|
%end;
|
||||||
|
%end;
|
||||||
%else %if &engine=TERADATA %then %do;
|
%else %if &engine=TERADATA %then %do;
|
||||||
%put NOTE: Obtaining &engine library details;
|
%put NOTE: Obtaining &engine library details;
|
||||||
data _null;
|
data _null;
|
||||||
@@ -24098,18 +24170,25 @@ run;
|
|||||||
msg=Cannot enter mfv_existfolder.sas with syscc=&syscc
|
msg=Cannot enter mfv_existfolder.sas with syscc=&syscc
|
||||||
)
|
)
|
||||||
|
|
||||||
%local fref rc;
|
%local fref rc var;
|
||||||
%let fref=%mf_getuniquefileref();
|
%let fref=%mf_getuniquefileref();
|
||||||
|
|
||||||
%if %sysfunc(filename(fref,,filesrvc,folderPath="&path"))=0 %then %do;
|
%if %sysfunc(filename(fref,,filesrvc,folderPath="&path"))=0 %then %do;
|
||||||
1
|
1
|
||||||
|
%let var=_FILESRVC_&fref._URI;
|
||||||
%let rc=%sysfunc(filename(fref));
|
%let rc=%sysfunc(filename(fref));
|
||||||
|
%symdel &var;
|
||||||
%end;
|
%end;
|
||||||
%else %do;
|
%else %do;
|
||||||
0
|
0
|
||||||
%let syscc=0;
|
%let syscc=0;
|
||||||
%end;
|
%end;
|
||||||
|
|
||||||
|
%mf_abort(
|
||||||
|
iftrue=(&syscc ne 0),
|
||||||
|
msg=Cannot leave mfv_existfolder.sas with syscc=&syscc
|
||||||
|
)
|
||||||
|
|
||||||
%mend mfv_existfolder;/**
|
%mend mfv_existfolder;/**
|
||||||
@file mfv_existsashdat.sas
|
@file mfv_existsashdat.sas
|
||||||
@brief Checks whether a CAS sashdat dataset exists in persistent storage.
|
@brief Checks whether a CAS sashdat dataset exists in persistent storage.
|
||||||
@@ -24273,6 +24352,10 @@ run;
|
|||||||
%let syscc=0;
|
%let syscc=0;
|
||||||
%end;
|
%end;
|
||||||
|
|
||||||
|
%mf_abort(
|
||||||
|
iftrue=(&syscc ne 0),
|
||||||
|
msg=Cannot leave &sysmacroname with syscc=&syscc
|
||||||
|
)
|
||||||
%mend mfv_getpathuri;/**
|
%mend mfv_getpathuri;/**
|
||||||
@file
|
@file
|
||||||
@brief Creates a file in SAS Drive using the API method
|
@brief Creates a file in SAS Drive using the API method
|
||||||
@@ -24344,6 +24427,7 @@ run;
|
|||||||
@li mp_base64copy.sas
|
@li mp_base64copy.sas
|
||||||
@li mp_replace.sas
|
@li mp_replace.sas
|
||||||
@li mv_createfolder.sas
|
@li mv_createfolder.sas
|
||||||
|
@li mv_getviyafileextparms.sas
|
||||||
|
|
||||||
<h4> Related Macros</h4>
|
<h4> Related Macros</h4>
|
||||||
@li mv_createfile.sas
|
@li mv_createfile.sas
|
||||||
@@ -24370,6 +24454,11 @@ run;
|
|||||||
%end;
|
%end;
|
||||||
%else %let dbg=*;
|
%else %let dbg=*;
|
||||||
|
|
||||||
|
%mp_abort(
|
||||||
|
iftrue=(&syscc ne 0),
|
||||||
|
msg=Cannot enter &sysmacroname with syscc=&syscc
|
||||||
|
)
|
||||||
|
|
||||||
%local oauth_bearer;
|
%local oauth_bearer;
|
||||||
%if &grant_type=detect %then %do;
|
%if &grant_type=detect %then %do;
|
||||||
%if %symexist(&access_token_var) %then %let grant_type=authorization_code;
|
%if %symexist(&access_token_var) %then %let grant_type=authorization_code;
|
||||||
@@ -24423,7 +24512,7 @@ run;
|
|||||||
|
|
||||||
options noquotelenmax;
|
options noquotelenmax;
|
||||||
%local base_uri; /* location of rest apis */
|
%local base_uri; /* location of rest apis */
|
||||||
%let base_uri=%mf_getplatform(VIYARESTAPI);
|
%let base_uri=%trim(%mf_getplatform(VIYARESTAPI));
|
||||||
|
|
||||||
/* create folder if it does not already exist */
|
/* create folder if it does not already exist */
|
||||||
%local folderds self_uri;
|
%local folderds self_uri;
|
||||||
@@ -24442,11 +24531,16 @@ run;
|
|||||||
/* abort or delete if file already exists */
|
/* abort or delete if file already exists */
|
||||||
%let force=%upcase(&force);
|
%let force=%upcase(&force);
|
||||||
%local fileuri ;
|
%local fileuri ;
|
||||||
%let fileuri=%mfv_getpathuri(&path/&name);
|
%let fileuri=%trim(%mfv_getpathuri(&path/&name));
|
||||||
%mp_abort(iftrue=(%mf_isblank(&fileuri)=0 and &force ne YES)
|
%mp_abort(iftrue=(%mf_isblank(&fileuri)=0 and &force ne YES)
|
||||||
,mac=MV_CREATEFILE
|
,mac=MV_CREATEFILE
|
||||||
,msg=%str(File &path/&name already exists and force=&force)
|
,msg=%str(File &path/&name already exists and force=&force)
|
||||||
)
|
)
|
||||||
|
%mp_abort(
|
||||||
|
iftrue=(&syscc ne 0),
|
||||||
|
mac=MV_CREATEFILE182
|
||||||
|
msg=syscc=&syscc after mfv_getpathuri
|
||||||
|
)
|
||||||
|
|
||||||
%if %mf_isblank(&fileuri)=0 and &force=YES %then %do;
|
%if %mf_isblank(&fileuri)=0 and &force=YES %then %do;
|
||||||
proc http method="DELETE" url="&base_uri&fileuri" &oauth_bearer;
|
proc http method="DELETE" url="&base_uri&fileuri" &oauth_bearer;
|
||||||
@@ -24464,7 +24558,13 @@ run;
|
|||||||
|
|
||||||
%local url mimetype ext;
|
%local url mimetype ext;
|
||||||
%let url=&base_uri/files/files?parentFolderUri=&self_uri;
|
%let url=&base_uri/files/files?parentFolderUri=&self_uri;
|
||||||
%let ext=%upcase(%scan(&name,-1,.));
|
%let ext=%upcase(%trim(%scan(&name,-1,.)));
|
||||||
|
|
||||||
|
/* Get Viya file-extension details into some macro variables */
|
||||||
|
%mv_getViyaFileExtParms(&ext
|
||||||
|
,propertiesVar=viyaProperties
|
||||||
|
,typeDefNameVar=viyaTypeDefName
|
||||||
|
,mdebug=&mdebug);
|
||||||
|
|
||||||
/* fetch job info */
|
/* fetch job info */
|
||||||
%local fname1;
|
%local fname1;
|
||||||
@@ -24477,12 +24577,23 @@ proc http method='POST' out=&fname1 &oauth_bearer in=&fref
|
|||||||
%else %do;
|
%else %do;
|
||||||
ct="&ctype"
|
ct="&ctype"
|
||||||
%end;
|
%end;
|
||||||
%if "&ext"="HTML" or "&ext"="CSS" or "&ext"="JS" or "&ext"="PNG"
|
|
||||||
or "&ext"="SVG" %then %do;
|
/* typeDefName */
|
||||||
url="&url%str(&)typeDefName=file";
|
%if not %mf_isBlank(&viyaTypeDefName) %then %do;
|
||||||
|
url="&url%str(&)typeDefName=&viyaTypeDefName";
|
||||||
%end;
|
%end;
|
||||||
%else %do;
|
%else %do;
|
||||||
url="&url";
|
%if "&ext"="HTM" or "&ext"="HTML" or "&ext"="XHTML" %then %do;
|
||||||
|
url="&url%str(&)typeDefName=file_html";
|
||||||
|
%end;
|
||||||
|
%else %do;
|
||||||
|
%if "&ext"="CSS" or "&ext"="JS" or "&ext"="PNG" or "&ext"="SVG" %then %do;
|
||||||
|
url="&url%str(&)typeDefName=file_%lowcase(&ext)";
|
||||||
|
%end;
|
||||||
|
%else %do;
|
||||||
|
url="&url";
|
||||||
|
%end;
|
||||||
|
%end;
|
||||||
%end;
|
%end;
|
||||||
|
|
||||||
headers "Accept"="application/json"
|
headers "Accept"="application/json"
|
||||||
@@ -24504,6 +24615,7 @@ run;
|
|||||||
,mac=MV_CREATEFILE
|
,mac=MV_CREATEFILE
|
||||||
,msg=%str(&SYS_PROCHTTP_STATUS_CODE &SYS_PROCHTTP_STATUS_PHRASE)
|
,msg=%str(&SYS_PROCHTTP_STATUS_CODE &SYS_PROCHTTP_STATUS_PHRASE)
|
||||||
)
|
)
|
||||||
|
|
||||||
%local libref2;
|
%local libref2;
|
||||||
%let libref2=%mf_getuniquelibref();
|
%let libref2=%mf_getuniquelibref();
|
||||||
libname &libref2 JSON fileref=&fname1;
|
libname &libref2 JSON fileref=&fname1;
|
||||||
@@ -24511,13 +24623,73 @@ libname &libref2 JSON fileref=&fname1;
|
|||||||
data &outds;
|
data &outds;
|
||||||
set &libref2..links end=last;
|
set &libref2..links end=last;
|
||||||
if rel='createChild' then do;
|
if rel='createChild' then do;
|
||||||
call symputx('href',quote(cats("&base_uri",href)),'l');
|
|
||||||
&dbg put (_all_)(=);
|
&dbg put (_all_)(=);
|
||||||
end;
|
end;
|
||||||
run;
|
run;
|
||||||
|
|
||||||
%put &sysmacroname: &name created at %mfv_getpathuri(&path/&name);%put;
|
/* URI of the created file */
|
||||||
%put &base_uri/SASJobExecution?_file=&path/&name;%put;
|
%let fileuri=%trim(%mfv_getpathuri(&path/&name));
|
||||||
|
|
||||||
|
/* If properties were found then patch the file to include them */
|
||||||
|
%if not %mf_isBlank(%superq(viyaProperties)) %then %do;
|
||||||
|
/* Wrap the properties object in a root object also containing the file name */
|
||||||
|
%local viyapatch;
|
||||||
|
%let viyapatch = %sysfunc(pathname(work))/%mf_getuniquename(prefix=patch_json_);
|
||||||
|
data _null_;
|
||||||
|
length line $32767;
|
||||||
|
file "&viyapatch" lrecl=32767;
|
||||||
|
put '{ "name": "' "&name" '",';
|
||||||
|
line = cat('"properties": ',symget("viyaProperties"));
|
||||||
|
put line;
|
||||||
|
put '}';
|
||||||
|
stop;
|
||||||
|
run;
|
||||||
|
|
||||||
|
%if &mdebug=1 %then %do;
|
||||||
|
data _null_;
|
||||||
|
if (_n_ eq 1) then put 'DEBUG: ** PATCH JSON **';
|
||||||
|
infile "&viyapatch" end=last;
|
||||||
|
input;
|
||||||
|
put _infile_;
|
||||||
|
run;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
/* And apply the properties to the newly created file, using the PATCH method */
|
||||||
|
%let fref=%mf_getuniquefileref();
|
||||||
|
filename &fref "&viyapatch";
|
||||||
|
%let url=&base_uri&fileuri;
|
||||||
|
|
||||||
|
proc http method='PATCH' oauth_bearer=sas_services in=&fref
|
||||||
|
url="&url";
|
||||||
|
headers "Accept"="application/json"
|
||||||
|
"Content-Type"="application/json"
|
||||||
|
"If-Match"="*";
|
||||||
|
%if &mdebug=1 %then %do;
|
||||||
|
debug level=2;
|
||||||
|
%end;
|
||||||
|
run;
|
||||||
|
%if &mdebug=1 %then %put &sysmacroname PATCH &=url
|
||||||
|
&=SYS_PROCHTTP_STATUS_CODE &=SYS_PROCHTTP_STATUS_PHRASE;
|
||||||
|
%mp_abort(iftrue=(&SYS_PROCHTTP_STATUS_CODE ne 200)
|
||||||
|
,mac=MV_CREATEFILE
|
||||||
|
,msg=%str(&SYS_PROCHTTP_STATUS_CODE &SYS_PROCHTTP_STATUS_PHRASE)
|
||||||
|
)
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%put &sysmacroname: &base_uri&fileuri;
|
||||||
|
%put /SASJobExecution?_file=&path/&name;%put;
|
||||||
|
|
||||||
|
%if &mdebug=0 %then %do;
|
||||||
|
/* clear refs */
|
||||||
|
filename &fname1 clear;
|
||||||
|
filename &fref clear;
|
||||||
|
libname &libref2 clear;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%mp_abort(
|
||||||
|
iftrue=(&syscc ne 0),
|
||||||
|
msg=Cannot leave &sysmacroname with syscc=&syscc
|
||||||
|
)
|
||||||
|
|
||||||
%mend mv_createfile;/**
|
%mend mv_createfile;/**
|
||||||
@file mv_createfolder.sas
|
@file mv_createfolder.sas
|
||||||
@@ -24567,6 +24739,11 @@ run;
|
|||||||
%end;
|
%end;
|
||||||
%else %let dbg=*;
|
%else %let dbg=*;
|
||||||
|
|
||||||
|
%mp_abort(
|
||||||
|
iftrue=(&syscc ne 0),
|
||||||
|
msg=Cannot enter &sysmacroname with syscc=&syscc
|
||||||
|
)
|
||||||
|
|
||||||
%if %mfv_existfolder(&path)=1 %then %do;
|
%if %mfv_existfolder(&path)=1 %then %do;
|
||||||
%&dbg.put &sysmacroname: &path already exists;
|
%&dbg.put &sysmacroname: &path already exists;
|
||||||
data &outds;
|
data &outds;
|
||||||
@@ -24576,6 +24753,7 @@ run;
|
|||||||
run;
|
run;
|
||||||
%return;
|
%return;
|
||||||
%end;
|
%end;
|
||||||
|
%mp_abort(iftrue=(&syscc ne 0),msg=syscc=&syscc when folder checking)
|
||||||
|
|
||||||
%local oauth_bearer;
|
%local oauth_bearer;
|
||||||
%if &grant_type=detect %then %do;
|
%if &grant_type=detect %then %do;
|
||||||
@@ -24629,6 +24807,17 @@ options noquotelenmax;
|
|||||||
headers "Authorization"="Bearer &&&access_token_var";
|
headers "Authorization"="Bearer &&&access_token_var";
|
||||||
%end;
|
%end;
|
||||||
run;
|
run;
|
||||||
|
%if &SYS_PROCHTTP_STATUS_CODE=401 %then %do;
|
||||||
|
/* relates to: https://github.com/sasjs/core/issues/400 */
|
||||||
|
%put 401 thrown in &sysmacroname;
|
||||||
|
%put sleeping: %sysfunc(sleep(12,1)) secs - will try again;
|
||||||
|
proc http method='GET' out=&fname1 &oauth_bearer
|
||||||
|
url="&base_uri/folders/folders/@item?path=&newpath";
|
||||||
|
%if &grant_type=authorization_code %then %do;
|
||||||
|
headers "Authorization"="Bearer &&&access_token_var";
|
||||||
|
%end;
|
||||||
|
run;
|
||||||
|
%end;
|
||||||
%local libref1;
|
%local libref1;
|
||||||
%let libref1=%mf_getuniquelibref();
|
%let libref1=%mf_getuniquelibref();
|
||||||
libname &libref1 JSON fileref=&fname1;
|
libname &libref1 JSON fileref=&fname1;
|
||||||
@@ -24636,7 +24825,7 @@ options noquotelenmax;
|
|||||||
iftrue=(
|
iftrue=(
|
||||||
&SYS_PROCHTTP_STATUS_CODE ne 200 and &SYS_PROCHTTP_STATUS_CODE ne 404
|
&SYS_PROCHTTP_STATUS_CODE ne 200 and &SYS_PROCHTTP_STATUS_CODE ne 404
|
||||||
)
|
)
|
||||||
,mac=&sysmacroname
|
,mac=mv_createfolder124
|
||||||
,msg=%str(&SYS_PROCHTTP_STATUS_CODE &SYS_PROCHTTP_STATUS_PHRASE)
|
,msg=%str(&SYS_PROCHTTP_STATUS_CODE &SYS_PROCHTTP_STATUS_PHRASE)
|
||||||
)
|
)
|
||||||
%if &mdebug=1 %then %do;
|
%if &mdebug=1 %then %do;
|
||||||
@@ -24685,7 +24874,7 @@ options noquotelenmax;
|
|||||||
'Content-Type'='application/vnd.sas.content.folder+json'
|
'Content-Type'='application/vnd.sas.content.folder+json'
|
||||||
'Accept'='application/vnd.sas.content.folder+json';
|
'Accept'='application/vnd.sas.content.folder+json';
|
||||||
run;
|
run;
|
||||||
%if &SYS_PROCHTTP_STATUS_CODE ne 200 %then %do;
|
%if &SYS_PROCHTTP_STATUS_CODE ne 201 %then %do;
|
||||||
%put &=SYS_PROCHTTP_STATUS_CODE &=SYS_PROCHTTP_STATUS_PHRASE;
|
%put &=SYS_PROCHTTP_STATUS_CODE &=SYS_PROCHTTP_STATUS_PHRASE;
|
||||||
%end;
|
%end;
|
||||||
%mp_abort(iftrue=(&SYS_PROCHTTP_STATUS_CODE ne 201)
|
%mp_abort(iftrue=(&SYS_PROCHTTP_STATUS_CODE ne 201)
|
||||||
@@ -24715,6 +24904,10 @@ options noquotelenmax;
|
|||||||
filename &fname1 clear;
|
filename &fname1 clear;
|
||||||
libname &libref1 clear;
|
libname &libref1 clear;
|
||||||
%end;
|
%end;
|
||||||
|
%mp_abort(
|
||||||
|
iftrue=(&syscc ne 0),
|
||||||
|
msg=Cannot leave &sysmacroname with syscc=&syscc
|
||||||
|
)
|
||||||
%mend mv_createfolder;/**
|
%mend mv_createfolder;/**
|
||||||
@file
|
@file
|
||||||
@brief Creates a Viya Job
|
@brief Creates a Viya Job
|
||||||
@@ -26305,7 +26498,8 @@ libname &libref1a JSON fileref=&fname1a;
|
|||||||
%let found=0;
|
%let found=0;
|
||||||
/* %put Getting object uri from &libref1a..items; */
|
/* %put Getting object uri from &libref1a..items; */
|
||||||
data _null_;
|
data _null_;
|
||||||
length contenttype name $1000;
|
length contenttype name uri $1000;
|
||||||
|
call missing(of _all_);
|
||||||
set &libref1a..items;
|
set &libref1a..items;
|
||||||
if contenttype='jobDefinition' and upcase(name)="%upcase(&name)" then do;
|
if contenttype='jobDefinition' and upcase(name)="%upcase(&name)" then do;
|
||||||
call symputx('uri',cats("&base_uri",uri),'l');
|
call symputx('uri',cats("&base_uri",uri),'l');
|
||||||
@@ -27280,7 +27474,7 @@ data _null_;
|
|||||||
uri=symget('uri');
|
uri=symget('uri');
|
||||||
if length(uri)<12 then do;
|
if length(uri)<12 then do;
|
||||||
call symputx('errflg',1);
|
call symputx('errflg',1);
|
||||||
call symputx('errmsg',"URI is invalid (too short) - '&uri'",'l');
|
call symputx('errmsg',"URI is too short - "!!uri,'l');
|
||||||
end;
|
end;
|
||||||
if scan(uri,-1)='state' or scan(uri,1) ne 'jobExecution' then do;
|
if scan(uri,-1)='state' or scan(uri,1) ne 'jobExecution' then do;
|
||||||
call symputx('errflg',1);
|
call symputx('errflg',1);
|
||||||
@@ -27345,7 +27539,7 @@ data _null_;
|
|||||||
uri=symget('loglocation');
|
uri=symget('loglocation');
|
||||||
if length(uri)<12 then do;
|
if length(uri)<12 then do;
|
||||||
call symputx('errflg',1);
|
call symputx('errflg',1);
|
||||||
call symputx('errmsg',"URI is invalid (too short) - '&uri'",'l');
|
call symputx('errmsg',"URI is too short - "!!uri,'l');
|
||||||
end;
|
end;
|
||||||
else if (scan(uri,1,'/') ne 'compute' or scan(uri,2,'/') ne 'sessions')
|
else if (scan(uri,1,'/') ne 'compute' or scan(uri,2,'/') ne 'sessions')
|
||||||
and (scan(uri,1,'/') ne 'files' or scan(uri,2,'/') ne 'files')
|
and (scan(uri,1,'/') ne 'files' or scan(uri,2,'/') ne 'files')
|
||||||
@@ -28057,6 +28251,254 @@ filename &fname1 clear;
|
|||||||
libname &libref1 clear;
|
libname &libref1 clear;
|
||||||
|
|
||||||
%mend mv_getusers;/**
|
%mend mv_getusers;/**
|
||||||
|
@file mv_getviyafileextparms.sas
|
||||||
|
@brief Reads the VIYA file-extension type definition and returns selected
|
||||||
|
values in SAS macro variables
|
||||||
|
|
||||||
|
@details Content is derived from the following endpoint:
|
||||||
|
"https://${serverUrl}/types/types?limit=999999"
|
||||||
|
|
||||||
|
@param [in] ext File extension to retrieve property info for.
|
||||||
|
@param [out] propertiesVar= SAS macro variable name that will contain
|
||||||
|
the 'properties' object json, if found, else blank.
|
||||||
|
@param [out] typeDefNameVar= SAS macro variable name that will contain
|
||||||
|
the 'typeDefName' property value, if found, else blank.
|
||||||
|
@param [out] mediaTypeVar= SAS macro variable name that will contain
|
||||||
|
the 'mediaType' property value, if found, else blank.
|
||||||
|
@param [out] viyaFileExtRespLibDs (work.mv_getViyaFileExtParmsResponse)
|
||||||
|
Library.name of the dataset to receive the local working copy of the initial
|
||||||
|
response that requests all file extension details. Created once per session
|
||||||
|
to avoid multiple api calls.
|
||||||
|
@param [in] mdebug= (0) Set to 1 to enable DEBUG messages
|
||||||
|
|
||||||
|
<h4> SAS Macros </h4>
|
||||||
|
@li mf_existds.sas
|
||||||
|
@li mf_getplatform.sas
|
||||||
|
@li mf_getuniquefileref.sas
|
||||||
|
@li mf_getuniquename.sas
|
||||||
|
@li mf_getvalue.sas
|
||||||
|
@li mf_getvarlist.sas
|
||||||
|
@li mf_getvartype.sas
|
||||||
|
@li mf_isblank.sas
|
||||||
|
@li mf_nobs.sas
|
||||||
|
@li mp_abort.sas
|
||||||
|
|
||||||
|
**/
|
||||||
|
|
||||||
|
%macro mv_getViyaFileExtParms(
|
||||||
|
ext,
|
||||||
|
typeDefNameVar=,
|
||||||
|
propertiesVar=,
|
||||||
|
mediaTypeVar=,
|
||||||
|
viyaFileExtRespLibDs=work.mv_getViyaFileExtParmsResponse,
|
||||||
|
mdebug=0
|
||||||
|
);
|
||||||
|
%local base_uri; /* location of rest apis */
|
||||||
|
%local url; /* File extension info end-point */
|
||||||
|
|
||||||
|
%mp_abort(
|
||||||
|
iftrue=(%mf_isBlank(&ext))
|
||||||
|
,msg=%str(No file extension provided.)
|
||||||
|
,mac=MV_GETVIYAFILEEXTPARMS
|
||||||
|
);
|
||||||
|
|
||||||
|
%mp_abort(
|
||||||
|
iftrue=(%mf_isBlank(&typeDefNameVar) and
|
||||||
|
%mf_isBlank(&propertiesVar) and
|
||||||
|
%mf_isBlank(&mediaTypeVar))
|
||||||
|
,msg=%str(MV_GETVIYAFILEEXTPARMS - No parameter was requested.)
|
||||||
|
,mac=MV_GETVIYAFILEEXTPARMS
|
||||||
|
);
|
||||||
|
|
||||||
|
%mp_abort(
|
||||||
|
iftrue=(%mf_isBlank(&viyaFileExtRespLibDs))
|
||||||
|
,msg=%str(No <libname.>dataset name provided to cache inital response.)
|
||||||
|
,mac=MV_GETVIYAFILEEXTPARMS
|
||||||
|
);
|
||||||
|
|
||||||
|
/* Declare requested parameters as global macro vars and initialize blank */
|
||||||
|
%if not %mf_isBlank(&typeDefNameVar) %then %do;
|
||||||
|
%global &typeDefNameVar;
|
||||||
|
%let &typeDefNameVar = %str();
|
||||||
|
%end;
|
||||||
|
%if not %mf_isBlank(&propertiesVar) %then %do;
|
||||||
|
%global &propertiesVar;
|
||||||
|
%let &propertiesVar = %str();
|
||||||
|
%end;
|
||||||
|
%if not %mf_isBlank(&mediaTypeVar) %then %do;
|
||||||
|
%global &mediaTypeVar;
|
||||||
|
%let &mediaTypeVar = %str();
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%let base_uri=%mf_getplatform(VIYARESTAPI);
|
||||||
|
%if &mdebug=1 %then %do;
|
||||||
|
%put DEBUG: &=base_uri;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%let ext=%lowcase(&ext);
|
||||||
|
|
||||||
|
/* Create a local copy of the Viya response containing all file type info, if
|
||||||
|
it does not already exist. */
|
||||||
|
%if not %mf_existds(&viyaFileExtRespLibDs) %then %do;
|
||||||
|
/* Create a temp file and fill with JSON that declares */
|
||||||
|
/* VIYA file-type details for the given file extension */
|
||||||
|
%local viyatypedefs;
|
||||||
|
%let viyatypedefs=%mf_getuniquefileref();
|
||||||
|
filename &viyatypedefs temp;
|
||||||
|
|
||||||
|
%let url = &base_uri/types/types?limit=999999;
|
||||||
|
|
||||||
|
proc http oauth_bearer=sas_services out=&viyatypedefs
|
||||||
|
url="&url";
|
||||||
|
run;
|
||||||
|
|
||||||
|
%if &mdebug=1 %then %put DEBUG: &sysmacroname &=url
|
||||||
|
&=SYS_PROCHTTP_STATUS_CODE &=SYS_PROCHTTP_STATUS_PHRASE;
|
||||||
|
|
||||||
|
%if (&SYS_PROCHTTP_STATUS_CODE ne 200) %then %do;
|
||||||
|
/* To avoid a breaking change, exit early if the request failed.
|
||||||
|
The calling process will proceed with empty requested macro variables. */
|
||||||
|
%put INFO: &sysmacroname File extension details were not retrieved.;
|
||||||
|
filename &viyatypedefs clear;
|
||||||
|
%return;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%if &mdebug=1 %then %do;
|
||||||
|
/* Dump the response to the log */
|
||||||
|
data _null_;
|
||||||
|
length line $120;
|
||||||
|
null=byte(0);
|
||||||
|
infile &viyatypedefs dlm=null lrecl=120 recfm=n;
|
||||||
|
input line $120.;
|
||||||
|
if _n_ = 1 then put "DEBUG:";
|
||||||
|
put line;
|
||||||
|
run;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
/* Convert the content of that JSON into SAS datasets */
|
||||||
|
/* First prepare a new WORK-based folder to receive the datasets */
|
||||||
|
%local jsonworkfolder jsonlib opt_dlcreatedir;
|
||||||
|
%let jsonworkfolder=%sysfunc(pathname(work))/%mf_getuniquename(prefix=json_);
|
||||||
|
%let jsonlib=%mf_getuniquelibref(prefix=json);
|
||||||
|
/* And point a libname at it */
|
||||||
|
%let opt_dlcreatedir = %sysfunc(getoption(dlcreatedir));
|
||||||
|
options dlcreatedir; libname &jsonlib "&jsonworkfolder"; options &opt_dlcreatedir;
|
||||||
|
|
||||||
|
/* Read the json output once and copy datasets to its work folder */
|
||||||
|
%local libref1;
|
||||||
|
%let libref1=%mf_getuniquelibref();
|
||||||
|
libname &libref1 JSON fileref=&viyatypedefs automap=create;
|
||||||
|
proc copy in=&libref1 out=&jsonlib; run;
|
||||||
|
|
||||||
|
libname &libref1 clear;
|
||||||
|
|
||||||
|
/* Now give all rows belonging to the same items array a grouping value */
|
||||||
|
data &viyaFileExtRespLibDs;
|
||||||
|
length _viyaItemIdx 8;
|
||||||
|
set &jsonlib..alldata;
|
||||||
|
retain _viyaItemIdx 0;
|
||||||
|
/* Increment the row group index when a new 'items' group is observed */
|
||||||
|
if P=1 and P1='items' then _viyaItemIdx + 1;
|
||||||
|
run;
|
||||||
|
|
||||||
|
%if &mdebug=0 %then %do;
|
||||||
|
/* Tidy up, unless debug=1 */
|
||||||
|
proc datasets library=&jsonlib nolist kill; quit;
|
||||||
|
libname &jsonlib clear;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
filename &viyatypedefs clear;
|
||||||
|
|
||||||
|
%end; /* If initial filetype query response didn't exist */
|
||||||
|
|
||||||
|
/* Find the row-group for the current file extension */
|
||||||
|
%local itemRowGroup;
|
||||||
|
%let itemRowGroup =
|
||||||
|
%mf_getValue(
|
||||||
|
&viyaFileExtRespLibDs
|
||||||
|
,_viyaItemIdx
|
||||||
|
,filter=%quote(p1='items' and p2='extensions' and value="&ext")
|
||||||
|
);
|
||||||
|
|
||||||
|
%if &mdebug %then %put DEBUG: &=itemRowGroup;
|
||||||
|
|
||||||
|
%if %mf_isBlank(&itemRowGroup) %then %do;
|
||||||
|
/* extension was not found */
|
||||||
|
%if(&mdebug=1) %then %put DEBUG: No type details found for extension "&ext".;
|
||||||
|
%return;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
/* Filter the cached response data down to the required file extension */
|
||||||
|
%local dsItems;
|
||||||
|
%let dsItems = %mf_getuniquename(prefix=dsItems_);
|
||||||
|
data work.&dsItems;
|
||||||
|
set &viyaFileExtRespLibDs;
|
||||||
|
where _viyaItemIdx = &itemRowGroup;
|
||||||
|
run;
|
||||||
|
|
||||||
|
/* Populate typeDefName, if requested */
|
||||||
|
%if (not %mf_isBlank(&typeDefNameVar)) %then %do;
|
||||||
|
%let &typeDefNameVar = %mf_getvalue(&dsItems,value,filter=%quote(p1="items" and p2="name"));
|
||||||
|
%if &mdebug=1 %then %put DEBUG: &=typeDefNameVar &typeDefNameVar=&&&typeDefNameVar;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
/* Populate mediaType, if requested */
|
||||||
|
%if (not %mf_isBlank(&mediaTypeVar)) %then %do;
|
||||||
|
%let &mediaTypeVar = %mf_getvalue(&dsItems,value,filter=%quote(p1="items" and p2="mediaType"));
|
||||||
|
%if &mdebug=1 %then %put DEBUG: &=mediaTypeVar &mediaTypeVar=&&&mediaTypeVar;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
/* Populate properties macro variable, if requested */
|
||||||
|
%if not %mf_isBlank(&propertiesVar) %then %do;
|
||||||
|
|
||||||
|
/* Filter dsItems down to the properties */
|
||||||
|
%local dsProperties;
|
||||||
|
%let dsProperties = %mf_getuniquename(prefix=dsProperties_);
|
||||||
|
data work.&dsProperties ( rename=(p3 = propertyName) );
|
||||||
|
set work.&dsItems;
|
||||||
|
where p2="properties" and v=1;
|
||||||
|
run;
|
||||||
|
|
||||||
|
/* Check for 1+ properties */
|
||||||
|
%if ( %mf_nobs(&dsProperties) = 0 ) %then %do;
|
||||||
|
%let &propertiesVar = %str();
|
||||||
|
%if &mdebug=1 %then %put DEBUG: &SYSMACRONAME - No Viya properties found for file suffix %str(%')&ext%str(%');
|
||||||
|
%end;
|
||||||
|
%else %do;
|
||||||
|
/* Properties potentially span multiple rows in the input table */
|
||||||
|
data _null_;
|
||||||
|
length
|
||||||
|
line $32767
|
||||||
|
properties $32767
|
||||||
|
;
|
||||||
|
retain properties;
|
||||||
|
set &dsProperties end=last;
|
||||||
|
if _n_ = 1 then properties = '{';
|
||||||
|
|
||||||
|
line = cats(quote(trim(propertyName)),':');
|
||||||
|
/* Only strings and bools appear in properties */
|
||||||
|
if value not in ("true","false") then value = quote(trim(value));
|
||||||
|
line = catx(' ',line,value);
|
||||||
|
/* Add a comma separator to all except the last line */
|
||||||
|
if not last then line = cats(line,',');
|
||||||
|
|
||||||
|
/* Add this line to the output value */
|
||||||
|
properties = catx(' ',properties,line);
|
||||||
|
|
||||||
|
if last then do;
|
||||||
|
/* Close off the properties object and output to the macro variable */
|
||||||
|
properties=catx(' ',properties,'}');
|
||||||
|
call symputx("&propertiesVar",properties);
|
||||||
|
end;
|
||||||
|
run;
|
||||||
|
|
||||||
|
%if &mdebug=1 %then %put DEBUG: &=propertiesVar &propertiesVar=&&&propertiesVar;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%mend mv_getViyaFileExtParms;
|
||||||
|
/**
|
||||||
@file
|
@file
|
||||||
@brief Executes a SAS Viya Job
|
@brief Executes a SAS Viya Job
|
||||||
@details Triggers a SAS Viya Job, with optional URL parameters, using
|
@details Triggers a SAS Viya Job, with optional URL parameters, using
|
||||||
@@ -28433,6 +28875,8 @@ run;
|
|||||||
%if %mf_existvarList(&inds,FLOW_ID)=0 %then %do;
|
%if %mf_existvarList(&inds,FLOW_ID)=0 %then %do;
|
||||||
retain FLOW_ID 0;
|
retain FLOW_ID 0;
|
||||||
%end;
|
%end;
|
||||||
|
/* https://github.com/sasjs/adapter/pull/845#issuecomment-2956589644 */
|
||||||
|
retain _omitSessionResults "false";
|
||||||
set &inds;
|
set &inds;
|
||||||
&dbg. putlog (_all_)(=);
|
&dbg. putlog (_all_)(=);
|
||||||
run;
|
run;
|
||||||
@@ -30538,6 +30982,117 @@ endsub;
|
|||||||
%end;
|
%end;
|
||||||
|
|
||||||
%mend mcf_string2file;/**
|
%mend mcf_string2file;/**
|
||||||
|
@file mx_createjob.sas
|
||||||
|
@brief Create a job in SAS 9, Viya or SASjs
|
||||||
|
@details Creates a Stored Process in SAS 9, a Job Execution Service in SAS
|
||||||
|
Viya, or a Stored Program on SASjs Server - depending on the executing
|
||||||
|
environment.
|
||||||
|
|
||||||
|
Usage:
|
||||||
|
|
||||||
|
%* compile macros ;
|
||||||
|
filename mc url "https://raw.githubusercontent.com/sasjs/core/main/all.sas";
|
||||||
|
%inc mc;
|
||||||
|
|
||||||
|
%* write some code;
|
||||||
|
filename ft15f001 temp;
|
||||||
|
parmcards4;
|
||||||
|
data example1;
|
||||||
|
set sashelp.class;
|
||||||
|
run;
|
||||||
|
;;;;
|
||||||
|
|
||||||
|
%* create the job;
|
||||||
|
%mx_createjob(path=/Public/app/jobs,name=myjob,replace=YES)
|
||||||
|
|
||||||
|
<h4> SAS Macros </h4>
|
||||||
|
@li mf_getplatform.sas
|
||||||
|
@li mm_createstp.sas
|
||||||
|
@li ms_createfile.sas
|
||||||
|
@li mv_createjob.sas
|
||||||
|
|
||||||
|
@param [in,out] path= The full folder path where the job will be created
|
||||||
|
@param [in,out] name= Job name. Avoid spaces.
|
||||||
|
@param [in] desc= The description of the job (optional)
|
||||||
|
@param [in] precode= Space separated list of filerefs, pointing to the code
|
||||||
|
that needs to be attached to the beginning of the job (optional)
|
||||||
|
@param [in] code= (ft15f001) Space seperated fileref(s) of the actual code to
|
||||||
|
be added
|
||||||
|
@param [in] replace= (YES) Select YES to replace any existing job in that
|
||||||
|
location
|
||||||
|
@param [in] mDebug= (0) set to 1 to show debug messages in the log
|
||||||
|
|
||||||
|
@author Allan Bowe
|
||||||
|
|
||||||
|
<h4> Related Macros </h4>
|
||||||
|
@li mx_createjob.test.sas
|
||||||
|
@li mx_createwebservice.sas
|
||||||
|
|
||||||
|
**/
|
||||||
|
|
||||||
|
%macro mx_createjob(path=HOME
|
||||||
|
,name=initJob
|
||||||
|
,precode=
|
||||||
|
,code=ft15f001
|
||||||
|
,desc=This job was created by the mx_createjob macro
|
||||||
|
,replace=YES
|
||||||
|
,mdebug=0
|
||||||
|
)/*/STORE SOURCE*/;
|
||||||
|
|
||||||
|
%if &syscc ge 4 %then %do;
|
||||||
|
%put syscc=&syscc - &sysmacroname will not execute in this state;
|
||||||
|
%return;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
/* combine precode and code into a single file */
|
||||||
|
%local tempref x fref freflist;
|
||||||
|
%let tempref=%mf_getuniquefileref();
|
||||||
|
%local work tmpfile;
|
||||||
|
%let work=%sysfunc(pathname(work));
|
||||||
|
%let tmpfile=&tempref..sas;
|
||||||
|
filename &tempref "&work/&tmpfile";
|
||||||
|
%let freflist=&precode &code ;
|
||||||
|
%do x=1 %to %sysfunc(countw(&freflist));
|
||||||
|
%let fref=%scan(&freflist,&x);
|
||||||
|
%put &sysmacroname: adding &fref;
|
||||||
|
data _null_;
|
||||||
|
file &tempref lrecl=3000 termstr=crlf mod;
|
||||||
|
infile &fref lrecl=3000;
|
||||||
|
input;
|
||||||
|
put _infile_;
|
||||||
|
run;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%local platform; %let platform=%mf_getplatform();
|
||||||
|
%if &platform=SASVIYA %then %do;
|
||||||
|
%if "&path"="HOME" %then %let path=/Users/&sysuserid/My Folder;
|
||||||
|
%mv_createjob(path=&path
|
||||||
|
,name=&name
|
||||||
|
,code=&tempref
|
||||||
|
,desc=&desc
|
||||||
|
,replace=&replace
|
||||||
|
)
|
||||||
|
%end;
|
||||||
|
%else %if &platform=SASJS %then %do;
|
||||||
|
%if "&path"="HOME" %then %let path=/Users/&_sasjs_username/My Folder;
|
||||||
|
%ms_createfile(&path/&name..sas
|
||||||
|
,inref=&tempref
|
||||||
|
,mdebug=&mdebug
|
||||||
|
)
|
||||||
|
%end;
|
||||||
|
%else %do;
|
||||||
|
%if "&path"="HOME" %then %let path=/User Folders/&_METAPERSON/My Folder;
|
||||||
|
%mm_createstp(stpname=&name
|
||||||
|
,filename=&tmpfile
|
||||||
|
,directory=&work
|
||||||
|
,tree=&path
|
||||||
|
,stpdesc=&desc
|
||||||
|
,mDebug=&mdebug
|
||||||
|
)
|
||||||
|
%end;
|
||||||
|
filename &tempref clear;
|
||||||
|
%mend mx_createjob;
|
||||||
|
/**
|
||||||
@file mx_createwebservice.sas
|
@file mx_createwebservice.sas
|
||||||
@brief Create a web service in SAS 9, Viya or SASjs
|
@brief Create a web service in SAS 9, Viya or SASjs
|
||||||
@details Creates a SASJS ready Stored Process in SAS 9, a Job Execution
|
@details Creates a SASJS ready Stored Process in SAS 9, a Job Execution
|
||||||
@@ -30587,6 +31142,9 @@ Usage:
|
|||||||
|
|
||||||
@author Allan Bowe
|
@author Allan Bowe
|
||||||
|
|
||||||
|
<h4> Related Macros </h4>
|
||||||
|
@li mx_createjob.sas
|
||||||
|
|
||||||
**/
|
**/
|
||||||
|
|
||||||
%macro mx_createwebservice(path=HOME
|
%macro mx_createwebservice(path=HOME
|
||||||
|
|||||||
@@ -25,8 +25,9 @@
|
|||||||
%local dsid vnum rc schema;
|
%local dsid vnum rc schema;
|
||||||
/* in case the parameter is a libref.tablename, pull off just the libref */
|
/* in case the parameter is a libref.tablename, pull off just the libref */
|
||||||
%let libref = %upcase(%scan(&libref, 1, %str(.)));
|
%let libref = %upcase(%scan(&libref, 1, %str(.)));
|
||||||
|
/* sysname can be 'Schema/Owner' or just 'Schema' (eg snowflake) */
|
||||||
%let dsid=%sysfunc(open(sashelp.vlibnam(where=(
|
%let dsid=%sysfunc(open(sashelp.vlibnam(where=(
|
||||||
libname="%upcase(&libref)" and sysname='Schema/Owner'
|
libname="%upcase(&libref)" and sysname=:'Schema'
|
||||||
)),i));
|
)),i));
|
||||||
%if (&dsid ^= 0) %then %do;
|
%if (&dsid ^= 0) %then %do;
|
||||||
%let vnum=%sysfunc(varnum(&dsid,SYSVALUE));
|
%let vnum=%sysfunc(varnum(&dsid,SYSVALUE));
|
||||||
|
|||||||
@@ -26,7 +26,7 @@
|
|||||||
@author Allan Bowe
|
@author Allan Bowe
|
||||||
**/
|
**/
|
||||||
|
|
||||||
%macro mf_getuniquelibref(prefix=mclib,maxtries=1000);
|
%macro mf_getuniquelibref(prefix=mc,maxtries=1000);
|
||||||
%local x;
|
%local x;
|
||||||
|
|
||||||
%if ( %length(&prefix) gt 7 ) %then %do;
|
%if ( %length(&prefix) gt 7 ) %then %do;
|
||||||
|
|||||||
@@ -1,7 +1,8 @@
|
|||||||
/**
|
/**
|
||||||
@file
|
@file
|
||||||
@brief Retrieves a value from a dataset. If no filter supplied, then first
|
@brief Retrieves a value from a dataset. Returned value is fetched from the
|
||||||
record is used.
|
'fetchobs=' record (row 1 by default), after applying the optional filter.
|
||||||
|
|
||||||
@details Be sure to <code>%quote()</code> your where clause. Example usage:
|
@details Be sure to <code>%quote()</code> your where clause. Example usage:
|
||||||
|
|
||||||
%put %mf_getvalue(sashelp.class,name,filter=%quote(age=15));
|
%put %mf_getvalue(sashelp.class,name,filter=%quote(age=15));
|
||||||
@@ -16,21 +17,39 @@
|
|||||||
@param [in] libds dataset to query
|
@param [in] libds dataset to query
|
||||||
@param [in] variable the variable which contains the value to return.
|
@param [in] variable the variable which contains the value to return.
|
||||||
@param [in] filter= (1) contents of where clause
|
@param [in] filter= (1) contents of where clause
|
||||||
|
@param [in] fetchobs= (1) observation to fetch. NB: Filter applies first.
|
||||||
|
|
||||||
@version 9.2
|
@version 9.2
|
||||||
@author Allan Bowe
|
@author Allan Bowe
|
||||||
**/
|
**/
|
||||||
|
|
||||||
%macro mf_getvalue(libds,variable,filter=1
|
%macro mf_getvalue(libds,variable,filter=1,fetchobs=1
|
||||||
)/*/STORE SOURCE*/;
|
)/*/STORE SOURCE*/;
|
||||||
%if %mf_getattrn(&libds,NLOBS)>0 %then %do;
|
%local dsid;
|
||||||
%local dsid rc &variable;
|
|
||||||
%let dsid=%sysfunc(open(&libds(where=(&filter))));
|
%let dsid=%sysfunc(open(&libds(where=(&filter))));
|
||||||
|
%if (&dsid) %then %do;
|
||||||
|
%local rc &variable;
|
||||||
%syscall set(dsid);
|
%syscall set(dsid);
|
||||||
%let rc = %sysfunc(fetch(&dsid));
|
%let rc = %sysfunc(fetchobs(&dsid,&fetchobs));
|
||||||
|
%if (&rc ne 0) %then %do;
|
||||||
|
%put NOTE: Problem reading obs &fetchobs from &libds..;
|
||||||
|
%put %sysfunc(sysmsg());
|
||||||
|
/* Coerce an rc value of -1 (read past end of data) to a 4
|
||||||
|
that, in SAS condition code terms, represents the sysmsg
|
||||||
|
w@rning it generates. */
|
||||||
|
%if &rc eq -1 %then %let rc = 4;
|
||||||
|
/* And update SYSCC if the &rc value is higher */
|
||||||
|
%let syscc = %sysfunc(max(&syscc,&rc));
|
||||||
|
%end;
|
||||||
%let rc = %sysfunc(close(&dsid));
|
%let rc = %sysfunc(close(&dsid));
|
||||||
|
|
||||||
%trim(&&&variable)
|
%trim(&&&variable)
|
||||||
|
|
||||||
%end;
|
%end;
|
||||||
%mend mf_getvalue;
|
%else %do;
|
||||||
|
%put %sysfunc(sysmsg());
|
||||||
|
%let syscc = %sysfunc(max(&syscc,%sysfunc(sysrc())));
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%mend mf_getvalue;
|
||||||
|
|||||||
@@ -73,10 +73,6 @@
|
|||||||
ignorelist=,
|
ignorelist=,
|
||||||
outds=work.test_results
|
outds=work.test_results
|
||||||
)/*/STORE SOURCE*/;
|
)/*/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
|
* this sets up the global vars, it will also enter STRICT mode. If this
|
||||||
* behaviour is not desired, simply initiate the following global macro
|
* behaviour is not desired, simply initiate the following global macro
|
||||||
@@ -84,6 +80,10 @@
|
|||||||
*/
|
*/
|
||||||
%mp_init()
|
%mp_init()
|
||||||
|
|
||||||
|
%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);
|
||||||
|
|
||||||
/* get current variables */
|
/* get current variables */
|
||||||
%if &action=SNAPSHOT %then %do;
|
%if &action=SNAPSHOT %then %do;
|
||||||
proc sql;
|
proc sql;
|
||||||
|
|||||||
@@ -148,6 +148,6 @@ data _null_;
|
|||||||
run;
|
run;
|
||||||
|
|
||||||
/* END */
|
/* END */
|
||||||
%put &sysmacroname took %sysevalf(%sysfunc(datetime())-&dttm) seconds to run;
|
/* %put &sysmacroname took %sysevalf(%sysfunc(datetime())-&dttm) secs to run; */
|
||||||
|
|
||||||
%mend mp_replace;
|
%mend mp_replace;
|
||||||
|
|||||||
@@ -36,6 +36,7 @@
|
|||||||
@li mf_islibds.sas
|
@li mf_islibds.sas
|
||||||
@li mf_wordsinstr1butnotstr2.sas
|
@li mf_wordsinstr1butnotstr2.sas
|
||||||
@li mp_abort.sas
|
@li mp_abort.sas
|
||||||
|
@li mp_ds2squeeze.sas
|
||||||
|
|
||||||
<h4> Related Macros </h4>
|
<h4> Related Macros </h4>
|
||||||
@li mddl_dc_difftable.sas
|
@li mddl_dc_difftable.sas
|
||||||
|
|||||||
52
meta/mm_assigndirectlib.sas
Executable file → Normal file
52
meta/mm_assigndirectlib.sas
Executable file → Normal file
@@ -213,7 +213,8 @@ run;
|
|||||||
%else %if &engine=ODBC %then %do;
|
%else %if &engine=ODBC %then %do;
|
||||||
%&mD.put NOTE: Retrieving ODBC connection details;
|
%&mD.put NOTE: Retrieving ODBC connection details;
|
||||||
data _null_;
|
data _null_;
|
||||||
length connx_uri conprop_uri value datasource up_uri schema domprop_uri authdomain $256.;
|
length connx_uri conprop_uri value datasource up_uri schema domprop_uri
|
||||||
|
authdomain $256.;
|
||||||
call missing (of _all_);
|
call missing (of _all_);
|
||||||
/* get source connection ID */
|
/* get source connection ID */
|
||||||
rc=metadata_getnasn("&liburi",'LibraryConnection',1,connx_uri);
|
rc=metadata_getnasn("&liburi",'LibraryConnection',1,connx_uri);
|
||||||
@@ -413,6 +414,55 @@ run;
|
|||||||
|
|
||||||
libname &libref SQLSVR datasrc=&path schema=&schema user="&user" pass="&pass";
|
libname &libref SQLSVR datasrc=&path schema=&schema user="&user" pass="&pass";
|
||||||
%end;
|
%end;
|
||||||
|
%else %if &engine=SASIOSNF or &engine=SNOW %then %do;
|
||||||
|
%&mD.put NOTE: Retrieving SNOW connection details;
|
||||||
|
data _null_;
|
||||||
|
length connx_uri conprop_uri value server up_uri schema domprop_uri
|
||||||
|
authdomain database $256.;
|
||||||
|
call missing (of _all_);
|
||||||
|
/* get source connection ID */
|
||||||
|
rc=metadata_getnasn("&liburi",'LibraryConnection',1,connx_uri);
|
||||||
|
/* get connection properties */
|
||||||
|
i=0;
|
||||||
|
do until (rc2<0);
|
||||||
|
i+1;
|
||||||
|
rc2=metadata_getnasn(connx_uri,'Properties',i,conprop_uri);
|
||||||
|
rc3=metadata_getattr(conprop_uri,'Name',value);
|
||||||
|
if value='Connection.DBMS.Property.SERVER.Name.xmlKey.txt' then do;
|
||||||
|
rc4=metadata_getattr(conprop_uri,'DefaultValue',server);
|
||||||
|
rc2=-1;
|
||||||
|
end;
|
||||||
|
end;
|
||||||
|
|
||||||
|
/* get auth domain */
|
||||||
|
autrc=metadata_getnasn(connx_uri,"Domain",1,domprop_uri);
|
||||||
|
arc=metadata_getattr(domprop_uri,"Name",authdomain);
|
||||||
|
if not missing(authdomain) then authdomain=cats('AUTHDOMAIN=',authdomain);
|
||||||
|
call symputx('authdomain',authdomain,'l');
|
||||||
|
|
||||||
|
/* get SCHEMA */
|
||||||
|
rc6=metadata_getnasn("&liburi",'UsingPackages',1,up_uri);
|
||||||
|
rc7=metadata_getattr(up_uri,'SchemaName',schema);
|
||||||
|
&mD.put rc= connx_uri= rc2= conprop_uri= rc3= value= rc4= server=
|
||||||
|
rc6= up_uri= rc7= schema=;
|
||||||
|
|
||||||
|
/* get database value */
|
||||||
|
prop='Connection.DBMS.Property.DB.Name.xmlKey.txt';
|
||||||
|
rc=metadata_getprop("&liburi",prop,database,"");
|
||||||
|
if database^='' then database='database='!!quote(trim(database));
|
||||||
|
call symputx('database',database,'l');
|
||||||
|
|
||||||
|
call symputx('snow_schema',schema,'l');
|
||||||
|
call symputx('snow_server',server,'l');
|
||||||
|
run;
|
||||||
|
|
||||||
|
libname &libref SNOW SERVER="&snow_server" SCHEMA=&snow_schema &authdomain
|
||||||
|
&database;
|
||||||
|
%if %length(&open_passthrough)>0 %then %do;
|
||||||
|
proc sql;
|
||||||
|
connect using &libref as &open_passthrough;
|
||||||
|
%end;
|
||||||
|
%end;
|
||||||
%else %if &engine=TERADATA %then %do;
|
%else %if &engine=TERADATA %then %do;
|
||||||
%put NOTE: Obtaining &engine library details;
|
%put NOTE: Obtaining &engine library details;
|
||||||
data _null;
|
data _null;
|
||||||
|
|||||||
@@ -5,7 +5,10 @@
|
|||||||
"ddl",
|
"ddl",
|
||||||
"fcmp",
|
"fcmp",
|
||||||
"lua",
|
"lua",
|
||||||
|
"meta",
|
||||||
|
"metax",
|
||||||
"server",
|
"server",
|
||||||
|
"viya",
|
||||||
"xplatform",
|
"xplatform",
|
||||||
"tests/base",
|
"tests/base",
|
||||||
"tests/ddlonly",
|
"tests/ddlonly",
|
||||||
@@ -42,7 +45,6 @@
|
|||||||
"deployScripts": []
|
"deployScripts": []
|
||||||
},
|
},
|
||||||
"macroFolders": [
|
"macroFolders": [
|
||||||
"viya",
|
|
||||||
"tests/viyaonly"
|
"tests/viyaonly"
|
||||||
],
|
],
|
||||||
"contextName": "SAS Job Execution compute context"
|
"contextName": "SAS Job Execution compute context"
|
||||||
@@ -56,8 +58,6 @@
|
|||||||
},
|
},
|
||||||
"appLoc": "/Shared Data/temp/macrocore",
|
"appLoc": "/Shared Data/temp/macrocore",
|
||||||
"macroFolders": [
|
"macroFolders": [
|
||||||
"meta",
|
|
||||||
"metax",
|
|
||||||
"tests/sas9only"
|
"tests/sas9only"
|
||||||
],
|
],
|
||||||
"programFolders": [],
|
"programFolders": [],
|
||||||
@@ -82,7 +82,6 @@
|
|||||||
"deployScripts": []
|
"deployScripts": []
|
||||||
},
|
},
|
||||||
"macroFolders": [
|
"macroFolders": [
|
||||||
"server",
|
|
||||||
"tests/serveronly"
|
"tests/serveronly"
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
|||||||
91
tests/base/mf_getvalue.test.sas
Normal file
91
tests/base/mf_getvalue.test.sas
Normal file
@@ -0,0 +1,91 @@
|
|||||||
|
/**
|
||||||
|
@file
|
||||||
|
@brief Testing mf_getvalue macro
|
||||||
|
|
||||||
|
<h4> SAS Macros </h4>
|
||||||
|
@li mf_getvalue.sas
|
||||||
|
@li mp_assert.sas
|
||||||
|
@li mp_assertscope.sas
|
||||||
|
|
||||||
|
**/
|
||||||
|
|
||||||
|
data work.test_data;
|
||||||
|
do i = 1 to 10;
|
||||||
|
output;
|
||||||
|
end;
|
||||||
|
stop;
|
||||||
|
run;
|
||||||
|
|
||||||
|
/* - Test 1 -
|
||||||
|
Get value from default first observation.
|
||||||
|
No filter.
|
||||||
|
*/
|
||||||
|
%mp_assertscope(SNAPSHOT)
|
||||||
|
%let test_value=%mf_getvalue(work.test_data,i);
|
||||||
|
%mp_assertscope(COMPARE,ignorelist=test_value)
|
||||||
|
|
||||||
|
%mp_assert(
|
||||||
|
iftrue=(&test_value=1 and &syscc eq 0),
|
||||||
|
desc=Basic test fetching value from default first obs,
|
||||||
|
outds=work.test_results
|
||||||
|
)
|
||||||
|
|
||||||
|
/* - Test 2 -
|
||||||
|
Get value from 10th observation.
|
||||||
|
No filter.
|
||||||
|
*/
|
||||||
|
%let test_value=%mf_getvalue(work.test_data,i,fetchobs=10);
|
||||||
|
%mp_assert(
|
||||||
|
iftrue=(&test_value=10 and &syscc eq 0),
|
||||||
|
desc=Test fetching value from specifically the 10th row,
|
||||||
|
outds=work.test_results
|
||||||
|
)
|
||||||
|
|
||||||
|
/* - Test 3 -
|
||||||
|
Get value from default first observation.
|
||||||
|
With filter.
|
||||||
|
*/
|
||||||
|
%let test_value=%mf_getvalue(work.test_data,i,filter=(i>4));
|
||||||
|
%mp_assert(
|
||||||
|
iftrue=(&test_value=5 and &syscc eq 0),
|
||||||
|
desc=Test fetching value from default row of filtered data,
|
||||||
|
outds=work.test_results
|
||||||
|
)
|
||||||
|
|
||||||
|
/* - Test 4 -
|
||||||
|
Get value from specified observation.
|
||||||
|
With filter.
|
||||||
|
*/
|
||||||
|
%let test_value=%mf_getvalue(work.test_data,i,filter=(i>4),fetchobs=5);
|
||||||
|
%mp_assert(
|
||||||
|
iftrue=(&test_value=9 and &syscc eq 0),
|
||||||
|
desc=Test fetching value from 5th row of filtered data,
|
||||||
|
outds=work.test_results
|
||||||
|
)
|
||||||
|
|
||||||
|
/* - Test 5 -
|
||||||
|
Get value from default observation.
|
||||||
|
Filter removes all rows. This simulates providing an empty dataset
|
||||||
|
or specifying an observation number beyond the set returned by the filter.
|
||||||
|
*/
|
||||||
|
%let test_value=%mf_getvalue(work.test_data,i,filter=(i>10));
|
||||||
|
%mp_assert(
|
||||||
|
iftrue=(&test_value=%str() and &syscc eq 4),
|
||||||
|
desc=Test fetching value from 1st row of empty (filtered) data,
|
||||||
|
outds=work.test_results
|
||||||
|
)
|
||||||
|
|
||||||
|
%let syscc = 0; /* Reset w@rning To ensure confidence in next test */
|
||||||
|
|
||||||
|
/* - Test 6 -
|
||||||
|
Get value from default observation.
|
||||||
|
Dataset does not exist.
|
||||||
|
*/
|
||||||
|
%let test_value=%mf_getvalue(work.test_data_x,i);
|
||||||
|
%mp_assert(
|
||||||
|
iftrue=(&test_value=%str() and &syscc gt 0),
|
||||||
|
desc=Test fetching value from 1st row of non-existent data,
|
||||||
|
outds=work.test_results
|
||||||
|
)
|
||||||
|
|
||||||
|
%let syscc = 0; /* To reset expected error and allow test job to exit clean. */
|
||||||
30
tests/sas9only/mm_assigndirectlib.test.sas
Normal file
30
tests/sas9only/mm_assigndirectlib.test.sas
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
/**
|
||||||
|
@file
|
||||||
|
@brief Testing mm_assigndirectlib macro
|
||||||
|
@details A valid library must first be configured in metadata.
|
||||||
|
To test success, also define a table for which we can test the existence.
|
||||||
|
This is a unit test - not part of the full test run, as it would be a
|
||||||
|
lot of overhead to create an external DB and metadata setup each time.
|
||||||
|
|
||||||
|
<h4> SAS Macros </h4>
|
||||||
|
@li mf_existds.sas
|
||||||
|
@li mp_assert.sas
|
||||||
|
@li mp_assertscope.sas
|
||||||
|
@li mm_assigndirectlib.sas
|
||||||
|
|
||||||
|
**/
|
||||||
|
|
||||||
|
%let runtest=0;
|
||||||
|
%let libref=XXX;
|
||||||
|
%let ds=XXXX;
|
||||||
|
|
||||||
|
|
||||||
|
%mp_assertscope(SNAPSHOT)
|
||||||
|
%mm_assigndirectlib(&libref)
|
||||||
|
%mp_assertscope(COMPARE)
|
||||||
|
|
||||||
|
|
||||||
|
%mp_assert(
|
||||||
|
iftrue=(&runtest=1 and %mf_existds(&libref..&ds)),
|
||||||
|
desc=Check if &libref..&ds exists
|
||||||
|
)
|
||||||
96
tests/viyaonly/mv_getviyafileextparms.test.sas
Normal file
96
tests/viyaonly/mv_getviyafileextparms.test.sas
Normal file
@@ -0,0 +1,96 @@
|
|||||||
|
/**
|
||||||
|
@file
|
||||||
|
@brief Testing mv_getviyafileextparms macro
|
||||||
|
|
||||||
|
<h4> SAS Macros </h4>
|
||||||
|
@li mf_isblank.sas
|
||||||
|
@li mp_assert.sas
|
||||||
|
@li mp_assertscope.sas
|
||||||
|
@li mv_getviyafileextparms.sas
|
||||||
|
|
||||||
|
**/
|
||||||
|
|
||||||
|
options mprint;
|
||||||
|
|
||||||
|
%let mvarIgnoreList =
|
||||||
|
MC0_JADP1LEN MC0_JADP2LEN MC0_JADP3LEN MC0_JADPNUM MC0_JADVLEN
|
||||||
|
SASJSPROCESSMODE SASJS_STPSRV_HEADER_LOC;
|
||||||
|
|
||||||
|
%put TEST 1 - Test with common extension, requesting only typeDefName parameter;
|
||||||
|
%mp_assertscope(SNAPSHOT)
|
||||||
|
%mv_getviyafileextparms(ext=txt, typeDefNameVar=viyaTypeDefName)
|
||||||
|
%mp_assertscope(COMPARE
|
||||||
|
,ignorelist=&mvarIgnoreList viyaTypeDefName
|
||||||
|
)
|
||||||
|
|
||||||
|
%mp_assert(
|
||||||
|
iftrue=(not %mf_isBlank(&viyaTypeDefName)),
|
||||||
|
desc=Check the requested macro variable viyaTypeDefName is not blank.
|
||||||
|
)
|
||||||
|
|
||||||
|
%put TEST 2 - Test with common extension, requesting only properties parameter;
|
||||||
|
%mp_assertscope(SNAPSHOT)
|
||||||
|
%mv_getviyafileextparms(ext=html, propertiesVar=viyaProperties)
|
||||||
|
%mp_assertscope(COMPARE
|
||||||
|
,ignorelist=&mvarIgnoreList viyaProperties
|
||||||
|
)
|
||||||
|
|
||||||
|
%mp_assert(
|
||||||
|
iftrue=(not %mf_isBlank(%superq(viyaProperties))),
|
||||||
|
desc=Check the requested macro variable viyaProperties is not blank.
|
||||||
|
)
|
||||||
|
|
||||||
|
%put TEST 3 - Test with common extension, requesting only mediaType parameter;
|
||||||
|
%mp_assertscope(SNAPSHOT)
|
||||||
|
%mv_getviyafileextparms(ext=mp3, mediaTypeVar=viyaMediaType)
|
||||||
|
%mp_assertscope(COMPARE
|
||||||
|
,ignorelist=&mvarIgnoreList viyaMediaType
|
||||||
|
)
|
||||||
|
|
||||||
|
%mp_assert(
|
||||||
|
iftrue=(not %mf_isBlank(&viyaMediaType)),
|
||||||
|
desc=Check the requested macro variable viyaMediaType is not blank.
|
||||||
|
)
|
||||||
|
|
||||||
|
%put TEST 4 - Test with common extension, requesting all parameters;
|
||||||
|
%mp_assertscope(SNAPSHOT)
|
||||||
|
%mv_getviyafileextparms(
|
||||||
|
ext=css,
|
||||||
|
typeDefNameVar=cssViyaTypeDefName,
|
||||||
|
propertiesVar=cssViyaProperties,
|
||||||
|
mediaTypeVar=cssViyaMediaType
|
||||||
|
)
|
||||||
|
%mp_assertscope(COMPARE
|
||||||
|
,ignorelist=
|
||||||
|
&mvarIgnoreList cssViyaTypeDefName cssViyaProperties cssViyaMediaType
|
||||||
|
)
|
||||||
|
|
||||||
|
%mp_assert(
|
||||||
|
iftrue=(not ( %mf_isBlank(&cssViyaTypeDefName) or
|
||||||
|
%mf_isBlank(%superq(cssViyaProperties)) or
|
||||||
|
%mf_isBlank(&cssViyaMediaType) ) ),
|
||||||
|
desc=Check a full set of requested macro variables are not blank.
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
%put TEST 5 - Test with invalid extension - requested parameters will be blank;
|
||||||
|
%mp_assertscope(SNAPSHOT)
|
||||||
|
%mv_getviyafileextparms(
|
||||||
|
ext=xxxINVALIDxxx,
|
||||||
|
typeDefNameVar=invalidTypeDefName,
|
||||||
|
propertiesVar=invalidProperties,
|
||||||
|
mediaTypeVar=invalidMediaType
|
||||||
|
)
|
||||||
|
%mp_assertscope(COMPARE
|
||||||
|
,ignorelist=
|
||||||
|
&mvarIgnoreList invalidTypeDefName invalidProperties invalidMediaType
|
||||||
|
)
|
||||||
|
|
||||||
|
%mp_assert(
|
||||||
|
iftrue=(
|
||||||
|
%mf_isBlank(&invalidTypeDefName) and
|
||||||
|
%mf_isBlank(%superq(invalidProperties)) and
|
||||||
|
%mf_isBlank(&invalidMediaType)
|
||||||
|
),
|
||||||
|
desc=Check the requested macro variables are all blank.
|
||||||
|
)
|
||||||
304
tests/x-platform/mx_createjob.test.sas
Normal file
304
tests/x-platform/mx_createjob.test.sas
Normal file
@@ -0,0 +1,304 @@
|
|||||||
|
/**
|
||||||
|
@file
|
||||||
|
@brief Testing mx_createjob.sas macro
|
||||||
|
|
||||||
|
Be sure to run <code>%let mcTestAppLoc=/Public/temp/macrocore;</code> when
|
||||||
|
running in Studio
|
||||||
|
|
||||||
|
<h4> SAS Macros </h4>
|
||||||
|
@li mx_createjob.sas
|
||||||
|
@li mp_assert.sas
|
||||||
|
@li mf_getuniquefileref.sas
|
||||||
|
@li mp_assertscope.sas
|
||||||
|
|
||||||
|
**/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test 1 - Basic job creation with default parameters
|
||||||
|
* Also checking for scope leakage
|
||||||
|
*/
|
||||||
|
filename ft15f001 temp;
|
||||||
|
parmcards4;
|
||||||
|
data example1;
|
||||||
|
set sashelp.class;
|
||||||
|
run;
|
||||||
|
%put Job executed successfully;
|
||||||
|
;;;;
|
||||||
|
%mp_assertscope(SNAPSHOT)
|
||||||
|
%mx_createjob(path=&mcTestAppLoc/jobs,name=testjob1,replace=YES)
|
||||||
|
%mp_assertscope(COMPARE)
|
||||||
|
|
||||||
|
%mp_assert(
|
||||||
|
iftrue=(&syscc=0),
|
||||||
|
desc=Test 1: No errors after basic job creation,
|
||||||
|
outds=work.test_results
|
||||||
|
)
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test 2 - Job creation with custom description
|
||||||
|
*/
|
||||||
|
filename ft15f001 temp;
|
||||||
|
parmcards4;
|
||||||
|
data example2;
|
||||||
|
set sashelp.cars;
|
||||||
|
run;
|
||||||
|
;;;;
|
||||||
|
%mx_createjob(
|
||||||
|
path=&mcTestAppLoc/jobs,
|
||||||
|
name=testjob2,
|
||||||
|
desc=Custom job description for testing,
|
||||||
|
replace=YES
|
||||||
|
)
|
||||||
|
|
||||||
|
%mp_assert(
|
||||||
|
iftrue=(&syscc=0),
|
||||||
|
desc=Test 2: Job created with custom description,
|
||||||
|
outds=work.test_results
|
||||||
|
)
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test 3 - Job creation with precode
|
||||||
|
*/
|
||||||
|
filename precode1 temp;
|
||||||
|
data _null_;
|
||||||
|
file precode1;
|
||||||
|
put '%let testvar=PreCodeValue;';
|
||||||
|
put '%put &=testvar;';
|
||||||
|
run;
|
||||||
|
|
||||||
|
filename ft15f001 temp;
|
||||||
|
parmcards4;
|
||||||
|
data example3;
|
||||||
|
set sashelp.class;
|
||||||
|
precode_var="&testvar";
|
||||||
|
run;
|
||||||
|
;;;;
|
||||||
|
%mx_createjob(
|
||||||
|
path=&mcTestAppLoc/jobs,
|
||||||
|
name=testjob3,
|
||||||
|
precode=precode1,
|
||||||
|
replace=YES
|
||||||
|
)
|
||||||
|
|
||||||
|
%mp_assert(
|
||||||
|
iftrue=(&syscc=0),
|
||||||
|
desc=Test 3: Job created with precode parameter,
|
||||||
|
outds=work.test_results
|
||||||
|
)
|
||||||
|
|
||||||
|
filename precode1 clear;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test 4 - Job creation with multiple code filerefs
|
||||||
|
*/
|
||||||
|
%let code1=%mf_getuniquefileref();
|
||||||
|
%let code2=%mf_getuniquefileref();
|
||||||
|
|
||||||
|
filename &code1 temp;
|
||||||
|
data _null_;
|
||||||
|
file &code1;
|
||||||
|
put 'data work.part1;';
|
||||||
|
put ' set sashelp.class(obs=5);';
|
||||||
|
put 'run;';
|
||||||
|
run;
|
||||||
|
|
||||||
|
filename &code2 temp;
|
||||||
|
data _null_;
|
||||||
|
file &code2;
|
||||||
|
put 'data work.part2;';
|
||||||
|
put ' set sashelp.class(firstobs=6);';
|
||||||
|
put 'run;';
|
||||||
|
run;
|
||||||
|
|
||||||
|
%mx_createjob(
|
||||||
|
path=&mcTestAppLoc/jobs,
|
||||||
|
name=testjob4,
|
||||||
|
code=&code1 &code2,
|
||||||
|
replace=YES
|
||||||
|
)
|
||||||
|
|
||||||
|
%mp_assert(
|
||||||
|
iftrue=(&syscc=0),
|
||||||
|
desc=Test 4: Job created with multiple code filerefs,
|
||||||
|
outds=work.test_results
|
||||||
|
)
|
||||||
|
|
||||||
|
filename &code1 clear;
|
||||||
|
filename &code2 clear;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test 5 - Job creation with both precode and multiple code files
|
||||||
|
*/
|
||||||
|
%let pre1=%mf_getuniquefileref();
|
||||||
|
%let pre2=%mf_getuniquefileref();
|
||||||
|
%let main1=%mf_getuniquefileref();
|
||||||
|
|
||||||
|
filename &pre1 temp;
|
||||||
|
data _null_;
|
||||||
|
file &pre1;
|
||||||
|
put '%let globalvar1=Value1;';
|
||||||
|
run;
|
||||||
|
|
||||||
|
filename &pre2 temp;
|
||||||
|
data _null_;
|
||||||
|
file &pre2;
|
||||||
|
put '%let globalvar2=Value2;';
|
||||||
|
run;
|
||||||
|
|
||||||
|
filename &main1 temp;
|
||||||
|
data _null_;
|
||||||
|
file &main1;
|
||||||
|
put 'data work.combined;';
|
||||||
|
put ' var1="&globalvar1";';
|
||||||
|
put ' var2="&globalvar2";';
|
||||||
|
put ' output;';
|
||||||
|
put 'run;';
|
||||||
|
run;
|
||||||
|
|
||||||
|
%mx_createjob(
|
||||||
|
path=&mcTestAppLoc/jobs,
|
||||||
|
name=testjob5,
|
||||||
|
precode=&pre1 &pre2,
|
||||||
|
code=&main1,
|
||||||
|
desc=Job with multiple precode and code files,
|
||||||
|
replace=YES
|
||||||
|
)
|
||||||
|
|
||||||
|
%mp_assert(
|
||||||
|
iftrue=(&syscc=0),
|
||||||
|
desc=Test 5: Job created with multiple precode and code files,
|
||||||
|
outds=work.test_results
|
||||||
|
)
|
||||||
|
|
||||||
|
filename &pre1 clear;
|
||||||
|
filename &pre2 clear;
|
||||||
|
filename &main1 clear;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test 6 - Job creation with special characters in code
|
||||||
|
*/
|
||||||
|
filename ft15f001 temp;
|
||||||
|
parmcards4;
|
||||||
|
data example6;
|
||||||
|
length text $200;
|
||||||
|
text='Special chars: & % $ # @ !';
|
||||||
|
output;
|
||||||
|
text="Quotes: 'single' and ""double""";
|
||||||
|
output;
|
||||||
|
run;
|
||||||
|
%put Test with special characters;
|
||||||
|
;;;;
|
||||||
|
%mx_createjob(
|
||||||
|
path=&mcTestAppLoc/jobs,
|
||||||
|
name=testjob6,
|
||||||
|
desc=Job with special characters in code,
|
||||||
|
replace=YES
|
||||||
|
)
|
||||||
|
|
||||||
|
%mp_assert(
|
||||||
|
iftrue=(&syscc=0),
|
||||||
|
desc=Test 6: Job created with special characters in code,
|
||||||
|
outds=work.test_results
|
||||||
|
)
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test 7 - Job creation with macro code
|
||||||
|
*/
|
||||||
|
filename ft15f001 temp;
|
||||||
|
parmcards4;
|
||||||
|
%macro testmacro();
|
||||||
|
data example7;
|
||||||
|
set sashelp.class;
|
||||||
|
where age > 12;
|
||||||
|
run;
|
||||||
|
%mend testmacro;
|
||||||
|
|
||||||
|
%testmacro()
|
||||||
|
;;;;
|
||||||
|
%mx_createjob(
|
||||||
|
path=&mcTestAppLoc/jobs,
|
||||||
|
name=testjob7,
|
||||||
|
desc=Job containing macro definitions,
|
||||||
|
replace=YES
|
||||||
|
)
|
||||||
|
|
||||||
|
%mp_assert(
|
||||||
|
iftrue=(&syscc=0),
|
||||||
|
desc=Test 7: Job created with macro code,
|
||||||
|
outds=work.test_results
|
||||||
|
)
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test 8 - Job creation with empty code (edge case)
|
||||||
|
*/
|
||||||
|
filename ft15f001 temp;
|
||||||
|
data _null_;
|
||||||
|
file ft15f001;
|
||||||
|
put '/* Empty job for testing */';
|
||||||
|
run;
|
||||||
|
|
||||||
|
%mx_createjob(
|
||||||
|
path=&mcTestAppLoc/jobs,
|
||||||
|
name=testjob8,
|
||||||
|
desc=Job with minimal code,
|
||||||
|
replace=YES
|
||||||
|
)
|
||||||
|
|
||||||
|
%mp_assert(
|
||||||
|
iftrue=(&syscc=0),
|
||||||
|
desc=Test 8: Job created with minimal code,
|
||||||
|
outds=work.test_results
|
||||||
|
)
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test 9 - Job creation with long code block
|
||||||
|
*/
|
||||||
|
filename ft15f001 temp;
|
||||||
|
data _null_;
|
||||||
|
file ft15f001;
|
||||||
|
put 'data work.longtest;';
|
||||||
|
do i=1 to 50;
|
||||||
|
put ' var' i +(-1) '=' i ';';
|
||||||
|
end;
|
||||||
|
put ' output;';
|
||||||
|
put 'run;';
|
||||||
|
run;
|
||||||
|
|
||||||
|
%mx_createjob(
|
||||||
|
path=&mcTestAppLoc/jobs,
|
||||||
|
name=testjob9,
|
||||||
|
desc=Job with many variables,
|
||||||
|
replace=YES
|
||||||
|
)
|
||||||
|
|
||||||
|
%mp_assert(
|
||||||
|
iftrue=(&syscc=0),
|
||||||
|
desc=Test 9: Job created with long code block,
|
||||||
|
outds=work.test_results
|
||||||
|
)
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test 10 - Replace existing job (replace=YES)
|
||||||
|
*/
|
||||||
|
filename ft15f001 temp;
|
||||||
|
parmcards4;
|
||||||
|
data example10_v1;
|
||||||
|
set sashelp.class;
|
||||||
|
run;
|
||||||
|
;;;;
|
||||||
|
%mx_createjob(path=&mcTestAppLoc/jobs,name=testjob10,replace=YES)
|
||||||
|
|
||||||
|
/* Now replace it */
|
||||||
|
filename ft15f001 temp;
|
||||||
|
parmcards4;
|
||||||
|
data example10_v2;
|
||||||
|
set sashelp.cars;
|
||||||
|
run;
|
||||||
|
;;;;
|
||||||
|
%mx_createjob(path=&mcTestAppLoc/jobs,name=testjob10,replace=YES)
|
||||||
|
|
||||||
|
%mp_assert(
|
||||||
|
iftrue=(&syscc=0),
|
||||||
|
desc=Test 10: Job replaced successfully with replace=YES,
|
||||||
|
outds=work.test_results
|
||||||
|
)
|
||||||
@@ -3,10 +3,10 @@
|
|||||||
@brief Testing mx_testservice.sas macro
|
@brief Testing mx_testservice.sas macro
|
||||||
|
|
||||||
Be sure to run <code>%let mcTestAppLoc=/Public/temp/macrocore;</code> when
|
Be sure to run <code>%let mcTestAppLoc=/Public/temp/macrocore;</code> when
|
||||||
runnin in Studio
|
running in Studio
|
||||||
|
|
||||||
<h4> SAS Macros </h4>
|
<h4> SAS Macros </h4>
|
||||||
@li mp_createwebservice.sas
|
@li mx_createwebservice.sas
|
||||||
@li mx_testservice.sas
|
@li mx_testservice.sas
|
||||||
@li mp_assert.sas
|
@li mp_assert.sas
|
||||||
|
|
||||||
|
|||||||
@@ -33,16 +33,23 @@
|
|||||||
msg=Cannot enter mfv_existfolder.sas with syscc=&syscc
|
msg=Cannot enter mfv_existfolder.sas with syscc=&syscc
|
||||||
)
|
)
|
||||||
|
|
||||||
%local fref rc;
|
%local fref rc var;
|
||||||
%let fref=%mf_getuniquefileref();
|
%let fref=%mf_getuniquefileref();
|
||||||
|
|
||||||
%if %sysfunc(filename(fref,,filesrvc,folderPath="&path"))=0 %then %do;
|
%if %sysfunc(filename(fref,,filesrvc,folderPath="&path"))=0 %then %do;
|
||||||
1
|
1
|
||||||
|
%let var=_FILESRVC_&fref._URI;
|
||||||
%let rc=%sysfunc(filename(fref));
|
%let rc=%sysfunc(filename(fref));
|
||||||
|
%symdel &var;
|
||||||
%end;
|
%end;
|
||||||
%else %do;
|
%else %do;
|
||||||
0
|
0
|
||||||
%let syscc=0;
|
%let syscc=0;
|
||||||
%end;
|
%end;
|
||||||
|
|
||||||
|
%mf_abort(
|
||||||
|
iftrue=(&syscc ne 0),
|
||||||
|
msg=Cannot leave mfv_existfolder.sas with syscc=&syscc
|
||||||
|
)
|
||||||
|
|
||||||
%mend mfv_existfolder;
|
%mend mfv_existfolder;
|
||||||
@@ -49,4 +49,8 @@
|
|||||||
%let syscc=0;
|
%let syscc=0;
|
||||||
%end;
|
%end;
|
||||||
|
|
||||||
|
%mf_abort(
|
||||||
|
iftrue=(&syscc ne 0),
|
||||||
|
msg=Cannot leave &sysmacroname with syscc=&syscc
|
||||||
|
)
|
||||||
%mend mfv_getpathuri;
|
%mend mfv_getpathuri;
|
||||||
@@ -69,6 +69,7 @@
|
|||||||
@li mp_base64copy.sas
|
@li mp_base64copy.sas
|
||||||
@li mp_replace.sas
|
@li mp_replace.sas
|
||||||
@li mv_createfolder.sas
|
@li mv_createfolder.sas
|
||||||
|
@li mv_getviyafileextparms.sas
|
||||||
|
|
||||||
<h4> Related Macros</h4>
|
<h4> Related Macros</h4>
|
||||||
@li mv_createfile.sas
|
@li mv_createfile.sas
|
||||||
@@ -95,6 +96,11 @@
|
|||||||
%end;
|
%end;
|
||||||
%else %let dbg=*;
|
%else %let dbg=*;
|
||||||
|
|
||||||
|
%mp_abort(
|
||||||
|
iftrue=(&syscc ne 0),
|
||||||
|
msg=Cannot enter &sysmacroname with syscc=&syscc
|
||||||
|
)
|
||||||
|
|
||||||
%local oauth_bearer;
|
%local oauth_bearer;
|
||||||
%if &grant_type=detect %then %do;
|
%if &grant_type=detect %then %do;
|
||||||
%if %symexist(&access_token_var) %then %let grant_type=authorization_code;
|
%if %symexist(&access_token_var) %then %let grant_type=authorization_code;
|
||||||
@@ -148,7 +154,7 @@
|
|||||||
|
|
||||||
options noquotelenmax;
|
options noquotelenmax;
|
||||||
%local base_uri; /* location of rest apis */
|
%local base_uri; /* location of rest apis */
|
||||||
%let base_uri=%mf_getplatform(VIYARESTAPI);
|
%let base_uri=%trim(%mf_getplatform(VIYARESTAPI));
|
||||||
|
|
||||||
/* create folder if it does not already exist */
|
/* create folder if it does not already exist */
|
||||||
%local folderds self_uri;
|
%local folderds self_uri;
|
||||||
@@ -167,11 +173,16 @@ run;
|
|||||||
/* abort or delete if file already exists */
|
/* abort or delete if file already exists */
|
||||||
%let force=%upcase(&force);
|
%let force=%upcase(&force);
|
||||||
%local fileuri ;
|
%local fileuri ;
|
||||||
%let fileuri=%mfv_getpathuri(&path/&name);
|
%let fileuri=%trim(%mfv_getpathuri(&path/&name));
|
||||||
%mp_abort(iftrue=(%mf_isblank(&fileuri)=0 and &force ne YES)
|
%mp_abort(iftrue=(%mf_isblank(&fileuri)=0 and &force ne YES)
|
||||||
,mac=MV_CREATEFILE
|
,mac=MV_CREATEFILE
|
||||||
,msg=%str(File &path/&name already exists and force=&force)
|
,msg=%str(File &path/&name already exists and force=&force)
|
||||||
)
|
)
|
||||||
|
%mp_abort(
|
||||||
|
iftrue=(&syscc ne 0),
|
||||||
|
mac=MV_CREATEFILE182
|
||||||
|
msg=syscc=&syscc after mfv_getpathuri
|
||||||
|
)
|
||||||
|
|
||||||
%if %mf_isblank(&fileuri)=0 and &force=YES %then %do;
|
%if %mf_isblank(&fileuri)=0 and &force=YES %then %do;
|
||||||
proc http method="DELETE" url="&base_uri&fileuri" &oauth_bearer;
|
proc http method="DELETE" url="&base_uri&fileuri" &oauth_bearer;
|
||||||
@@ -189,7 +200,13 @@ run;
|
|||||||
|
|
||||||
%local url mimetype ext;
|
%local url mimetype ext;
|
||||||
%let url=&base_uri/files/files?parentFolderUri=&self_uri;
|
%let url=&base_uri/files/files?parentFolderUri=&self_uri;
|
||||||
%let ext=%upcase(%scan(&name,-1,.));
|
%let ext=%upcase(%trim(%scan(&name,-1,.)));
|
||||||
|
|
||||||
|
/* Get Viya file-extension details into some macro variables */
|
||||||
|
%mv_getViyaFileExtParms(&ext
|
||||||
|
,propertiesVar=viyaProperties
|
||||||
|
,typeDefNameVar=viyaTypeDefName
|
||||||
|
,mdebug=&mdebug);
|
||||||
|
|
||||||
/* fetch job info */
|
/* fetch job info */
|
||||||
%local fname1;
|
%local fname1;
|
||||||
@@ -202,12 +219,23 @@ proc http method='POST' out=&fname1 &oauth_bearer in=&fref
|
|||||||
%else %do;
|
%else %do;
|
||||||
ct="&ctype"
|
ct="&ctype"
|
||||||
%end;
|
%end;
|
||||||
%if "&ext"="HTML" or "&ext"="CSS" or "&ext"="JS" or "&ext"="PNG"
|
|
||||||
or "&ext"="SVG" %then %do;
|
/* typeDefName */
|
||||||
url="&url%str(&)typeDefName=file";
|
%if not %mf_isBlank(&viyaTypeDefName) %then %do;
|
||||||
|
url="&url%str(&)typeDefName=&viyaTypeDefName";
|
||||||
%end;
|
%end;
|
||||||
%else %do;
|
%else %do;
|
||||||
url="&url";
|
%if "&ext"="HTM" or "&ext"="HTML" or "&ext"="XHTML" %then %do;
|
||||||
|
url="&url%str(&)typeDefName=file_html";
|
||||||
|
%end;
|
||||||
|
%else %do;
|
||||||
|
%if "&ext"="CSS" or "&ext"="JS" or "&ext"="PNG" or "&ext"="SVG" %then %do;
|
||||||
|
url="&url%str(&)typeDefName=file_%lowcase(&ext)";
|
||||||
|
%end;
|
||||||
|
%else %do;
|
||||||
|
url="&url";
|
||||||
|
%end;
|
||||||
|
%end;
|
||||||
%end;
|
%end;
|
||||||
|
|
||||||
headers "Accept"="application/json"
|
headers "Accept"="application/json"
|
||||||
@@ -229,6 +257,7 @@ run;
|
|||||||
,mac=MV_CREATEFILE
|
,mac=MV_CREATEFILE
|
||||||
,msg=%str(&SYS_PROCHTTP_STATUS_CODE &SYS_PROCHTTP_STATUS_PHRASE)
|
,msg=%str(&SYS_PROCHTTP_STATUS_CODE &SYS_PROCHTTP_STATUS_PHRASE)
|
||||||
)
|
)
|
||||||
|
|
||||||
%local libref2;
|
%local libref2;
|
||||||
%let libref2=%mf_getuniquelibref();
|
%let libref2=%mf_getuniquelibref();
|
||||||
libname &libref2 JSON fileref=&fname1;
|
libname &libref2 JSON fileref=&fname1;
|
||||||
@@ -236,12 +265,72 @@ libname &libref2 JSON fileref=&fname1;
|
|||||||
data &outds;
|
data &outds;
|
||||||
set &libref2..links end=last;
|
set &libref2..links end=last;
|
||||||
if rel='createChild' then do;
|
if rel='createChild' then do;
|
||||||
call symputx('href',quote(cats("&base_uri",href)),'l');
|
|
||||||
&dbg put (_all_)(=);
|
&dbg put (_all_)(=);
|
||||||
end;
|
end;
|
||||||
run;
|
run;
|
||||||
|
|
||||||
%put &sysmacroname: &name created at %mfv_getpathuri(&path/&name);%put;
|
/* URI of the created file */
|
||||||
%put &base_uri/SASJobExecution?_file=&path/&name;%put;
|
%let fileuri=%trim(%mfv_getpathuri(&path/&name));
|
||||||
|
|
||||||
|
/* If properties were found then patch the file to include them */
|
||||||
|
%if not %mf_isBlank(%superq(viyaProperties)) %then %do;
|
||||||
|
/* Wrap the properties object in a root object also containing the file name */
|
||||||
|
%local viyapatch;
|
||||||
|
%let viyapatch = %sysfunc(pathname(work))/%mf_getuniquename(prefix=patch_json_);
|
||||||
|
data _null_;
|
||||||
|
length line $32767;
|
||||||
|
file "&viyapatch" lrecl=32767;
|
||||||
|
put '{ "name": "' "&name" '",';
|
||||||
|
line = cat('"properties": ',symget("viyaProperties"));
|
||||||
|
put line;
|
||||||
|
put '}';
|
||||||
|
stop;
|
||||||
|
run;
|
||||||
|
|
||||||
|
%if &mdebug=1 %then %do;
|
||||||
|
data _null_;
|
||||||
|
if (_n_ eq 1) then put 'DEBUG: ** PATCH JSON **';
|
||||||
|
infile "&viyapatch" end=last;
|
||||||
|
input;
|
||||||
|
put _infile_;
|
||||||
|
run;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
/* And apply the properties to the newly created file, using the PATCH method */
|
||||||
|
%let fref=%mf_getuniquefileref();
|
||||||
|
filename &fref "&viyapatch";
|
||||||
|
%let url=&base_uri&fileuri;
|
||||||
|
|
||||||
|
proc http method='PATCH' oauth_bearer=sas_services in=&fref
|
||||||
|
url="&url";
|
||||||
|
headers "Accept"="application/json"
|
||||||
|
"Content-Type"="application/json"
|
||||||
|
"If-Match"="*";
|
||||||
|
%if &mdebug=1 %then %do;
|
||||||
|
debug level=2;
|
||||||
|
%end;
|
||||||
|
run;
|
||||||
|
%if &mdebug=1 %then %put &sysmacroname PATCH &=url
|
||||||
|
&=SYS_PROCHTTP_STATUS_CODE &=SYS_PROCHTTP_STATUS_PHRASE;
|
||||||
|
%mp_abort(iftrue=(&SYS_PROCHTTP_STATUS_CODE ne 200)
|
||||||
|
,mac=MV_CREATEFILE
|
||||||
|
,msg=%str(&SYS_PROCHTTP_STATUS_CODE &SYS_PROCHTTP_STATUS_PHRASE)
|
||||||
|
)
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%put &sysmacroname: &base_uri&fileuri;
|
||||||
|
%put /SASJobExecution?_file=&path/&name;%put;
|
||||||
|
|
||||||
|
%if &mdebug=0 %then %do;
|
||||||
|
/* clear refs */
|
||||||
|
filename &fname1 clear;
|
||||||
|
filename &fref clear;
|
||||||
|
libname &libref2 clear;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%mp_abort(
|
||||||
|
iftrue=(&syscc ne 0),
|
||||||
|
msg=Cannot leave &sysmacroname with syscc=&syscc
|
||||||
|
)
|
||||||
|
|
||||||
%mend mv_createfile;
|
%mend mv_createfile;
|
||||||
@@ -46,6 +46,11 @@
|
|||||||
%end;
|
%end;
|
||||||
%else %let dbg=*;
|
%else %let dbg=*;
|
||||||
|
|
||||||
|
%mp_abort(
|
||||||
|
iftrue=(&syscc ne 0),
|
||||||
|
msg=Cannot enter &sysmacroname with syscc=&syscc
|
||||||
|
)
|
||||||
|
|
||||||
%if %mfv_existfolder(&path)=1 %then %do;
|
%if %mfv_existfolder(&path)=1 %then %do;
|
||||||
%&dbg.put &sysmacroname: &path already exists;
|
%&dbg.put &sysmacroname: &path already exists;
|
||||||
data &outds;
|
data &outds;
|
||||||
@@ -55,6 +60,7 @@
|
|||||||
run;
|
run;
|
||||||
%return;
|
%return;
|
||||||
%end;
|
%end;
|
||||||
|
%mp_abort(iftrue=(&syscc ne 0),msg=syscc=&syscc when folder checking)
|
||||||
|
|
||||||
%local oauth_bearer;
|
%local oauth_bearer;
|
||||||
%if &grant_type=detect %then %do;
|
%if &grant_type=detect %then %do;
|
||||||
@@ -108,6 +114,17 @@ options noquotelenmax;
|
|||||||
headers "Authorization"="Bearer &&&access_token_var";
|
headers "Authorization"="Bearer &&&access_token_var";
|
||||||
%end;
|
%end;
|
||||||
run;
|
run;
|
||||||
|
%if &SYS_PROCHTTP_STATUS_CODE=401 %then %do;
|
||||||
|
/* relates to: https://github.com/sasjs/core/issues/400 */
|
||||||
|
%put 401 thrown in &sysmacroname;
|
||||||
|
%put sleeping: %sysfunc(sleep(12,1)) secs - will try again;
|
||||||
|
proc http method='GET' out=&fname1 &oauth_bearer
|
||||||
|
url="&base_uri/folders/folders/@item?path=&newpath";
|
||||||
|
%if &grant_type=authorization_code %then %do;
|
||||||
|
headers "Authorization"="Bearer &&&access_token_var";
|
||||||
|
%end;
|
||||||
|
run;
|
||||||
|
%end;
|
||||||
%local libref1;
|
%local libref1;
|
||||||
%let libref1=%mf_getuniquelibref();
|
%let libref1=%mf_getuniquelibref();
|
||||||
libname &libref1 JSON fileref=&fname1;
|
libname &libref1 JSON fileref=&fname1;
|
||||||
@@ -115,7 +132,7 @@ options noquotelenmax;
|
|||||||
iftrue=(
|
iftrue=(
|
||||||
&SYS_PROCHTTP_STATUS_CODE ne 200 and &SYS_PROCHTTP_STATUS_CODE ne 404
|
&SYS_PROCHTTP_STATUS_CODE ne 200 and &SYS_PROCHTTP_STATUS_CODE ne 404
|
||||||
)
|
)
|
||||||
,mac=&sysmacroname
|
,mac=mv_createfolder124
|
||||||
,msg=%str(&SYS_PROCHTTP_STATUS_CODE &SYS_PROCHTTP_STATUS_PHRASE)
|
,msg=%str(&SYS_PROCHTTP_STATUS_CODE &SYS_PROCHTTP_STATUS_PHRASE)
|
||||||
)
|
)
|
||||||
%if &mdebug=1 %then %do;
|
%if &mdebug=1 %then %do;
|
||||||
@@ -164,7 +181,7 @@ options noquotelenmax;
|
|||||||
'Content-Type'='application/vnd.sas.content.folder+json'
|
'Content-Type'='application/vnd.sas.content.folder+json'
|
||||||
'Accept'='application/vnd.sas.content.folder+json';
|
'Accept'='application/vnd.sas.content.folder+json';
|
||||||
run;
|
run;
|
||||||
%if &SYS_PROCHTTP_STATUS_CODE ne 200 %then %do;
|
%if &SYS_PROCHTTP_STATUS_CODE ne 201 %then %do;
|
||||||
%put &=SYS_PROCHTTP_STATUS_CODE &=SYS_PROCHTTP_STATUS_PHRASE;
|
%put &=SYS_PROCHTTP_STATUS_CODE &=SYS_PROCHTTP_STATUS_PHRASE;
|
||||||
%end;
|
%end;
|
||||||
%mp_abort(iftrue=(&SYS_PROCHTTP_STATUS_CODE ne 201)
|
%mp_abort(iftrue=(&SYS_PROCHTTP_STATUS_CODE ne 201)
|
||||||
@@ -194,4 +211,8 @@ options noquotelenmax;
|
|||||||
filename &fname1 clear;
|
filename &fname1 clear;
|
||||||
libname &libref1 clear;
|
libname &libref1 clear;
|
||||||
%end;
|
%end;
|
||||||
|
%mp_abort(
|
||||||
|
iftrue=(&syscc ne 0),
|
||||||
|
msg=Cannot leave &sysmacroname with syscc=&syscc
|
||||||
|
)
|
||||||
%mend mv_createfolder;
|
%mend mv_createfolder;
|
||||||
@@ -117,7 +117,8 @@ libname &libref1a JSON fileref=&fname1a;
|
|||||||
%let found=0;
|
%let found=0;
|
||||||
/* %put Getting object uri from &libref1a..items; */
|
/* %put Getting object uri from &libref1a..items; */
|
||||||
data _null_;
|
data _null_;
|
||||||
length contenttype name $1000;
|
length contenttype name uri $1000;
|
||||||
|
call missing(of _all_);
|
||||||
set &libref1a..items;
|
set &libref1a..items;
|
||||||
if contenttype='jobDefinition' and upcase(name)="%upcase(&name)" then do;
|
if contenttype='jobDefinition' and upcase(name)="%upcase(&name)" then do;
|
||||||
call symputx('uri',cats("&base_uri",uri),'l');
|
call symputx('uri',cats("&base_uri",uri),'l');
|
||||||
|
|||||||
@@ -126,7 +126,7 @@ data _null_;
|
|||||||
uri=symget('uri');
|
uri=symget('uri');
|
||||||
if length(uri)<12 then do;
|
if length(uri)<12 then do;
|
||||||
call symputx('errflg',1);
|
call symputx('errflg',1);
|
||||||
call symputx('errmsg',"URI is invalid (too short) - '&uri'",'l');
|
call symputx('errmsg',"URI is too short - "!!uri,'l');
|
||||||
end;
|
end;
|
||||||
if scan(uri,-1)='state' or scan(uri,1) ne 'jobExecution' then do;
|
if scan(uri,-1)='state' or scan(uri,1) ne 'jobExecution' then do;
|
||||||
call symputx('errflg',1);
|
call symputx('errflg',1);
|
||||||
@@ -191,7 +191,7 @@ data _null_;
|
|||||||
uri=symget('loglocation');
|
uri=symget('loglocation');
|
||||||
if length(uri)<12 then do;
|
if length(uri)<12 then do;
|
||||||
call symputx('errflg',1);
|
call symputx('errflg',1);
|
||||||
call symputx('errmsg',"URI is invalid (too short) - '&uri'",'l');
|
call symputx('errmsg',"URI is too short - "!!uri,'l');
|
||||||
end;
|
end;
|
||||||
else if (scan(uri,1,'/') ne 'compute' or scan(uri,2,'/') ne 'sessions')
|
else if (scan(uri,1,'/') ne 'compute' or scan(uri,2,'/') ne 'sessions')
|
||||||
and (scan(uri,1,'/') ne 'files' or scan(uri,2,'/') ne 'files')
|
and (scan(uri,1,'/') ne 'files' or scan(uri,2,'/') ne 'files')
|
||||||
|
|||||||
248
viya/mv_getviyafileextparms.sas
Normal file
248
viya/mv_getviyafileextparms.sas
Normal file
@@ -0,0 +1,248 @@
|
|||||||
|
/**
|
||||||
|
@file mv_getviyafileextparms.sas
|
||||||
|
@brief Reads the VIYA file-extension type definition and returns selected
|
||||||
|
values in SAS macro variables
|
||||||
|
|
||||||
|
@details Content is derived from the following endpoint:
|
||||||
|
"https://${serverUrl}/types/types?limit=999999"
|
||||||
|
|
||||||
|
@param [in] ext File extension to retrieve property info for.
|
||||||
|
@param [out] propertiesVar= SAS macro variable name that will contain
|
||||||
|
the 'properties' object json, if found, else blank.
|
||||||
|
@param [out] typeDefNameVar= SAS macro variable name that will contain
|
||||||
|
the 'typeDefName' property value, if found, else blank.
|
||||||
|
@param [out] mediaTypeVar= SAS macro variable name that will contain
|
||||||
|
the 'mediaType' property value, if found, else blank.
|
||||||
|
@param [out] viyaFileExtRespLibDs (work.mv_getViyaFileExtParmsResponse)
|
||||||
|
Library.name of the dataset to receive the local working copy of the initial
|
||||||
|
response that requests all file extension details. Created once per session
|
||||||
|
to avoid multiple api calls.
|
||||||
|
@param [in] mdebug= (0) Set to 1 to enable DEBUG messages
|
||||||
|
|
||||||
|
<h4> SAS Macros </h4>
|
||||||
|
@li mf_existds.sas
|
||||||
|
@li mf_getplatform.sas
|
||||||
|
@li mf_getuniquefileref.sas
|
||||||
|
@li mf_getuniquename.sas
|
||||||
|
@li mf_getvalue.sas
|
||||||
|
@li mf_getvarlist.sas
|
||||||
|
@li mf_getvartype.sas
|
||||||
|
@li mf_isblank.sas
|
||||||
|
@li mf_nobs.sas
|
||||||
|
@li mp_abort.sas
|
||||||
|
|
||||||
|
**/
|
||||||
|
|
||||||
|
%macro mv_getViyaFileExtParms(
|
||||||
|
ext,
|
||||||
|
typeDefNameVar=,
|
||||||
|
propertiesVar=,
|
||||||
|
mediaTypeVar=,
|
||||||
|
viyaFileExtRespLibDs=work.mv_getViyaFileExtParmsResponse,
|
||||||
|
mdebug=0
|
||||||
|
);
|
||||||
|
%local base_uri; /* location of rest apis */
|
||||||
|
%local url; /* File extension info end-point */
|
||||||
|
|
||||||
|
%mp_abort(
|
||||||
|
iftrue=(%mf_isBlank(&ext))
|
||||||
|
,msg=%str(No file extension provided.)
|
||||||
|
,mac=MV_GETVIYAFILEEXTPARMS
|
||||||
|
);
|
||||||
|
|
||||||
|
%mp_abort(
|
||||||
|
iftrue=(%mf_isBlank(&typeDefNameVar) and
|
||||||
|
%mf_isBlank(&propertiesVar) and
|
||||||
|
%mf_isBlank(&mediaTypeVar))
|
||||||
|
,msg=%str(MV_GETVIYAFILEEXTPARMS - No parameter was requested.)
|
||||||
|
,mac=MV_GETVIYAFILEEXTPARMS
|
||||||
|
);
|
||||||
|
|
||||||
|
%mp_abort(
|
||||||
|
iftrue=(%mf_isBlank(&viyaFileExtRespLibDs))
|
||||||
|
,msg=%str(No <libname.>dataset name provided to cache inital response.)
|
||||||
|
,mac=MV_GETVIYAFILEEXTPARMS
|
||||||
|
);
|
||||||
|
|
||||||
|
/* Declare requested parameters as global macro vars and initialize blank */
|
||||||
|
%if not %mf_isBlank(&typeDefNameVar) %then %do;
|
||||||
|
%global &typeDefNameVar;
|
||||||
|
%let &typeDefNameVar = %str();
|
||||||
|
%end;
|
||||||
|
%if not %mf_isBlank(&propertiesVar) %then %do;
|
||||||
|
%global &propertiesVar;
|
||||||
|
%let &propertiesVar = %str();
|
||||||
|
%end;
|
||||||
|
%if not %mf_isBlank(&mediaTypeVar) %then %do;
|
||||||
|
%global &mediaTypeVar;
|
||||||
|
%let &mediaTypeVar = %str();
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%let base_uri=%mf_getplatform(VIYARESTAPI);
|
||||||
|
%if &mdebug=1 %then %do;
|
||||||
|
%put DEBUG: &=base_uri;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%let ext=%lowcase(&ext);
|
||||||
|
|
||||||
|
/* Create a local copy of the Viya response containing all file type info, if
|
||||||
|
it does not already exist. */
|
||||||
|
%if not %mf_existds(&viyaFileExtRespLibDs) %then %do;
|
||||||
|
/* Create a temp file and fill with JSON that declares */
|
||||||
|
/* VIYA file-type details for the given file extension */
|
||||||
|
%local viyatypedefs;
|
||||||
|
%let viyatypedefs=%mf_getuniquefileref();
|
||||||
|
filename &viyatypedefs temp;
|
||||||
|
|
||||||
|
%let url = &base_uri/types/types?limit=999999;
|
||||||
|
|
||||||
|
proc http oauth_bearer=sas_services out=&viyatypedefs
|
||||||
|
url="&url";
|
||||||
|
run;
|
||||||
|
|
||||||
|
%if &mdebug=1 %then %put DEBUG: &sysmacroname &=url
|
||||||
|
&=SYS_PROCHTTP_STATUS_CODE &=SYS_PROCHTTP_STATUS_PHRASE;
|
||||||
|
|
||||||
|
%if (&SYS_PROCHTTP_STATUS_CODE ne 200) %then %do;
|
||||||
|
/* To avoid a breaking change, exit early if the request failed.
|
||||||
|
The calling process will proceed with empty requested macro variables. */
|
||||||
|
%put INFO: &sysmacroname File extension details were not retrieved.;
|
||||||
|
filename &viyatypedefs clear;
|
||||||
|
%return;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%if &mdebug=1 %then %do;
|
||||||
|
/* Dump the response to the log */
|
||||||
|
data _null_;
|
||||||
|
length line $120;
|
||||||
|
null=byte(0);
|
||||||
|
infile &viyatypedefs dlm=null lrecl=120 recfm=n;
|
||||||
|
input line $120.;
|
||||||
|
if _n_ = 1 then put "DEBUG:";
|
||||||
|
put line;
|
||||||
|
run;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
/* Convert the content of that JSON into SAS datasets */
|
||||||
|
/* First prepare a new WORK-based folder to receive the datasets */
|
||||||
|
%local jsonworkfolder jsonlib opt_dlcreatedir;
|
||||||
|
%let jsonworkfolder=%sysfunc(pathname(work))/%mf_getuniquename(prefix=json_);
|
||||||
|
%let jsonlib=%mf_getuniquelibref(prefix=json);
|
||||||
|
/* And point a libname at it */
|
||||||
|
%let opt_dlcreatedir = %sysfunc(getoption(dlcreatedir));
|
||||||
|
options dlcreatedir; libname &jsonlib "&jsonworkfolder"; options &opt_dlcreatedir;
|
||||||
|
|
||||||
|
/* Read the json output once and copy datasets to its work folder */
|
||||||
|
%local libref1;
|
||||||
|
%let libref1=%mf_getuniquelibref();
|
||||||
|
libname &libref1 JSON fileref=&viyatypedefs automap=create;
|
||||||
|
proc copy in=&libref1 out=&jsonlib; run;
|
||||||
|
|
||||||
|
libname &libref1 clear;
|
||||||
|
|
||||||
|
/* Now give all rows belonging to the same items array a grouping value */
|
||||||
|
data &viyaFileExtRespLibDs;
|
||||||
|
length _viyaItemIdx 8;
|
||||||
|
set &jsonlib..alldata;
|
||||||
|
retain _viyaItemIdx 0;
|
||||||
|
/* Increment the row group index when a new 'items' group is observed */
|
||||||
|
if P=1 and P1='items' then _viyaItemIdx + 1;
|
||||||
|
run;
|
||||||
|
|
||||||
|
%if &mdebug=0 %then %do;
|
||||||
|
/* Tidy up, unless debug=1 */
|
||||||
|
proc datasets library=&jsonlib nolist kill; quit;
|
||||||
|
libname &jsonlib clear;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
filename &viyatypedefs clear;
|
||||||
|
|
||||||
|
%end; /* If initial filetype query response didn't exist */
|
||||||
|
|
||||||
|
/* Find the row-group for the current file extension */
|
||||||
|
%local itemRowGroup;
|
||||||
|
%let itemRowGroup =
|
||||||
|
%mf_getValue(
|
||||||
|
&viyaFileExtRespLibDs
|
||||||
|
,_viyaItemIdx
|
||||||
|
,filter=%quote(p1='items' and p2='extensions' and value="&ext")
|
||||||
|
);
|
||||||
|
|
||||||
|
%if &mdebug %then %put DEBUG: &=itemRowGroup;
|
||||||
|
|
||||||
|
%if %mf_isBlank(&itemRowGroup) %then %do;
|
||||||
|
/* extension was not found */
|
||||||
|
%if(&mdebug=1) %then %put DEBUG: No type details found for extension "&ext".;
|
||||||
|
%return;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
/* Filter the cached response data down to the required file extension */
|
||||||
|
%local dsItems;
|
||||||
|
%let dsItems = %mf_getuniquename(prefix=dsItems_);
|
||||||
|
data work.&dsItems;
|
||||||
|
set &viyaFileExtRespLibDs;
|
||||||
|
where _viyaItemIdx = &itemRowGroup;
|
||||||
|
run;
|
||||||
|
|
||||||
|
/* Populate typeDefName, if requested */
|
||||||
|
%if (not %mf_isBlank(&typeDefNameVar)) %then %do;
|
||||||
|
%let &typeDefNameVar = %mf_getvalue(&dsItems,value,filter=%quote(p1="items" and p2="name"));
|
||||||
|
%if &mdebug=1 %then %put DEBUG: &=typeDefNameVar &typeDefNameVar=&&&typeDefNameVar;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
/* Populate mediaType, if requested */
|
||||||
|
%if (not %mf_isBlank(&mediaTypeVar)) %then %do;
|
||||||
|
%let &mediaTypeVar = %mf_getvalue(&dsItems,value,filter=%quote(p1="items" and p2="mediaType"));
|
||||||
|
%if &mdebug=1 %then %put DEBUG: &=mediaTypeVar &mediaTypeVar=&&&mediaTypeVar;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
/* Populate properties macro variable, if requested */
|
||||||
|
%if not %mf_isBlank(&propertiesVar) %then %do;
|
||||||
|
|
||||||
|
/* Filter dsItems down to the properties */
|
||||||
|
%local dsProperties;
|
||||||
|
%let dsProperties = %mf_getuniquename(prefix=dsProperties_);
|
||||||
|
data work.&dsProperties ( rename=(p3 = propertyName) );
|
||||||
|
set work.&dsItems;
|
||||||
|
where p2="properties" and v=1;
|
||||||
|
run;
|
||||||
|
|
||||||
|
/* Check for 1+ properties */
|
||||||
|
%if ( %mf_nobs(&dsProperties) = 0 ) %then %do;
|
||||||
|
%let &propertiesVar = %str();
|
||||||
|
%if &mdebug=1 %then %put DEBUG: &SYSMACRONAME - No Viya properties found for file suffix %str(%')&ext%str(%');
|
||||||
|
%end;
|
||||||
|
%else %do;
|
||||||
|
/* Properties potentially span multiple rows in the input table */
|
||||||
|
data _null_;
|
||||||
|
length
|
||||||
|
line $32767
|
||||||
|
properties $32767
|
||||||
|
;
|
||||||
|
retain properties;
|
||||||
|
set &dsProperties end=last;
|
||||||
|
if _n_ = 1 then properties = '{';
|
||||||
|
|
||||||
|
line = cats(quote(trim(propertyName)),':');
|
||||||
|
/* Only strings and bools appear in properties */
|
||||||
|
if value not in ("true","false") then value = quote(trim(value));
|
||||||
|
line = catx(' ',line,value);
|
||||||
|
/* Add a comma separator to all except the last line */
|
||||||
|
if not last then line = cats(line,',');
|
||||||
|
|
||||||
|
/* Add this line to the output value */
|
||||||
|
properties = catx(' ',properties,line);
|
||||||
|
|
||||||
|
if last then do;
|
||||||
|
/* Close off the properties object and output to the macro variable */
|
||||||
|
properties=catx(' ',properties,'}');
|
||||||
|
call symputx("&propertiesVar",properties);
|
||||||
|
end;
|
||||||
|
run;
|
||||||
|
|
||||||
|
%if &mdebug=1 %then %put DEBUG: &=propertiesVar &propertiesVar=&&&propertiesVar;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%mend mv_getViyaFileExtParms;
|
||||||
@@ -188,6 +188,8 @@
|
|||||||
%if %mf_existvarList(&inds,FLOW_ID)=0 %then %do;
|
%if %mf_existvarList(&inds,FLOW_ID)=0 %then %do;
|
||||||
retain FLOW_ID 0;
|
retain FLOW_ID 0;
|
||||||
%end;
|
%end;
|
||||||
|
/* https://github.com/sasjs/adapter/pull/845#issuecomment-2956589644 */
|
||||||
|
retain _omitSessionResults "false";
|
||||||
set &inds;
|
set &inds;
|
||||||
&dbg. putlog (_all_)(=);
|
&dbg. putlog (_all_)(=);
|
||||||
run;
|
run;
|
||||||
|
|||||||
111
xplatform/mx_createjob.sas
Normal file
111
xplatform/mx_createjob.sas
Normal file
@@ -0,0 +1,111 @@
|
|||||||
|
/**
|
||||||
|
@file mx_createjob.sas
|
||||||
|
@brief Create a job in SAS 9, Viya or SASjs
|
||||||
|
@details Creates a Stored Process in SAS 9, a Job Execution Service in SAS
|
||||||
|
Viya, or a Stored Program on SASjs Server - depending on the executing
|
||||||
|
environment.
|
||||||
|
|
||||||
|
Usage:
|
||||||
|
|
||||||
|
%* compile macros ;
|
||||||
|
filename mc url "https://raw.githubusercontent.com/sasjs/core/main/all.sas";
|
||||||
|
%inc mc;
|
||||||
|
|
||||||
|
%* write some code;
|
||||||
|
filename ft15f001 temp;
|
||||||
|
parmcards4;
|
||||||
|
data example1;
|
||||||
|
set sashelp.class;
|
||||||
|
run;
|
||||||
|
;;;;
|
||||||
|
|
||||||
|
%* create the job;
|
||||||
|
%mx_createjob(path=/Public/app/jobs,name=myjob,replace=YES)
|
||||||
|
|
||||||
|
<h4> SAS Macros </h4>
|
||||||
|
@li mf_getplatform.sas
|
||||||
|
@li mm_createstp.sas
|
||||||
|
@li ms_createfile.sas
|
||||||
|
@li mv_createjob.sas
|
||||||
|
|
||||||
|
@param [in,out] path= The full folder path where the job will be created
|
||||||
|
@param [in,out] name= Job name. Avoid spaces.
|
||||||
|
@param [in] desc= The description of the job (optional)
|
||||||
|
@param [in] precode= Space separated list of filerefs, pointing to the code
|
||||||
|
that needs to be attached to the beginning of the job (optional)
|
||||||
|
@param [in] code= (ft15f001) Space seperated fileref(s) of the actual code to
|
||||||
|
be added
|
||||||
|
@param [in] replace= (YES) Select YES to replace any existing job in that
|
||||||
|
location
|
||||||
|
@param [in] mDebug= (0) set to 1 to show debug messages in the log
|
||||||
|
|
||||||
|
@author Allan Bowe
|
||||||
|
|
||||||
|
<h4> Related Macros </h4>
|
||||||
|
@li mx_createjob.test.sas
|
||||||
|
@li mx_createwebservice.sas
|
||||||
|
|
||||||
|
**/
|
||||||
|
|
||||||
|
%macro mx_createjob(path=HOME
|
||||||
|
,name=initJob
|
||||||
|
,precode=
|
||||||
|
,code=ft15f001
|
||||||
|
,desc=This job was created by the mx_createjob macro
|
||||||
|
,replace=YES
|
||||||
|
,mdebug=0
|
||||||
|
)/*/STORE SOURCE*/;
|
||||||
|
|
||||||
|
%if &syscc ge 4 %then %do;
|
||||||
|
%put syscc=&syscc - &sysmacroname will not execute in this state;
|
||||||
|
%return;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
/* combine precode and code into a single file */
|
||||||
|
%local tempref x fref freflist;
|
||||||
|
%let tempref=%mf_getuniquefileref();
|
||||||
|
%local work tmpfile;
|
||||||
|
%let work=%sysfunc(pathname(work));
|
||||||
|
%let tmpfile=&tempref..sas;
|
||||||
|
filename &tempref "&work/&tmpfile";
|
||||||
|
%let freflist=&precode &code ;
|
||||||
|
%do x=1 %to %sysfunc(countw(&freflist));
|
||||||
|
%let fref=%scan(&freflist,&x);
|
||||||
|
%put &sysmacroname: adding &fref;
|
||||||
|
data _null_;
|
||||||
|
file &tempref lrecl=3000 termstr=crlf mod;
|
||||||
|
infile &fref lrecl=3000;
|
||||||
|
input;
|
||||||
|
put _infile_;
|
||||||
|
run;
|
||||||
|
%end;
|
||||||
|
|
||||||
|
%local platform; %let platform=%mf_getplatform();
|
||||||
|
%if &platform=SASVIYA %then %do;
|
||||||
|
%if "&path"="HOME" %then %let path=/Users/&sysuserid/My Folder;
|
||||||
|
%mv_createjob(path=&path
|
||||||
|
,name=&name
|
||||||
|
,code=&tempref
|
||||||
|
,desc=&desc
|
||||||
|
,replace=&replace
|
||||||
|
)
|
||||||
|
%end;
|
||||||
|
%else %if &platform=SASJS %then %do;
|
||||||
|
%if "&path"="HOME" %then %let path=/Users/&_sasjs_username/My Folder;
|
||||||
|
%ms_createfile(&path/&name..sas
|
||||||
|
,inref=&tempref
|
||||||
|
,mdebug=&mdebug
|
||||||
|
)
|
||||||
|
%end;
|
||||||
|
%else %do;
|
||||||
|
%if "&path"="HOME" %then %let path=/User Folders/&_METAPERSON/My Folder;
|
||||||
|
%mm_createstp(stpname=&name
|
||||||
|
,filename=&tmpfile
|
||||||
|
,directory=&work
|
||||||
|
,tree=&path
|
||||||
|
,stpdesc=&desc
|
||||||
|
,mDebug=&mdebug
|
||||||
|
)
|
||||||
|
%end;
|
||||||
|
filename &tempref clear;
|
||||||
|
%mend mx_createjob;
|
||||||
@@ -48,6 +48,9 @@ Usage:
|
|||||||
|
|
||||||
@author Allan Bowe
|
@author Allan Bowe
|
||||||
|
|
||||||
|
<h4> Related Macros </h4>
|
||||||
|
@li mx_createjob.sas
|
||||||
|
|
||||||
**/
|
**/
|
||||||
|
|
||||||
%macro mx_createwebservice(path=HOME
|
%macro mx_createwebservice(path=HOME
|
||||||
|
|||||||
Reference in New Issue
Block a user