Delphi_調用DLL方法[調用對象下面的方法]


unit PayIntf_MisDll;

interface

uses
  Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,
  Dialogs, StdCtrls;

type
  { /* 交易類型: '00'消費'01'撤銷'02'退貨'03'查余
                  '04'重打印'05'簽到'06'結算'08'小費'09'預授權
                  '10'預授權追加'11'預授權完成'12'預授權撤銷
                  '13'預授權完成撤銷'14'商場分期'15'專用分期
                  '16'分期撤銷'18'分期額度查詢'19'汽車卡洗車
                  '20'快速支付'21'指定賬戶圈存'22'電子現金退貨
                  '23'電子現金查余'24'積分兌換消費'25'積分兌換撤銷
                  '26'積分查詢'28'權益積分查詢'29'權益積分消費
                  '30'權益積分撤銷'32'卡轉出轉賬'33'財務報銷'34'財務還款
                  ‘60’-聚合支付被掃
                  ‘61’-聚合支付主掃
                  ‘62’-聚合支付退貨
                  ‘63’-獲取 POS 掃碼槍二維碼數據
                  ‘64’-單筆聚合支付異常訂單查詢
                  '72'-聚合支付異常訂單查詢
                  ‘90’-惠兜圈優惠立減銀行卡
                  ‘94’-惠兜圈立減優惠銀聯二維碼
                  ‘95’-雲閃付優惠立減交易
                  ‘96’-商場分期優惠立減
                  */  }
  TInputData = record
    posid: array[0..7] of char;         // /* 收銀機號(8 字節,左對齊,不足部分補空格)*/
    operid: array[0..7] of char;        // /* 操作員號(8 字節,左對齊,不足部分補空格)*/
    trans: array[0..1] of char;         //交易類型 見上面
    amount: array[0..11] of char;       //  /* 金額(12 字節,無小數點,左補 0,單位:分)*/
    old_date: array[0..7] of char;      //  /* 原交易日期(8 字節,yyyymmdd 格式,隔日退貨時用)*/
    old_reference: array[0..11] of char;  // /* 原交易參考號*/
    old_trace: array[0..5] of char;     // /* 流水號(6 字節,右對齊,左補 0,退貨或重打印等用)*/
    old_batch: array[0..5] of char;     // /*57~62 位  原批次號*/
    old_auth: array[0..5] of char;      //  /*63~68 位  原授權碼*/
    old_terno: array[0..7] of char;     // /*69~76 位  原交易終端號*/
    szFenqiNum: array[0..1] of char;    // /*77~78 位 分期數 03,06,09,12,18,24,36,42,48,60*/
    szServiceNum: array[0..1] of char;  //  /*79~80 位 享受服務人數*/
    szGoodsNo: array[0..11] of char;    //  /*81~92 位  商品項目編碼*/
    trk2: array[0..36] of char;         // /*  二磁道數據(37 字節,左對齊,不足部分補空格)*/
    trk3: array[0..103] of char;        // /* 三磁道數據(104 字節, 左對齊,不足部分補空格)*/
    lrc: array[0..2] of char;           //  /* 交易校驗數據(3 位從 0~9 的隨機字符)*/
    szOrderTrace: array[0..19] of char; // 6/16/2015 新增 收銀流水(訂單)號 左對齊 不足補空格
    szPrefer: array[0..49] of char;     // 優惠券左對齊 不足補空格
    szRsv: array[0..299] of char;       // 保留字段 在此輸入二維碼的條碼數據 左對齊 不足補空格
  end;

  TOutPutData = record
    resp_code: array[0..1] of char;    //  /*1~2 位  返回碼 (2 字節,"00"交易成功,其他失敗)*/
    bank_code: array[0..3] of char;    //  /*3~6 位  銀行行號(4 字節)*/
    card_no: array[0..29] of char;     //  /*7~36 位  卡號:622280*********4860 */
    expr: array[0..3] of char;         //  /*37~40 位 有效期 (4 字節) */
    amount: array[0..11] of char;      //  /*141~52 位 金額(12 字節,無小數點,左補 0,單位:分)*/
    trace: array[0..5] of char;        //  /*53~58 位 流水號 (6 字節,左對齊)*/
    refer: array[0..11] of char;       //  /*59~70 位 交易參考號*/
    auth: array[0..5] of char;         //  /*71~76 位 授權號*/
    batch: array[0..5] of char;        //  /*77~82 位 批次號*/
    date: array[0..13] of char;        //  /*83~96 位 交易日期(8 字節,yyyymmddhhmmss 格式)*/
    userno: array[0..14] of char;      //  /*97~111 位 商戶號*/
    terno: array[0..7] of char;        //  /*112~119 位 終端號*/
    old_terno: array[0..7] of char;    //  /*120~127 位  原終端號*/
    resp_chin: array[0..49] of char;   //  /* 錯誤說明(左對齊,不足部分補空格)*/
    OrderNo: array[0..49] of char;     //  /* 二維碼交易訂單訂單號 */
    trans: array[0..1] of char;        //  /* 交易類型 */
    ChannelType: array[0..1] of char;  //  /* '0'銀聯二維碼 '1'微信 '2' 支付寶 '3'龍支付 */
    lrc: array[0..2] of char;          //  /* 交易數據校驗碼(3 字節)*/
  end;

function PayIntf_Trans(ParkCode, OptorCode, Trans, Money, OldTrandID: string; var OrderNo: string): Boolean;

function GetInputStr(ParkCode, OptorCode, Trans, Money, OldTrandID: string; var InStr: string): Boolean;

procedure GetOutputData(OutStr: string);

procedure FillArrayChar(FillType, ParamVaule: string; var ArrVaule: array of char);

function GetOldTradeInfo(OldTrandID: string; var old_date, old_reference, old_trace: string): boolean;

function CheckDataValid(ParkCode, OptorCode, Trans, Money, OldTrandID: string): boolean;

function Obj2Json_InputData(InputData: TInputData): string;

function Obj2Json_OutputData(OutputData: TOutputData): string;

procedure MyWriteLog(const mStr: string);

function BankTrans(PInStr: Pchar; POutStr: Pchar): integer; stdcall; external 'C:\landiccbmispos\MisPos.dll';   //0- 成功 , 其他-失敗

implementation

function CheckDataValid(ParkCode, OptorCode, Trans, Money, OldTrandID: string): boolean;
var
  sMsg,sFileDLL: string;
begin
  Result := False;
  sFileDLL := 'C:\landiccbmispos\MisPos.dll';
  if not FileExists(sFileDLL) then
  begin
    sMsg := '調用支付接口文件不存在!';
    MyWriteLog(sMsg + '[' + sFileDLL + ']');
    Application.MessageBox(Pchar(sMsg), '提示', MB_OK);
    Exit;
  end;

  if (Trans <> '00') and (Trans <> '01') and (Trans <> '02')
  and (Trans <> '05') and (Trans <> '06') then    //'00'消費'01'撤銷'02'退貨'03'查余 '05'簽到'06'結算
  begin
    sMsg := '非法的交易類型!';
    MyWriteLog(sMsg);
    Application.MessageBox(Pchar(sMsg), '提示', MB_OK);
    Exit;
  end;
  if (Trans = '01') or (Trans = '02') then
  begin
    if OldTrandID = '' then
    begin
      sMsg := '撤銷和退貨時,請傳入原交易單號!';
      MyWriteLog(sMsg);
      Application.MessageBox(Pchar(sMsg), '提示', MB_OK);
      Exit;
    end;
  end;
  if StrToCurrDef(Money,0) <= 0 then
  begin
    sMsg := '金額不允許為負數或零!';
    MyWriteLog(sMsg);
    Application.MessageBox(Pchar(sMsg), '提示', MB_OK);
    Exit;
  end;

  Result := True;
end;

function PayIntf_Trans(ParkCode, OptorCode, Trans, Money, OldTrandID: string; var OrderNo: string): Boolean;
var
  lv_nRet: integer;
  InStr: string;
  OutStr: string;
  sMsg: string;
  resp_code: string;
begin
  {*  1、檢查傳入參數:根據文檔規則
   *  2、參數組合成定長字符串
   *  3、調用DLL方法
   *  4、解析返回定長字符串:判斷結果
   *  5、記錄日志
   *}
   
  //檢查傳入參數
  if not CheckDataValid(ParkCode, OptorCode, Trans, Money, OldTrandID) then
  begin
    Exit;
  end;

  //組合傳入參數,形成定長字符串InStr
  if not GetInputStr(ParkCode, OptorCode, Trans, Money, OldTrandID, InStr) then
  begin
    Exit;
  end;

  MyWriteLog('發送定長字符串:[' + InStr + ']');
  SetLength(OutStr, SizeOf(TOutPutData));
  Application.ProcessMessages;
  lv_nRet := BankTrans(Pchar(InStr), Pchar(OutStr));   //調用DLL中的BankTrans方法
  Application.ProcessMessages;
  MyWriteLog('返回定長字符串:[' + OutStr + ']');
  GetOutputData(OutStr);

  if (lv_nRet <> 0) or (Copy(OutStr, 1, 2) <> '00') then
  begin
    sMsg := '交易失敗!';
    MyWriteLog(sMsg);
    Application.MessageBox(Pchar(sMsg), '提示', MB_OK);
    Exit;
  end;

  OrderNo := Trim(Copy(OrderNo, 158, 50));
  if Copy(OrderNo, 1, 8) <> 'EEEEEEEE' then   // 特殊,當退貨出現異常情況, 如網絡異常時, 返回頭的輸出參數頭2個字節為00其余字符會以大寫字母EE填充,
  begin                                       //同時DLL會彈出對話框。請商戶及時和銀行核實是否退貨成功!!!
    sMsg := '交易成功!';
    MyWriteLog(sMsg + '[' + OrderNo + ']');
    Application.MessageBox(Pchar(sMsg), '提示', MB_OK);
  end else
  begin
    sMsg := '交易成功,單需要手工核對!';
    MyWriteLog(sMsg + '[' + OrderNo + ']');
    Application.MessageBox(Pchar(sMsg), '提示', MB_OK);
  end;
  
end;

function GetInputStr(ParkCode, OptorCode, Trans, Money, OldTrandID: string; var InStr: string): Boolean;
var
  old_date, old_reference, old_trace, sMsg: string;
  InputData: TInputData;
begin
  {交易說明:
            消費 :傳入必要信息為 (交易類型,金額)
            退貨 :傳入必要信息為 (交易類型,金額,原交易參考號,原交易日期)
            撤銷 : 傳入必要信息為 (交易類型,金額,原流水號)
            查余 : 傳入必要信息為 (交易類型)
            注: 查詢金額 操作 卡內余額顯示在 密碼鍵盤。並且不會返回對應的余額.
            重打印 : 傳入必要信息為 (交易類型,原流水號)
            注: 重打印上一筆交易 原流水號:”000000”
            簽到 : 傳入必要信息為 (交易類型)
            結算 : 傳入必要信息為 (交易類型)  }
  Result := False;
  try
    FillChar(InputData,SizeOf(TInputData),' ');
    FillArrayChar('RN', ParkCode, InputData.posid);
    FillArrayChar('RN', OptorCode, InputData.operid);
    FillArrayChar('R0', Trans, InputData.trans);
    FillArrayChar('L0', IntToStr(Trunc(StrToCurrDef(Money, 0) * 100)), InputData.amount);

    if (Trans = '01') or (Trans = '02') then    //'00'消費'01'撤銷'02'退貨'03'查余
    begin
      //獲取原交易信息
      if not GetOldTradeInfo(OldTrandID, old_date, old_reference, old_trace) then
      begin
        Exit;
      end;

      if Trans = '01' then
      begin
        FillArrayChar('L0', old_trace, InputData.old_trace);           // /* 原流水號(6 字節,右對齊,左補 0,退貨或重打印等用)*/
        FillArrayChar('RN', '', InputData.old_date);
        FillArrayChar('RN', '', InputData.old_reference);
      end;
      if Trans = '02' then
      begin
        FillArrayChar('L0', '', InputData.old_trace);
        FillArrayChar('RN', old_date, InputData.old_date);             // /* 原交易日期(8 字節,yyyymmdd 格式,隔日退貨時用)*/
        FillArrayChar('RN', old_reference, InputData.old_reference);   // /* 原交易參考號*/
      end;
    end
    else        //非撤銷和退貨
    begin
      FillArrayChar('L0', '', InputData.old_trace);
      FillArrayChar('RN', '', InputData.old_date);
      FillArrayChar('RN', '', InputData.old_reference);
    end;

    //以下為默認填充
    FillArrayChar('RN', '', InputData.old_batch);    // /*57~62 位  原批次號*/
    FillArrayChar('RN', '', InputData.old_auth);     //  /*63~68 位  原授權碼*/
    FillArrayChar('RN', '', InputData.old_terno);    // /*69~76 位  原交易終端號*/
    FillArrayChar('RN', '', InputData.szFenqiNum);   // /*77~78 位 分期數 03,06,09,12,18,24,36,42,48,60*/
    FillArrayChar('RN', '', InputData.szServiceNum); //  /*79~80 位 享受服務人數*/
    FillArrayChar('RN', '', InputData.szGoodsNo);    //  /*81~92 位  商品項目編碼*/
    FillArrayChar('RN', '', InputData.trk2);         // /*  二磁道數據(37 字節,左對齊,不足部分補空格)*/
    FillArrayChar('RN', '', InputData.trk3);         // /* 三磁道數據(104 字節, 左對齊,不足部分補空格)*/
    FillArrayChar('RN', '', InputData.lrc);          //  /* 交易校驗數據(3 位從 0~9 的隨機字符)*/
    FillArrayChar('RN', '', InputData.szOrderTrace); // 6/16/2015 新增 收銀流水(訂單)號 左對齊 不足補空格
    FillArrayChar('RN', '', InputData.szPrefer);     // 優惠券左對齊 不足補空格
    FillArrayChar('RN', '', InputData.szRsv);        // 保留字段 在此輸入二維碼的條碼數據 左對齊 不足補空格

    //取傳參的累加的定長字符串
    SetLength(InStr,SizeOf(TInputData));
    StrLCopy(Pchar(InStr), @InputData, SizeOf(TInputData));
    sMsg := '發送信息[轉換為JSON]:' + Obj2Json_InputData(InputData);
    MyWriteLog(sMsg);
    Result := True;
  except
    on e: exception do
    begin
      sMsg := '獲取傳入參數字符串異常:' + e.message;
      MyWriteLog(sMsg);
      Application.MessageBox(Pchar(sMsg), '提示', MB_OK);
      Exit;
    end;
  end;
end;

procedure GetOutputData(OutStr: string);
var
  sMsg, sTemp: string;
  OutputData: TOutPutData;
begin
  try
    if OutStr = '' then
    begin
      sMsg := '交易返回參數為空!';
      MyWriteLog(sMsg);
      Application.MessageBox(Pchar(sMsg), '提示', MB_OK);
      Exit;
    end;

    //字符串賦值給記錄
    FillChar(OutputData,SizeOf(TOutputData),' ');
    StrLCopy(@OutputData, Pchar(OutStr), SizeOf(TOutputData));

    //記錄返回的信息
    sTemp := '返回信息[轉換為JSON]:' + Obj2Json_OutputData(OutputData);
    MyWriteLog(sTemp);
  except
    on e: exception do
    begin
      sMsg := '轉換返回參數異常:' + e.message;
      MyWriteLog(sMsg);
      Application.MessageBox(Pchar(sMsg), '提示', MB_OK);
      Exit;
    end;
  end;
end;

function Obj2Json_InputData(InputData: TInputData): string;
begin
  Result := '{';
  Result := Result + '"posid":"' + Trim(InputData.posid) + '",';
  Result := Result + '"operid":"' + Trim(InputData.operid) + '",';
  Result := Result + '"trans":"' + Trim(InputData.trans) + '",';
  Result := Result + '"amount":"' + Trim(InputData.amount) + '",';
  Result := Result + '"old_date":"' + Trim(InputData.old_date) + '",';
  Result := Result + '"old_reference":"' + Trim(InputData.old_reference) + '",';
  Result := Result + '"old_trace":"' + Trim(InputData.old_trace) + '",';
  Result := Result + '"old_batch":"' + Trim(InputData.old_batch) + '",';
  Result := Result + '"old_auth":"' + Trim(InputData.old_auth) + '",';
  Result := Result + '"old_terno":"' + Trim(InputData.old_terno) + '",';
  Result := Result + '"szFenqiNum":"' + Trim(InputData.szFenqiNum) + '",';
  Result := Result + '"szServiceNum":"' + Trim(InputData.szServiceNum) + '",';
  Result := Result + '"szGoodsNo":"' + Trim(InputData.szGoodsNo) + '",';
  Result := Result + '"trk2":"' + Trim(InputData.trk2) + '",';
  Result := Result + '"trk3":"' + Trim(InputData.trk3) + '",';
  Result := Result + '"lrc":"' + Trim(InputData.lrc) + '",';
  Result := Result + '"szOrderTrace":"' + Trim(InputData.szOrderTrace) + '",';
  Result := Result + '"szPrefer":"' + Trim(InputData.szPrefer) + '",';
  Result := Result + '"szRsv":"' + Trim(InputData.szRsv) + '"';
  Result := Result + '}';
end;

function Obj2Json_OutputData(OutputData: TOutputData): string;
begin
  Result := '{';
  Result := Result + '"resp_code":"' + Trim(OutputData.resp_code) + '",';
  Result := Result + '"bank_code":"' + Trim(OutputData.bank_code) + '",';
  Result := Result + '"card_no":"' + Trim(OutputData.card_no) + '",';
  Result := Result + '"expr":"' + Trim(OutputData.expr) + '",';
  Result := Result + '"amount":"' + Trim(OutputData.amount) + '",';
  Result := Result + '"trace":"' + Trim(OutputData.trace) + '",';
  Result := Result + '"refer":"' + Trim(OutputData.refer) + '",';
  Result := Result + '"auth":"' + Trim(OutputData.auth) + '",';
  Result := Result + '"batch":"' + Trim(OutputData.batch) + '",';
  Result := Result + '"date":"' + Trim(OutputData.date) + '",';
  Result := Result + '"userno":"' + Trim(OutputData.userno) + '",';
  Result := Result + '"terno":"' + Trim(OutputData.terno) + '",';
  Result := Result + '"old_terno":"' + Trim(OutputData.old_terno) + '",';
  Result := Result + '"resp_chin":"' + Trim(OutputData.resp_chin) + '",';
  Result := Result + '"OrderNo":"' + Trim(OutputData.OrderNo) + '",';
  Result := Result + '"trans":"' + Trim(OutputData.trans) + '",';
  Result := Result + '"ChannelType":"' + Trim(OutputData.ChannelType) + '",';
  Result := Result + '"lrc":"' + Trim(OutputData.lrc) + '"';
  Result := Result + '}';
end;

function GetOldTradeInfo(OldTrandID: string; var old_date, old_reference, old_trace: string): boolean;
var
  sMsg: string;
begin
  Result := False;
  try
    old_date := '20190805';
    old_reference := '123456789987456';
    old_trace := '123456';

    Result := True;
  except
    on e: exception do
    begin
      sMsg := '獲取原交易支付信息異常[原交易號:' + OldTrandID + ']:' + e.message;
      MyWriteLog(sMsg);
      Application.MessageBox(Pchar(sMsg), '提示', MB_OK);
      Exit;
    end;
  end;
end;

procedure FillArrayChar(FillType, ParamVaule: string; var ArrVaule: array of char);
var
  I, Len, Diff: integer;
  sTemp: string;
begin
  Len := Length(ArrVaule);
  if Length(ParamVaule) < Len then
  begin
    Diff := Len - Length(ParamVaule);
    for I := 1 to Diff do
    begin
      if Copy(FillType, 2, 1) = '0' then    //填充"0"
      begin
        sTemp := sTemp + '0';
      end
      else
      begin
        sTemp := sTemp + ' ';             //填充空格
      end;
    end;

    if Copy(FillType, 1, 1) = 'L' then  //左填充
    begin
      sTemp := sTemp + ParamVaule;
    end
    else if Copy(FillType, 1, 1) = 'R' then   //右填充
    begin
      sTemp := ParamVaule + sTemp;
    end;
  end else
  begin
    sTemp := ParamVaule;
  end;

  for I := 1 to Len do
  begin
    ArrVaule[I - 1] := sTemp[I];
  end;
end;

procedure MyWriteLog(const mStr: string);
var
  f: textfile;
  myDir, myFileName: string;
  FileHandle: Integer;
  LogType, LogDate, ModuleID: string;
begin
  LogType := 'INFO';
  LogDate := FormatDateTime('YYYY-MM-DD hh:nn:ss zzz', Now);
  ModuleID := '';
//  if not (CanLogFile in FLogFlags) then exit;
  //------寫入文件部分的實現--開始
  myDir := ExtractFilePath(Paramstr(0));
  //確定文件名稱
  myFileName := FormatDateTime('"PayIntf"yyyymmdd".log"', Date);
  //如果可執行目錄下不存在log目錄創建之
  if not DirectoryExists(myDir + '\log') then
    CreateDir(myDir + '\log');
  //如果當日日志文件不存在,則創建文件並釋放句柄
  if not FileExists(myDir + '\log\' + myFileName) then
  begin
    FileHandle := FileCreate(myDir + '\log\' + myFileName); //創建文件
    FileClose(FileHandle); //釋放句柄
  end;
  try//try...except...statements
    AssignFile(f, myDir + '\log\' + myFileName);
    Append(f);
    Writeln(f, '$$' + Format('%6s', [LogType]) + '$$ ' + Format('%12s', [ModuleID]) + '$$' + LogDate + chr(9) + mStr);
    Flush(f);
    CloseFile(f);
    //-----寫入文件部分的實現--結束
  except//try...except...statements
  end; //try...except...statements
end;

end.

  


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM