Getting a Library’s Engine Name
Creating Macro Variables from SYSPARM
Displaying Macro Notes for Debugging
I have a set of utility macros that I usually have in my SASAUTOS library. These macros simplify tasks that I do regularly. This appendix provides documentation about each macro.
I use a number of standards when writing my utilities to keep them from conflicting with other programs:
● The macro names always start with an underscore.
● I use l_ at the beginning of all my local macro variable names, and I define them with a %LOCAL statement so that I don’t risk deleting any macro variables in the calling program.
● I create and use some global macro variables, and I begin the names of these macro variables with two underscores so that they won’t interfere with the rest of the environment.
● If I make temporary tables in the macro, their names begin with tmp_, and they are deleted at the end of the macro.
● Many of my utility and other macros use a debug flag variable that I call &__DEBUG. If &__DEBUG is blank or 0, then no debugging takes place. Values that are greater than zero mean that some debugging takes place. You can use a %LET statement to set this macro variable to anything greater than zero to turn on debugging before calling the utility macro.
● If there is only one parameter, I use a positional parameter (with no equal sign). If there is more than one parameter, then I use named parameters (with equal signs).
● I don’t call any other custom macros from within these macros, because nothing is more annoying than trying to use a macro and finding out that you also need five other macros to run it.
Some of these utilities are macro functions rather than standard macros. This means that they return a value to the calling program, so they must be called from a place where the return of a value makes sense. Generally this means that they are used in a %LET statement, a %IF statement, a DATA step assignment, or a DATA step IF statement.
The %_delete_tables utility will delete multiple tables from more than one library in a single macro call. It is useful for keeping programs from leaving temporary tables behind. It runs PROC DATASETS to delete the specified tables. The NOWARN option prevents warnings in the SAS log, even if a table doesn’t exist.
If the &__DEBUG global macro variable is greater than 0, then the tables are not deleted. This is helpful when you are trying to figure out why your program isn’t doing what it should. You just set &__DEBUG to 1 before calling running the program, and the %_delete_tables macro calls do not delete your tables.
The syntax is as follows:
%_delete_tables(tableList)
tableList
Specifies a list of data tables. The table names should be delimited with either a space or a comma. The table names can be one-level names (for WORK tables), two-level names, or a mixture of both.
The example in Program A.1 and SAS Log A.1 shows an attempt to delete two WORK tables and a SASUSER table. The WORK.files table does not exist, so it is ignored. The log reports which tables were deleted.
Program A.1: Running %_delete_tables
%_delete_tables(test, files, sasuser.test);
SAS Log A.1: Running %_delete_tables
1 %_delete_tables(test, files, sasuser.test);
NOTE: Deleting SASUSER.TEST (memtype=DATA).
NOTE: PROCEDURE DATASETS used (Total process time):
real time 0.01 seconds
cpu time 0.01 seconds
NOTE: Deleting WORK.TEST (memtype=DATA).
NOTE: PROCEDURE DATASETS used (Total process time):
real time 0.68 seconds
cpu time 0.67 seconds
The %_get_num_records macro function returns the number of records (active records, not records marked as deleted) in a table. If the table doesn’t exist, then it returns -1.
This function is very useful if you want to skip some code when a table doesn’t exist or doesn’t have any data: You can use this macro in a macro %IF statement like this:
%if (%_get_num_records(work.table) gt 0) %then...
Or you can use it in a %LET statement:
%let nobs = %_get_num_records(work.table);
It can also be used from a DATA step:
data _null_;
nobs = %_get_num_records(work.table);
putlog nobs=;
if (%_get_num_records(work.otherTable) eq -1) then
putlog "otherTable does not exist";
run;
The macro also accepts a table with a WHERE clause and returns the number of records in the subset.
The syntax is as follows:
%_get_num_records(table)
table
Specifies the name of the table from which to get the number of records. This can be a one-level (WORK) or a two-level table name, and it can include a WHERE clause in the following form: tableName(where=(where_clause)).
The function returns the following values:
−1 indicates that the table does not exist.
0 indicates that the table exists but has no records.
>0 indicates the number of records in the table.
In the Program A.2 and SAS Log A.2 example, the first %LET statement shows getting the records for a table, and the second %LET statement shows getting the records from a table with a WHERE clause applied to it.
Program A.2: Running %_get_num_records
%let nobs = %_get_num_records(sashelp.class);
%put &=nobs;
%let nobs = %_get_num_records(
%str(sashelp.class(where=(sex = "F"))));
%put &=nobs;
SAS Log A.2: Running %_get_num_records
1 %let nobs = %_get_num_records(sashelp.class);
2 %put &=nobs;
NOBS=19
3
4 %let nobs = %_get_num_records(%str(sashelp.class(where=(sex = "F"))));
5 %put &=nobs;
NOBS=9
The %_get_engine macro function returns the name of the engine of a library. This function is useful if you need to do different processing based on the type of engine (for example, you might need to use pass-through if this is a DBMS engine).
The syntax is as follows:
%_get_engine(libref)
libref
Specifies the libref that points to a library defined in your SAS session.
The function returns the name of the engine of the library (for example, V9 = the Base SAS engine for a SAS 9 library).
The function returns a blank if the libref hasn’t been defined.
The example in Program A.3 and SAS Log A.3 shows how to get the engine name from a Base SAS library and from an XML library. It also attempts to get an engine name from a library that hasn’t been defined.
Program A.3: Running %_get_engine
%let engine = %_get_engine(sashelp); /* base sas library */
%put &=engine;
%let engine = %_get_engine(syscfg); /* xml engine library */
%put &=engine;
%let engine = %_get_engine(xyz); /* xyz doesn't exist */
%put &=engine;
SAS Log A.3: Running %_get_engine
8 %let engine = %_get_engine(sashelp); /* base sas library */
2 %put &=engine;
ENGINE=V9
3
4 %let engine = %_get_engine(syscfg); /* xml engine library */
5 %put &=engine;
ENGINE=XMLV2
6
7 %let engine = %_get_engine(xyz); /* xyz doesn’t exist */
8 %put &=engine;
ENGINE=
The %_get_keeplist macro function gets a list of variable names from the specified table and returns the list in the format needed—either as a keep list (with spaces between the variables) or for a PROC SQL select clause (with an alias before each variable name and commas between variables). If you specify a PROC SQL alias, then the select clause style is returned. Otherwise, the standard keep list is returned.
The syntax is as follows:
%_get_keeplist(table=, sqlAlias=)
table
Specifies the name of the table to get the keep list from. This can be a one-level (WORK) or two-level name. This parameter is required.
sqlAlias
Specifies a SQL alias if you want a list of variable names for use in a PROC SQL SELECT statement. The alias is specified on the PROC SQL FROM statement: from table as sqlAlias.
The function returns a list of variables either in the format for a KEEP statement, where the names are delimited with spaces, or in the format for a PROC SQL SELECT statement, where the names are delimited by commas and each name is prefixed with sqlAlias.
If the table doesn’t exist or has no variables, then a blank is returned.
Program A.4 and SAS Log A.4 show how to get a list of variables to use in a KEEP statement.
Program A.4: Running %_get_keeplist for a KEEP Statement
%let keep = %_get_keeplist(table = sashelp.class);
data new (keep = &keep);
set sashelp.classfit;
run;
SAS Log A.4: Running %_get_keeplist for a KEEP Statement
1 data new
2 ! (keep = &keep);
SYMBOLGEN: Macro variable KEEP resolves to Name Sex Age Height Weight
3 set sashelp.classfit;
4 run;
Program A.5 and SAS Log A.5 show how to get a list of variables for a SELECT statement.
Program A.5: Running %_get_keeplist for a SELECT Statement
proc sql;
select %_get_keeplist(table = sashelp.class, sqlAlias = t),
"test" as newVar
from sashelp.class as t;
quit;
SAS Log A.5: Running %_get_keeplist for a SELECT Statement
1 proc sql;
2 select %_get_keeplist(table = sashelp.class, sqlAlias = t),
MPRINT(_GET_KEEPLIST): t.Name, t.Sex, t.Age, t.Height, t.Weight
3 "test" as newVar
4 from sashelp.class as t;
5 quit;
The %_make_attribs macro can be called from inside a DATA step to create ATTRIB statements based on the variables in a data table. If the table doesn’t exist or has no variables, then no statements are created, and a note is displayed in the SAS log.
The syntax is as follows:
%_make_attribs(table);
table
Specifies the name of the table from which to get the variables and attributes. This parameter is required.
Program A.6 and SAS Log A.6 show how to use the macro to create the ATTRIB statements based on the variables in sashelp.class.
Program A.6: Running %_make_attribs
options mprint;
data new;
%_make_attribs(sashelp.class);
call missing(of _all_);
run;
SAS Log A.6: Running %_make_attribs
1 options mprint;
2 data new;
3 %_make_attribs(sashelp.class);
MPRINT(_MAKE_ATTRIBS): attrib Name length = $8 ;
MPRINT(_MAKE_ATTRIBS): attrib Sex length = $1 ;
MPRINT(_MAKE_ATTRIBS): attrib Age length = 8 ;
MPRINT(_MAKE_ATTRIBS): attrib Height length = 8 ;
MPRINT(_MAKE_ATTRIBS): attrib Weight length = 8 ;
4 call missing(of _all_);
5 run;
The %_make_format macro creates a format from a data table.
The syntax is as follows:
%_make_format(inputTable=,
fmtLib=,
fmtName=,
start=,
label=,
otherLabel=,
type=);
inputTable
Specifies the name of the table to use as input. This parameter is required.
fmtLib
Specifies the library in which to store the format. If blank, then it defaults to WORK.
fmtName
Specifies the name of the format to be created. This parameter is required.
start
Specifies the variable that holds the start value. This parameter is required.
label
Specifies the variable that holds the label value. This parameter is required.
otherLabel
Specifies the value to use for "other" values. If blank, no "other" value is created.
type
Specifies the type of format: C (character format), N (numeric informat), J (character informat), or I (numeric informat). If blank, then it defaults to C.
Program A.7 creates a format from the sashelp.class table that will convert a name value to a sex value. It sets the “other” value to “X”, so any names that are not in the class table are set to “X”. The DATA step tests the format. “William” and “Jane” are both in the format, and “Martha” is not, so “William” has a sex of “M”, “Jane” is “F”, and “Martha” is “X”. The output is shown in SAS Log A.7.
Program A.7: Running %_make_format
%_make_format(inputTable = sashelp.class,
fmtName = $sex,
start = name,
label = sex,
otherLabel = 'X');
data _null_;
n = "William";
s = put(n, $sex.);
putlog n= s=;
n = "Jane";
s = put(n, $sex.);
putlog n= s=;
n = "Martha";
s = put(n, $sex.);
putlog n= s=;
run;
SAS Log A.7: Running %_MAKE_FORMAT%_make_format
n=William s=M
n=Jane s=F
n=Martha s=X
The %_make_directory macro creates the directories in the specified path. It can be used on either Windows or Unix. This macro will check each level of the supplied path and will create the levels that do not already exist. If the entire path already exists, nothing is created.
The syntax is as follows:
%_make_directory(path);
path
specifies the full path to be created. Windows path are expected in the drive:\level1\level2 format, and Unix paths are expected in the /level1/level2 format. This parameter is required.
Program A.8 is run on Windows, and none of the path levels exist before running the macro. Figure A.1 shows the directories that are created.
Program A.8: Running %_make_directory on Windows
%_make_directory(c:\one\two\three);
Figure A.1: Running %_make_directory on Windows
Program A.9 is run on Unix, and the first 2 levels of the path already exist before running the macro. Figure A.2 shows the directories on Unix after running the macro.
Program A.9: Running %_make_directory on Unix
%_make_directory(/home/user/one/two/three);
Figure A.2: Running %_make_directory on Unix
If you use the SYSPARM option to send values into a SAS program, then you can use the %_read_sysparm macro to read the contents of SYSPARM and create global macro variables from the contents. The macro assumes that the contents are space- or comma-delimited, and the values are either simple values or in name-value pairs (name=value). If just a value is found, then the macro variable is called parm#. An additional macro variable, __macroCount, is created to tell you how many parm# variables were created. Each time a macro variable is created, a note is written to the SAS log with the macro variable name and value.
The syntax is as follows:
%_read_sysparm(notes)
notes
Set this parameter to nonotes to turn off the macro variable name and value notes.
Program A.10 sets the SYSPARM option ( this option is normally set when you start SAS, but can also be set with an OPTIONS statement), and then calls the utility macro to read the values. It then puts out the values of the parameters and the &__macroCount variable. The SYSPARM values are separated by commas and spaces. Some of the values are name-value pairs, and some are just values. The output from the program is shown in SAS Log A.8.
Program A.10: Running %_ read_sysparm
options sysparm = "a=1 b=2, 3, 4, c=5";
%_read_sysparm ();
%put &=__macroCount;
SAS Log A.8: Running %_read_sysparm
1 options sysparm = "a=1 b=2, 3, 4, c=5";
2 %_read_sysparm ();
Macro variable created: &a = 1
Macro variable created: &b = 2
Macro variable created: &parm1 = 3
Macro variable created: &parm2 = 4
Macro variable created: &c = 5
3 %put &=__macroCount;
__MACROCOUNT=2
The %_turn_log_options_off, %_turn_log_options_on, and %_reset_log_options macros work together to turn on and off the system options that control what is displayed in the SAS log.
If you want to run some code that doesn’t write anything to the SAS log, you can use the %_turn_log_options_off macro to turn the following options off: SOURCE, SOURCE2, NOTES, MLOGIC, MLOGICNEST, MPRINT, MPRINTNEST, and SYMBOLGEN. The macro saves the original values of these options in global macro variables so that the options can be reset to their original values when you want to turn them on again. The %_reset_log_options macro uses those global macro variables to reset the options for you. So you can call %_turn_log_options_off at the beginning of a program, and then call %_reset_log_options at the end of the program to get almost nothing in the log from running the program.
For debugging, the %_turn_log_options_on macro sets all the log options on so that you get as much log information as SAS can provide. Because this macro also saves the original option settings in macro variables, you can use the %_reset_log_options macro to restore your standard settings.
Tip: You can save all options and their values in a SAS table by using PROC OPTSAVE, and you can reload them using PROC OPTLOAD.
I don’t use these procedures in my utility macros because I try to avoid as much “noise” in the SAS log as possible. But these procedures are great in other cases to save and reset options.
The syntax for the three macros is as follows:
%_turn_log_options_off
%_tun_log options_on
%_reset_log_options
There are no parameters for these macros.
Program A.11 shows a test macro that has some macro statements, followed by a DATA step. SAS Log A.9 shows the output when the macro is run using the default option settings in a SAS session.
Program A.11: Standard Output from a Macro
%macro test;
%let x = 1;
%if (&x eq 1) %then
%do;
data _null_;
putlog "In data step";
run;
%put mprint: %sysfunc(getoption(mprint));
%put mlogic: %sysfunc(getoption(mlogic));
%put symbolgen: %sysfunc(getoption(symbolgen));
%put source: %sysfunc(getoption(source));
%put notes: %sysfunc(getoption(notes));
%end;
%mend test;
%test;
SAS Log A.9: Standard Output from a Macro
15 %test
In data step
NOTE: DATA statement used (Total process time):
real time 0.00 seconds
cpu time 0.00 seconds
mprint: NOMPRINT
mlogic: NOMLOGIC
symbolgen: NOSYMBOLGEN
source: SOURCE
notes: NOTES
Program A.12 runs %_turn_log_options_on and then the same %TEST macro, and SAS Log A.10 shows the output that is created.
Program A.12: Running %_turn_log_options_on
%_turn_log_options_on;
%test;
SAS Log A.10: Running %_turn_log_options_on
17 %test;
MLOGIC(TEST): Beginning execution.
MLOGIC(TEST): %LET (variable name is X)
SYMBOLGEN: Macro variable X resolves to 1
MLOGIC(TEST): %IF condition (&x eq 1) is TRUE
MPRINT(TEST): data _null_;
MPRINT(TEST): putlog "In data step";
MPRINT(TEST): run;
In data step
NOTE: DATA statement used (Total process time):
real time 0.00 seconds
cpu time 0.00 seconds
MLOGIC(TEST): %PUT mprint: %sysfunc(getoption(mprint))
mprint: MPRINT
MLOGIC(TEST): %PUT mlogic: %sysfunc(getoption(mlogic))
mlogic: MLOGIC
MLOGIC(TEST): %PUT symbolgen: %sysfunc(getoption(symbolgen))
symbolgen: SYMBOLGEN
MLOGIC(TEST): %PUT source: %sysfunc(getoption(source))
source: SOURCE
MLOGIC(TEST): %PUT notes: %sysfunc(getoption(notes))
notes: NOTES
MLOGIC(TEST): Ending execution.
Program A.13 runs %_turn_log_options_off and then the same %TEST macro, and SAS Log A.11 shows the output that is created.
Program A.13: Running %_turn_log_options_off
%_turn_log_options_off;
%test;
SAS Log A.11: Running %_turn_log_options_off
In data step
mprint: NOMPRINT
mlogic: NOMLOGIC
symbolgen: NOSYMBOLGEN
source: NOSOURCE
notes: NONOTES
Finally, Program A.14 runs %_RESET_LOG_OPTIONS and then the same %test macro, and SAS Log A.12 shows that the output is back to the original output.
Program A.14: Running %_reset_log_options
%_reset_log_options;
%test;
SAS Log A.12: Running %_reset_log_options
21 %test;
In data step
NOTE: DATA statement used (Total process time):
real time 0.00 seconds
cpu time 0.00 seconds
mprint: NOMPRINT
mlogic: NOMLOGIC
symbolgen: NOSYMBOLGEN
source: SOURCE
notes: NOTES
The %_debug_macro_note macro writes a note in the log with the name of the macro and a list of the parameters and their values. I call this macro at the beginning of all the non-utility macros in my systems. This is just for debugging, so it only writes notes if the &__DEBUG variable is set to a number greater than 0.
Do not call this macro from a macro function; the function will fail.
The syntax is as follows:
%_debug_macro_note
There are no parameters for this macro.
Program A.15 and SAS Log A.13 show that the utility macro does nothing when &__DEBUG is set to 0.
Program A.15: Running %_debug_macro_note with &__DEBUG Set to 0
%macro test(a=, b=);
%_debug_macro_note;
%put In test macro;
%mend test;
%let __debug = 0;
%test (a = 1, b = 2);
SAS Log A.13: Running %_debug_macro_note with &__DEBUG Set to 0
7 %test (a = 1, b = 2);
In test macro
Program A.16 and SAS Log A.14 show the note that is displayed in the SAS log.
Program A.16: Running %_debug_macro_note with &__DEBUG Set to 1
%let __debug = 1;
%test (a = 1, b = 2);
SAS Log A.14: Running %_debug_macro_note with &__DEBUG Set to 1
9 %test (a = 1, b = 2);
DEBUG: =====
DEBUG: ===== Macro: TEST
DEBUG: ===== Parameters:
DEBUG: ===== &A = 1
DEBUG: ===== &B = 2
DEBUG: =====
In test macro
Tip: The code for this macro is a good example of figuring out where you are in the macro stack.
The %_refresh_macros macro can be run at any time to include all the SASAUTOS macros into the SAS session and submit them for you. It includes both external files and sas catalog source members.
This is a development macro. It is not meant to be used in a production process. It is meant to be used in a SAS session where you have a SASAUTOS library. If any of the macros are changed outside of the SAS session, then you have to include the updated macro into the SAS session and submit it in order to get the latest version. I run into this situation quite a bit when I’m working on Unix, because I prefer to use an editor other than the SAS Program Editor. Note that the macro does not include and submit macros that are part of the SAS system.
The macro uses the &__DEBUG macro variable. If it is blank or 0, then nothing is added to the SAS log (other than a note that tells you that the macros have been refreshed). If you set &__DEBUG to 1 or more, you get lots of information in the SAS log.
The syntax is as follows:
%_refresh_macros;
There are no parameters for this macro.
To use the macro, just call it from a SAS session as in Program A.17. SAS Log A.15 shows that only a note is displayed in the SAS log.
Program A.17: Running %_refresh_macros
%_refresh_macros;
SAS Log A.15: Running %_refresh_macros
1 %_refresh_macros;
===> SASAUTO macros have been refreshed