Creating an XML File using the XMLDoc object in PeopleCode

 The process in brief:

  1. Create and load a rowset
  2. Enrich the data in that rowset
  3. Create and XMLDoc object and load the data
  4. Use one of the XMLDoc’s built-in methods to generate an XML string.

For demonstration purposes, our premise is creating an XML file for a BI Publisher report.   However, there are other situations requiring XML Files.   The XMLDoc is extremely flexible and powerful enough to generate the XML file as required by the target specification.

Premise for XML File Creation Example

Demonstration premise: A letter to all employees summarizing their current job and showing all training taken through the company.  This is a complex data set with a parent/child relationship.  Each employee will get a separate page in the report.  For each employee, there will be zero or more training entries.  The class returns this Rowset object:

Employee Record: EE_RPT_LTR_VW (Parent Record) • EMPLID • NAME • DEPTID • LOCATION • JOBCODE • COMPRATE • CHANGE_PCT • LOCATION_DESCR • JOB_DESCR • COMPANY_DESCR • EMPLOYEE_PHOTO • Training Record: TRAINING_VW (Child Records) • EMPLID • COURSE_START_DT • COURSE • SESSION_NBR • COURSE_TITLE • COURSE_END_DT • ATTENDANCE • TRAINING_REASON • COURSE_GRADE • PREREQ_MET • DEPTID • BUSINESS_UNIT

XMLDoc method of creating an XML File

The XMLDoc object is PeopleTools utility to manipulate XML Files in their native format.  One of these objects methods is to generate a formatted XML string from the current XMLDoc structure.  The developer creates an XMLDoc structure and adds “Nodes” to the document to create tags and structure.  The level the node is attached to determines the structure of the XML document.

Though this code may look similar to a freehand method, realize that the object is maintaining the structure.  The developer is not concerned with end tags and completing parent child structures for the document.  As each node is added, the value of that node is set. 

When all the data has been moved to the XMLDoc object, use the “GenFormattedXmlString” method of the object to create the entire XML file in one string.  Then write that string to file.

import PSXP_RPTDEFNMANAGER:*;
import X_BI_PUB_PCODE:LoadTestData;

Local XmlDoc &inXMLDoc;
Local XmlNode &childNode, &textNode, &midNode, &rowNode;
Local File &XML_File;

Local X_BI_PUB_PCODE:LoadTestData &LoadTestData = create X_BI_PUB_PCODE:LoadTestData();
&RS_Employee = &LoadTestData.LoadTestDataSet();

/* Create & Load XMLDoc */
&inXMLDoc = CreateXmlDoc("<?xml version='1.0'?><root/>");

For &i = 1 To &RS_Employee.ActiveRowCount
   &rowNode = &inXMLDoc.DocumentElement.AddElement("EMPLOYEE_DATA");
   &EE_RPT_LTR_VW_REC = &RS_Employee(&i).GetRecord(Record.EE_RPT_LTR_VW);
   
   For &f = 1 To &EE_RPT_LTR_VW_REC.FieldCount
      &childNode = &rowNode.AddElement(&EE_RPT_LTR_VW_REC.GetField(&f).Name);
      &childNode.NodeValue = String(&EE_RPT_LTR_VW_REC.GetField(&f).Value);
   End-For; /* record fields */
   
   &RS_Training = &RS_Employee(&i).GetRowset(Scroll.TRAINING_VW);
   For &j = 1 To &RS_Training.ActiveRowCount
      &X_TRAINING_VW_REC = &RS_Training(&j).GetRecord(Record.TRAINING_VW);
      &midNode = &rowNode.AddElement("TRAINING_DATA");
      
      For &f = 1 To &TRAINING_VW_REC.FieldCount
         &childNode = &midNode.AddElement(&TRAINING_VW_REC.GetField(&f).Name);
         &childNode.NodeValue = String(&TRAINING_VW_REC.GetField(&f).Value);
      End-For; /* record fields */
   End-For; /* training rowset */
End-For; /* employee rowset */

&Xml_String = &inXMLDoc.GenFormattedXmlString();

&XML_File = GetFile("XMLDOC.xml", "W", "UTF8");
&XML_File.WriteLine(&Xml_String);

/* save file name and path for publishing */
&XML_Filename_path = &XML_File.Name;
&XML_File.Close();

XML File Generation Considerations

Concurrency: Generating multiple versions of the same report simultaneously 

PeopleSoft by its very nature may have multiple, different users attempting to create the same report at the same time.  Ensure that your code accounts for this probability by making the file names unique per each instance of the report.  One idea to accomplish this is to append the User or Employee ID to the file creating a unique file name. 

Another method is to append the current date/time stamp to the filename to force uniqueness. 

Schema Files

For PeopleTools version 8.48 & 8.49, BI Publisher requires XML Schema files for the correct mapping of tags to a PDF Template.  You could include these for RTF (MSWord) templates, but they were not necessary.

The easiest way to generate correct schemas for the generated XML Files is to use an XML File editing utility such as XML Spy or XML Fox.  XML Fox is a freeware utility that works very nicely.  These utilities can generate schemas from your final XML Files.

Starting with PeopleTools 8.50, BI Publisher does not require a schema file for PDF Template mapping.  Only a sample data file is required.

Sample Files for defining BI Publisher Data Sources

You must provide a Sample XML Data file when creating a new BI Publisher report.  A good idea is to execute a version of the XML File generation code using a sample or representative employee.  Another good idea is to, for the sample run, modify your code to put the XML Tag names as values in the XML File.  This will aid in mapping and formatting: do you have the right field in the right place on your report?

House Keeping

Depending on data policies of the implementations site, it may be a good idea to delete the BI Publisher Source files from the files directory after the report documents have been created.  The easiest way to delete the file is to re-open the file after the BI Publisher code is complete, then use the “delete()” method instead of the “close()” method.  This will remove the file from the servers.

Appendix A

Data set class for data source creation.  This function populates and returns a complex RowSet object.

class LoadTestData
   method LoadTestData();
   method LoadTestDataSet() Returns Rowset;
   
private
   
   method LoadEmployeeImage(&Emplid As string) Returns string;
   
end-class;

method LoadTestData
   
end-method;

method LoadTestDataSet
   /+ Returns Rowset +/
   
   Local Rowset &RS_Training, &RS_Employees;
   Local Record &JOB_REC, &LOCATION_TBL_REC, &COMPANY_TBL_REC, &JOBCODE_TBL_REC;
   Local integer &i;
   
   /* create records */
   &JOB_REC = CreateRecord(Record.JOB);
   &LOCATION_TBL_REC = CreateRecord(Record.LOCATION_TBL);
   &COMPANY_TBL_REC = CreateRecord(Record.COMPANY_TBL);
   &JOBCODE_TBL_REC = CreateRecord(Record.JOBCODE_TBL);
   
   /* create rowsets */
   &RS_Training = CreateRowset(Record.TRAINING_VW); /* child rowset */
   &RS_Employees = CreateRowset(Record.EE_RPT_LTR_VW, &RS_Training); /* parent rowset */
   
   /* Fill Parent */
   &RS_Employees.Fill("where emplid like 'KU00%' and exists (select 'x' from ps_training t where t.emplid = fill.emplid)");
   
   /* Loop through parent rowset for processing on each row */
   For &i = 1 To &RS_Employees.ActiveRowCount
      
      /* Fill child rowset */
      &RS_Training = &RS_Employees(&i).GetRowset(Scroll.TRAINING_VW);
      &RS_Training.Fill("where emplid = :1", &RS_Employees(&i).EE_RPT_LTR_VW.EMPLID.Value);
      
      /* Get job row for linking other data */
      &JOB_REC.EMPLID.Value = &RS_Employees(&i).EE_RPT_LTR_VW.EMPLID.Value;
      &JOB_REC.EMPL_RCD.Value = 0;
      /* get the current effdt & effseq for the EEs job row */
      SQLExec("select %dateout(j.effdt), j.effseq from ps_job j where j.emplid = :1 and j.empl_rcd = :2 and j.effdt = (select max(j2.effdt) from ps_job j2 where j2.emplid = j.emplid and j2.empl_rcd = j.empl_rcd and j2.effdt <= %datein(:3)) and j.effseq = (select max(j3.effseq) from ps_job j3 where j3.emplid = j.emplid and j3.empl_rcd = j.empl_rcd and j3.effdt = j.effdt)", &JOB_REC.EMPLID.Value, &JOB_REC.EMPL_RCD.Value, %Date, &JOB_REC.EFFDT.Value, &JOB_REC.EFFSEQ.Value);
      &JOB_REC.SelectByKey();
      
      /* retrieve specific location data for description in report */
      &LOCATION_TBL_REC.SETID.Value = &JOB_REC.SETID_LOCATION.Value;
      &LOCATION_TBL_REC.LOCATION.Value = &JOB_REC.LOCATION.Value;
      &LOCATION_TBL_REC.SelectByKeyEffDt(%Date);
      &RS_Employees(&i).EE_RPT_LTR_VW.LOCATION_DESCR.Value = &LOCATION_TBL_REC.DESCR.Value;
      
      /* retrieve specific company data for description in report */
      &COMPANY_TBL_REC.COMPANY.Value = &JOB_REC.COMPANY.Value;
      &COMPANY_TBL_REC.SelectByKeyEffDt(%Date);
      &RS_Employees(&i).EE_RPT_LTR_VW.COMPANY_DESCR.Value = &COMPANY_TBL_REC.DESCR.Value;
      
      /* retrieve specific jobcode data for description in report */
      &JOBCODE_TBL_REC.SETID.Value = &JOB_REC.SETID_JOBCODE.Value;
      &JOBCODE_TBL_REC.JOBCODE.Value = &JOB_REC.JOBCODE.Value;
      &JOBCODE_TBL_REC.SelectByKeyEffDt(%Date);
      &RS_Employees(&i).EE_RPT_LTR_VW.JOB_DESCR.Value = &JOBCODE_TBL_REC.DESCR.Value;
      
      /* get employee image */
      rem  &RS_Employees(&i).EE_RPT_LTR_VW.EMPLOYEE_PHOTO.Value = %This.LoadEmployeeImage(&JOB_REC.EMPLID.Value);
      
   End-For;
   
   Return &RS_Employees;
   
end-method;

method LoadEmployeeImage
   /+ &Emplid as String +/
   /+ Returns String +/
   
   Local File &Image_File;
   Local string &Base64String, &NewFileName, &FQ_Filename_path;
   Local integer &retcode;
   
   &NewFileName = %UserId | %Datetime | ".jpg";
   &Image_File = GetFile(&NewFileName, "W");
   &FQ_Filename_path = &Image_File.Name;
   &Image_File.Close();
   
   &retcode = GetAttachment("record://X_EPHOTO_VW", &Emplid, &FQ_Filename_path);
   
   rem MessageBox(0, "", 0, 0, "result: %1", &retcode);
   
   If &retcode < 2 Then
      &Image_File = GetFile(&FQ_Filename_path, "R", %FilePath_Absolute);
      &Base64String = &Image_File.GetBase64StringFromBinary();
      &Image_File.Close();
   End-If;
   
   /* delete file */
   &Image_File = GetFile(&FQ_Filename_path, "R", %FilePath_Absolute);
   &Image_File.Delete();
   
   Return &Base64String;
end-method;

Appendix B

XMLDoc XML File Example

<?xml version="1.0"?>
<root>
      <EMPLOYEE_DATA>
          <EMPLID>KU0001</EMPLID>
          <NAME>Douglas Lewis</NAME>
          <DEPTID>ADMIN</DEPTID>
          <LOCATION>KUNY00</LOCATION>
          <JOBCODE>700005</JOBCODE>
          <COMPRATE>21666.666667</COMPRATE>
          <CHANGE_PCT>0</CHANGE_PCT>
          <LOCATION_DESCR>Corporation Headquarters</LOCATION_DESCR>
          <JOB_DESCR>President & CEO</JOB_DESCR>
          <COMPANY_DESCR>Global Business Institute</COMPANY_DESCR>
          <TRAINING_DATA>
               <EMPLID>KU0001</EMPLID>
               <COURSE_START_DT>2005-01-13</COURSE_START_DT>
               <COURSE>M2005</COURSE>
               <SESSION_NBR>0001</SESSION_NBR>
               <COURSE_TITLE>course with 2000 hours</COURSE_TITLE>
               <COURSE_END_DT/>
               <ATTENDANCE>E</ATTENDANCE>
               <TRAINING_REASON/>
               <COURSE_GRADE/>
               <PREREQ_MET>N</PREREQ_MET
               ><DEPTID>ADMIN</DEPTID>
               <BUSINESS_UNIT>GBIBU</BUSINESS_UNIT>
         </TRAINING_DATA>
      </EMPLOYEE_DATA>
</root>

BI Publisher

From Application Engine

For printing .XLS/.XLSX format output reports in process monitor logs/report repository in BI Publisher reports , We need to add below piece of code in your App Engine,

SELECT PRCSOUTPUTDIR FROM PSPRCSPARMS WHERE PRCSINSTANCE = AETRecord.ProcessInstance

Next, Assign the value of PRCSOUTPUTDIR to Report definition object.

&oRptDefn = create PSXP_RPTDEFNMANAGER:ReportDefn(“Report_ID”);

&oRptDefn.Get();

&oRptDefn.OutDestination = “Value from PRCSOUTPUTDIR”;

PeopleCode to Create XML file – BI Publisher Report

In Bi Publisher there are three methods to create XML files as data source

  • PS Query Based
  • Row-set Based
  • Manually Creating XML file

To create XML files through PS Query and Row-set methods, need to import delivered App package ”PSXP_XMLGEN”

PeopleCode to create XML file through Rowset:

Local Rowset &rsRowset;

Local File &fXMLFile, &fXSDFile;

Local string &strXSDFileName, &strXMLData, &strXMLSchema, &strPSHome;

Local PSXP_XMLGEN:RowSetDS &objRowsetDS = create PSXP_XMLGEN:RowSetDS();

&strFileName = &filename;

&rsRowset = &rs;

&strXMLFileName = %This.XMLFilePath | &strFileName | “.xml”;

&strXSDFileName = %This.XMLFilePath | &strFileName | “.xsd”;

/*******************Create XML Schema File *****************/

&strXMLSchema = &objRowsetDS.getXSDSchema(&rsRowset);

&fXSDFile = GetFile(&strXSDFileName, “w”, %FilePath_Absolute);

&fXSDFile.WriteLine(&strXMLSchema);

&fXSDFile.Close();

/*******************Create XML File *****************/

&strXMLData = &objRowsetDS.getXMLData(&rsRowset, &strXSDFileName);

&fXMLFile = GetFile(&strXMLFileName, “w”, %FilePath_Absolute);

&fXMLFile.WriteLine(&strXMLData);

&fXMLFile.Close();


PeopleCode to create XML file manually:


Local XmlDoc &inXMLDoc;

Local XmlNode &xRootNode, &xMonthNode, &xAmountNode, &xSummaryNode,;

&inXMLDoc = CreateXmlDoc(“<?xml version=’1.0′, encoding=’UTF-8’STANDALONE=’yes’?><ROOT/>”);

&inXMLDoc.DocumentElement.AddAttribute(“xmlns:xsi”,“http://www.w3.org/2001/XMLSchema-instance&#8221;);

&xRootNode = &inXMLDoc.DocumentElement.AddElement(“MAIN”);

&xSummaryNode = &xRootNode.AddElement(“SUMMARY”);

&xMonthNode1 = &xSummaryNode.AddElement(“JAN”);

&xMonthNode1.NodeValue = String(&nbrAmount);

&xMonthNode2 = &xSummaryNode.AddElement(“DEC”);

&xMonthNode2.NodeValue = String(&nbrAmount);

/*To add Child Nodes dynamically either through For loop or While Loop*/

&sqlSummary=CreateSQL(“SELECT MONTH,AMOUNT FROM PS_TEST_TBL”);

While &sqlSummary.Fetch(&nbrMonth, &nbrAmount)

&xDataNode = &xRootNode.AddElement(“SUMMARYDATA”);

&xMonthNode = &xDataNode.AddElement(“MONTH”);

&xMonthNode.NodeValue = &strMonth;

&xAmountNode = &xDataNode.AddElement(“AMOUNT”);

&xAmountNode.NodeValue = String(&nbrAmount);

End-While;


&strXMLFileName = %This.FilePath | &strFileName | “.xml”;

&fXMLFile = GetFile(&strXMLFileName, “w”, %FilePath_Absolute);

If &fXMLFile.IsOpen Then

&strXMLData = &inXMLDoc.GenFormattedXmlString();

&fXMLFile.WriteLine(&strXMLData);

&fXMLFile.Close();

Else

Error (“Temporary XML file creation failed. Please contact your System Administrator”);

End-If;


PeopleCode to display a report in PDF format:


Import PSXP_RPTDEFNMANAGER:*;

Local string &strRptDefnId, &strTemplateId, &strLanguageCd, &strOutputfmt, &strOutDest;

Local date &dtAsofdate;

Local Rowset &rsOutput;

Local PSXP_RPTDEFNMANAGER:ReportDefn &objReportdefn;


&strFileName = “APPaycheck”;

&strRptDefnId =” APPAYCHECK”;;

&strTemplateId = “APPAYCHECK_1”;

&strOutputfmt = “PDF”;

&strLanguageCd = “ENG”;

&dtAsofdate = %Date;

&strXMLFileName = %This.XMLFilePath | &strFileName | “.xml”;

&strOutDest = %This.XMLFilePath;

&objReportdefn = create PSXP_RPTDEFNMANAGER:ReportDefn(&strRptDefnId);

&objReportdefn.Get();

&objReportdefn.SetRuntimeDataXMLFile(&strXMLFileName);


/*To store the PDF report with the given name, provide the following properties to the report definition object*/


&objReportdefn.OutDestination = &strOutDest;

&objReportdefn.ReportFileName = &strFileName;


/*Generate and Display PDF Report*/

&objReportdefn.ProcessReport(&strTemplateId, &strLanguageCd, &dtAsofdate, &strOutputfmt); CommitWork();

/*To display the report on line*/

&objReportdefn.DisplayOutput();


/*To print the report*/

&objReportdefn.PrintReport();

/*PeopleCode to delete the XML file created after displaying the report */

try

Local File &fTempFile = GetFile(&strXMLFileName, “W”, %FilePath_Absolute);

&fTempFile.Delete();

catch Exception &ExCleanup

Warning (“Clean-Up of temporary XML File failed with Exception:” | &ExCleanup.ToString());

end-try;

Excel to Component Interface Utility

  To use the Excel to Component Interface utility, you must grant access to the iScript WEBLIB_SOAPTOCI in the permission list of the user w...