- Overview
- Prerequisites
- Installation
- How PIV Works
- Command Syntax
- Example
- List of PIV Commands
- Built-in functions
PIV is a REXX procedure that you can run in batch to verify that the system is in the state you expect it to be after making a change.
How is this better than simply running SDSF in batch?
- There is less clutter because there is no SDSF screen rendering
- You can programmatically examine the SDSF command response lines (or file content) and set an appropriate condition code.
Because of this it can be used as an audit tool to verify, for example, that PARMLIB members contain the correct values, or system software is at the expected maintenance level etc. This is much quicker than doing a manual audit check on possibly many systems.
PIV can be used to:
- issue z/OS commands and Unix command and verify that the response is as expected
- read z/OS datasets or Unix files and verify that they contain the expected strings
- read JES sysout datasets (including those in the job executing the PIV REXX) and verify that the content is as expected.
The result of this checking is indicated in the job step condition code of the step running the PIV EXEC. Usually COND CODE 0 means "all good" and COND CODE 4 means "PIV failed" - but you can set whatever condition code you want. This enables you to take further automated action depending on the condition code.
PIV also prints a log of its own activity, including all commands issued and responses received, which makes for an excellent audit artifact.
- To fully benefit from PIV you will need appropriate RACF authorisations to:
- Issue SDSF console commands
- Read SDSF spool files
- Read Unix System Services files
- Issue Unix System Services commands
However, you will still be able to use a subset of its features (for example, examining your own SDSF job output) if your site allows that.
-
Install the following REXX file into your RECFM=VB LRECL=255 REXX library (FB80 is sooooo last century):
File Description PIV The PIV processor
The PIV REXX procedure uses the SDSF REXX interface to:
- Extract information from SDSF
- Read specified spool files
- Issue z/OS commands
It can also:
- Issue Unix commands and retrieve the resulting output
- Read z/OS datasets and Unix files
As each command is executed, the response (e.g. console command output,
or file contents) is stored in REXX line.n
variables for subsequent
testing (for example, by ASSERT
commands).
The PIV SDSF commands are pretty much identical to the SDSF primary commands that everyone is familiar with, so the learning curve is minimal.
However, there are some additional PIV commands that can be used to verify
the system state (ASSERT
, IF
and USING
, PASSIF
, FAILIF
) and set the
job step condition code accordingly.
The PIV REXX procedure prints a simple report (to SYSTSPRT) of its activities which include:
- The command issued (e.g
/D ETR
, orREAD SYS1.PARMLIB(IEASYS00)
, etc) - The command response received (or the file contents read)
- The result of any verification commands (e.g.
ASSERT
)
At the end of the report, a simple one-line statement is written to indicate success or failure of the PIV process:
- Result: PIV successful
- Result: PIV failed with maxrc=n
All commands and tests are read from DD INPUT.
The syntax rules for the input file are:
-
CONTINUATIONS
A line continuation is indicated by a comma (,) as the last non-blank character (i.e. the same as a REXX continuation).
-
COMMENTS
Lines with a blank in column 1 are treated as comments and are simply printed to SYSTSPRT as-is.
-
COMMANDS
A line NOT beginning with a blank in column 1 is a command to be processed. All commands are case-insensitive. The response from each command is returned in REXX variables called
line.n
where n ranges from 1 to the number of response lines andline.0
contains the number of response lines.
This example checks the health of a fictitious software product:
//PIVTEST JOB ,,CLASS=A,MSGCLASS=X,NOTIFY=&SYSUID
//PIV EXEC PGM=IKJEFT1A,PARM='%PIV'
//SYSEXEC DD DISP=SHR,DSN=your.exec.lib
//SYSTSPRT DD SYSOUT=*
//INPUT DD *
Ensure the product started tasks are all running. We expect
that both MYSTCA and MYSTCB are up:
prefix MY*
da
assert find('MYSTCA')
assert find('MYSTCB')
Check that the software version is correct and is licensed:
select mystca sysprint
assert find('MY0001I','VERSION', '1.0.39')
assert \find('MY0002E','LICENSE EXPIRED')
assert \find('MY0003W','LICENSE ABOUT TO EXPIRE')
Check that the software configuration settings are correct:
read SYS1.PARMLIB(MYCONFIG)
assert isPresent('STC=MYSTCA')
Check that the software is responding to commands:
/F MYSTCA,DISPLAY,VERSION
assert isPresent('MY0001I')
Just for shiggles, check that the Server Time Protocol is active:
/D ETR
using keyword . . mode .
passif keyword = 'SYNCHRONIZATION' & mode = 'STP'
using . . . . keyword level .
passif keyword = 'STRATUM' & level < 3
/*
All commands must start in column 1 and are case insensitive. The uppercase letters are the minimum abbreviation for each of the commands shown below.
Command | Description |
---|---|
HELP | Display this help information (in batch) |
/zoscommand | Issue a z/OS command (e.g. /D ETR) |
Reply msgno replytext | Reply to a particular WTOR (identified by message number) after a z/OS command is issued |
PREfix [jobname] | Filter SPOOL files by job name. Reset the filter by omitting the job name |
OWNer [userid] | Filter SPOOL files by owner name. Reset the filter by omitting the owner name |
SYSname [sysname] | Filter SPOOL files by system name. Reset the filter by omitting the system name |
DEST [destname] | Filter SPOOL files by destination name. Reset the filter by omitting the destination name |
SORT sortspec | Set the column sorting order for tabular output from an SDSF primary command (DA, I, O, etc) |
sdsfcommand | Issue an SDSF primary command (e.g. DA, I, O, etc) - optionally filtered by prior PREFIX , OWNER , SYSNAME and DEST commands |
? | List the spool datasets for this job. You must first issue an SDSF primary command (DA, I, O, etc) |
Select jobname [sysoutdd [stepname [procstep]] | Read the specified JES SPOOL dataset(s) for a job into REXX line.n stem variables. The datasets to be read can optionally be filtered by sysoutdd etc). You must first issue an SDSF primary command (DA, I, O, etc) |
USS usscommand | Issue a USS command (e.g. cat /etc/profile) and capture the output into REXX line.n stem variables |
READ {dsn | ddname | pathname} | Read a dataset or PDS member, a DD name, or a Unix file into REXX line.n stem variables |
SET var = value | Set a REXX variable (usually, an SDSF special variable such as ISFPRTDSNAME) prior to issuing an SDSF export command (XD, XDC, XF, XFC) |
XD jobname [sysoutdd [stepname [procstep]] | Export and append the specified JES sysout dataset to a dynamically allocated output dataset (identified by the ISFPRTDSNAME REXX variable) |
XDC jobname [sysoutdd [stepname [procstep]] | Same as XD but Close the dynamically allocated output dataset afterwards |
XF jobname [sysoutdd [stepname [procstep]] | Export and append the specified JES sysout dataset to a pre-allocated output DD name (identified by the ISFPRTDDNAME REXX variable) |
XFC jobname [sysoutdd [stepname [procstep]] | Same as XF but Close the pre-allocated output DD name afterwards |
SHow ? | Display the gamut of column names for the most recently issued SDSF primary command (DA, I, O, etc) |
SHow column [column ...] | Display the values in just the specified columns (after issuing an SDSF primary command (DA, I, O, etc) |
SHow ON | Enable automatic display of command responses |
SHow OFF | Suppress display of command responses |
SHow | Display the response lines for the previous command (even if automatic display is suppressed) |
SHow nnn | Limit command response output to nnn lines |
SHow heading[,m[,n]] | Display a command response heading of heading, and limit command response output to lines m to n |
SHow 'str [str ...]' | Display only those command response lines at contain at least one of the specified strings |
ASSERT expression | Evaluate the REXX expression expression and set return code (rc) 0 if true, or 4 if false |
IF expression THEN rc = n; ELSE ... | Evaluate the REXX expression expression and set rc to a user-specified return code n |
USING template | Define a REXX parsing template for use by the PASSIF and FAILIF commands |
PASSIF expression | Set return code 0 if a response line is found where the expression evaluates to 1 (true), else sets return code 4 |
FAILIF expression | Set return code 4 if a response line is found where the expression evaluates to 1 (true), else sets return code 0 |
Prints this help information in batch.
Any command prefixed with a /
character is submitted to z/OS
for execution as an operator command. For example,
/D ETR
You will need RACF authorisation for this to be successful.
Reply msgno replytext
If *nn msgno
is found in the command response then reply automatically by issuing:
/R nn,replytext
You can specify many REPLY
commands and the first matching msgno will be actioned.
Filter SPOOL files by job name. Reset the filter by omitting the job name.
Filter SPOOL files by owner name. Reset the filter by omitting the owner name.
Filter SPOOL files by system name. Reset the filter by omitting the system name.
Filter SPOOL files by destination name. Reset the filter by omitting the destination name.
Set the column sorting order for tabular output from an SDSF primary command (DA, I, O, etc). You must enter column names rather than column titles (e.g. SORT DATEE D TIMEE D). Issue the SHOW ? command to list the gamut of column names after you have selected the SDSF primary command (DA, I, O etc).
Issue an SDSF primary command (DA
, O
, H
, I
, ST
, PS
, etc). The response
is stored in line.n
REXX variables. Only a subset of the columns
is returned, but you can adjust this by updating the PIV REXX
procedure to set the g.0COLS.
variables as required.
List spool datasets for a job (when issued after DA, O, H, I or ST primary command).
This is a PIV command to read the sysout of the
specified jobname into the line.n
variables - after you have issued
an SDSF primary command (DA, O, I etc).
You can refine the sysout to be selected by sysoutdd, step name, and proc step name.
This is a PIV command to issue a Unix System Services
shell command and retrieve the command output into line.n
variables.
For example:
USS cat /etc/profile
This is a PIV command to read the contents of the
specified dataset into the line.n
variables.
If dsn
contains a /
then it is treated as a (case-sensitive)
Unix path name. The content of the file will be
converted to EBCDIC (IBM-1047) if necessary.
If dsn
contains a .
then it is treated as a (case-insensitive)
fully qualified dataset name that will be
dynamically allocated.
Otherwise it is assumed to be the name of a pre-allocated DD in your JCL.
For example,
Command | Description |
---|---|
read sys1.parmlib(ieasys00) | Reads the IEASYS00 member of SYS1.PARMLIB |
read /etc/profile | Reads the contents of the Unix /etc/profile file |
read MYDD | Reads the dataset allocated as DD: MYDD |
This is a PIV command to set an SDSF/REXX interface special variable.
The complete list of valid variables is described in "SDSF Operation and Customization" (manual SA23-2274) in Table 180 "Special REXX Variables".
Some useful ones (when using XD
, XDC
, XF
, or XFC
) are:
Variable | Meaning |
---|---|
ISFPRTDDNAME | Export DD name (for XF/XFC) |
ISFPRTDSNAME | Export data set name (for XD/XDC) |
ISFPRTDISP | Disposition (NEW/OLD/SHR/MOD) |
ISFPRTBLKSIZE | Default 3120 (27994 is better) |
ISFPRTLRECL | Default 240 |
ISFPRTRECFM | Default VBA |
ISFPRTSPACETYPE | Default BLKS |
ISFPRTPRIMARY | Default 500 |
ISFPRTSECONDARY | Default 500 |
Export and append the sysout of the specified jobname to the dataset specified in REXX special variable ISFPRTDSNAME (see the SET command above).
You can refine the sysout to be selected by sysoutdd, step name, and proc step name.
You must have already issued either DA
, I
, H
, O
or ST
before you can export spool files.
For example,
Command | Description |
---|---|
o xd myjob |
Displays the SDSF output panel columns (subject to filtering by PREFIX etc)Reads all output spool datasets from MYJOB |
xd myjob sysprint | Reads only SYSPRINT output from all steps of MYJOB |
xd myjob sysprint asmlink | Reads only SYSPRINT output from step ASMLINK of MYJOB |
xd myjob sysprint asmlink asm | Reads only SYSPRINT output from procstep ASM of step ASMLINK of MYJOB |
This is the same as XD
command except that the output
dataset is also Closed.
Export and append the sysout of the specified jobname to the pre-allocated DD specified in REXX special variable ISFPRTDDNAME (see the SET command above).
You can refine the sysout to be selected by sysoutdd, step name, and proc step name.
You must have already issued either DA
, I
, H
, O
or ST
before you can export spool files.
This is the same as XF
command except that the output
ddname is also Closed.
List the possible column names for the SDSF primary command last issued (e.g. DA, O, etc)
For example, to list all the available column names for the DA command:
DA
SHOW ?
This is a convenience command so that you don't have to refer to the manual.
Display specific columns (for example, JNAME, JOBID, PNAME). Use SHOW ? to list all the valid column names that you can specify for a particular ISPF panel (DA, O, ST, etc).
Automatically print command output. This is the default.
Suppress printing of command output.
Display either the contents of the default columns appropriate for
the SDSF primary command last issued or the current
contents of the line.n
variables.
This is useful when you have suppressed displaying responses (using SHOW OFF) and want to display the response for a particular command.
Limit the acquired output to n lines maximum. The default is 1000
lines. This applies to output from all commands (i.e. z/OS command
output, USS commands, and lines read from datasets or files). It
essentially limits the number of line.n
stem variables used.
Prints the heading text followed by lines with line numbers between m and n. The default is to print ALL lines. A negative number is relative to the last line, so for example:
SHOW last few lines,-5
...will print the heading LAST FEW LINES
followed
by the last 5 lines of output.
Prints only output lines that contain at least one of the specified strings.
This is useful when you only want to see messages relating to a limited subject, for example:
show 'LICENSE EXPIR'
...would show lines containing any of 'LICENSE', 'LICENSED', 'UNLICENSED', 'EXPIRED' 'EXPIRES' 'EXPIRATION'.
The REXX expression expression is evaluated and if true then the return code is set to 0 (pass), else the return code is set to 4 (fail).
The test expression can be any expression that is
valid after a REXX if
statement. For example,
ASSERT isPresent('SOMETHING')
or
ASSERT find('SOMETHING')
...means:
"Set return code 0 if the string 'SOMETHING' is present in the command response, else set return code 4"
This is similar to ASSERT
but lets you set whatever return code
you want. Note that the PIV step will exit with the highest
return code you set by any of the IF
, ASSERT
, or FAILIF
PIV commands.
The IF
command has exactly the same syntax as the REXX if
statement. The ASSERT example above could be
equivalently written as:
if \isPresent('SOMETHING') then rc = 4
...which means:
"If the string 'SOMETHING' is not () present in the command response then set return code 4, else set return code 0"
The ELSE clause is rarely needed and is only included for completeness.
You can use a combination of either USING
and PASSIF
, or
USING
and FAILIF
to check for a response value that
may vary from time to time (i.e. is not simply a
constant string) and therefore needs to be
extracted from the command response and assigned to a REXX
variable in order to test it.
-
USING sets the parsing template to be applied to each line of the command response. The template can be anything that is valid after a REXX
parse var line.n
statement. -
PASSIF sets the return code to 0 if the REXX expression on PASSIF evaluates to 1 (true), or 4 if the expression evaluates to 0 (false).
-
FAILIF sets the return code to 4 if the REXX expression on FAILIF evaluates to 1 (true), or 0 if the expression evaluates to 0 (false).
For example, you may want to verify that spool
usage is acceptable by checking the output of a
$DSPOOL
JES command, which may look like:
$HASP893 VOLUME(SPOOL1) STATUS=ACTIVE,PERCENT=23
$HASP646 23.6525 PERCENT SPOOL UTILIZATION
To verify this, you could use:
/$DSPOOL
USING msgno percent .
PASSIF msgno = '$HASP646' & percent < 80
This will cause each line of the command response to be
parsed (using parse var line.n msg percent .
) causing
the first and second words of each
line to be assigned to the REXX variables msgno
and percent
respectively. If a line is found
where msgno
is '$HASP646' and the percent
value is less than 80, then return code 0 is set.
If no line containing $HASP646 is found, or the
percent value is more than 80, then return code
4 is set.
The processing of command response lines stops
when the expression on PASSIF
or FAILIF
is true.
Alternatively, you could examine the $HASP893 message (which has a lower resolution percentage):
/$DSPOOL
USING msgno . 'PERCENT='percent .
PASSIF msgno = '$HASP893' & percent < 80
This will cause each line of the command response to be parsed and the message number and percent value to be extracted. Processing is similar to the previous example.
There are a few convenience functions in the PIV
REXX procedure that you can use on the ASSERT
, IF
,
PASSIF
and FAILIF
commands if you want.
You can, of course, use any of the REXX built-in functions too. For example:
subword(line.3,2) = 'HI'
Function | Description |
---|---|
find(str) | Returns 1 if str is found in any line (else returns 0) |
find(str[,str ...]) | Returns 1 if ALL strings are found on one line (else returns 0) |
isPresent(str) | Returns 1 if str is found in any line (else returns 0) |
isPresent(str,line.n) | Returns 1 if str is found in line.n (else returns 0) |
isAbsent(str) | Returns 1 if str is not found in any line (else returns 0) |
isAbsent(str,line.n) | Returns 1 if str is not found in line.n (else returns 0) |
isNum(str) | Returns 1 if str is an integer (else returns 0) |
isWhole(str) | Returns 1 if str is an integer (else returns 0) |
isHex(str) | Returns 1 if str is non-blank hexadecimal (else returns 0) |
isAlpha(str) | Returns 1 if str is alphanumeric (else returns 0) |
isUpper(str) | Returns 1 if str is upper case alphabetic (else returns 0) |
isLower(str) | Returns 1 if str is lower case alphabetic (else returns 0) |
isMixed(str) | Returns 1 if str is mixed case alphabetic (else returns 0) |