Home   Package List   Routine Alphabetical List   Global Alphabetical List   FileMan Files List   FileMan Sub-Files List   Package Component Lists   Package-Namespace Mapping  
Routine: ABSPOSQ2

ABSPOSQ2.m

Go to the documentation of this file.
  1. ABSPOSQ2 ; IHS/FCS/DRS - form transmission packets ; [ 11/07/2002 6:57 AM ]
  1. ;;1.0;PHARMACY POINT OF SALE;**3,31,42**;JUN 01, 2001;Build 38
  1. ;
  1. ; Status comes in as 30. Or as 31.
  1. ; Status set to 40 which PACKET() is in progress.
  1. ; When PACKET() is done, the status is changed to 50
  1. ; The transmission packet is in ^ABSPECX("POS",DIALOUT,"C",*)
  1. ; and ABSPOSQ3 is scheduled to do the transmission.
  1. ;
  1. ; Split-off code:
  1. ; STATUS31^ABSPOSQF - Insurer asleep condition is handled here
  1. ; PACKET^ABSPOSQG - packetizing RXILIST(*)
  1. ;
  1. ;----------------------------------------------------------------
  1. ; IHS/SD/lwj 8/5/02 NCPDP 5.1 changes
  1. ; Prior Authorization claims must now be sent one by one - no
  1. ; bundling allowed. New logic added before bundling to check
  1. ; the overrides for field 498.01 (Request Type), 498.02 (Request
  1. ; period date-begin), 498.03 (Request period date-end), and 498.04
  1. ; (basis of request). These four fields are required for a prior
  1. ; authorization claim, and if we encounter even one of the fields
  1. ; we will keep the claim from bundling.
  1. ;
  1. ;
  1. PACKETS ; construct packets for transmission to NDC
  1. ; your work list is ^ABSPT("AD",30)
  1. ;
  1. ; LOGGING: Do NOT keep a "current" slot.
  1. ; When you need to log something, use one of these routines:
  1. ; LOG59^ABSPOSQ(MSG,IEN59) - log MSG to this one slot
  1. ; LOG2LIST^ABSPOSQ(MSG) - log MSG to all on RXILIST(*)
  1. ; LOG2CLM^ABSPOSQ(MSG,IEN02) - log MSG to all on this 9002313.02 claim
  1. ;
  1. N ERROR,SILENT S SILENT=1
  1. N RXILIST,STATUS30
  1. D STATUS31^ABSPOSQF ; deal with insurer alseep waiting
  1. F STATUS30=30 I $D(^ABSPT("AD",STATUS30)) D STATUS30
  1. ; If there are still any claims with status 30,
  1. ; perhaps due to failed LOCK59, queue up ABSPOSQ2 to run again
  1. N NEEDQ1 S NEEDQ1=$O(^ABSPT("AD",30,0))
  1. ; Don't worry about status 31 - retry after insurer asleep
  1. ; was already scheduled, when the response packet came.
  1. ;I 'NEEDQ1 S NEEDQ1=$O(^ABSPT("AD",31,0))
  1. I NEEDQ1 H 60 D TASK^ABSPOSQ1
  1. Q
  1. SETSTAT(NEWSTAT) ;EP - ABSPOSQF ; given IEN59
  1. N ABSBRXI S ABSBRXI=IEN59 ; unfortunate variable naming convention
  1. D SETSTAT^ABSPOSU(NEWSTAT)
  1. Q
  1. SETRESU(RESCODE,TEXT) ; given IEN59
  1. N ABSBRXI S ABSBRXI=IEN59 ; unfortunate variable naming convention
  1. D SETRESU^ABSPOSU(RESCODE,TEXT)
  1. Q
  1. ; Locking and unlocking the list of claims with
  1. ; with STATUS30=30 or 31. The lock should always succeed.
  1. ; But since it's required to be a timed lock, LOCK59 is a $$,
  1. ; true if successful, false if failure.
  1. LOCK59() ;EP - ABSPOSQF
  1. L +^ABSPT("AD",STATUS30):60 Q $T
  1. UNLOCK59 ;EP - ABSPOSQF
  1. L -^ABSPT("AD",STATUS30) Q
  1. NEXT59(X) ;EP - ABSPOSQF
  1. N INS,T
  1. N59A S X=$O(^ABSPT("AD",STATUS30,X))
  1. I X="" Q X ; end of list, return ""
  1. ; but if the insurer is asleep, don't take this one
  1. S INS=$P(^ABSPT(X,1),U,6)
  1. I $G(INS)="" Q X ;IHS/OIT/SCR 05/12/09 patch 31 avoid undefined
  1. I '$D(^ABSPEI(INS)) Q X ;IHS/OIT/RAN 03/31/2011 patch 42 avoid undefined
  1. S T=$P($G(^ABSPEI(INS,101)),U) ; insurer asleep retry time
  1. I 'T Q X ; insurer is not asleep
  1. ; - below here - insurer is asleep -
  1. ; If cancellation is requested, let it through, regardless of sleep.
  1. ; This will speed it on its way to cancellation
  1. I $G(^ABSPT(X,3)) Q X
  1. ; If necessary, update the .59's record of when to retry
  1. I $P($G(^ABSPT(X,8)),U)'=T D ;
  1. . S $P(^ABSPT(X,8),U)=T ; stamp with latest retry time
  1. . N IEN59 S IEN59=X D SETSTAT(31) ; force screen update, too
  1. . D LOG59^ABSPOSQ("Insurer still asleep - retry at "_T,IEN59)
  1. I T<$$NOW Q X ; time to retry, so yes, we do this one
  1. ;. don't clear this - a successful non-sleep response will clear it
  1. ;. S $P(^ABSPEI(INS,101),U)="" ; clear the sleep-until time
  1. ;. but don't clear the piece 5 current interval, as we may increment
  1. ; Else still in waiting - if it's status 30, change it to 31
  1. I STATUS30=30 D
  1. . N IEN59 S IEN59=X D SETSTAT(31)
  1. . S $P(^ABSPT(IEN59,8),U)=$P(^ABSPEI(INS,101),U)
  1. . S $P(^ABSPT(IEN59,8),U,3)=INS
  1. G N59A ; still in wait time; don't look at this claim
  1. NOW() N %,%H,%I,X D NOW^%DTC Q %
  1. STATUS30 ; given STATUS30=30
  1. N IEN59 S IEN59=""
  1. Q:'$$LOCK59
  1. F S IEN59=$$NEXT59(IEN59) Q:IEN59="" D
  1. . K RXILIST ; init list each time through this loop
  1. . S RXILIST(IEN59)=""
  1. . D SETSTAT(40) ; set its status to "packetizing"
  1. .; Reversals go in a packet alone
  1. . I $G(^ABSPT(IEN59,4)) G POINTX
  1. .;
  1. .;IHS/SD/lwj 8/5/02 NCPDP 5.1 - claim cannot be bundled if
  1. .; there is prior authorization information
  1. .; use CHKPA when the processor are ready to use the actual
  1. .; prior authorization segment - for now, use the CHKPA2
  1. .; G:$$CHKPA() POINTX
  1. . G:$$CHKPA2() POINTX
  1. .;
  1. .; Who is the patient? Find all other prescriptions for this
  1. .; patient which have status 30, and add them to the RXILIST, too
  1. .; Must also have same VISIT and same DIVISION
  1. .;
  1. . N RA0,RA1 S RA0=^ABSPT(IEN59,0),RA1=^(1)
  1. . N IEN59 S IEN59="" ; preserve the top-level index!
  1. . F S IEN59=$$NEXT59(IEN59) Q:'IEN59 D
  1. . . N RB0,RB1 S RB0=^ABSPT(IEN59,0),RB1=^(1)
  1. . . ; Only bundle when you have the same:
  1. . . ; Patient, Visit, Division, Division Source, Insurer, Pharmacy
  1. . . I $P(RA0,U,6,7)'=$P(RB0,U,6,7) Q
  1. . . I $P(RA1,U,4,7)'=$P(RB1,U,4,7) Q
  1. . . I $P(RB0,U,2)'=30 Q ; might have been canceled, or maybe 31'd
  1. . . I $P(RB0,U,2)'=STATUS30 D Q
  1. . . . D IMPOSS^ABSPOSUE("P","TI","IEN59 status "_$P(RB0,U,2)_" but must be 30",,"STATUS30",$T(+0))
  1. . . D SETSTAT(40)
  1. . . S RXILIST(IEN59)=""
  1. POINTX . ; (reversals branch here around multi-claim packeting)
  1. . ;
  1. . ; so now we have a big list of prescriptions, RXILIST(*)
  1. . ; they're all marked with status = 40 PACKETIZING
  1. . ; Get going - packetize them!
  1. . ;
  1. . S ERROR=$$PACKET^ABSPOSQG ;
  1. . ;
  1. . ; Having packetized, reset status to 50, Waiting for transmit
  1. . ; There should never be an error returned by $$PACKET,
  1. . ; although it could happen.
  1. . ; But if there was an error, reset the status to 99 right now.
  1. . ; And set that status for every prescription in the RXILIST,
  1. . ; even if maybe only one of them caused all the trouble.
  1. . ;
  1. . S IEN59="" F S IEN59=$O(RXILIST(IEN59)) Q:IEN59="" D
  1. . . I ERROR D
  1. . . . D SETSTAT(99) ;
  1. . . . D SETRESU($P(ERROR,U),$P(ERROR,U,2,$L(ERROR,U))) ;
  1. . . E D
  1. . . . D SETSTAT(50) ; "Waiting for transmit"
  1. D UNLOCK59
  1. Q
  1. ;
  1. ; TASK^ABSPOSQ2 starts up a sender-receiver in ABSPOSQ3.
  1. ; TASK is an entry point used by POKE^ABSPOS2D
  1. ; It may well be called from other places in the future,
  1. ; if we try to be clever and start the dialing while we're getting
  1. ; claim information together, for instance.
  1. ;
  1. TASK ;EP - ABSPOS2D,ABSPOSAP,ABSPOSC3,ABSPOSQG
  1. N X,Y,%DT
  1. S X="N",%DT="ST" D ^%DT
  1. D TASKAT(Y)
  1. Q
  1. TASKAT(ZTDTH) ;EP - ABSPOSQJ
  1. ;ZTDTH = time when you want COMMS^ABSPOSQ3 to run
  1. ; called from TASK, above, normally
  1. ; called here from ABSPOSQ3 when it's requeueing itself for
  1. ; retry after a dial-out error condition
  1. ;N (DUZ,DIALOUT,TIME,ZTDTH)
  1. N ZTRTN,ZTIO,ZTSAVE
  1. S ZTRTN="COMMS^ABSPOSQ3",ZTIO=""
  1. S ZTSAVE("DIALOUT")="" ; which entry in 9002313.55
  1. D ^%ZTLOAD
  1. Q
  1. KSCRATCH ;EP - ABSPOSQG ; Kill scratch globals
  1. K ^ABSPECX($J,"R")
  1. K ^ABSPECX($J,"C")
  1. Q
  1. ;
  1. CHKPA() ;---------------------------------------------------------------
  1. ;IHS/SD/lwj 8/30/02 NCPDP 5.1 - we aren't quite ready for this
  1. ; yet - the below logic is going to work with the actual
  1. ; prior authorization segment, and no one is going to use that
  1. ; or support that segment for a while. SO.....I'm leaving this
  1. ; here for future reference - BUT for now, we're going to use
  1. ; the CHKPA2 routine to simply check for field 461 and 461.
  1. ;*
  1. ;IHS/SD/lwj 8/5/02 NCPDP 5.1 If any of the following fields
  1. ; appear as an override on this claim, we must consider it a prior
  1. ; authorization claim, and it cannot be bundled with other claims.
  1. ; 498.01 Request Type
  1. ; 498.02 Request Period Date Begin
  1. ; 498.03 Request Period Date End
  1. ; 498.04 Basis of Request
  1. ;
  1. N OVRREC,OVRFLD,NCPDPF,NCPDPFN,PACLM
  1. S PACLM=0
  1. ;
  1. S OVRREC=$P($G(^ABSPT(IEN59,1)),U,13) ;grab the overrides
  1. Q:OVRREC="" 0 ;no overrides - can't be a prior auth claim
  1. ;
  1. ; loop through the overrides and look for the prior auth fields
  1. S OVRFLD=0
  1. F S OVRFLD=$O(^ABSP(9002313.511,OVRREC,1,OVRFLD)) Q:'+OVRFLD D
  1. . S NCPDPF=$P($G(^ABSP(9002313.511,OVRREC,1,OVRFLD,0)),U) ;int fld
  1. . S NCPDPFN=$P($G(^ABSPF(9002313.91,NCPDPF,0)),U) ;fld number
  1. . Q:(NCPDPFN<498.01)!(NCPDPFN>498.04)
  1. . S PACLM=1
  1. ;
  1. Q PACLM
  1. ;
  1. CHKPA2() ;---------------------------------------------------------------
  1. ; IHS/SD/lwj 8/30/02 NCPDP 5.1
  1. ;Until processors are ready to use the prior authorization
  1. ; segment, we are going to use this routine to check for
  1. ; a prior authorization. To do this we will simply check
  1. ; for field 461 (Prior Authorization type code) and field
  1. ; 462 (Prior Authorization Number Submitted)
  1. ; If either exist, we will not bundle the claim. Prior
  1. ; authorization claims must be sent on their own.
  1. ;*
  1. ;
  1. N PATYP,PANUM
  1. S PACLM=0
  1. ;
  1. S PATYP=$P($G(^ABSPT(IEN59,1)),U,15) ;prior auth type code
  1. S PANUM=$P($G(^ABSPT(IEN59,1)),U,9) ;prior auth number
  1. I ($G(PATYP)'="")!($G(PANUM)'="") S PACLM=1
  1. ;
  1. Q PACLM