305 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
			
		
		
	
	
			305 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
| Function WM_IN_Actions(Action, CalcColName, FSList, Handle, Name, FMC, Record, Status, OrigRecord, Param1, Param2, Param3, Param4, Param5, Param6, Param7, Param8, Param9, Param10)
 | |
| /***********************************************************************************************************************
 | |
| 
 | |
|     This program is proprietary and is not to be used by or disclosed to others, nor is it to be copied without written
 | |
|     permission from Infineon.
 | |
| 
 | |
|     Name        :   WM_IN_Actions
 | |
| 
 | |
|     Description :   Handles calculated columns and MFS calls for the current table.
 | |
| 
 | |
|     Notes       :   This function uses @ID, @RECORD, and @DICT to make sure {ColumnName} references work correctly.
 | |
|                     If called from outside of a calculated column these will need to be set and restored.
 | |
| 
 | |
|     Parameters  :
 | |
|         Action          [in] -- Name of the action to be taken
 | |
|         CalcColName     [in] -- Name of the calculated column that needs to be processed. Normally this should only be
 | |
|                                 populated when the CalcField action is being used.
 | |
|         FSList          [in] -- The list of MFSs and the BFS name for the current file or volume. This is an @SVM
 | |
|                                 delimited array, with the current MFS name as the first value in the array, and the BFS
 | |
|                                 name as the last value. Normally set by a calling MFS.
 | |
|         Handle          [in] -- The file handle of the file or media map being accessed. Note, this does contain the
 | |
|                                 entire handle structure that the Basic+ Open statement would provide. Normally set by a
 | |
|                                 calling MFS.
 | |
|         Name            [in] -- The name (key) of the record or file being accessed. Normally set by a calling MFS.
 | |
|         FMC             [in] -- Various functions. Normally set by a calling MFS.
 | |
|         Record          [in] -- The entire record (for record-oriented functions) or a newly-created handle (for
 | |
|                                 "get handle" functions). Normally set by a calling MFS.
 | |
|         Status      [in/out] -- Indicator of the success or failure of an action. Normally set by the calling MFS but
 | |
|                                 for some actions can be set by the action handler to indicate failure.
 | |
|         OrigRecord      [in] -- Original content of the record being processed by the current action. This is
 | |
|                                 automatically being assigned by the WRITE_RECORD and DELETE_RECORD actions within
 | |
|                                 BASE_MFS.
 | |
|         Param1-10   [in/out] -- Additional request parameter holders
 | |
|         ActionFlow     [out] -- Used to control the action chain (see the ACTION_SETUP insert for more information.)
 | |
|                                 Can also be used to return a special value, such as the results of the CalcField
 | |
|                                 method.
 | |
| 
 | |
|     History     :   (Date, Initials, Notes)
 | |
|         06/05/18    djs     Original programmer.
 | |
|         03/07/19    djs     Updated WRITE_RECORD and DELETE_RECORD actions such that calls to obj_NCR('RejQty') are
 | |
|                             now wrapped in a Sum() function call because EpiPro wafers return @VM delimited reject
 | |
|                             quantity values for each distinct RDS in the NCR record. This Sum() function call will not
 | |
|                             affect Non-EpiPro NCR records or EpiPro NCRs with only a single reject quanitity (i.e. the
 | |
|                             NCR only contains one distinct RDS).
 | |
| 
 | |
| 
 | |
| ***********************************************************************************************************************/
 | |
| 
 | |
| #pragma precomp SRP_PreCompiler
 | |
| 
 | |
| $insert APP_INSERTS
 | |
| $insert FILE.SYSTEM.EQUATES
 | |
| $insert ACTION_SETUP
 | |
| $insert WM_IN_EQUATES
 | |
| $insert RLIST_EQUATES
 | |
| $insert SCHED_DET_NG_EQUATES
 | |
| 
 | |
| Equ Comma$ to ','
 | |
| 
 | |
| Declare function    Database_Services, obj_NCR, obj_SAP, Environment_Services, Logging_Services, Datetime
 | |
| Declare subroutine  Error_Services, Database_Services, obj_NCR, obj_SAP, Material_Services, Work_Order_Services
 | |
| Declare subroutine  Logging_Services, obj_Notes
 | |
| 
 | |
| LogPath      = Environment_Services('GetApplicationRootPath') : '\LogFiles\WM_IN'
 | |
| LogDate      = Oconv(Date(), 'D4/')
 | |
| LogTime      = Oconv(Time(), 'MTS')
 | |
| LogFileName  = LogDate[7, 4] : '-' : LogDate[1, 2] : '-' : LogDate[4, 2] : ' Delete Attempt Log.csv'
 | |
| Headers      = 'Logging DTM' : @FM : 'WM_IN Key' : @FM : 'Notes'
 | |
| objDeleteLog = Logging_Services('NewLog', LogPath, LogFileName, CRLF$, Comma$, Headers, '', False$, False$)
 | |
| LoggingDTM   = LogDate : ' ' : LogTime   ; // Logging DTM
 | |
| 
 | |
| If KeyID then GoSub Initialize_System_Variables
 | |
| 
 | |
| Begin Case
 | |
| 	
 | |
| 	Case Action _EQC 'CalcField'            ;   GoSub Calc_Field
 | |
| 	Case Action _EQC 'READ_RECORD_PRE'      ;   GoSub READ_RECORD_PRE
 | |
| 	Case Action _EQC 'READ_RECORD'          ;   GoSub READ_RECORD
 | |
| 	Case Action _EQC 'READONLY_RECORD_PRE'  ;   GoSub READONLY_RECORD_PRE
 | |
| 	Case Action _EQC 'READONLY_RECORD'      ;   GoSub READONLY_RECORD
 | |
| 	Case Action _EQC 'WRITE_RECORD_PRE'     ;   GoSub WRITE_RECORD_PRE
 | |
| 	Case Action _EQC 'WRITE_RECORD'         ;   GoSub WRITE_RECORD
 | |
| 	Case Action _EQC 'DELETE_RECORD_PRE'    ;   GoSub DELETE_RECORD_PRE
 | |
| 	Case Action _EQC 'DELETE_RECORD'        ;   GoSub DELETE_RECORD
 | |
| 	Case Otherwise$                         ;   Status = 'Invalid Action'
 | |
| 		
 | |
| End Case
 | |
| 
 | |
| If KeyID then GoSub Restore_System_Variables
 | |
| 
 | |
| If Assigned(ActionFlow) else ActionFlow = ACTION_CONTINUE$
 | |
| 
 | |
| Return ActionFlow
 | |
| 
 | |
| 
 | |
| // ----- Calculated Columns --------------------------------------------------------------------------------------------
 | |
| //
 | |
| // The typical structure of a calculated column will look like this:
 | |
| //
 | |
| // Declare function TableName_Actions
 | |
| //
 | |
| // A = {COL1} ; * Reference as many data columns in this way to ensure the dictionary dependency is generated.
 | |
| //
 | |
| // @ANS = TableName_Actions('CalcField', 'CalcColName')
 | |
| //
 | |
| // ---------------------------------------------------------------------------------------------------------------------
 | |
| 
 | |
| Calc_Field:
 | |
| 	// Make sure the ActionFlow return variable is cleared in case nothing is calculated.
 | |
| 	ActionFlow = ''
 | |
| 	
 | |
| return
 | |
| 
 | |
| 
 | |
| // ----- MFS calls -----------------------------------------------------------------------------------------------------
 | |
| 
 | |
| READ_RECORD_PRE:
 | |
| 	// In order to stop a record from being read in this action these lines of code must be used:
 | |
| 	//
 | |
| 	//    OrigFileError   = 100 : @FM : KeyID
 | |
| 	//    Status          = 0
 | |
| 	//    Record          = ''
 | |
| 	//    ActionFlow      = ACTION_STOP$
 | |
| return
 | |
| 
 | |
| READ_RECORD:
 | |
| 	// In order to stop a record from being read in this action these lines of code must be used:
 | |
| 	//
 | |
| 	//    OrigFileError   = 100 : @FM : KeyID
 | |
| 	//    Status          = 0
 | |
| 	//    Record          = ''
 | |
| 	*    LockOwner = Record<WM_IN_LOCKED_BY$>
 | |
| 	*    If LockOwner EQ '' then Record<WM_IN_LOCKED_BY$> = @User4
 | |
| 	
 | |
| return
 | |
| 
 | |
| READONLY_RECORD_PRE:
 | |
| 	// In order to stop a record from being read in this action these lines of code must be used:
 | |
| 	//
 | |
| 	//    OrigFileError   = 100 : @FM : KeyID
 | |
| 	//    Status          = 0
 | |
| 	//    Record          = ''
 | |
| 	//    ActionFlow      = ACTION_STOP$
 | |
| return
 | |
| 
 | |
| READONLY_RECORD:
 | |
| 	// In order to stop a record from being read in this action these lines of code must be used:
 | |
| 	//
 | |
| 	//    OrigFileError   = 100 : @FM : KeyID
 | |
| 	//    Status          = 0
 | |
| 	//    Record          = ''
 | |
| return
 | |
| 
 | |
| WRITE_RECORD_PRE:
 | |
| return
 | |
| 
 | |
| WRITE_RECORD:
 | |
| 	
 | |
| 	// Look for schedule events that may need to be updated.
 | |
| 	WONo          = Field(Name, '*', 1)
 | |
| 	CassNo        = Field(Name, '*', 3)
 | |
| 	ProcessedDTM  = Datetime()
 | |
| 	// Find the SCHED_DET_NG event record that contains the cassette number.
 | |
| 	Query         = 'SELECT SCHED_DET_NG WITH WO_NO EQ ':WONo:' AND WITH UNPROCESSED_CASS CONTAINING ':CassNo
 | |
| 	GoSub ClearCursors
 | |
| 	RList(Query, TARGET_ACTIVELIST$, '', '', '')
 | |
| 	EOF           = False$
 | |
| 	Done          = False$
 | |
| 	Loop
 | |
| 		Readnext SchedDetNGKey else EOF = True$
 | |
| 	Until EOF = True$
 | |
| 		SchedDetNGRec      = Database_Services('ReadDataRow', 'SCHED_DET_NG', SchedDetNGKey)
 | |
| 		ProcessedCassettes = SchedDetNGRec<SCHED_DET_NG.PROCESSED_CASS$>
 | |
| 		ProcessedWfrQty    = DCount(ProcessedCassettes, @VM) * 25
 | |
| 		TotalWfrs          = SchedDetNGRec<SCHED_DET_NG.EVENT_TOTAL_WFRS$>
 | |
| 		CassRemWfrs        = Xlate('WM_IN', Name, 'REM_WFRS', 'X')
 | |
| 		ProcessedCassQty   = 25 - CassRemWfrs
 | |
| 		If (ProcessedWfrQty + ProcessedCassQty ) EQ TotalWfrs then
 | |
| 			// Mark cassette complete.
 | |
| 			UnprocessedCassettes = SchedDetNGRec<SCHED_DET_NG.UNPROCESSED_CASS$>
 | |
| 			Locate CassNo in UnprocessedCassettes using @VM setting vPos then
 | |
| 				UnprocessedCassettes                          = Delete(UnprocessedCassettes, 0, vPos, 0)
 | |
| 				ProcessedCassettes                            = SchedDetNGRec<SCHED_DET_NG.PROCESSED_CASS$>
 | |
| 				ProcessedCassettes<0, -1>                     = CassNo
 | |
| 				SchedDetNGRec<SCHED_DET_NG.UNPROCESSED_CASS$> = UnprocessedCassettes
 | |
| 				SchedDetNGRec<SCHED_DET_NG.PROCESSED_CASS$>   = ProcessedCassettes
 | |
| 				Database_Services('WriteDataRow', 'SCHED_DET_NG', SchedDetNGKey, SchedDetNGRec, True$, False$, True$)
 | |
| 				Done                                          = True$
 | |
| 			end
 | |
| 			// Mark previous cassette processed if it isn't already. This covers situations where users fail to
 | |
| 			// sign the unload stage for a cassette.
 | |
| 			PrevCassNo = CassNo - 1
 | |
| 			Locate PrevCassNo in UnprocessedCassettes using @VM setting vPos then
 | |
| 				UnprocessedCassettes                          = Delete(UnprocessedCassettes, 0, vPos, 0)
 | |
| 				ProcessedCassettes                            = SchedDetNGRec<SCHED_DET_NG.PROCESSED_CASS$>
 | |
| 				ProcessedCassettes<0, -1>                     = PrevCassNo
 | |
| 				SchedDetNGRec<SCHED_DET_NG.UNPROCESSED_CASS$> = UnprocessedCassettes
 | |
| 				SchedDetNGRec<SCHED_DET_NG.PROCESSED_CASS$>   = ProcessedCassettes
 | |
| 				Database_Services('WriteDataRow', 'SCHED_DET_NG', SchedDetNGKey, SchedDetNGRec, True$, False$, True$)
 | |
| 			end
 | |
| 		end
 | |
| 		
 | |
| 	Until Done EQ True$
 | |
| 	Repeat
 | |
| 	
 | |
| 	WfrCount = DCount(Record<WM_IN_SLOT_NO$>, @VM)
 | |
| 	If WfrCount LT 25 then
 | |
| 		// Notify FI as this is unusual and can cause systematic issues.
 | |
| 		Recipients   =  XLATE('SEC_GROUPS', 'OI_ADMIN', 'USER', 'X')
 | |
| 		SentFrom     = 'OI Admin'
 | |
| 		Subject      = 'WM_IN cassette quantity less than 25 detected!'
 | |
| 		Message      = 'WM_IN ':@ID:' quantity is ':WfrCount:'. Correct if needed!'
 | |
| 		AttachWindow = 'WM_IN'
 | |
| 		AttachKey    = @ID   
 | |
| 		SendToGroup  = ''   
 | |
| 		Parms        = Recipients:@RM:SentFrom:@RM:Subject:@RM:Message:@RM:AttachWindow:@RM:AttachKey:@RM:SendToGroup
 | |
| 		obj_Notes('Create',Parms) 
 | |
| 	end
 | |
| 	
 | |
| return
 | |
| 
 | |
| DELETE_RECORD_PRE:
 | |
| 	
 | |
| 	// Log which user and computer station attempted to delete the record
 | |
| 	*    LogData    = ''
 | |
| 	*    LogData<1> = LoggingDTM
 | |
| 	*    LogData<2> = Name
 | |
| 	*    LogData<3> = @User4
 | |
| 	*    Logging_Services('AppendLog', objDeleteLog, LogData, @RM, @FM)
 | |
| 	*    
 | |
| 	*    // Send an LSL message to FI admins to alert them
 | |
| 	*    Recipients   = Xlate('SEC_GROUPS', 'OI_ADMIN', 'USER', 'X')
 | |
| 	*    SentFrom     = 'SYSTEM'
 | |
| 	*    Subject      = 'WM_IN Delete Attempt'
 | |
| 	*    Message      = 'An attempt to delete WM_IN record ':Name:' was made by ':@User4:'.'
 | |
| 	*    AttachWindow = ''
 | |
| 	*    AttachKey    = ''
 | |
| 	*    SendToGroup  = ''
 | |
| 	* 
 | |
| 	*    Parms = Recipients:@RM:SentFrom:@RM:Subject:@RM:Message:@RM:AttachWindow:@RM:AttachKey:@RM:SendToGroup
 | |
| 	*    obj_Notes('Create',Parms)
 | |
| 	
 | |
| 	// Stop the system from deleting the record
 | |
| 	*    ActionFlow = ACTION_STOP$
 | |
| 	
 | |
| return
 | |
| 
 | |
| DELETE_RECORD:
 | |
| 	
 | |
| return
 | |
| 
 | |
| 
 | |
| // ----- Internal Methods ----------------------------------------------------------------------------------------------
 | |
| 
 | |
| ClearCursors:
 | |
| 	For counter = 0 to 8
 | |
| 		ClearSelect counter
 | |
| 	Next counter
 | |
| return
 | |
| 
 | |
| Initialize_System_Variables:
 | |
| 	// Save these for restoration later
 | |
| 	SaveDict        = @DICT
 | |
| 	SaveID          = @ID
 | |
| 	SaveRecord      = @RECORD
 | |
| 	OrigFileError   = @FILE.ERROR
 | |
| 	
 | |
| 	// Now make sure @DICT, ID, and @RECORD are populated
 | |
| 	CurrentDictName = ''
 | |
| 	If @DICT then
 | |
| 		DictHandle = @DICT<1, 2>
 | |
| 		Locate DictHandle in @TABLES(5) Using @FM Setting fPos then
 | |
| 			CurrentDictName = Field(@TABLES(0), @FM, fPos, 1)
 | |
| 		end
 | |
| 	end
 | |
| 	
 | |
| 	If CurrentDictName NE DictName then
 | |
| 		Open DictName to @DICT else Status = 'Unable to initialize @DICT'
 | |
| 	end    
 | |
| 	
 | |
| 	@ID = KeyID
 | |
| 	If Record else
 | |
| 		// Record might not have been passed in. Read the record from the database table just to make sure.
 | |
| 		@FILE.ERROR     = ''
 | |
| 		Open TableName to hTable then
 | |
| 			FullFSList  = hTable[1, 'F' : @VM]
 | |
| 			BFS         = FullFSList[-1, 'B' : @SVM]
 | |
| 			LastHandle  = hTable[-1, 'B' : \0D\]
 | |
| 			FileHandle  = \0D\ : LastHandle[1, @VM]
 | |
| 			
 | |
| 			Call @BFS(READO.RECORD, BFS, FileHandle, KeyID, FMC, Record, ReadOStatus)
 | |
| 		end
 | |
| 	end
 | |
| 	@RECORD = Record
 | |
| return
 | |
| 
 | |
| Restore_System_Variables:
 | |
| 	Transfer SaveDict   to @DICT
 | |
| 	Transfer SaveID     to @ID
 | |
| 	Transfer SaveRecord to @RECORD
 | |
| 	@FILE.ERROR = OrigFileError
 | |
| return
 | |
| 
 | |
| 
 |