| [1595] | 1 | C0PWS2    ; ERX/GPL - Web Service utilities; 8/31/09; 12/08/2010 ; 5/9/12 12:29am | 
|---|
|  | 2 | ;;1.0;C0P;;Apr 25, 2012;Build 103 | 
|---|
|  | 3 | ;Copyright 2009 George Lilly.  Licensed under the terms of the GNU | 
|---|
|  | 4 | ;General Public License See attached copy of the License. | 
|---|
|  | 5 | ; Modified by Chris Richardson, November, 2010. | 
|---|
|  | 6 | ; Code has been modified to accept very large XML documents and block them logically. | 
|---|
|  | 7 | ; 3101208 - RCR - Correct end of buffer condition, BF=">" | 
|---|
|  | 8 | QUIT | 
|---|
|  | 9 | ; | 
|---|
|  | 10 | ; TEST Lines below not intended for End Users. Programmers only. | 
|---|
|  | 11 | ; BEWARE ZWRITE SYNTAX. It may not work in other M Implementations. | 
|---|
|  | 12 | TEST(C0PDUZ,C0PDFN)     ; TEST RETRIEVAL OF PATIENT1 MEDS | 
|---|
|  | 13 | ;S DEBUG=1 ; | 
|---|
|  | 14 | D SOAP("C0POUT",6,C0PDUZ,C0PDFN) | 
|---|
|  | 15 | ZWRITE C0POUT ; Should use ^%ZOSV Node, this is very GT.M Specific | 
|---|
|  | 16 | QUIT | 
|---|
|  | 17 | ; | 
|---|
|  | 18 | ACCOUNTF()       QUIT 113059002  ; file number for account file | 
|---|
|  | 19 | ; | 
|---|
|  | 20 | XMLFN()  QUIT 113059001  ; XML TEMPLATE FILE NUMBER | 
|---|
|  | 21 | ; | 
|---|
|  | 22 | BINDFN()         QUIT 113059001.04 ; FILE NUMBER FOR BINDING SUBFILE | 
|---|
|  | 23 | ; | 
|---|
|  | 24 | ; | 
|---|
|  | 25 | GETTID(C0PWS,C0PTNAME)  ; EXTRINSIC WHICH RETURNS THE TEMPLATE ID FOR | 
|---|
|  | 26 | ; TEMPLATE NAMED C0PTNAME BELONGING TO WEB SERVICE NAMED C0PWS | 
|---|
|  | 27 | ; ALSO WORKS IF THE ACCOUNT NUMBER IS PASSED IN C0PWS | 
|---|
|  | 28 | S C0PXF=113059001 ; FILE NUMBER FOR THE C0P XML TEMPLATE FILE | 
|---|
|  | 29 | S C0PAF=113059002 ; FILE NUMBER FOR THE C0P WS ACCT FILE | 
|---|
|  | 30 | N C0PA,C0PT       ; C0P ACCOUNT AND C0P TEMPLATE | 
|---|
|  | 31 | DO | 
|---|
|  | 32 | . I C0PWS>0 S C0PA=C0PWS QUIT | 
|---|
|  | 33 | . ; | 
|---|
|  | 34 | . DO  ; NAME NOT RECORD NUMBER IS PASSED FOR ACCOUNT | 
|---|
|  | 35 | . . S C0PA=$O(^C0PS("B",C0PWS,"")) ; RECORD NUMBER OF ACCOUNT | 
|---|
|  | 36 | . . I C0PA="" D  Q  ; OOPS ACCOUNT NOT FOUND | 
|---|
|  | 37 | . . . W "ACCOUNT "_C0PWS_" NOT FOUND",! | 
|---|
|  | 38 | . . .QUIT | 
|---|
|  | 39 | . .QUIT | 
|---|
|  | 40 | .QUIT | 
|---|
|  | 41 | S C0PT=$O(^C0PX("C",C0PA,C0PTNAME,"")) ; RECORD NUMBER OF TEMPLATE | 
|---|
|  | 42 | ; WE USE THE C INDEX TO INSURE THAT THE TEMPLATE BELONGS TO THE WEB SERVICE | 
|---|
|  | 43 | Q C0PT | 
|---|
|  | 44 | ; | 
|---|
|  | 45 | RESTID(C0PDUZ,C0PTID)   ; RESOLVE TEMPLATE ID FROM SUBSCRIPTION | 
|---|
|  | 46 | ; | 
|---|
|  | 47 | N C0PAIEN,COPACCT,COPWBS,COPUTID | 
|---|
|  | 48 | S C0PAIEN=$$SUBINIT^C0PSUB(C0PDUZ) ;IEN OF SUBSCRIPTION | 
|---|
|  | 49 | ; N C0PACCT | 
|---|
|  | 50 | S C0PACCT=$$GET1^DIQ(C0PSUBF,C0PAIEN_","_C0PDUZ_",",1,"I") ;ACCT | 
|---|
|  | 51 | ; N C0PWBS | 
|---|
|  | 52 | S C0PWBS=$$GET1^DIQ(C0PAF,C0PACCT_",",4,"I") ;WEB SERVICE IEN | 
|---|
|  | 53 | ; N C0PUTID | 
|---|
|  | 54 | S C0PUTID=$$GETTID(C0PWBS,C0PTID) ;TEMPLATE ID | 
|---|
|  | 55 | Q C0PUTID | 
|---|
|  | 56 | ; | 
|---|
|  | 57 | SOAP(C0PRTN,C0PTID,C0PDUZ,C0PDFN,C0PVOR)        ; MAKES A SOAP CALL FOR | 
|---|
|  | 58 | ; TEMPLATE ID C0PTID | 
|---|
|  | 59 | ; RETURNS THE XML RESULT IN C0PRTN, PASSED BY NAME | 
|---|
|  | 60 | ; C0PVOR IS THE NAME OF A VARIABLE OVERRIDE ARRAY, WHICH IS APPLIED | 
|---|
|  | 61 | ; BEFORE MAPPING | 
|---|
|  | 62 | ; | 
|---|
|  | 63 | ; ARTIFACTS SECTION | 
|---|
|  | 64 | ; THE FOLLOWING WILL SET UP DEBUGGING ARTIFACTS FOR A POSSIBLE FUTURE | 
|---|
|  | 65 | ; ONLINE DEBUGGER. IF DEBUG=1, VARIABLES CONTAINING INTERMEDIATE RESULTS | 
|---|
|  | 66 | ; WILL NOT BE NEWED. | 
|---|
|  | 67 | I $G(DEBUG)="" N C0PV ; CATALOG OF ARTIFACT VARIABLES AND ARRAYS | 
|---|
|  | 68 | S C0PV(100,"C0PXF","XML TEMPLATE FILE NUMBER")="" | 
|---|
|  | 69 | S C0PV(200,"C0PHEAD","SOAP HEADER VARIABLE NAME")="" | 
|---|
|  | 70 | S C0PV(300,"header","SOAP HEADER")="" | 
|---|
|  | 71 | S C0PV(400,"C0PMIME","MIME TYPE")="" | 
|---|
|  | 72 | S C0PV(500,"C0PURL","WS URL")="" | 
|---|
|  | 73 | S C0PV(550,"C0PPURL","PROXY URL")="" | 
|---|
|  | 74 | S C0PV(600,"C0PXML","XML VARIABLE NAME")="" | 
|---|
|  | 75 | S C0PV(700,"xml","OUTBOUND XML")="" | 
|---|
|  | 76 | S C0PV(800,"C0PRSLT","RAW XML RESULT RETURNED FROM WEB SERVICE")="" | 
|---|
|  | 77 | S C0PV(900,"C0PRHDR","RETURNED HEADER")="" | 
|---|
|  | 78 | S C0PV(1000,"C0PRXML","XML RESULT NORMALIZED")="" | 
|---|
|  | 79 | S C0PV(1100,"C0PR","REPLY TEMPLATE")="" | 
|---|
|  | 80 | S C0PV(1200,"C0PREDUX","REDUX STRING")="" | 
|---|
|  | 81 | S C0PV(1300,"C0PIDX","RESULT XPATH INDEX")="" | 
|---|
|  | 82 | S C0PV(1400,"C0PARY","RESULT XPATH ARRAY")="" | 
|---|
|  | 83 | S C0PV(1500,"C0PNOM","RESULT DOM DOCUMENT NAME")="" | 
|---|
|  | 84 | S C0PV(1600,"C0PID","RESULT DOM ID")="" | 
|---|
|  | 85 | N ZI,ZN,ZS | 
|---|
|  | 86 | S ZN="" | 
|---|
|  | 87 | D:$G(DEBUG)=""   ; G NOTNEW ; SKIP NEWING THE VARIABLES IF IN DEBUG | 
|---|
|  | 88 | . S ZI="",ZN="",ZS="" | 
|---|
|  | 89 | . F  S ZI=$O(COPV(ZI)) Q:ZI=""  D | 
|---|
|  | 90 | . . ; S ZJ=$O(C0PV(ZI,"")) ; SET UP NEW COMMAND | 
|---|
|  | 91 | . . S ZN=ZN_ZS_$O(C0PV(ZI,"")),ZS="," | 
|---|
|  | 92 | . .QUIT | 
|---|
|  | 93 | .QUIT | 
|---|
|  | 94 | I $L(ZN) N @ZN  ; Apply collected NEW Variables 1 time | 
|---|
|  | 95 | ; NEW | 
|---|
|  | 96 | ; S ZI=$O(C0PV(ZI)) | 
|---|
|  | 97 | ; S ZJ=$O(C0PV(ZI,"")) ; SET UP NEW COMMAND | 
|---|
|  | 98 | ;W ZJ,! | 
|---|
|  | 99 | ; N @ZJ ; NEW THE VARIABLE | 
|---|
|  | 100 | ; I $O(C0PV(ZI))'="" G NEW ;LOOP TO GET NEW IN CONTEXT | 
|---|
|  | 101 | ;NOTNEW | 
|---|
|  | 102 | ; END ARTIFACTS | 
|---|
|  | 103 | ; | 
|---|
|  | 104 | D INITXPF("C0PF") ; SET FILE NUMBER AND PARAMATERS | 
|---|
|  | 105 | S C0PXF=C0PF("XML FILE NUMBER") ; FILE NUMBER FOR THE C0P XML TEMPLATE FILE | 
|---|
|  | 106 | D | 
|---|
|  | 107 | . I +C0PTID=0 D  Q  ; A STRING WAS PASSED FOR THE TEMPLATE NAME | 
|---|
|  | 108 | . . S C0PUTID=$$RESTID(C0PDUZ,C0PTID) ;RESOLVE TEMPLATE IEN FROM NAME | 
|---|
|  | 109 | . .QUIT | 
|---|
|  | 110 | . ; | 
|---|
|  | 111 | . S C0PUTID=C0PTID ; AN IEN WAS PASSED | 
|---|
|  | 112 | .QUIT | 
|---|
|  | 113 | N xml,template,header | 
|---|
|  | 114 | S C0PHEAD=$$GET1^DIQ(C0PXF,C0PUTID_",",2.2,,"header") | 
|---|
|  | 115 | S C0PMIME=$$GET1^DIQ(C0PXF,C0PUTID_",","MIME TYPE") | 
|---|
|  | 116 | S C0PPURL=$$GET1^DIQ(C0PXF,C0PUTID_",","PROXY SERVER") | 
|---|
|  | 117 | ;S C0PURL=$$GET1^DIQ(C0PXF,C0PUTID_",","URL") ;GPL CHANGE TO USE PROD FLAG | 
|---|
|  | 118 | D SETUP^C0PMAIN() ; INITIALIZE C0PACCT IEN OF WS ACCOUNT | 
|---|
|  | 119 | S C0PURL=$$WSURL^C0PMAIN(C0PACCT) ; RESOLVES PRODUCTION VS TEST | 
|---|
|  | 120 | S C0PXML=$$GET1^DIQ(C0PXF,C0PUTID_",",2.1,,"xml") | 
|---|
|  | 121 | S C0PTMPL=$$GET1^DIQ(C0PXF,C0PUTID_",",3,,"template") | 
|---|
|  | 122 | I C0PTMPL="template" D  ; there is a template to process | 
|---|
|  | 123 | . K xml ; going to replace the xml array | 
|---|
|  | 124 | . D EN^C0PMAIN("xml","url",C0PDUZ,C0PDFN,C0PUTID,$G(C0PVOR)) | 
|---|
|  | 125 | . ;N ZZG M ZZG(1)=xml | 
|---|
|  | 126 | . ;S ZDIR=^TMP("C0CCCR","ODIR") | 
|---|
|  | 127 | . ;ZWR ZZG(1) | 
|---|
|  | 128 | . ;W $$OUTPUT^C0CXPATH("xml(1)","GPLTEST-"_ZDFN_".xml",ZDIR) | 
|---|
|  | 129 | .QUIT | 
|---|
|  | 130 | I $G(C0PPROXY) S C0PURL=C0PPURL | 
|---|
|  | 131 | K C0PRSLT,C0PRHDR | 
|---|
|  | 132 | S ok=$$httpPOST^%zewdGTM(C0PURL,.xml,C0PMIME,.C0PRSLT,.header,"",.gpl5,.C0PRHDR) | 
|---|
|  | 133 | K C0PRXML | 
|---|
|  | 134 | I $D(GPLTEST) D  ; WAY TO TEST WITH DATA FROM LIVE | 
|---|
|  | 135 | . K C0PSRLT ; GPL HACK TO TEST XML FROM LIVE | 
|---|
|  | 136 | . I GPLTEST=1 M C0PRSLT=^C0PG ; THIS IS THE BIG STATUS EMBEDDED XML FROM LIVE | 
|---|
|  | 137 | . I GPLTEST=2 M C0PRSLT=^C0PG2 ; THIS IS THE BIG REFILL XML  FROM LIVE | 
|---|
|  | 138 | . Q | 
|---|
|  | 139 | ; The following is a temporary fix to keep eRx working while a better | 
|---|
|  | 140 | ; solution is developed. Template ID 6 is GETMEDS for eRx and it needs | 
|---|
|  | 141 | ; to handle xml files that are too big for NORMAL to handle. So, I wrote | 
|---|
|  | 142 | ; CHUNK which will allow us to handle any size xml file bound for the | 
|---|
|  | 143 | ; EWD parser. | 
|---|
|  | 144 | ; However, all the other templates in eRx need NORMAL to find the | 
|---|
|  | 145 | ; embedded XML file in their web service responses. So, we will use | 
|---|
|  | 146 | ; CHUNK for template 6 and continue to use NORMAL for all other templates | 
|---|
|  | 147 | ; we can handle big med lists, but not big web service calls. | 
|---|
|  | 148 | ; What is needed is a better NORMAL (see NORMAL2) or another routine | 
|---|
|  | 149 | ; to detect, extract, and decode embeded XML files of any size. gpl 10/8/10 | 
|---|
|  | 150 | ; | 
|---|
|  | 151 | I $D(C0PRSLT(1)) D  ; | 
|---|
|  | 152 | . D CHUNK("C0PRXML","C0PRSLT",1000) ;RETURN IN AN ARRAY | 
|---|
|  | 153 | . I $G(C0PRSLT("RELOC",1,1))'="" D  ; THERE WAS EMBEDED XML | 
|---|
|  | 154 | . . K C0PRXML ; THROW AWAY WRAPPER | 
|---|
|  | 155 | . . M C0PRXML=C0PRSLT("RELOC",1) ; REPLACE WITH EMBEDDED DOCUMENT | 
|---|
|  | 156 | ; D:C0PUTID=6 | 
|---|
|  | 157 | ;. I $D(C0PRSLT(1)) D CHUNK("C0PRXML","C0PRSLT",2000) QUIT  ;RETURN IN AN ARRAY | 
|---|
|  | 158 | ;. ; | 
|---|
|  | 159 | ;. I $D(C0PRSLT(1)) D NORMAL("C0PRXML","C0PRSLT(1)") ;RETURN XML IN AN ARRAY | 
|---|
|  | 160 | ;.QUIT | 
|---|
|  | 161 | S C0PR=$$GET1^DIQ(C0PXF,C0PUTID_",",.03,"I") ; REPLY TEMPLATE | 
|---|
|  | 162 | ; reply templates are optional and are specified by populating a | 
|---|
|  | 163 | ; template pointer in field 2.5 of the request template | 
|---|
|  | 164 | ; if specified, the reply template is the source of the REDUX string | 
|---|
|  | 165 | ; used for XPath on the reply, and for UNBIND processing | 
|---|
|  | 166 | ; if no reply template is specified, REDUX is obtained from the request | 
|---|
|  | 167 | ; template and no UNBIND processing is performed. The XPath array is | 
|---|
|  | 168 | ; returned without variable bindings | 
|---|
|  | 169 | I C0PR'="" D  ; REPLY TEMPLATE EXISTS | 
|---|
|  | 170 | . I +$G(DEBUG)'=0 W "REPLY TEMPLATE:"_C0PR,! | 
|---|
|  | 171 | . S C0PTID=C0PR ; | 
|---|
|  | 172 | .QUIT | 
|---|
|  | 173 | S C0PREDUX=$$GET1^DIQ(C0PXF,C0PUTID_",",2.5) ;XPATH REDUCTION STRING | 
|---|
|  | 174 | K C0PIDX,C0PARY ; XPATH INDEX AND ARRAY VARS | 
|---|
|  | 175 | S C0PNOM="C0PMEDS"_$J ; DOCUMENT NAME FOR THE DOM | 
|---|
|  | 176 | N ZBIG S ZBIG=0 | 
|---|
|  | 177 | ;I C0PUTID'=6 D  ; | 
|---|
|  | 178 | ;. S ZBIG=$$TOOBIG("C0PRXML") ; PATCH BY GPL WHICH ASSUMES ONLY | 
|---|
|  | 179 | ;. ; TEMPLATE 1 IS A REGULAR XML FILE.. EVERYTHING ELSE HAS EMBEDDED XML | 
|---|
|  | 180 | ;.QUIT | 
|---|
|  | 181 | ;D | 
|---|
|  | 182 | ;. I ZBIG>0 D    QUIT  ; PROBABLY AN EMBEDDED XML DOCUMENT | 
|---|
|  | 183 | ;. . S C0PID=$$UNWRAP("C0PRXML",ZBIG,C0PNOM) ; DECODE AND PARSE THE EMBEDED XML | 
|---|
|  | 184 | ;. .QUIT | 
|---|
|  | 185 | ;. ; | 
|---|
|  | 186 | ;. ; ELSE | 
|---|
|  | 187 | ;. S C0PID=$$PARSE^C0PXEWD("C0PRXML",C0PNOM) ;CALL THE PARSER | 
|---|
|  | 188 | ;.QUIT | 
|---|
|  | 189 | ; I $D(GPLTEST) B  ; STOP TO LOOK AT C0PRXML --> use ZB SOAP+137^C0PWS2 //SMH | 
|---|
|  | 190 | S C0PID=$$PARSE^C0PXEWD("C0PRXML",C0PNOM) ;CALL THE PARSER | 
|---|
|  | 191 | S C0PID=$$FIRST^C0PXEWD($$ID^C0PXEWD(C0PNOM)) ;ID OF FIRST NODE | 
|---|
|  | 192 | D XPATH^C0PXEWD(C0PID,"/","C0PIDX","C0PARY","",C0PREDUX) ;XPATH GENERATOR | 
|---|
|  | 193 | S OK=$$DELETE^C0PXEWD(C0PNOM) ; REMOVE PARSED XML FROM THE EWD DOM | 
|---|
|  | 194 | ; Next, call UNBIND to map the reply XPath array to variables | 
|---|
|  | 195 | ; This is only done if a Reply Template is provided | 
|---|
|  | 196 | D DEMUXARY(C0PRTN,"C0PARY") | 
|---|
|  | 197 | ; M @C0PRTN=C0PARY | 
|---|
|  | 198 | QUIT | 
|---|
|  | 199 | ; | 
|---|
|  | 200 | TOOBIG(ZXML)    ; EXTRINSIC WHICH RETURNS TRUE IF ANY NODE IS OVER 2000 CHARS | 
|---|
|  | 201 | ; RETURNS THE INDEX OF THE LARGE NODE . IF NO LARGE NODE, RETURNS ZERO | 
|---|
|  | 202 | N ZI,ZR | 
|---|
|  | 203 | S ZI="" | 
|---|
|  | 204 | S ZR=0 ; DEFAULT FALSE | 
|---|
|  | 205 | ; First time we go over 1,000, we can stop. | 
|---|
|  | 206 | F  S ZI=$O(@ZXML@(ZI)) Q:ZI=""  I $L(@ZXML@(ZI))>1000 S ZR=ZI Q   ; First oversize stops | 
|---|
|  | 207 | QUIT ZR | 
|---|
|  | 208 | ; =================== | 
|---|
|  | 209 | NORMAL(OUTXML,INXML)    ;NORMALIZES AN XML STRING PASSED BY NAME IN INXML | 
|---|
|  | 210 | ; INTO AN XML ARRAY RETURNED IN OUTXML, ALSO PASSED BY NAME | 
|---|
|  | 211 | ; | 
|---|
|  | 212 | N INBF,ZI,ZN,ZTMP | 
|---|
|  | 213 | S ZN=1,INBF=@INXML | 
|---|
|  | 214 | S @OUTXML@(ZN)=$P(INBF,"><",ZN)_">" | 
|---|
|  | 215 | ; S ZN=ZN+1 | 
|---|
|  | 216 | ; F  S @OUTXML@(ZN)="<"_$P(@INXML,"><",ZN) Q:$P(@INXML,"><",ZN+1)=""  D  ; | 
|---|
|  | 217 | ; Should speed up, and not leave a dangling node, and doesn't stop at first NULL | 
|---|
|  | 218 | F ZN=2:1:$L(INBF,"><") S @OUTXML@(ZN)="<"_$P(INBF,"><",ZN)_">" | 
|---|
|  | 219 | ; . ; S ZN=ZN+1 | 
|---|
|  | 220 | ; .QUIT | 
|---|
|  | 221 | QUIT | 
|---|
|  | 222 | ;  ================ | 
|---|
|  | 223 | ; The goal of this block has changed a little bit.  Most modern MUMPS engines can | 
|---|
|  | 224 | ; handle a 1,000,000 byte string.  We will use BF to hold hunks that big so that | 
|---|
|  | 225 | ; we can logically suck up a big hunk of the input to supply the reblocking of the XML | 
|---|
|  | 226 | ; into more logical blocks less than 2000 bytes in length blocks. | 
|---|
|  | 227 | ; A series of signals will be needed, Source (INXML) is exhausted (INEND), | 
|---|
|  | 228 | ; BF is less than 2200 bytes (BFLD, BuFfer reLoaD) | 
|---|
|  | 229 | ; BF is Full (BF contains 998,000 bytes or more, BFULL) | 
|---|
|  | 230 | ; BF and Process is Complete (BFEND) | 
|---|
|  | 231 | ; ZSIZE defaults to 2,000 now, but can be set lower or higher | 
|---|
|  | 232 | ; | 
|---|
|  | 233 | CHUNK(OUTXML,INXML,ZSIZE)       ; BREAKS INXML INTO ZSIZE BLOCKS | 
|---|
|  | 234 | ; INXML IS AN ARRAY PASSED BY NAME OF STRINGS | 
|---|
|  | 235 | ; OUTXML IS ALSO PASSED BY NAME | 
|---|
|  | 236 | ; IF ZSIZE IS NOT PASSED, 2000 IS USED | 
|---|
|  | 237 | I '$D(ZSIZE) S ZSIZE=2000 ; DEFAULT BLOCK SIZE | 
|---|
|  | 238 | N BF,BFEND,BFLD,BFMAX,BFULL,INEND,ZB,ZI,ZJ,ZK,ZL,ZN | 
|---|
|  | 239 | ; S ZB=ZSIZE-1 | 
|---|
|  | 240 | S ZN=1 | 
|---|
|  | 241 | S BFMAX=998000 | 
|---|
|  | 242 | S ZI=0 ; BEGINNING OF INDEX TO INXML | 
|---|
|  | 243 | S (BFLD,BFEND,BFULL,INEND)=0,BF="" | 
|---|
|  | 244 | ; Major loop loads the buffer, BF, and unloads it into the Output Array | 
|---|
|  | 245 | ;  in | 
|---|
|  | 246 | F  D  Q:BFEND | 
|---|
|  | 247 | . ; Input LOADER | 
|---|
|  | 248 | . D:'INEND | 
|---|
|  | 249 | . . F  S ZI=$O(@INXML@(ZI)) S INEND=(ZI="")  Q:INEND!BFULL  D   ; LOAD EACH STRING IN INXML | 
|---|
|  | 250 | . . . S BF=BF_@INXML@(ZI)                                       ; ADD TO THE BF STRING | 
|---|
|  | 251 | . . . S BFULL=($L(BF)>BFMAX) | 
|---|
|  | 252 | . . .QUIT | 
|---|
|  | 253 | . .QUIT | 
|---|
|  | 254 | . ;  Full Buffer, BF, now check for Encryption and Unpack | 
|---|
|  | 255 | . D TEST4COD(.BF,"C0PRSLT(""RELOC"")") | 
|---|
|  | 256 | . ; Output BREAKER | 
|---|
|  | 257 | . F  Q:BFLD  D   ; ZJ=1:ZSIZE:ZL D  ; | 
|---|
|  | 258 | . . ; ZK=$S(ZJ+ZB<ZL:ZJ+ZB,1:ZL) ; END FOR EXTRACT | 
|---|
|  | 259 | . . F ZK=ZSIZE:-1:0  Q:$E(BF,ZK)=">" | 
|---|
|  | 260 | . . I ZK=0 S ZK=ZSIZE | 
|---|
|  | 261 | . . S @OUTXML@(ZN)=$E(BF,1,ZK) ; PULL OUT THE PIECE | 
|---|
|  | 262 | . . S ZN=ZN+1 ; INCREMENT OUT ARRAY INDEX | 
|---|
|  | 263 | . . S BF=$E(BF,ZK+1,BFMAX) | 
|---|
|  | 264 | . . S BFLD=($L(BF)<(ZSIZE*2)) | 
|---|
|  | 265 | . .QUIT | 
|---|
|  | 266 | . S BFEND=(INEND&BFLD)!(">"[BF) | 
|---|
|  | 267 | . I $L(BF)&BFEND S @OUTXML@(ZN)=BF,BF="" | 
|---|
|  | 268 | .QUIT | 
|---|
|  | 269 | QUIT | 
|---|
|  | 270 | ;  ============== | 
|---|
|  | 271 | ; Test for Encryption, extract it and decode it. | 
|---|
|  | 272 | TEST4COD(INBF,RELOC) | 
|---|
|  | 273 | N DBF,I,MSK,TBF,TRG,RCNT | 
|---|
|  | 274 | S RCNT=0 | 
|---|
|  | 275 | ;  Segments expected <seg 1>DATA</seg 1><seg 2>DATA</seg 2> | 
|---|
|  | 276 | ;                           ^   ^ | 
|---|
|  | 277 | S MSK=""   ; It turns out that some of the characters used were not reliable | 
|---|
|  | 278 | F I=32:1:42,44:1:47,62:1:64,91:1:96 S MSK=MSK_$C(I) | 
|---|
|  | 279 | F I=1:1:$L(INBF,"</")-1 D | 
|---|
|  | 280 | . S TBF=$RE($P($RE($P(INBF,"</",I)),">")) | 
|---|
|  | 281 | . ; Remove sample for testing | 
|---|
|  | 282 | . ; Set the trigger, mostly included to show intent and associated code | 
|---|
|  | 283 | . ;  this could be refined later if determined already obvious enough | 
|---|
|  | 284 | . S TRG=0 | 
|---|
|  | 285 | . ;DO:$L(TBF)>20  ; If $TR doesn't remove anything, then these characters are not there | 
|---|
|  | 286 | . ; gpl  trying to keep refills from crashing.. 20 chars is not enough | 
|---|
|  | 287 | . DO:$L(TBF)>100  ; If $TR doesn't remove anything, then these characters are not there | 
|---|
|  | 288 | . . I (TBF=$TR(TBF,MSK))   S TRG=1 | 
|---|
|  | 289 | . . ; I (TBF=$TR(TBF," <->@*!?.,:;#$%&[/|\]={}~")) S TRG=1 | 
|---|
|  | 290 | . . ;   <>!"#$%&'()*,-./67:;<>?@[\]^_`fqr{|}~  <<= Ignore 6,7,f,q, and r | 
|---|
|  | 291 | . . ; Now we set up for the DECODE and replacement in INBF | 
|---|
|  | 292 | . . DO:TRG | 
|---|
|  | 293 | . . . N A,C,CC,CV,CCX,K,XBF,T,V | 
|---|
|  | 294 | . . . DO | 
|---|
|  | 295 | . . . . N I | 
|---|
|  | 296 | . . . . S DBF=$$DECODER(TBF) | 
|---|
|  | 297 | . . . .QUIT | 
|---|
|  | 298 | . . . ; | 
|---|
|  | 299 | . . . S CCX="" | 
|---|
|  | 300 | . . . F K=1:1:$L(DBF) S CC=$E(DBF,K) S:CC?1C C=$A(CC),A(C)=$G(A(C))+1 | 
|---|
|  | 301 | . . . S C="",V="" | 
|---|
|  | 302 | . . . F  S C=$O(A(C)) Q:C=""  S CCX=CCX_$C(C) S:A(C)>V V=A(C),CV=C | 
|---|
|  | 303 | . . . S CC=$C(CV) | 
|---|
|  | 304 | . . . ;  The "_$C(13,10)_" may need to be generalized, tested and set earlier | 
|---|
|  | 305 | . . . ;    Expand embedded XML in XBF | 
|---|
|  | 306 | . . . F K=1:1:$L(DBF,CC) S T=$P(DBF,CC,K),XBF(K)=$TR(T,CCX) | 
|---|
|  | 307 | . . . S RCNT=RCNT+1 | 
|---|
|  | 308 | . . . M @RELOC@(RCNT)=XBF | 
|---|
|  | 309 | . . . ;   Curley braces and = makes it so it won't trigger a second time by retest. | 
|---|
|  | 310 | . . . S INBF=$P(INBF,TBF)_"<{REPLACED}="_RCNT_$P(INBF,TBF,2,999) | 
|---|
|  | 311 | . . .QUIT | 
|---|
|  | 312 | . .QUIT | 
|---|
|  | 313 | .QUIT | 
|---|
|  | 314 | ;  Now shorten the INBF so it gets smaller | 
|---|
|  | 315 | ;S INBF=$P(INBF,">",I+1,99999) | 
|---|
|  | 316 | QUIT | 
|---|
|  | 317 | ; | 
|---|
|  | 318 | DECODER(BF)     ; Decrypts the Encrypted Strings | 
|---|
|  | 319 | QUIT $$DECODE^RGUTUU(BF) | 
|---|
|  | 320 | ; | 
|---|
|  | 321 | NORMAL2(OUTXML,INXML)   ;NORMALIZES AN ARRAY OF XML STRINGS PASSED BY NAME INXML | 
|---|
|  | 322 | ; AS @INXML@(1) TO @INXML@(x) ALL NUMERIC | 
|---|
|  | 323 | ; INTO AN XML ARRAY RETURNED IN OUTXML, ALSO PASSED BY NAME | 
|---|
|  | 324 | ; this routine doesn't work unless the blocks are on xml tag boundaries - gpl | 
|---|
|  | 325 | ; which is hard to do... this routine is left here awaiting future development | 
|---|
|  | 326 | N ZI,ZN,ZJ | 
|---|
|  | 327 | S ZJ=0 | 
|---|
|  | 328 | S ZN=1 | 
|---|
|  | 329 | F  S ZJ=$O(@INXML@(ZJ)) Q:+ZJ=0  D  ; FOR EACH XML STRING IN ARRAY | 
|---|
|  | 330 | . S @OUTXML@(ZN)=$P(@INXML@(ZJ),"><",ZN)_">" | 
|---|
|  | 331 | . S ZN=ZN+1 | 
|---|
|  | 332 | . F  S @OUTXML@(ZN)="<"_$P(@INXML@(ZJ),"><",ZN) Q:$P(@INXML@(ZJ),"><",ZN+1)=""  D  ; | 
|---|
|  | 333 | . . S @OUTXML@(ZN)=@OUTXML@(ZN)_">" | 
|---|
|  | 334 | . . S ZN=ZN+1 | 
|---|
|  | 335 | . .QUIT | 
|---|
|  | 336 | .QUIT | 
|---|
|  | 337 | QUIT | 
|---|
|  | 338 | ;  =============== | 
|---|
|  | 339 | ; | 
|---|
|  | 340 | UNWRAP(ZXML,ZI,ZNOM)    ; EXTRINSIC TO LOCATE, DECODE AND PARSE AN EMBEDED XML DOC | 
|---|
|  | 341 | ; RETURNS THE DOCID OF THE DOM | 
|---|
|  | 342 | N ZS,ZX | 
|---|
|  | 343 | S ZS=$P($P(@ZXML@(ZI),">",2),"<",1) ; PULL OUT THE ENCODED STRING | 
|---|
|  | 344 | S ZX=$$DECODE^RGUTUU(ZS) | 
|---|
|  | 345 | N ZZ | 
|---|
|  | 346 | N ZY S ZY="<?xml version=""1.0"" encoding=""utf-8""?>" | 
|---|
|  | 347 | I $E(ZX,1,5)'="<?xml" S ZZ(1)=ZY_ZX | 
|---|
|  | 348 | E  S ZZ(1)=ZX | 
|---|
|  | 349 | N ZI | 
|---|
|  | 350 | ;F ZI=1:1 Q:$$REDUCE(.ZZ,ZI) ; CHOP THE STRING INTO 4000 CHAR ARRAY | 
|---|
|  | 351 | S ZI=$$REDUCRCR(.ZZ,1) ; RECURSIVE VERSION OF REDUCE | 
|---|
|  | 352 | S G=$$PARSE^C0PXEWD("ZZ",C0PNOM) | 
|---|
|  | 353 | ; GTM Specific | 
|---|
|  | 354 | ; I G=0 ZWR ^TMP("MXMLERR",$J,*) B | 
|---|
|  | 355 | QUIT G | 
|---|
|  | 356 | ;  ============= | 
|---|
|  | 357 | REDUCE(ZARY,ZN) ; WILL REDUCE ZARY(ZN) BY CHOPPING IT TO 4000 CHARS | 
|---|
|  | 358 | ; AND PUTTING THE REST IN ZARY(ZN+1) | 
|---|
|  | 359 | ; ZARY IS PASSED BY REFERENCE | 
|---|
|  | 360 | ; EXTRINSIC WHICH RETURNS FALSE IF THERE IS NOTHING TO REDUCE | 
|---|
|  | 361 | I $L(ZARY(ZN))<4001   QUIT 0 ;NOTHING TO REDUCE | 
|---|
|  | 362 | ; | 
|---|
|  | 363 | S ZARY(ZN+1)=$E(ZARY(ZN),4001,$L(ZZ(ZN))) ;BREAK IT UP | 
|---|
|  | 364 | S ZARY(ZN)=$E(ZARY(ZN),1,4000) ; | 
|---|
|  | 365 | QUIT 1  ;ACTUALLY REDUCED | 
|---|
|  | 366 | ;  =========== | 
|---|
|  | 367 | REDUCRCR(ZARY,ZN)       ; RECURSIVE VERSION OF REDUCE ABOVE | 
|---|
|  | 368 | ; WILL REDUCE ZARY(ZN) BY CHOPPING IT TO 4000 CHARS | 
|---|
|  | 369 | ; AND PUTTING THE REST IN ZARY(ZN+1) | 
|---|
|  | 370 | ; ZARY IS PASSED BY REFERENCE | 
|---|
|  | 371 | ; EXTRINSIC WHICH RETURNS FALSE IF THERE IS NOTHING TO REDUCE | 
|---|
|  | 372 | I $L(ZARY(ZN))<4001 Q 0 ;NOTHING TO REDUCE | 
|---|
|  | 373 | ; | 
|---|
|  | 374 | S ZARY(ZN+1)=$E(ZARY(ZN),4001,$L(ZZ(ZN))) ;BREAK IT UP | 
|---|
|  | 375 | S ZARY(ZN)=$E(ZARY(ZN),1,4000) ; | 
|---|
|  | 376 | I '$$REDUCRCR(.ZARY,ZN+1) Q 1 ; CALL RECURSIVELY | 
|---|
|  | 377 | ; | 
|---|
|  | 378 | QUIT 1  ;ACTUALLY REDUCED | 
|---|
|  | 379 | ; | 
|---|
|  | 380 | DEMUXARY(OARY,IARY)     ;CONVERT AN XPATH ARRAY PASSED AS IARY TO | 
|---|
|  | 381 | ; FORMAT @OARY@(x,xpath) where x is the first multiple | 
|---|
|  | 382 | N ZI,ZJ,ZK,ZL S ZI="" | 
|---|
|  | 383 | F  S ZI=$O(@IARY@(ZI)) Q:ZI=""  D  ; | 
|---|
|  | 384 | . D DEMUX^C0CMXP("ZJ",ZI) | 
|---|
|  | 385 | . S ZK=$P(ZJ,"^",3) | 
|---|
|  | 386 | . S ZK=$RE($P($RE(ZK),"/",1)) | 
|---|
|  | 387 | . S ZL=$P(ZJ,"^",1) | 
|---|
|  | 388 | . I ZL="" S ZL=1 | 
|---|
|  | 389 | . S @OARY@(ZL,ZK)=@IARY@(ZI) | 
|---|
|  | 390 | .QUIT | 
|---|
|  | 391 | QUIT | 
|---|
|  | 392 | ; | 
|---|
|  | 393 | PARSE(INXML,INDOC)      ;CALL THE EWD PARSER ON INXML, PASSED BY NAME | 
|---|
|  | 394 | ; INDOC IS PASSED AS THE DOCUMENT NAME TO EWD | 
|---|
|  | 395 | ; EXTRINSIC WHICH RETURNS THE DOCID ASSIGNED BY EWD | 
|---|
|  | 396 | N ZR | 
|---|
|  | 397 | M ^CacheTempEWD($j)=@INXML ; | 
|---|
|  | 398 | S ZR=$$parseDocument^%zewdHTMLParser(INDOC) | 
|---|
|  | 399 | K ^CacheTempEWD($j) ;clean up after | 
|---|
|  | 400 | QUIT ZR | 
|---|
|  | 401 | ; | 
|---|
|  | 402 | TBLD(INT)       ; TEMPLATE BUILD OF TEMPLATE INT | 
|---|
|  | 403 | ; want to break this up into pieces -  gpl | 
|---|
|  | 404 | ; THE TEMPLATE NEEDS TO EXIST AND THE DEFINING XML URL MUST BE POPULATED | 
|---|
|  | 405 | ; THEN THE DEFINING XML WILL BE RETRIVED AND STORED INTO THE RAW XML FIELD | 
|---|
|  | 406 | ; IT WILL BE TRANSFORMED INTO A TEMPLATE AND STORED IN THE TEMPLATE FIELD | 
|---|
|  | 407 | ; ALL THE XPATHs WILL BE EXTRACTED AND A BINDING MULTIPLE CREATED FOR EACH | 
|---|
|  | 408 | ; ALL IN ONE SIMPLE ROUTINE | 
|---|
|  | 409 | ; WHAT REMAINS IS FOR MANUAL ENTRY OF THE OTHER FIELDS IN THE BINDINGS | 
|---|
|  | 410 | N C0PXTF S C0PXTF=113059001 ; XML TEMPLATE FILE | 
|---|
|  | 411 | N C0PURL ; URL TO RETRIEVE THE DEFINING XML FOR THE TEMPLATE | 
|---|
|  | 412 | S C0PURL=$$GET1^DIQ(C0PXTF,INT,2) | 
|---|
|  | 413 | D GET1URL^C0PEWD2(C0PURL) | 
|---|
|  | 414 | D CLEAN^DILF | 
|---|
|  | 415 | ; D WP^DIE(ZF,ZIEN_",",1,,$NA(@ZOR@(ZD,ZI,"TX"))) ; WP OF ORDER TXT | 
|---|
|  | 416 | D WP^DIE(C0PXTF,INT_",",2.1,,$NA(gpl)) | 
|---|
|  | 417 | D WP^DIE(C0PXTF,INT_",",3,,$NA(gplTEMP)) | 
|---|
|  | 418 | ;N C0PFDA ; DON'T NEW FOR TESTING | 
|---|
|  | 419 | D ADDXP("gpl2",INT) | 
|---|
|  | 420 | QUIT | 
|---|
|  | 421 | ;  ========== | 
|---|
|  | 422 | COMPILE(INTID)  ;COMPILE A XML TEMPLATE IN RECORD INTID | 
|---|
|  | 423 | D INITXPF("C0PF") ;FILE ARRAY TO POINT TO C0P FILES | 
|---|
|  | 424 | D COMPILE^C0CMXP(INTID,"C0PF") ;COMPILE THE TEMPLATE | 
|---|
|  | 425 | QUIT | 
|---|
|  | 426 | ;  ========== | 
|---|
|  | 427 | CPBIND(INID,OUTID,FORCE)        ; COPIES XPATH BINDINGS FROM TEMPLATE INID | 
|---|
|  | 428 | ; TO TEMPLATE OUTID - ONLY BINDINGS FOR MATCHING XPATHS ARE COPIED | 
|---|
|  | 429 | ; NOTE - REDO THIS TO USE FILEMAN CALLS GPL | 
|---|
|  | 430 | ; WILL NOT OVERWRITE UNLESS FORCE=1 | 
|---|
|  | 431 | N FARY,ZI | 
|---|
|  | 432 | S FARY="C0PF" | 
|---|
|  | 433 | D INITXPF("C0PF") | 
|---|
|  | 434 | I +OUTID=0 S OUTID=$$RESTID^C0CSOAP(OUTID,FARY) ;RESOLVE TEMPLATE NAME | 
|---|
|  | 435 | I +INID=0 S INID=$$RESTID^C0CSOAP(INID,FARY) ;RESOLVE TEMPLATE NAME | 
|---|
|  | 436 | S ZI=0 | 
|---|
|  | 437 | F  S ZI=$O(^C0PX(OUTID,5,ZI)) Q:+ZI=0  D  ; FOR EACH XPATH IN OUTID | 
|---|
|  | 438 | . W !,ZI," ",^C0PX(OUTID,5,ZI,0) | 
|---|
|  | 439 | . S ZN=^C0PX(OUTID,5,ZI,0) | 
|---|
|  | 440 | . I $D(^C0PX(OUTID,5,ZI,1)) D  ;Q  ; | 
|---|
|  | 441 | . . W !,"ERROR XPATH BINDING EXISTS ",ZI | 
|---|
|  | 442 | . .QUIT | 
|---|
|  | 443 | . D  ; LOOK FOR MATCHING XPATH IN SOURCE | 
|---|
|  | 444 | . . S ZJ=$O(^C0PX(INID,5,"B",ZN,"")) | 
|---|
|  | 445 | . . ;W " FOUND:",ZJ | 
|---|
|  | 446 | . . I ZJ'="" D  ; | 
|---|
|  | 447 | . . . ;W !,"SETTING ",$G(^C0PX(INID,5,ZJ,1)) | 
|---|
|  | 448 | . . . S ^C0PX(OUTID,5,ZI,0)=^C0PX(INID,5,ZJ,0) ;GET BOTH FIELDS | 
|---|
|  | 449 | . . . S ^C0PX(OUTID,5,ZI,1)=$G(^C0PX(INID,5,ZJ,1)) | 
|---|
|  | 450 | QUIT | 
|---|
|  | 451 | ; | 
|---|
|  | 452 | INITXPF(ARY)    ;INITIAL XML/XPATH FILE ARRAY | 
|---|
|  | 453 | ; | 
|---|
|  | 454 | S @ARY@("XML FILE NUMBER")=113059001 | 
|---|
|  | 455 | S @ARY@("BINDING SUBFILE NUMBER")=113059001.04 | 
|---|
|  | 456 | S @ARY@("MIME TYPE")="2.3" | 
|---|
|  | 457 | S @ARY@("PROXY SERVER")="2.4" | 
|---|
|  | 458 | S @ARY@("REPLY TEMPLATE")=".03" | 
|---|
|  | 459 | S @ARY@("TEMPLATE NAME")=".01" | 
|---|
|  | 460 | S @ARY@("TEMPLATE XML")="3" | 
|---|
|  | 461 | S @ARY@("URL")="1" | 
|---|
|  | 462 | S @ARY@("WSDL URL")="2" | 
|---|
|  | 463 | S @ARY@("XML")="2.1" | 
|---|
|  | 464 | S @ARY@("XML HEADER")="2.2" | 
|---|
|  | 465 | S @ARY@("XPATH REDUCTION STRING")="2.5" | 
|---|
|  | 466 | S @ARY@("CCR VARIABLE")="4" | 
|---|
|  | 467 | S @ARY@("FILEMAN FIELD NAME")="1" | 
|---|
|  | 468 | S @ARY@("FILEMAN FIELD NUMBER")="1.2" | 
|---|
|  | 469 | S @ARY@("FILEMAN FILE POINTER")="1.1" | 
|---|
|  | 470 | S @ARY@("INDEXED BY")=".05" | 
|---|
|  | 471 | S @ARY@("SQLI FIELD NAME")="3" | 
|---|
|  | 472 | S @ARY@("VARIABLE NAME")="2" | 
|---|
|  | 473 | QUIT | 
|---|
|  | 474 | ; | 
|---|
|  | 475 | ADDXP(INARY,TID)        ;ADD XPATH .01 FIELD TO BINDING SUBFILE OF TEMPLATE TID | 
|---|
|  | 476 | N FARY | 
|---|
|  | 477 | S FARY="C0PFILES" | 
|---|
|  | 478 | D INITXPF(FARY) | 
|---|
|  | 479 | D ADDXP^C0CMXP(INARY,TID,FARY) ; | 
|---|
|  | 480 | QUIT | 
|---|
|  | 481 | ; | 
|---|
|  | 482 | ADDXML(INXML,TEMPID)    ;ADD XML TO A TEMPLATE ID TEMPID | 
|---|
|  | 483 | ; INXML IS PASSED BY NAME | 
|---|
|  | 484 | N FARY S FARY="C0PFILES" | 
|---|
|  | 485 | D INITXPF(FARY) | 
|---|
|  | 486 | D ADDXML^C0CMXP(INXML,TEMPID,FARY) ;CALL C0C ROUTINE TO ADD TO THE FILE | 
|---|
|  | 487 | QUIT | 
|---|
|  | 488 | ; | 
|---|
|  | 489 | ADDTEMP(INXML,TEMPID,FARY)      ;ADD XML TEMPLATE TO TEMPLATE RECORD TEMPID FIELD 3 | 
|---|
|  | 490 | ; | 
|---|
|  | 491 | N FARY | 
|---|
|  | 492 | S FARY="C0PFILES" | 
|---|
|  | 493 | D INITXPF(FARY) | 
|---|
|  | 494 | D ADDTEMP^C0CMXP(INXML,TEMPID,FARY) | 
|---|
|  | 495 | QUIT | 
|---|
|  | 496 | ; | 
|---|
|  | 497 | GETXML(OUTXML,TEMPID,FARY)      ;GET THE XML FROM TEMPLATE TEMPID | 
|---|
|  | 498 | ; | 
|---|
|  | 499 | N FARY | 
|---|
|  | 500 | S FARY="C0PFILES" | 
|---|
|  | 501 | D INITXPF(FARY) | 
|---|
|  | 502 | N C0PUTID ; TEMPLATE IEN TO USE | 
|---|
|  | 503 | D GETXML^C0CMXP(OUTXML,TEMPID,FARY) | 
|---|
|  | 504 | QUIT | 
|---|
|  | 505 | ; | 
|---|
|  | 506 | GETTEMP(OUTXML,TEMPID,FARY)     ;GET THE TEMPLATE XML FROM TEMPLATE TEMPID | 
|---|
|  | 507 | ; | 
|---|
|  | 508 | N FARY | 
|---|
|  | 509 | S FARY="C0PFILES" | 
|---|
|  | 510 | D INITXPF(FARY) | 
|---|
|  | 511 | N C0PUTID ; TEMPLATE IEN TO USE | 
|---|
|  | 512 | D GETTEMP^C0CMXP(OUTXML,TEMPID,FARY) | 
|---|
|  | 513 | QUIT | 
|---|
|  | 514 | ; | 
|---|
|  | 515 | COPYHDR(ZS,ZD)  ; COPY XML HEADER FROM RECORD ZS TO ZD | 
|---|
|  | 516 | ; ASSUMES C0P XML TEMPLATE FILE | 
|---|
|  | 517 | N FARY | 
|---|
|  | 518 | D INITXPF("FARY") | 
|---|
|  | 519 | D COPYWP^C0CMXP("XML HEADER",ZS,ZD,"FARY") | 
|---|
|  | 520 | QUIT | 
|---|
|  | 521 | ; | 
|---|
|  | 522 | UPDIE     ; INTERNAL ROUTINE TO CALL UPDATE^DIE AND CHECK FOR ERRORS | 
|---|
|  | 523 | K ZERR | 
|---|
|  | 524 | D CLEAN^DILF | 
|---|
|  | 525 | D UPDATE^DIE("","C0PFDA","","ZERR") | 
|---|
|  | 526 | I $D(ZERR) D ERROR^C0PMAIN(",U113059008,",$ST($ST,"PLACE"),"ERX-UPDIE-FAIL","Fileman Data Update Failure") QUIT | 
|---|
|  | 527 | K C0PFDA | 
|---|
|  | 528 | QUIT | 
|---|