zmwire ; M/Wire Protocol for M Systems (eg GT.M, Cache)
 ;
 ; ----------------------------------------------------------------------------
 ; | M/Wire                                                                   |
 ; | Copyright (c) 2004-11 M/Gateway Developments Ltd,                        |
 ; | Reigate, Surrey UK.                                                      |
 ; | All rights reserved.                                                     |
 ; |                                                                          |
 ; | http://www.mgateway.com                                                  |
 ; | Email: rtweed@mgateway.com                                               |
 ; |                                                                          |
 ; | This program is free software: you can redistribute it and/or modify     |
 ; | it under the terms of the GNU Affero General Public License as           |
 ; | published by the Free Software Foundation, either version 3 of the       |
 ; | License, or (at your option) any later version.                          |
 ; |                                                                          |
 ; | This program is distributed in the hope that it will be useful,          |
 ; | but WITHOUT ANY WARRANTY; without even the implied warranty of           |
 ; | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the            |
 ; | GNU Affero General Public License for more details.                      |
 ; |                                                                          |
 ; | You should have received a copy of the GNU Affero General Public License |
 ; | along with this program.  If not, see .    |
 ; ----------------------------------------------------------------------------
 ;
 QUIT
 ;
 ; By default this server code runs on port 6330
 ;
 ; For GT.M systems it is invoked via xinetd:
 ;   Edit /etc/services and add the line:
 ; 
 ;    mwire  6330/tcp  # Service for M/Wire Protocol
 ;
 ;   Copy the file mwire to /etc/xinetd.d/mwire
 ;   Copy the file zmwire to /usr/local/gtm/zmwire and change 
 ;    its permissions to executable (eg 755)
 ;
 ;   These files may be edited to change the paths or port number
 ;   Restart xinetd using: sudo /etc/init.d/xinetd restart
 ;
 ;   On GT.M systems you must also have installed MGWSI or m_apache
 ;     in order to provide the MD5 hashing function for passwords
 ;     Alternatively substitute the MD5 callout to the MD5 function of your choice
 ;
 ; For Cache systems, it is invoked via the M/Wire Daemon routine
 ;   which should be running as a jobbed process:
 ;
 ;     job start^zmwireDaemon
 ;
 ;   You can change the port number by simply editing the line
 ; 
 ;      port+1^zmwireDaemon
 ;
 ;    Stop the Daemon process using ^RESJOB and restart it.
 ;
mwireVersion
 ;;Build 22
 ;
mwireDate
 ;;06 July 2011
 ;
version
 ;
 s output="+M/Wire "_$p($t(mwireVersion+1),";;",2,2000)_crlf
 w output
 QUIT
 ;
build
 ;
 n crlf,response
 ;
 s crlf=$c(13,10)
 s response="*3"_crlf
 w response
 i $g(^zewd("trace"))=1 d trace("build: "_response_" sent")
 ;
 s response=$p($t(mwireVersion+1),";;",2,2000)
 s response="$"_$l(response)_crlf_response_crlf
 w response
 i $g(^zewd("trace"))=1 d trace("build: "_response_" sent")
 ;
 s response=$p($t(mwireDate+1),";;",2,2000)
 s response="$"_$l(response)_crlf_response_crlf
 w response
 i $g(^zewd("trace"))=1 d trace("build: "_response_" sent")
 ;
 s response=$zv
 s response="$"_$l(response)_crlf_response_crlf
 w response
 i $g(^zewd("trace"))=1 d trace("build: "_response_" sent")
 QUIT
 ;
command ;
 n authNeeded,c,crlf,input,output
 ;
 d cleardown
 i $zv["GT.M" s $zint="d monitoroutput"
 s ^zmwire("connected",$j)=""
 l +^zmwire("connected",$j)
 s crlf=$c(13,10)
 s authNeeded=0
 s role="user"
 i $d(^zmwire("auth")) s authNeeded=1
loop
 i $g(^zmwire("relink"))=1,'$d(^zmwire("relink","process",$j)) s ok=$$relinkRoutines()
 r *c
 i $c(c)="*" d
 . s input=$$multiBulkRequest()
 . i $g(^zewd("trace"))=1 d trace($j_": "_$h_": mwire input: "_input)
 e  d
 . r input s input=$c(c)_input 
 . i $g(^zewd("trace"))=1 d trace($h_": mwire input: "_input)
 . i input'="" d
 . . i $zv["GT.M" s input=$e(input,1,$l(input)-2)
 . . i $zv["Cache" s input=$e(input,1,$l(input)-1)
 ;i input="PING" w "+PONG"_crlf g loop
 i input="PING" s output="+PONG"_crlf w output g loop
 i $d(^zmwire("monitor","listener")) d log(input)
 i input="" g loop
 i input="EXIT" g halt
 i input="QUIT" g quit
 i input="HALT" g halt
 ;
 i authNeeded,$e(input,1,4)'="AUTH" s output="-Authentication required"_crlf w output g loop
 i 'authNeeded,$e(input,1,4)="AUTH" s output="-Authentication ignored"_crlf w output g loop
 i $e(input,1,5)="AUTH " d auth($e(input,6,$l(input))) g loop
 i 'authNeeded!(role="admin"),$e(input,1,12)="SETPASSWORD " d setpassword($e(input,13,$l(input))) g loop
 ;
 s input=$$utfConvert(input)
 i $e(input,1,4)="SET " d set($e(input,5,$l(input))) g loop
 i $e(input,1,10)="GETGLOBAL " d getGlobal($e(input,11,$l(input))) g loop
 i $e(input,1,14)="GETJSONSTRING " d getJSON($e(input,15,$l(input))) g loop
 i $e(input,1,14)="SETJSONSTRING " d setJSON($e(input,15,$l(input))) g loop
 i $e(input,1,15)="RUNTRANSACTION " d runTransaction($e(input,16,$l(input))) g loop
 i $e(input,1,4)="GET " d get($e(input,5,$l(input))) g loop
 i $e(input,1,7)="INCRBY " d incrby($e(input,8,$l(input))) g loop
 i $e(input,1,7)="DECRBY " d decrby($e(input,8,$l(input))) g loop
 i $e(input,1,14)="NEXTSUBSCRIPT " d nextSubscript($e(input,15,$l(input)),1) g loop
 i $e(input,1,18)="PREVIOUSSUBSCRIPT " d nextSubscript($e(input,19,$l(input)),-1) g loop
 i $e(input,1,5)="KILL " d kill($e(input,6,$l(input))) g loop
 i $e(input,1,4)="DEL " d kill($e(input,5,$l(input))) g loop
 i $e(input,1,5)="DATA " d data($e(input,6,$l(input))) g loop
 i $e(input,1,7)="EXISTS " d data($e(input,8,$l(input))) g loop
 i $e(input,1,5)="INCR " d incr($e(input,6,$l(input))) g loop
 i $e(input,1,5)="DECR " d decr($e(input,6,$l(input))) g loop
 i $e(input,1,11)="COPYGLOBAL " d copy($e(input,12,$l(input))) g loop
 i $e(input,1,5)="LOCK " d lock($e(input,6,$l(input))) g loop
 i $e(input,1,7)="UNLOCK " d unlock($e(input,8,$l(input))) g loop
 i $e(input,1,6)="ORDER " d order($e(input,7,$l(input))) g loop
 i $e(input,1,5)="NEXT " d order($e(input,6,$l(input))) g loop
 i $e(input,1,9)="ORDERALL " d orderall($e(input,10,$l(input))) g loop
 i $e(input,1,11)="GETGLOBALS2" d getGlobals() g loop
 i $e(input,1,10)="GETGLOBALS" d getGlobalList() g loop
 i $e(input,1,9)="MULTIGET " d multiGet($e(input,10,$l(input))) g loop
 i $e(input,1,11)="GETALLSUBS " d orderall($e(input,12,$l(input))) g loop
 i $e(input,1,14)="GETSUBSCRIPTS " d getAllSubscripts($e(input,15,$l(input))) g loop
 i $e(input,1,9)="REVORDER " d reverseorder($e(input,10,$l(input))) g loop
 i $e(input,1,9)="PREVIOUS " d reverseorder($e(input,10,$l(input))) g loop
 i $e(input,1,6)="QUERY " d query($e(input,7,$l(input))) g loop
 i $e(input,1,9)="QUERYGET " d queryget($e(input,10,$l(input))) g loop
 i $e(input,1,10)="MERGEFROM " d mergefrom($e(input,11,$l(input))) g loop
 i $e(input,1,11)="GETSUBTREE " d mergefrom($e(input,12,$l(input))) g loop
 i $e(input,1,8)="MERGETO " d mergeto($e(input,9,$l(input))) g loop
 i $e(input,1,11)="SETSUBTREE " d mergeto($e(input,12,$l(input))) g loop
 i $e(input,1,9)="FUNCTION " d function($e(input,10,$l(input))) g loop
 i $e(input,1,8)="EXECUTE " d function($e(input,9,$l(input))) g loop
 i $e(input,1,6)="TSTART" d tstart g loop
 i $e(input,1,7)="TCOMMIT" d tcommit g loop
 i $e(input,1,9)="TROLLBACK" d trollback g loop
 i $e(input,1,5)="MDATE" d mdate g loop
 i $e(input,1,9)="PROCESSID" d processid g loop
 i $e(input,1,7)="VERSION" d version g loop
 i $e(input,1,8)="GETBUILD" d build g loop
 i $e(input,1,8)="MVERSION" d zv g loop
 i $e(input,1,4)="INFO" d info g loop
 i $e(input,1,7)="MONITOR" d monitor g loop
 s output="-"_input_" not recognized"_crlf w output
 g loop
 ;
multiBulkRequest()
 ;
 n buff,c,i,input,j,len,noOfCommands,param,space
 ;
 s noOfCommands=""
 f  d  q:c=13
 . r *c q:c=13
 . ;d trace("0: "_$c(c))
 . s noOfCommands=noOfCommands_$c(c)
 r *x
 ;d trace("1: "_$c(x))
 ;
 f i=1:1:noOfCommands d
 . ;d trace("i="_i)
 . s len=""
 . f  d  q:c=13
 . . r *c
 . . ;d trace("2: "_$c(c))
 . . i $c(c)="$",len="" q
 . . q:c=13
 . . s len=len_$c(c)
 . r *c
 . ;d trace("3: "_$c(c))
 . s input=""
 . i len=0 d
 . . s param(i)=""
 . . r *c
 . . ;d trace("4: "_$c(c))
 . e  d
 . . r input#len
 . . ;d trace("input="_input)
 . . s param(i)=input
 . . ;d trace("param "_i_" = "_input)
 . . r *c,*c
 ;
 s param(1)=$zconvert(param(1),"U")
 ;QUIT "PING"
 i param(1)="PING" QUIT param(1)
 ;s no=$increment(^rob) m ^rob(no)=param
 i param(1)="SET" QUIT param(1)_" "_param(2)_" "_$l(param(3))_crlf_param(3)
 i param(1)="SETJSONSTRING" QUIT param(1)_" "_param(2)_crlf_param(3)_crlf_param(4)
 i param(1)="COPYGLOBAL"!(param(1)="GETSUBSCRIPTS") d  QUIT input
 . s space="",input=""
 . f i=1:1:noOfCommands d
 . . s input=input_space_param(i)
 . . i space="" s space=" " q
 . . i space=" " s space=$c(1)
 i param(1)="EXECUTE" d  QUIT input
 . ;d trace("Execute: param(3)="_param(3))
 . s param(3)=$$replaceAll(param(3),"\""","""""")
 . i $e(param(3),1)="[" s input=param(1)_" "_param(2)_"("_$e(param(3),2,$l(param(3))-1)_")" q
 . s input=param(1)_" "_param(2)
 ;
 s space="",input=""
 f i=1:1:noOfCommands d
 . s input=input_space_param(i)
 . s space=" " 
 ;
 QUIT input
 ;
halt
 k ^zmwire("connected",$j)
 HALT
 ;
quit
 ;
 i '$d(^zmwire("monitor","listener",$j)) g halt
 k ^zmwire("monitor","listener",$j)
 g loop
 ;
cleardown
 ;
 n ignore,pid
 ;
 s pid=""
 f  s pid=$o(^zmwire("connected",pid)) q:pid=""  d
 . i pid=$j q
 . s ignore=1
 . l +^zmwire("connected",pid):0 e  s ignore=0
 . i ignore d
 . . l -^zmwire("connected",pid)
 . . k ^zmwire("connected",pid)
 . . k ^zmwire("monitor","listener",pid)
 . . k ^zmwire("monitor","output",pid)
 s pid=""
 f  s pid=$o(^zmwire("monitor","output",pid)) q:pid=""  d
 . i pid=$j q
 . s ignore=1
 . l +^zmwire("connected",pid):0 e  s ignore=0
 . l -^zmwire("connected",pid)
 . i ignore d
 . . k ^zmwire("monitor","output",pid)
 QUIT
 ;
monitor
 ;
 i $zv'["GT.M" w "-Command unavailable"_crlf QUIT
 n quit
 ;
 s ^zmwire("monitor","listener",$j)=""
 s output="+OK"_crlf w output
 f  h 1 r quit:0  i $e(quit,1,4)="QUIT" q
 k ^zmwire("monitor","listener",$j)
 s output="+OK"_crlf w output
 QUIT
 ;
log(input)
 ;
 i $zv'["GT.M" QUIT
 ;
 QUIT:'$d(^zmwire("monitor","listener"))
 ;
 n dev,inputr,io,lineNo,pid
 ;
 i input["AUTH" QUIT
 i input["QUIT" QUIT
 i input["EXIT" QUIT
 i input["HALT" QUIT
 s inputr=$re(input)
 i $e(inputr,1,2)'=$c(10,13) s input=input_crlf
 s pid=""
 f  s pid=$o(^zmwire("monitor","listener",pid)) q:pid=""  d
 . i pid=$j q
 . s lineNo=$o(^zmwire("monitor","output",pid,""),-1)+1
 . s ^zmwire("monitor","output",pid,lineNo)=input
 . ;zsy "mupip intrpt "_pid_" >/dev/null"
 . zsy "kill -USR1 "_pid
 ;
 QUIT
 ;
monitoroutput
 ;
 n lineNo
 ;
 s lineNo=""
 f  s lineNo=$o(^zmwire("monitor","output",$j,lineNo)) q:lineNo=""  d
 . w ^zmwire("monitor","output",$j,lineNo)
 . k ^zmwire("monitor","output",$j,lineNo)
 QUIT
 ;
logger(command,initialise)
 ;
 n tot,count
 ;
 i $g(initialise) k ^mwireLogger
 s tot=$increment(^mwireLogger)
 s count=$increment(^mwireLogger(command))
 ;
 QUIT
 ;
info
 ;
 n count,ignore,pid,response
 ;
 s response="m_wire_version:"_$p($t(mwireVersion+1),";;",2,2000)_crlf
 s pid="",count=0
 f  s pid=$o(^zmwire("connected",pid)) q:pid=""  d
 . s ignore=1
 . i pid=$j d
 . . s ignore=0
 . e  d
 . . l +^zmwire("connected",pid):0 e  s ignore=0
 . i ignore d
 . . l -^zmwire("connected",pid)
 . . k ^zmwire("connected",pid)
 . . k ^zmwire("monitor","listener",pid)
 . e  d
 . . s count=count+1
 s response=response_"connected_clients:"_count ;_crlf
 s output="$"_$l(response)_crlf_response_crlf
 w output
 QUIT
 ;
auth(input)
 ;
 n pass
 s pass=$$MD5(input)
 i $d(^zmwire("auth",pass)) d
 . s authNeeded=0
 . s role=^zmwire("auth",pass)
 . s output="+OK"_crlf
 . w output
 e  d
 . s output="-Invalid password"_crlf
 . w output
 QUIT
 ;
setpassword(input)
 ;
 ; SETPASSWORD secret
 ; +OK 
 ;
 ; SETPASSWORD secret admin
 ; +OK
 ;
 n pass,newrole
 ;
 i $d(^zmwire("auth")),role'="admin" s output="-Invalid command"_crlf w output QUIT
 i $$stripSpaces(input)="" s output="-Invalid command"_crlf w output QUIT
 s newrole="user"
 i input[" " d
 . s newrole=$p(input," ",2)
 . s input=$p(input," ",1)
 i '$d(^zmwire("auth")) s newrole="admin"
 i newrole'="user",newrole'="admin" s output="-Invalid role"_crlf w output QUIT
 ;
 s pass=$$MD5(input)
 s ^zmwire("auth",pass)=newrole
 s output="+OK"_crlf
 w output
 QUIT
 ;
getGloRef(input)
 ;
 n gloName,gloRef,nb,subs
 ;
 s gloRef=input
 i $e(gloRef,1)'="^" s gloRef="^"_gloRef
 i $e(gloRef,$l(gloRef))="]" s gloRef=$e(gloRef,1,$l(gloRef)-1)
 s gloName=$p(gloRef,"[",1)
 s nb=$l(gloRef,"[")+2
 s subs=$p(gloRef,"[",2,nb)
 i subs="" QUIT gloName
 QUIT gloName_"("_subs_")"
 ;
set(input)
 ;
 n c123,data,gloName,gloRef,i,inputr,json,len,nb,nsp,ok,quot,subs,x
 ;
 ; SET myglobal["1","xx yy",3] 5
 ; hello
 ; +OK
 ; SET myGlo 5
 ; hello
 ; +OK
 ;
 ;n n
 ;s n=$increment(^rob("set"))
 ;s ^rob("set",n)=input
 i input[crlf d
 . s data=$p(input,crlf,2,$l(input))
 . s input=$p(input,crlf,1)
 s nsp=$l(input," ")
 s len=$p(input," ",nsp)
 i len'=0,+len=0 s output="-Data length was not specified"_crlf w output QUIT
 s gloRef=$p(input," ",1,nsp-1)
 i $e(gloRef,1)'="^" s gloRef="^"_gloRef
 i $e(gloRef,$l(gloRef))="]" s gloRef=$e(gloRef,1,$l(gloRef)-1)
 s gloName=$p(gloRef,"[",1)
 s nb=$l(gloRef,"[")+2
 s subs=$p(gloRef,"[",2,nb)
 ; Process Javascript escaping
 i subs'="" s subs=subs_")"
 s quot=""""
 i subs'="" f i=1:1:$l(subs)-2 d
 . s c123=$e(subs,i,i+2)
 . i $e(c123,1)'="\" q
 . i $e(c123,2)="\" q
 . i $e(c123,2)=quot d  q
 . . i $e(c123,3)="," q
 . . i $e(c123,3)=")" q
 . . s subs=$e(subs,1,i-1)_quot_quot_$e(subs,i+2,$l(subs))
 i subs["\\" d
 . s subs=$$replaceAll(subs,"\\",$c(5))
 . s subs=$$replaceAll(subs,$c(5),"\")
 s gloRef=gloName
 i gloRef["^zmwire" s output="-No access allowed to this global"_crlf w output QUIT
 i subs'="" s gloRef=gloRef_"("_subs
 i '$d(data)  d
 . s data=$$readChars(len)
 . r ok 
 . i $d(^zmwire("monitor","listener")) d log(data)
 i data["""" s data=$$replaceAll(data,"""","""""")
 i data="zmwire_null_value" s data=""
 s x="s "_gloRef_"="""_data_""""
 i $g(^zewd("trace"))=1 d trace("SET x = "_x)
 s $zt=$$zt()
 x x
 s $zt=""
 s json="{""ok"":true}"
 s output="$"_$l(json)_crlf_json_crlf
 w output
 i $g(^zewd("trace"))=1 d trace("set: ok:true sent")
 i $g(^mwire("logger"))=1 d logger("set")
 QUIT
 ;
getGlobalList()
 ;
 n arrString,comma,count,glo,gloRef,list,response,x
 ;
 i $zv["GT.M" d
 . s x="^%"
 . i $d(@x) s list(x)=""
 . f  s x=$order(@x) q:x=""  s list(x)=""
 . ;
 e  d
 . d getGlobalList^MDBMCache(.list)
 ;
 ;s count=0,glo=""
 ;f  s glo=$o(list(glo)) q:glo=""  s count=count+1
 ;s response="*"_count_crlf
 ;w response
 ;i $g(^zewd("trace"))=1 d trace("getGlobalList: "_response_" sent") 
 ;s glo=""
 ;f  s glo=$o(list(glo)) q:glo=""  d
 ;. s gloRef=$e(glo,2,$l(glo))
 ;. s response="$"_$l(gloRef)_crlf_gloRef_crlf
 ;. w response
 ;. i $g(^zewd("trace"))=1 d trace("getGlobalList: "_response_" sent") 
 s arrString="["
 s glo="",comma=""
 f  s glo=$o(list(glo)) q:glo=""  d
 . s gloRef=$e(glo,2,$l(glo))
 . s arrString=arrString_comma_""""_gloRef_""""
 . s comma=","
 s arrString=arrString_"]"
 s response="$"_$l(arrString)_crlf_arrString_crlf
 w response
 i $g(^zewd("trace"))=1 d trace("getGlobalList: "_response_" sent")  ;
 QUIT
 ;
getGlobals()
 ;
 n arrString,comma,count,glo,gloRef,list,response,x
 ;
 i $zv["GT.M" d
 . s x="^%"
 . i $d(@x) s list(x)=""
 . f  s x=$order(@x) q:x=""  s list(x)=""
 . ;
 e  d
 . d getGlobalList^MDBMCache(.list)
 ;
 s count=0,glo=""
 f  s glo=$o(list(glo)) q:glo=""  s count=count+1
 s response="*"_count_crlf
 w response
 i $g(^zewd("trace"))=1 d trace("getGlobalList: "_response_" sent") 
 s glo=""
 f  s glo=$o(list(glo)) q:glo=""  d
 . s gloRef=$e(glo,2,$l(glo))
 . s response="$"_$l(gloRef)_crlf_gloRef_crlf
 . w response
 . i $g(^zewd("trace"))=1 d trace("getGlobalList: "_response_" sent") 
 QUIT
 ;
get(input)
 ;
 n data,exists,gloRef,response,x
 ;
 ; GET myglobal["1","xx yy",3]
 ; $6
 ; foobar
 ;
 s gloRef=$$getGloRef(input)
 i gloRef["^zmwire" s output="-No access allowed to this global"_crlf w output QUIT
 s x="s exists=$d("_gloRef_")"
 s $zt=$$zt()
 x x
 s $zt=""
 i exists'=1,exists'=11 s output="$-1"_crlf w output QUIT
 s x="s data="_gloRef
 s $zt=$$zt()
 x x
 s $zt=""
 ;
 s response="$"_$l(data)_crlf_data_crlf
 ;
 w response
 i $g(^mwire("logger"))=1 d logger("get")
 QUIT
 ;
getGlobal(input)
 ;
 n data,exists,gloRef,json,response,x
 ;
 ; GET myglobal["1","xx yy",3]
 ; $6
 ; foobar
 ;
 s gloRef=$$getGloRef(input)
 i gloRef["^zmwire" s output="-No access allowed to this global"_crlf w output QUIT
 s x="s exists=$d("_gloRef_"),"
 ;x x
 s x=x_"data=$g("_gloRef_")"
 s $zt=$$zt()
 x x
 s $zt=""
 s json="{""value"":"""_data_""",""dataStatus"":"_exists_"}"
 ;
 s response="$"_$l(json)_crlf_json_crlf
 ;
 w response
 i $g(^zewd("trace"))=1 d trace("getGlobal: response="_response)
 i $g(^mwire("logger"))=1 d logger("getGlobal")
 QUIT
 ;
multiGet(input)
 ;
 n comma,dataStatus,error,exists,globalName,i,json,props,ref,response,subs,value
 ;
 ; MULTIGET myglobal[
 ;  {globalName:'xxx', subscripts:["x1","y1"]},
 ;  {globalName:'xxx', subscripts:["x2",""]},
 ;  {globalName:'xxx', subscripts:["x2","y2"]}
 ;]
 ;
 ;
 s error=$$parseJSON(input,.props,1)
 i error'="" s output="-"_error_crlf w output QUIT
 ;
 s stop=0,error="",json="[",comma=""
 f i=1:1 q:'$d(props(i))  d
 . s dataStatus=0,value="",error=""
 . s globalName=$g(props(i,"globalName"))
 . i globalName="" s globalName=$g(props(i,"GlobalName"))
 . i globalName="" d
 . . s error="globalName not defined"
 . e  d
 . . i $e(globalName,1)'="^" s globalName="^"_globalName
 . . s ref="s exists=$d("_globalName
 . . s subs=""
 . . i $d(props(i,"subscripts")) d
 . . . n comma,j,stop,sub
 . . . s subs="(",comma="",stop=0
 . . . f j=1:1 q:'$d(props(i,"subscripts",j))  d  q:stop
 . . . . s sub=props(i,"subscripts",j)
 . . . . i sub="" s error="Subscript "_j_" is null",stop=1 q
 . . . . s subs=subs_comma_""""_props(i,"subscripts",j)_"""",comma=","
 . . . i error="" s subs=subs_")"
 . . i error="" d
 . . . s ref=ref_subs_")"
 . . . s $zt=$$zt()
 . . . x ref
 . . . s $zt=""
 . . . i exists d
 . . . . s ref="s value=$g("_globalName
 . . . . s ref=ref_subs_")"
 . . . . s $zt=$$zt()
 . . . . x ref
 . . . . s $zt=""
 . s json=json_comma_"{""value"":"""_value_""",""dataStatus"":"_exists_",""error"":"""_error_"""}"
 . s comma=","
 s json=json_"]"
 ;
 s response="$"_$l(json)_crlf_json_crlf
 ;
 w response
 i $g(^zewd("trace"))=1 d trace("multiGet: response="_response)
 QUIT
 ;
kill(input)
 ;
 n i,glo,gloRef,len,nsp,p1,p2,x
 ;
 ; KILL myglobal["1","xx yy",3]
 ; +OK
 ;
 s glo=input
 s p1=$p(glo,"[",1)
 s p2=$p(glo,"[",2,2000)
 s p2=$e(p2,1,$l(p2)-1)
 s glo=p1_"("_p2_")"
 i glo["()" d
 . s len=$l(glo)
 . i $e(glo,len-1,len)="()" s glo=$e(glo,1,len-2)
 e  d
 . ; Process Javascript escaping
 . n c123,quot
 . s quot=""""
 . f i=1:1:$l(glo)-2 d
 . . s c123=$e(glo,i,i+2)
 . . i $e(c123,1)'="\" q
 . . i $e(c123,2)="\" q
 . . i $e(c123,2)=quot d  q
 . . . i $e(c123,3)="," q
 . . . i $e(c123,3)=")" q
 . . . s glo=$e(glo,1,i-1)_quot_quot_$e(glo,i+2,$l(glo))
 . i glo["\\" d
 . . s glo=$$replaceAll(glo,"\\",$c(5))
 . . s glo=$$replaceAll(glo,$c(5),"\")
 ;
 i glo'["zmwire" s glo(glo)=""
 s x="k ^"_glo
 s $zt=$$zt()
 x x
 s $zt=""
 s response="+ok"_crlf
 i $g(^%zewd("trace"))=1 d trace("kill: response="_response)
 w response
 i $g(^mwire("logger"))=1 d logger("kill")
 QUIT
 ;
data(input)
 ;
 n data,gloRef,x
 ;
 ; DATA myglobal["1","xx yy",3]
 ; :10
 ;
 s gloRef=$$getGloRef(input)
 i gloRef["^zmwire" s output="-No access allowed to this global"_crlf w output QUIT
 s x="s data=$d("_gloRef_")"
 s $zt=$$zt()
 x x
 s $zt=""
 i $g(^zewd("trace"))=1 d trace("input="_input_"; data="_data)
 s output=":"_data_crlf
 w output
 i $g(^mwire("logger"))=1 d logger("data")
 QUIT
 ;
runTransaction(input)
 ;
 n error,globalName,i,json,props,ref,result,stop,subscripts
 ;
 s error=$$parseJSON(input,.props,1)
 i error'="" s output="-"_error_crlf w output QUIT
 ;
 s stop=0,error=""
 f i=1:1 q:'$d(props(i))  d  q:stop
 . s method=$g(props(i,"method"))
 . i method="" s stop=1,error="Missing method in JSON transaction document at step "_i q
 . i method="setJSON" d  q:stop
 . . n json
 . . m json=props(i,"json")
 . . i '$d(json) s stop=1,error="Missing JSON document in JSON transaction document at step "_i q
 . . s globalName=$g(props(i,"globalName"))
 . . i globalName="" s globalName=$g(props(i,"GlobalName"))
 . . i globalName="" s stop=1,error="Missing Global name in JSON transaction document at step "_i q
 . . i $e(globalName,1)'="^" s globalName="^"_globalName
 . . s ref="m "_globalName
 . . i $d(props(i,"subscripts")) d
 . . . n comma,j
 . . . s ref=ref_"(",comma=""
 . . . f j=1:1 q:'$d(props(i,"subscripts",j))  d
 . . . . s ref=ref_comma_""""_props(i,"subscripts",j)_"""",comma=","
 . . . s ref=ref_")"
 . . s ref=ref_"=json"
 . . x ref
 . i method="setGlobal" d  q:stop
 . . n value
 . . s globalName=$g(props(i,"globalName"))
 . . i globalName="" s globalName=$g(props(i,"GlobalName"))
 . . i globalName="" s stop=1,error="Missing Global name in JSON transaction document at step "_i q
 . . i $e(globalName,1)'="^" s globalName="^"_globalName
 . . s ref="s "_globalName
 . . i $d(props(i,"subscripts")) d
 . . . n comma,j
 . . . s ref=ref_"(",comma=""
 . . . f j=1:1 q:'$d(props(i,"subscripts",j))  d
 . . . . s ref=ref_comma_""""_props(i,"subscripts",j)_"""",comma=","
 . . . s value=$g(props(i,"value"))
 . . . i value="zmwire_null_value" s value=""
 . . . s ref=ref_")="""_value_""""
 . . x ref
 . i method="kill" d  q:stop
 . . s globalName=$g(props(i,"globalName"))
 . . i globalName="" s globalName=$g(props(i,"GlobalName"))
 . . i globalName="" s stop=1,error="Missing Global name in JSON transaction document at step "_i q
 . . i $e(globalName,1)'="^" s globalName="^"_globalName
 . . s ref="k "_globalName
 . . i $d(props(i,"subscripts")) d
 . . . n comma,j
 . . . s ref=ref_"(",comma=""
 . . . f j=1:1 q:'$d(props(i,"subscripts",j))  d
 . . . . s ref=ref_comma_""""_props(i,"subscripts",j)_"""",comma=","
 . . . s ref=ref_")"
 . . x ref
 ;
 i error'="" s output="-"_error_crlf w output QUIT
 s response="+ok"_crlf
 w response
 i $g(^zewd("trace"))=1 d trace("transaction: response="_response)
 QUIT
 ;
setJSON(input)
 ;
 n arr,del,error,flrc,inputr,gloRef,inputr,json,nb,nsp,props,ref,response,subs
 ;
 ; SETJSONSTRING myglobal["1","xx yy",3] CRLF {"a":123} CRLF 1
 ; +ok
 ;
 s flrc=$c(10,13)
 s gloRef=$p(input,crlf,1)
 s input=$p(input,crlf,2,10000)
 s inputr=$re(input)
 s del=$p(inputr,flrc,1),del=$re(del)
 s inputr=$p(inputr,flrc,2,10000) ; in case it contains crlfs
 s json=$re(inputr)
 i $zv["GT.M" d
 . s json=$$unEscape(json)
 e  d
 . s json=$$unEscape^MDBMCache(json)
 ;
 i $e(gloRef,1)'="^" s gloRef="^"_gloRef
 i $e(gloRef,$l(gloRef))="]" s gloRef=$e(gloRef,1,$l(gloRef)-1)
 s gloName=$p(gloRef,"[",1)
 s nb=$l(gloRef,"[")+2
 s subs=$p(gloRef,"[",2,nb)
 s gloRef=gloName
 i gloRef["^zmwire" s output="-No access allowed to this global"_crlf w output QUIT
 i subs'="" s gloRef=gloRef_"("_subs_")"
 s error=$$parseJSON(json,.props,1)
 i error'="" s output="-Invalid JSON in setJSON: "_json_crlf w output QUIT
 ;
 s ref=""
 i del s ref="k "_gloRef_" "
 ;
 s ref=ref_"m "_gloRef_"=props"
 x ref
 s response="+OK"_crlf
 w response
 i $g(^zewd("trace"))=1 d trace("setJSON: response="_response)
 i $g(^mwire("logger"))=1 d logger("setJSON")
 ;
 QUIT
 ;
getJSON(input)
 ;
 n arr,inputr,gloRef,json,nb,nsp,ref,response,subs
 ;
 ; GETJSONSTRING myglobal["1","xx yy",3]
 ; $5
 ; {x:1}
 ;
 s gloRef=input
 i $e(gloRef,1)'="^" s gloRef="^"_gloRef
 i $e(gloRef,$l(gloRef))="]" s gloRef=$e(gloRef,1,$l(gloRef)-1)
 s gloName=$p(gloRef,"[",1)
 s nb=$l(gloRef,"[")+2
 s subs=$p(gloRef,"[",2,nb)
 s gloRef=gloName
 i gloRef["^zmwire" s output="-No access allowed to this global"_crlf w output QUIT
 i subs'="" s gloRef=gloRef_"("_subs_")"
 s ref="m arr="_gloRef
 x ref
 i '$d(arr) d
 . s response="$-1"_crlf
 e  d
 . s json=$$arrayToJSON("arr")
 . s response="$"_$l(json)_crlf_json_crlf
 i $g(^zewd("trace"))=1 d trace("getJSON: response="_response)
 ;
 w response
 ;
 i $g(^mwire("logger"))=1 d logger("getJSON")
 QUIT
 ;
incr(input)
 ;
 n data,gloRef,x
 ;
 ; INCR myglobal["1","xx yy",3]
 ; :4
 ;
 s gloRef=$$getGloRef(input)
 i gloRef["^zmwire" s output="-No access allowed to this global"_crlf w output QUIT
 s x="s data=$increment("_gloRef_")"
 s $zt=$$zt()
 x x
 s $zt=""
 s output=":"_data_crlf
 w output
 i $g(^mwire("logger"))=1 d logger("incr")
 QUIT
 ;
incrby(input)
 ;
 n by,data,gloName,gloRef,inputr,len,nb,nsp,ok,subs,x
 ;
 ; INCRBY myglobal["1","xx yy",3] 3
 ; :7
 ;
 s inputr=$re(input)
 s by=$re($p(inputr," ",1))
 s nsp=$l(input," ")+2
 s gloRef=$p(inputr," ",2,nsp)
 s gloRef=$re(gloRef)
 i $e(gloRef,1)'="^" s gloRef="^"_gloRef
 i $e(gloRef,$l(gloRef))="]" s gloRef=$e(gloRef,1,$l(gloRef)-1)
 s gloName=$p(gloRef,"[",1)
 s nb=$l(gloRef,"[")+2
 s subs=$p(gloRef,"[",2,nb)
 s gloRef=gloName
 i gloRef["^zmwire" s output="-No access allowed to this global"_crlf w output QUIT
 i subs'="" s gloRef=gloRef_"("_subs_")"
 s x="s data=$increment("_gloRef_","_by_")"
 s $zt=$$zt() x x
 s $zt=""
 s output=":"_data_crlf
 w output
 i $g(^mwire("logger"))=1 d logger("incrbr")
 QUIT
 ;
function(input)
 ;
 n data,func,x
 ;
 ; FUNCTION label^rou("1","xx yy")
 ; $5
 ; hello
 ;
 s func=input
 i func["^",$e(func,1,2)'="$$" s func="$$"_func
 i func["class(",$e(func,1,2)'="##" s func="##"_func
 s x="s data="_func
 s $zt=$$zt()
 x x
 s $zt=""
 s output="$"_$l(data)_crlf_data_crlf
 w output
 i $g(^mwire("logger"))=1 d logger("function")
 QUIT
 ;
decr(input)
 ;
 n data,gloRef,x
 ;
 ; DECR myglobal["1","xx yy",3]
 ; :3
 ;
 s gloRef=$$getGloRef(input)
 i gloRef["^zmwire" s output="-No access allowed to this global"_crlf w output QUIT
 s x="s data=$increment("_gloRef_",-1)"
 s $zt=$$zt() x x
 s $zt=""
 s output=":"_data_crlf
 w output
 i $g(^mwire("logger"))=1 d logger("decr")
 QUIT
 ;
decrby(input)
 ;
 n by,data,gloName,gloRef,inputr,nb,nsp,ok,subs,x
 ;
 ; DECRBY myglobal["1","xx yy",3] 3
 ; :4
 ;
 s inputr=$re(input)
 s by=$re($p(inputr," ",1))
 s nsp=$l(input," ")+2
 s gloRef=$p(inputr," ",2,nsp)
 s gloRef=$re(gloRef)
 i $e(gloRef,1)'="^" s gloRef="^"_gloRef
 i $e(gloRef,$l(gloRef))="]" s gloRef=$e(gloRef,1,$l(gloRef)-1)
 s gloName=$p(gloRef,"[",1)
 s nb=$l(gloRef,"[")+2
 s subs=$p(gloRef,"[",2,nb)
 s gloRef=gloName
 i gloRef["^zmwire" s output="-No access allowed to this global"_crlf w output QUIT
 i subs'="" s gloRef=gloRef_"("_subs_")"
 s x="s data=$increment("_gloRef_",-"_by_")"
 s $zt=$$zt()
 x x
 s $zt=""
 s output=":"_data_crlf
 w output
 i $g(^mwire("logger"))=1 d logger("decrby")
 QUIT
 ;
nextSubscript(input,direction)
 ;
 n data,gloRef,response,subscript,x,value
 ;
 ; NEXTSUBSCRIPT myglobal["1","xx yy",""]
 ; +abc
 ;
 s gloRef=$$getGloRef(input)
 i gloRef["^zmwire" s output="-No access allowed to this global"_crlf w output QUIT
 s x="s subscript=$o("_gloRef_",direction)"
 s $zt=$$zt()
 x x
 s $zt=""
 s value="",data=0
 i subscript'="" d
 . s value=$g(^(subscript))
 . s data=$d(^(subscript))
 ;
 s response="{""subscriptValue"":"""_subscript_""","
 s response=response_"""dataStatus"":"_data_","
 s response=response_"""dataValue"":"""_value_"""}"
 ;
 s response="$"_$l(response)_crlf_response_crlf
 i $g(^zewd("trace"))=1 d trace("nextsubscript: response="_response)
 w response
 i $g(^mwire("logger"))=1 d logger("nextsubscript")
 ;
 QUIT
 ;
order(input)
 ;
 n data,gloRef,x
 ;
 ; ORDER myglobal["1","xx yy",""]
 ; +abc
 ;
 s gloRef=$$getGloRef(input)
 i gloRef["^zmwire" s output="-No access allowed to this global"_crlf w output QUIT
 s x="s data=$o("_gloRef_")"
 s $zt=$$zt()
 x x
 s $zt=""
 i data="" s output="$-1"_crlf w output QUIT
 s output="$"_$l(data)_crlf_data_crlf
 w output
 i $g(^mwire("logger"))=1 d logger("order")
 QUIT
 ;
reverseorder(input)
 ;
 n data,gloRef,x
 ;
 ; REVORDER myglobal["1","xx yy",""]
 ; +abc
 ;
 s gloRef=$$getGloRef(input)
 i gloRef["^zmwire" s output="-No access allowed to this global"_crlf w output QUIT
 s x="s data=$o("_gloRef_",-1)"
 s $zt=$$zt()
 x x
 s $zt=""
 i data="" s output="$-1"_crlf w output QUIT
 s output="$"_$l(data)_crlf_data_crlf
 w output
 i $g(^mwire("logger"))=1 d logger("reverseorder")
 QUIT
 ;
query(input)
 ;
 n data,gloRef,nb,p1,p2,x
 ;
 ; QUERY myglobal["1","xx yy"]
 ;
 s gloRef=$$getGloRef(input)
 i gloRef["^zmwire" s output="-No access allowed to this global"_crlf w output QUIT
 s x="s data=$q("_gloRef_")"
 s $zt=$$zt()
 ;d trace("query x="_x)
 x x
 i data="" s output="$-1"_crlf w output QUIT
 s data=$e(data,2,$l(data))
 ;d trace("data="_data)
 s p1=$p(data,"(",1)
 s nb=$l(data,"(")+2
 ;d trace("nb="_nb)
 s p2=$p(data,"(",2,nb)
 ;d trace("1 p2="_p2)
 s p2=$e(p2,1,$l(p2)-1)
 ;d trace("2 p2="_p2)
 s data=p1_"["_p2_"]"
 s output="$"_$l(data)_crlf_data_crlf
 w output
 s $zt=""
 i $g(^mwire("logger"))=1 d logger("query")
 QUIT
 ;
queryget(input)
 ;
 n data,gloRef,nb,odata,p1,p2,value,x
 ;
 ; QUERYGET myglobal["1","xx yy"]
 ;
 s gloRef=$$getGloRef(input)
 i gloRef["^zmwire" s output="-No access allowed to this global"_crlf w output QUIT
 s x="s data=$q("_gloRef_")"
 s $zt=$$zt()
 x x
 i data="" s output="$-1"_crlf w output QUIT
 s odata=data
 s output="*2"_crlf
 w output
 s data=$e(data,2,$l(data))
 s p1=$p(data,"(",1)
 s nb=$l(data,"(")+2
 s p2=$p(data,"(",2,nb)
 s p2=$e(p2,1,$l(p2)-1)
 s data=p1_"["_p2_"]"
 s output="$"_$l(data)_crlf_data_crlf
 w output
 s value=@odata
 s output="$"_$l(value)_crlf_value_crlf
 w output
 s $zt=""
 i $g(^mwire("logger"))=1 d logger("queryget")
 QUIT
 ;
lock(input)
 ;
 n gloName,gloRef,inputr,nb,nsp,ok,subs,time,x
 ;
 ; LOCK myglobal["1","xxyy"] 5
 ; +OK
 ;
 s inputr=$re(input)
 s time=$re($p(inputr," ",1))
 i time?1N.N d
 . s nsp=$l(input," ")+2
 . s gloRef=$p(inputr," ",2,nsp)
 e  d
 . s time=5
 . s gloRef=inputr
 s gloRef=$re(gloRef)
 i $e(gloRef,1)'="^" s gloRef="^"_gloRef
 i $e(gloRef,$l(gloRef))="]" s gloRef=$e(gloRef,1,$l(gloRef)-1)
 s gloName=$p(gloRef,"[",1)
 i gloRef["^zmwire" s output="-No access allowed to this global"_crlf w output QUIT
 s nb=$l(gloRef,"[")+2
 s subs=$p(gloRef,"[",2,nb)
 s gloRef=gloName
 i subs'="" s gloRef=gloRef_"("_subs_")"
 s x="s ok=1 l +"_gloRef_":"_time_" e  s ok=0"
 s $zt=$$zt()
 x x
 s $zt=""
 s output=":"_ok_crlf
 w output
 QUIT
 ;
unlock(input)
 ;
 n gloRef,x
 ;
 ; UNLOCK myglobal["1","xxyy"]
 ; +OK
 ;
 s gloRef=$$getGloRef(input)
 i gloRef["^zmwire" s output="-No access allowed to this global"_crlf w output QUIT
 s x="l -"_gloRef
 s $zt=$$zt()
 x x
 s $zt=""
 s output="+OK"_crlf
 w output
 QUIT
 ;
getAllSubscripts(input)
 ;
 n comma,data,exists,from,i,gloRef,len,numericEnd,rec,ref
 n stop,subscripts,subs,subs1,to,x
 ;
 ; GETSUBSCRIPTS myglobal["1","xx yy"] fromValue toValue
 ; 
 s gloRef=$p(input,$c(1),1)
 s gloRef=$$getGloRef(gloRef)
 s from=$p(input,$c(1),2)
 i from="zz-null" s from=""
 s to=$p(input,$c(1),3)
 i to="zz-null" s to=""
 s numericEnd=$$numeric(to)
 ;d trace("to="_to_": numeric = "_numericEnd)
 ;
 i gloRef["^zmwire" s output="-No access allowed to this global"_crlf w output QUIT
 i $e(gloRef,$l(gloRef))=")" d
 . s x="s exists=$d("_gloRef_")"
 . s gloRef=$e(gloRef,1,$l(gloRef)-1)_","
 e  d
 . s x="s exists=$d("_gloRef_")"
 . s gloRef=gloRef_"("
 s $zt=$$zt()
 x x
 s $zt=""
 i 'exists!(exists=1) s output="$2"_crlf_"[]"_crlf w output QUIT
 ;
 s subs=from
 s subs1=subs i subs1["""" s subs1=$$replaceAll(subs1,"""","""""")
 i from'="" d
 . s x="s subs1=$o("_gloRef_""""_subs1_"""),-1)"
 . ;d trace("1 x="_x)
 . x x
 s x="s subs=$o("_gloRef_""""_subs1_"""))"
 x x
 s len=3+$l(subs)
 s comma=",",stop=0
 i subs'="" d
 . f  s subs=$o(^(subs)) q:stop  d
 . . i subs="" s stop=1 q
 . . i to'="" d  q:stop
 . . . i numericEnd d
 . . . . ;d trace("numeric: subs="_subs_": to="_to)
 . . . . i $$numeric(subs),subs>to s stop=1
 . . . e  d
 . . . . i subs]to s stop=1
 . . s len=len+$l(comma)+2+$l(subs)
 ;d trace("3 len="_len)
 s len=len+1
 s response="$"_len_crlf
 w response
 ;
 s x="s subs=$o("_gloRef_""""_subs1_"""))"
 ;d trace("4 x="_x)
 ;d trace("xx to="_to)
 x x
 s response="["""_subs_""""
 w response
 i subs'="" d
 . s stop=0
 . f  s subs=$o(^(subs)) q:stop  d
 . . i subs="" s stop=1 q
 . . i to'="" d  q:stop
 . . . i numericEnd d
 . . . . ;d trace("numeric: subs="_subs_": to="_to)
 . . . . i $$numeric(subs),subs>to s stop=1
 . . . e  d
 . . . . i subs]to s stop=1
 . . s response=comma_""""_subs_""""
 . . w response
 ;d trace("5 response="_response)
 s response="]"_crlf
 w response
 i $g(^mwire("logger"))=1 d logger("getallsubscripts")
 ;
 QUIT
 ;
orderall(input)
 ;
 n data,exists,i,gloRef,rec,subs,subs1,x
 ;
 ; ORDERALL myglobal["1","xx yy"] 
 ; *6
 ; $2
 ; aa
 ; $5
 ; hello
 ; $2
 ; bb
 ; $5
 ; world
 ; $3
 ; bba
 ; $-1
 ; 
 s gloRef=$$getGloRef(input)
 i gloRef["^zmwire" s output="-No access allowed to this global"_crlf w output QUIT
 i $e(gloRef,$l(gloRef))=")" d
 . s x="s exists=$d("_gloRef_")"
 . s gloRef=$e(gloRef,1,$l(gloRef)-1)_","
 e  d
 . s x="s exists=$d("_gloRef_")"
 . s gloRef=gloRef_"("
 s $zt=$$zt()
 x x
 i 'exists s output="$-1"_crlf w output QUIT
 ;
 s subs="",rec=0
 k ^CacheTempEWD($j)
 f  d  q:subs=""
 . s subs1=subs i subs1["""" s subs1=$$replaceAll(subs1,"""","""""")
 . s x="s subs=$o("_gloRef_""""_subs1_"""))"
 . x x
 . i subs="" q
 . s rec=rec+1
 . s ^CacheTempEWD($j,rec)="$"_$l(subs)_crlf_subs_crlf
 . s x="s exists=$d("_gloRef_""""_subs_"""))"
 . x x
 . i exists=1!(exists=11) d
 . . s x="s data="_gloRef_""""_subs_""")"
 . . x x
 . . s rec=rec+1
 . . s ^CacheTempEWD($j,rec)="$"_$l(data)_crlf_data_crlf
 . e  d
 . . s rec=rec+1
 . . s ^CacheTempEWD($j,rec)="$-1"_crlf
 s $zt=""
 s output="*"_rec_crlf
 w output
 f i=1:1:rec w ^CacheTempEWD($j,i)
 k ^CacheTempEWD($j)
 i $g(^mwire("logger"))=1 d logger("orderall")
 QUIT
 ;
mergefrom(input)
 ;
 n data,gloRef,i,params,resp,start,x
 ;
 ; MERGEFROM myglobal["1","a"]
 ; *6
 ; $1
 ; 1  
 ; $5
 ; hello  
 ; $9
 ; 1,"a\"aa"   note escaping
 ; $5
 ; world
 ; $8
 ; 2,"cccc"
 ; $3
 ; foo
 ;
 s gloRef=$$getGloRef(input)
 i gloRef["^zmwire" s output="-No access allowed to this global"_crlf w output QUIT
 k ^CacheTempEWD($j)
 s x="m ^CacheTempEWD($j)="_gloRef
 s $zt=$$zt()
 x x
 s $zt=""
 s x=$q(^CacheTempEWD($j,""))
 i x="" d  QUIT
 . s output="*-1"_crlf 
 . w output 
 . k ^CacheTempEWD($j)
 f i=1:1 s x=$q(@x) q:x=""
 i $d(^CacheTempEWD($j))=1!($d(^CacheTempEWD($j))=11) s i=i+1
 s output="*"_(i*2)_crlf
 w output
 i $d(^CacheTempEWD($j))=1!($d(^CacheTempEWD($j))=11) d
 . s output="$-1"_crlf
 . w output
 . s output="$"_$l(^CacheTempEWD($j))_crlf_^CacheTempEWD($j)_crlf
 . w output
 s x=$q(^CacheTempEWD($j,""))
 s start="^CacheTempEWD("_$j_","
 s params=$p(x,start,2,2000)
 s params=$e(params,1,$l(params)-1)
 s resp=params
 s output="$"_$l(resp)_crlf_resp_crlf
 w output
 s data=@x
 s output="$"_$l(data)_crlf_data_crlf  
 w output
 f i=1:1 s x=$q(@x) q:x=""  d
 . s params=$p(x,start,2,2000)
 . s params=$e(params,1,$l(params)-1)
 . s resp=params
 . s output="$"_$l(resp)_crlf_resp_crlf  
 . w output
 . s data=@x
 . i data="" d
 . . s output="$-1"_crlf
 . . w output
 . e  d
 . . s output="$"_$l(data)_crlf_data_crlf  
 . . w output
 k ^CacheTempEWD($j)
 QUIT
 ;
mergeto(input)
 ;
 n data,dataLength,error,gloRef,i,key,keyLength,noOfRecs,x
 ;
 ; MERGETO myglobal["1","a"]
 ; *6
 ; $1
 ; 1   
 ; $5
 ; hello  
 ; $7
 ; 1,"aaa"  
 ; $5
 ; world   
 ; $8
 ; 2,"cccc"
 ; $3
 ; foo
 ; +OK
 ;   note $-1 for key length means no key - data put at top level
 ;
 s $zt=$$zt()
 s gloRef=$$getGloRef(input)
 i gloRef["^zmwire" s output="-No access allowed to this global"_crlf w output QUIT
 i $e(gloRef,$l(gloRef))=")" s gloRef=$e(gloRef,1,$l(gloRef)-1)
 ;
 r noOfRecs
 i $e(noOfRecs,1)'="*" s output="-Invalid: expected number of records"_crlf w output QUIT
 s noOfRecs=+$e(noOfRecs,2,$l(noOfRecs))
 i noOfRecs'?1N.N s output="-Invalid format for number of records"_crlf w output QUIT
 i noOfRecs=0 QUIT "+OK"_crlf QUIT
 i (noOfRecs#2)=1 s output="-Invalid: no of records must be an even number"_crlf w output QUIT
 s noOfRecs=noOfRecs/2
 k ^CacheTempEWD($j)
 s error=""
 k ^CacheTempEWD($j)
 f i=1:1:noOfRecs d  q:error'=""
 . r keyLength
 . i $e(keyLength,1)'="$" s error="Invalid record "_i_": record length" q
 . s keyLength=+$e(keyLength,2,$l(keyLength))
 . i keyLength=-1 d
 . . s key=""
 . e  d
 . . i keyLength'?1N.N s error="Invalid record "_i_": bad format for record length" q
 . . i keyLength=0 s error="Invalid record "_i_": record length cannot be zero" q
 . . s key=$$readChars(keyLength)
 . . r ok d log(key)
 . . ;r key#keyLength,ok d log(key)
 . . i key["\""" s key=$$replaceAll(key,"\""","""""")
 . i error'="" q
 . r dataLength
 . i $e(dataLength,1)'="$" s error="Invalid record "_i_": expected data length" q
 . s dataLength=+$e(dataLength,2,$l(dataLength))
 . i dataLength'=-1,dataLength'?1N.N s error="Invalid record "_i_": bad format for data length" q
 . i dataLength=-1 d
 . . s data=""
 . e  d
 . . s data=$$readChars(dataLength)
 . . r ok
 . . d log(data)
 . . ;r data#dataLength,ok d log(data)
 . . i data["""" s data=$$replaceAll(data,"""","""""")
 . i key="" d
 . . n gloRef1
 . . s gloRef1=gloRef
 . . i gloRef["(" s gloRef1=gloRef1_")"
 . . s x="s "_gloRef1_"="""_data_""""
 . e  d
 . . n gloRef1
 . . s gloRef1=gloRef
 . . i gloRef'["(" d
 . . . s gloRef1=gloRef1_"("
 . . e  d
 . . . s gloRef1=gloRef1_","
 . . s x="s "_gloRef1_key_")="""_data_""""
 . s ^CacheTempEWD($j,i)=x
 i error'="" s output="-"_error_crlf w output QUIT
 f i=1:1:noOfRecs d
 . s x=^CacheTempEWD($j,i)
 . x x
 k ^CacheTempEWD($j)
 s output="+OK"_crlf
 w output
 s $zt=""
 QUIT
 ;
copy(input)
 ;
 n fromGlo,killToFirst,p2,response,toGlo,x
 ;
 ; COPY fromGlobal["1","a"] toGlobal["x"] 1
 i $g(^%zewd("trace"))=1 d trace("copy: input="_input)
 s $zt=$$zt()
 s fromGlo=$p(input,$c(1),1)
 s toGlo=$p(input,$c(1),2)
 s killToFirst=$p(input,$c(1),3)
 s p2=$p(fromGlo,"[",2,2000)
 s p2=$e(p2,1,$l(p2)-1)
 i p2'="" s p2="("_p2_")"
 s fromGlo="^"_$p(fromGlo,"[",1)_p2
 s p2=$p(toGlo,"[",2,2000)
 s p2=$e(p2,1,$l(p2)-1)
 i p2'="" s p2="("_p2_")"
 s toGlo="^"_$p(toGlo,"[",1)_p2
 s x=""
 i killToFirst s x="k "_toGlo_" "
 s x=x_"m "_toGlo_"="_fromGlo
 x x
 ;
 i $g(^%zewd("trace"))=1 d trace("x="_x)
 s response="+ok"_crlf
 i $g(^%zewd("trace"))=1 d trace("copy: response="_response)
 w response
 s $zt=""
 i $g(^mwire("logger"))=1 d logger("copy")
 ;
 QUIT
 ;
mdate
 ;
 n date,day,time
 ;
 s date=$h
 s day=+date
 s output="*2"_crlf_"$"_$l(day)_crlf_day_crlf
 w output
 s time=$p(date,",",2)
 s output="$"_$l(time)_crlf_time_crlf
 w output
 QUIT
 ;
tstart
 s $zt=$$zt()
 TSTART
 s output="+OK"_crlf
 w output
 s $zt=""
 QUIT
 ;
tcommit
 s $zt=$$zt()
 TCOMMIT
 s output="+OK"_crlf
 w output
 s $zt=""
 QUIT
 ;
trollback
 s $zt=$$zt()
 TROLLBACK
 s output="+OK"_crlf
 w output
 s $zt=""
 QUIT
 ;
zv
 s output="+"_$zv_crlf
 w output
 QUIT
 ;
zt()
 i $zv["GT.M" QUIT "g executeError^zmwire"
 QUIT "executeError^zmwire"
 ;
processid
 s output=":"_$j_crlf
 w output
 QUIT
 ;
ping
 s output="+PONG"_crlf
 w output
 QUIT
 ;
executeError
 s output="-Invalid Command"_crlf
 w output
 g loop
 ;
replaceAll(InText,FromStr,ToStr) ; Replace all occurrences of a substring
 ;
 n p
 ;
 s p=InText
 i ToStr[FromStr d  QUIT p
 . n i,stop,tempText,tempTo
 . s stop=0
 . f i=0:1:255 d  q:stop
 . . q:InText[$c(i)
 . . q:FromStr[$c(i)
 . . q:ToStr[$c(i)
 . . s stop=1
 . s tempTo=$c(i)
 . s tempText=$$replaceAll(InText,FromStr,tempTo)
 . s p=$$replaceAll(tempText,tempTo,ToStr)
 f  q:p'[FromStr  S p=$$replace(p,FromStr,ToStr)
 QUIT p
 ;
replace(InText,FromStr,ToStr) ; replace old with new in string
 ;
 n np,p1,p2
 ;
 i InText'[FromStr q InText
 s np=$l(InText,FromStr)+1
 s p1=$p(InText,FromStr,1),p2=$p(InText,FromStr,2,np)
 QUIT p1_ToStr_p2
 ;
readChars(length)
 ;
 n data,i,x
 ;
 s data=""
 f i=1:1:length r *x s data=data_$c(x)
 QUIT data
 ;
stripSpaces(string)
 i $zv["Cache" QUIT $$stripSpaces^MDBMCache(string)
 ;
 s string=$$stripLeadingSpaces(string)
 QUIT $$stripTrailingSpaces(string)
 ;
stripLeadingSpaces(string)
 ;
 n i
 ;
 f i=1:1:$l(string) QUIT:$e(string,i)'=" "
 QUIT $e(string,i,$l(string))
 ;
stripTrailingSpaces(string)
 ;
 n i,spaces,new
 ;
 s spaces=$$makeString(" ",100)
 s new=string_spaces
 QUIT $p(new,spaces,1)
 ;
makeString(char,len) ; create a string of len characters
 ;
 n str
 ;
 s str="",$p(str,char,len+1)=""
 QUIT str
 ;
MD5(string)
 ;
 ; n hash
 ;
 i $zv["Cache" QUIT $$MD5^MDBMCache(string)
 ;
 QUIT $$MD5^%ZMGWSIS(string,1,1)
 ;
unEscape(string)
 ;
 n buf,outstring,p1,p2,hex,asc
 ;
 s buf=string
 s outstring=""
 f  q:buf'["%"  d
 . s p1=$p(buf,"%",1)
 . s outstring=outstring_p1
 . s p2=$p(buf,"%",2,50000)
 . i $e(p2)="u" s buf=$e(p2,6,9999),hex=$e(p2,2,5),outstring=outstring_$c($$hex2Ascii(hex)-1264) q
 . s hex=$e(p2,1,2)
 . s buf=$e(p2,3,$l(p2))
 . s asc=$$hex2Ascii(hex)
 . s outstring=outstring_$c(asc)
 QUIT (outstring_buf)
 ;
hex2Ascii(string)
 ;
 n asc,c,conv,err,i,n,power
 ;
 s string=$zconvert(string,"U")
 s asc=0
 f i=0:1:9 S conv(i)=i
 s conv("A")=10
 s conv("B")=11
 s conv("C")=12
 s conv("D")=13
 s conv("E")=14
 s conv("F")=15
 s n=-1,err=0
 f i=$l(string):-1:1 d  q:err
 . s n=n+1
 . s power=16**n
 . s c=$e(string,i)
 . i '$d(conv(c)) s err=1 q
 . s asc=asc+(conv(c)*power)
 i err QUIT "-1"
 QUIT asc
 ;
ts()
 s last=$g(^zmwire("lastts"))
 n io,p,resp
 s io=$io
 s p="time"
 o p:(COMMAND="date +%s%N":READONLY)::"PIPE"
 u p
 r resp q:$ZEOF
 c p
 u io
 s ^zmwire("lastts")=resp
 QUIT ((resp-last)/1000000000)
 ;
relinkRoutines()
 n list,rou,xrou
 i $g(^zewd("trace"))=1 d trace("Process "_$j_": Relinking...")
 s rou=""
 f  s rou=$view("RTNNEXT",rou) q:rou=""  d
 . i rou="zmwire" q
 . i rou="%zewdGTMRuntime" q
 . i rou="%zewdPHP" q
 . i rou="MDB" q
 . i rou="ewdWLewdmgrrelink" q
 . i rou="%ZMGWSI" q
 . i rou="%ZMGWSIS" q
 . i rou="GTM$DMOD" q
 . s xrou=rou
 . i $e(xrou,1)="%" s xrou="_"_$e(xrou,2,$l(xrou))
 . zl xrou
 . i $g(^zewd("trace"))=1 d trace("relinked "_rou)
 s ^zmwire("relink","process",$j)=""
 i $g(^zewd("trace"))=1 d trace("Process "_$j_": Relinking complete")
 QUIT ""
 ;
relink ;
 s ^zmwire("relink")=1
 k ^zmwire("relink","process")
 QUIT
 ;
arrayToJSON(name)
 n a,buff,c,json,subscripts
 i '$d(@name) QUIT "[]"
 s json=$$walkArray("",name)
 ; Encode UTF-8 characters
 s buff=""
 f i=1:1:$l(json) d
 . s c=$e(json,i)
 . s a=$a(c)
 . i a>160 d
 . . i a<192 d
 . . . s buff=buff_$c(194)_c
 . . e  d
 . . . s buff=buff_$c(195)_$c(a-64)
 . e  d
 . . s buff=buff_c
 QUIT buff
 ;
walkArray(json,name,subscripts)
 ;
 n allNumeric,arrComma,brace,comma,count,cr,dd,i,no,numsub,dblquot,quot
 n ref,sub,subNo,subscripts1,type,valquot,value,xref,zobj
 ;
 s cr=$c(13,10),comma=","
 s (quot,dblquot,valquot)=""""
 s dd=$d(@name)
 i dd=1!(dd=11) d  i dd=1 QUIT json
 . s value=@name
 . i value'[">" q
 . s json=$$walkArray(json,value,.subscripts)
 s ref=name_"("
 s no=$o(subscripts(""),-1)
 i no>0 f i=1:1:no d
 . i subscripts(i)[quot s subscripts(i)=$$replaceAll(subscripts(i),quot,quot_quot)
 . i subscripts(i)?."-"1N.N s quot=""
 . s ref=ref_quot_subscripts(i)_quot_","
 . s quot=dblquot
 s ref=ref_"sub)"
 s sub="",numsub=0,subNo=0,count=0
 s allNumeric=1
 f  s sub=$o(@ref) q:sub=""  d  q:'allNumeric
 . i sub'?1N.N s allNumeric=0
 . s count=count+1
 . i sub'=count s allNumeric=0
 ;i allNumeric,count=1 s allNumeric=0
 s allNumeric=0
 i allNumeric d
 . s json=json_"["
 e  d
 . s json=json_"{"
 s sub=""
 f  s sub=$o(@ref) q:sub=""  d
 . s subscripts(no+1)=sub
 . s subNo=subNo+1
 . s dd=$d(@ref)
 . i dd=1 d
 . . s value=@ref
 . . ;i value["\" s value=$$replaceAll(value,"\","\\")
 . . s value=$$removeControlChars(value)
 . . i 'allNumeric d
 . . . ;i sub["\",sub'["\\",sub'["\"""  s sub=$$replaceAll(sub,"\","\\")
 . . . i sub["\" s sub=$$replaceAll(sub,"\","\\")
 . . . s sub=$$removeControlChars(sub)
 . . . s json=json_""""_sub_""":"
 . . s type="literal"
 . . i $$numeric(value) s type="numeric"
 . . i value="true"!(value="false") s type="boolean"
 . . i $e(value,1)="{",$e(value,$l(value))="}" s type="variable"
 . . i type="literal" d
 . . . ;i value[quot s value=$$replaceAll(value,quot,"\"_quot)
 . . . ;i value["\",value'["\\",value'["\"""  s value=$$replaceAll(value,"\","\\")
 . . . i value["\" s value=$$replaceAll(value,"\","\\")
 . . . i value[quot s value=$$replaceAll(value,quot,"\""")
 . . . s value=valquot_value_valquot
 . . d
 . . . s json=json_value_","
 . k subscripts1
 . m subscripts1=subscripts
 . i dd>9 d
 . . n subx
 . . ;i sub?1N.N d
 . . ;. i subNo=1 d
 . . ;. . s numsub=1
 . . ;. . s json=$e(json,1,$l(json)-1)_"["
 . . ;e  d
 . . ;. s json=json_""""_sub_""":"
 . . ;i sub["\",sub'["\\",sub'["\""" s sub=$$replaceAll(sub,"\","\\")
 . . s subx=sub
 . . i subx["\" s subx=$$replaceAll(sub,"\","\\")
 . . i subx[quot s subx=$$replaceAll(subx,quot,"\""")
 . . s subx=$$removeControlChars(subx)
 . . s json=json_""""_subx_""":"
 . . s json=$$walkArray(json,name,.subscripts1)
 . . d
 . . . s json=json_","
 ;
 s json=$e(json,1,$l(json)-1)
 i allNumeric d
 . s json=json_"]"
 e  d
 . s json=json_"}"
 QUIT json ; exit!
 ;
trace(text,clear) ; trace  ;
 n i
 s text=$g(text)
 i $g(clear)=1 k ^%zewdTrace
 s i=$increment(^%zewdTrace)
 s ^%zewdTrace(i)=text
 QUIT
 ;
parseJSON(jsonString,propertiesArray,mode)
 ;
 n array,arrRef,buff,c,error
 ;
 k propertiesArray
 s error=""
 s buff=$g(jsonString)
 s buff=$$replaceAll(buff,"\""","\'")
 s arrRef="array"
 s c=$e(buff,1)
 s buff=$e(buff,2,$l(buff))
 d
 . i c="{" d  q
 . . n prefix
 . . s prefix="""zobj1"""
 . . i $g(mode)=1 s prefix=""
 . . s error=$$parseJSONObject(.buff,prefix)
 . . q:error
 . . i buff'="" s error=1
 . i c="[" d  q
 . . n prefix
 . . s prefix=1
 . . i $g(mode)=1 s prefix=""
 . . s error=$$parseJSONArray(.buff,prefix)
 . . q:error
 . . i buff'="" s error=1
 . s error=1
 i error=1 QUIT "Invalid JSON"
 m propertiesArray=array
 QUIT ""
 ;
parseJSONObject(buff,subs)
 n c,error,name,stop,subs2,value,x
 s stop=0,name="",error=""
 f  d  q:stop
 . s c=$e(buff,1)
 . i c="" s error=1,stop=1 q
 . s buff=$e(buff,2,$l(buff))
 . i c="[" s error=1,stop=1 q
 . i c="}" d  q
 . . s stop=1
 . i c=":" d  q
 . . n subs2
 . . s value=$$getJSONValue(.buff)
 . . d  q:stop
 . . . i value="" q
 . . . i $e(value,1)="""",$e(value,$l(value))="""" q
 . . . i value="true"!(value="false") s value=""""_value_"""" q
 . . . i $$numeric(value) q
 . . . s error=1,stop=1
 . . i value="",$e(buff,1)="{" d  q
 . . . i $e(name,1)'="""",$e(name,$l(name))'="""" s name=""""_name_""""
 . . . s subs2=subs
 . . . i subs'="" s subs2=subs2_","
 . . . s subs2=subs2_name
 . . . i $g(mode)="" s subs2=subs2_",""zobj1"""
 . . . s buff=$e(buff,2,$l(buff))
 . . . s error=$$parseJSONObject(.buff,subs2)
 . . . i error=1 s stop=1 q
 . . i value="",$e(buff,1)="[" d  q
 . . . ;s subs2=subs_","""_name_""",""1"""
 . . . i $e(name,1)'="""",$e(name,$l(name))'="""" s name=""""_name_""""
 . . . s subs2=subs
 . . . i subs'="" s subs2=subs2_","
 . . . s subs2=subs2_name
 . . . s buff=$e(buff,2,$l(buff))
 . . . s error=$$parseJSONArray(.buff,subs2)
 . . . i error=1 s stop=1 q
 . . i $e(name,1)="""",$e(name,$l(name))'="""" s error=1,stop=1 q
 . . i $e(name,1)'="""",$e(name,$l(name))="""" s error=1,stop=1 q
 . . i $e(name,1)'="""",$e(name,$l(name))'="""" s name=""""_name_""""
 . . s subs2=subs
 . . i subs'="" s subs2=subs2_","
 . . s subs2=subs2_name
 . . i value["\'" s value=$$replaceAll(value,"\'","""""")
 . . s x="s "_arrRef_"("_subs2_")="_value
 . . x x
 . i c="," s name="" q
 . s name=name_c q
 QUIT error
 ;
parseJSONArray(buff,subs)
 n c,error,name,no,stop,subs2,value,x
 s stop=0,name="",no=0,error=""
 f  d  q:stop
 . s c=$e(buff,1)
 . i c="" s error=1,stop=1 q
 . s buff=$e(buff,2,$l(buff))
 . i c=":" s error=1,stop=1 q
 . i c="]" d  q
 . . s stop=1
 . . i name="" q
 . . s no=no+1
 . . s subs2=subs
 . . i subs'="" s subs2=subs2_","
 . . s subs2=subs2_no
 . . s x="s "_arrRef_"("_subs2_")="_name
 . . x x
 . i c="[" d  q
 . . s no=no+1
 . . s subs2=subs
 . . i subs'="" s subs2=subs2_","
 . . s subs2=subs2_no
 . . ;s buff=$e(buff,2,$l(buff))
 . . s error=$$parseJSONArray(.buff,subs2)
 . . i error=1 s stop=1 q
 . i c="{" d  q
 . . s no=no+1
 . . s subs2=subs
 . . i subs'="" s subs2=subs2_","
 . . s subs2=subs2_no
 . . i $g(mode)="" s subs2=subs2_",""zobj1"""
 . . ;s buff=$e(buff,2,$l(buff))
 . . s error=$$parseJSONObject(.buff,subs2)
 . . i error=1 s stop=1 q
 . s subs2=subs
 . i subs'="" s subs2=subs2_","
 . s subs2=subs2_""""_name_""""
 . i c="," d  q
 . . i name="" q
 . . d  q:stop
 . . . i $e(name,1)="""",$e(name,$l(name))="""" q
 . . . ;i value="true"!(value="false") s value=""""_value_"""" q
 . . . i $$numeric(name) q
 . . . s error=1,stop=1
 . . s no=no+1
 . . s subs2=subs
 . . i subs'="" s subs2=subs2_","
 . . s subs2=subs2_""""_no_""""
 . . s x="s "_arrRef_"("_subs2_")="_name
 . . x x
 . . s name=""
 . s name=name_c q
 QUIT error
 ;
getJSONValue(buff)
 n c,isLiteral,lc,stop,value
 s stop=0,value="",isLiteral=0,lc=""
 f  d  q:stop  q:buff=""
 . s c=$e(buff,1)
 . i value="",c="""" s isLiteral=1
 . i 'isLiteral,c="[" s stop=1 q
 . i 'isLiteral,c="{" s stop=1 q
 . i c="}" d  q:stop
 . . i isLiteral,lc'="""" q
 . . s stop=1
 . i c="," d  q:stop
 . . i isLiteral,lc'="""" q
 . . s stop=1
 . s buff=$e(buff,2,$l(buff))
 . s value=value_c
 . s lc=c
 QUIT value
 ;
numeric(value)
 i $e(value,1)=0,$l(value)>1 QUIT 0
 i $e(value,1,2)="-0",$l(value)>2,$e(value,1,3)'="-0." QUIT 0
 i value?1N.N QUIT 1
 i value?1"-"1N.N QUIT 1
 i value?1N.N1"."1N.N QUIT 1
 i value?1"-"1N.N1"."1N.N QUIT 1
 i value?1"."1N.N QUIT 1
 i value?1"-."1N.N QUIT 1
 QUIT 0
 ;
removeControlChars(string)
 n c,i,newString
 s newString=""
 f i=1:1:$l(string) d
 . s c=$e(string,i)
 . i $a(c)<32 s c="~"
 . s newString=newString_c
 QUIT newString
 ;
utfConvert(input)
 ; Unescape UTF-8 characters
 i input[$c(195) d
 . n buf,c1,i,no,p
 . s buf=$p(input,$c(195),1)
 . s no=$l(input,$c(195))
 . f i=2:1:no d
 . . s p=$p(input,$c(195),i)
 . . s c1=$e(p,1)
 . . s c1=$c($a(c1)+64)
 . . s buf=buf_c1_$e(p,2,$l(p))
 . s input=buf
 s input=$tr(input,$c(194),"")
 QUIT input
 ;