iboot-http-client 1.1.3 → 1.1.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +32 -11
- package/dist/iboot-http-client.cjs.js +1 -1
- package/dist/iboot-http-client.cjs.js.map +1 -1
- package/dist/iboot-http-client.es.js +342 -305
- package/dist/iboot-http-client.es.js.map +1 -1
- package/dist/iboot-http-client.umd.js +1 -1
- package/dist/iboot-http-client.umd.js.map +1 -1
- package/dist/types/http-client.d.ts +9 -21
- package/dist/types/types/http.d.ts +0 -35
- package/dist/types/types/user.d.ts +8 -8
- package/package.json +3 -1
package/README.md
CHANGED
|
@@ -45,17 +45,13 @@ yarn add iboot-http-client
|
|
|
45
45
|
```ts
|
|
46
46
|
import { HttpClient } from 'iboot-http-client'
|
|
47
47
|
|
|
48
|
-
const client = new HttpClient(
|
|
49
|
-
baseURL: 'https://api.your-icms.com',
|
|
50
|
-
headers: { 'Content-Type': 'application/json' },
|
|
51
|
-
token: 'your-access-token',
|
|
52
|
-
})
|
|
48
|
+
const client = new HttpClient()
|
|
53
49
|
|
|
54
50
|
// GET example
|
|
55
|
-
const list = await client.get('/articles', {
|
|
51
|
+
const list = await client.get('/articles', { data: { page: 1, pageSize: 10 } })
|
|
56
52
|
|
|
57
53
|
// POST example
|
|
58
|
-
const created = await client.post('/articles', { title: 'Example', body: 'Content' })
|
|
54
|
+
const created = await client.post('/articles', { data:title: 'Example', body: 'Content' })
|
|
59
55
|
|
|
60
56
|
console.log(list, created)
|
|
61
57
|
```
|
|
@@ -64,10 +60,11 @@ console.log(list, created)
|
|
|
64
60
|
|
|
65
61
|
## 配置项 / Options
|
|
66
62
|
|
|
67
|
-
- `
|
|
68
|
-
- `
|
|
69
|
-
- `
|
|
70
|
-
- `
|
|
63
|
+
- `deviceId`:客户端设备id(可选),不填会缓存一个随机数
|
|
64
|
+
- `lang`:语言(可选),不填默认zh-CN
|
|
65
|
+
- `websiteId`:websiteId(网站ID,icms系统调用时可用,不填为icms设置的默认网站)
|
|
66
|
+
- `websiteNo`:websiteNo(网站编号,icms系统调用时可用, 不填为icms设置的默认网站),
|
|
67
|
+
- `userType`:客户端用户类型0:营运用户,1:C端用户,2.B端用户
|
|
71
68
|
|
|
72
69
|
## 错误处理 / Error Handling
|
|
73
70
|
|
|
@@ -81,6 +78,30 @@ try {
|
|
|
81
78
|
}
|
|
82
79
|
```
|
|
83
80
|
|
|
81
|
+
## 环境变量
|
|
82
|
+
.env.development / .env.production
|
|
83
|
+
```
|
|
84
|
+
APP_NAME=iCMS
|
|
85
|
+
APP_ENV=production
|
|
86
|
+
APP_APIKEY=
|
|
87
|
+
APP_BASEURL=http://localhost:8081
|
|
88
|
+
#用户来源
|
|
89
|
+
#1.PCweb
|
|
90
|
+
#2.PC桌面应用
|
|
91
|
+
#3.微信小程序
|
|
92
|
+
#4.微信公众号
|
|
93
|
+
#5.企业微信
|
|
94
|
+
#6.手机APP
|
|
95
|
+
APP_USERFROM=1
|
|
96
|
+
```
|
|
97
|
+
#### 注意,如在生产环境不能获取以上环境变量,你可以在 shell 配置文件中设置环境变量,例如 .bashrc 或 .profile 导入环境变量
|
|
98
|
+
```
|
|
99
|
+
export APP_NAME="icms"
|
|
100
|
+
export APP_ENV="production"
|
|
101
|
+
export APP_APIKEY="xxx"
|
|
102
|
+
APP_BASEURL=http://localhost:8081
|
|
103
|
+
```
|
|
104
|
+
|
|
84
105
|
## 开发与贡献 / Contributing
|
|
85
106
|
|
|
86
107
|
- 欢迎提交 issue 或 PR。请在 issue 中描述重现步骤与期望行为。
|
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
"use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const M=require("pino"),L=require("pino-pretty"),k=require("js-md5"),E=require("crypto-js"),B={GENERAL:0,PHONE:1,EMAIL:2},A={TYPE_MGT:0,TYPE_C:1,TYPE_B:2},D={FROM_WEB:1,FROM_PC:2,FROM_WX_MINI_PRO:3,FROM_WX_PLU:4,FROM_WX_E:5,FROM_APP:6,FROM_DEVICE:7},x={unknown:0,male:1,female:2},H=L({colorize:!0,destination:process.stdout,translateTime:"SYS:yyyy-mm-dd HH:MM:ss",ignore:"pid,hostname",messageFormat:"{msg}",singleLine:!0}),u=M({name:process.env.APP_NAME??"iBoot",level:process.env.APP_ENV==="production"?"info":"debug",nestedKey:"payload"},H),b=o=>Array.isArray(o)||typeof o=="object"&&Object.prototype.toString.call(o)==="[object Array]",_=o=>{const t=o||32,e="ABCDEFGHJKMNPQRSTWXYZabcdefhijkmnprstwxyz2345678",s=e.length;let r="";for(let n=0;n<t;n++)r+=e.charAt(Math.floor(Math.random()*s));return r},C=(o,t)=>{const e={};return o.split("&").forEach(r=>{const n=r.split("=");n[1]&&(!t||t.indexOf(n[0])==-1)&&(e[n[0]]=n[1])}),e},y=o=>!o||o==""||o.length==0?"":encodeURIComponent(o).replace(/\(/g,"%28").replace(/\)/g,"%29").replace(/\'/g,"%27").replace(/\!/g,"%21").replace(/\~/g,"%7E"),W=(o,t)=>{const e={"M+":o.getMonth()+1,"d+":o.getDate(),"h+":o.getHours(),"m+":o.getMinutes(),"s+":o.getSeconds(),"q+":Math.floor((o.getMonth()+3)/3),S:o.getMilliseconds()};if(/(y+)/.test(t)){const r=new RegExp(/(y+)/).exec(t);let n=0;r&&r.length>0&&(n=r[0].length),t=t.replace(/(y+)/,o.getFullYear()+"").substring(4-n)}for(const s in e){const r="("+s+")",n=new RegExp(r);if(n.test(t)){const i=n.exec(t);let c=0;i&&i.length>0&&(c=i[0].length),t=t.replace(n,c==1?e[s].toString():("00"+e[s]).substring(e[s].toString().length))}}return t},R="zh-CN",w="_device_id_key",O="_current_website_key",v="Get request error!",S="Post request error!",I={"Device-Id":"Device-Id",Lang:"Lang","Website-Id":"Website-Id","Website-No":"Website-No"},g={IBOOT_DEVICE_ID:"IBOOT_DEVICE_ID",IBOOT_LOCALE:"IBOOT_LOCALE",IBOOT_WEBSITE_ID:"IBOOT_WEBSITE_ID",IBOOT_WEBSITE_NO:"IBOOT_WEBSITE_NO",token:"token",user:"user"},U=()=>{const o=localStorage.getItem(w)??"",t=localStorage.getItem(O),e=t!=null?JSON.parse(t):null,s={};return o.length>0&&(s["Device-Id"]=o),e&&(s.Lang=e.language,e.websiteId&&(s["Website-Id"]=e.websiteId),e.websiteNo&&(s["Website-No"]=e.websiteNo)),s},$=(o,t)=>{localStorage.setItem(w,o),t&&localStorage.setItem(O,JSON.stringify(t))},N=async(o,t)=>{if(t?.data){const i=new URLSearchParams(t.data);o.indexOf("?")!=-1?o+=`&${i}`:o+=`?${i}`}const e=U(),s=t?.headers;if(s)for(const i in s){const c=s[i];c&&(e[i]=c)}u.debug({url:o,headers:e},"GET");const r=await fetch(o,{method:"GET",headers:e,cache:t?.useCache?"force-cache":"default"});if(r.ok){const i=await r.json();return u.debug(i,"GET_RESULT"),i}const n={code:r.status,success:!1,msg:r.statusText};return u.error(n),n},P=async(o,t)=>{const e=t?.data??{},s=t?.headers;let r;const n=new Headers(U());if(e instanceof FormData?r=e:(r=JSON.stringify(e),n.set("Content-Type","application/json")),s)for(let a in s){const l=s[a];l&&n.set(a,l)}u.debug({url:o,headers:n},"POST");const i=await fetch(o,{method:"POST",headers:n,body:r});if(i.ok){const a=await i.json();return u.debug(a,"POST_RESULT"),a}const c={code:i.status,success:!1,msg:i.statusText};return u.error(c),c},F=async(o,t)=>{const e=await N(o,t);if(e.success)return e.data;if(t?.showError){t.showError(e.msg??v);return}throw Error(e.msg??v)},j=async(o,t)=>{const e=await P(o,t);if(e.success)return e.data;if(t?.showError){t.showError(e.msg??S);return}throw Error(e.msg??S)},K=async(o,t)=>{const e=await P(o,t);return e.success?(t?.showSuccess&&t.showSuccess(e.msg??"SUCCESS"),!0):(t?.showError&&t.showError(e.msg??S),!1)},G=o=>{const t=o.headers.get(I["Device-Id"]),e=o.headers.get(I.Lang),s=o.headers.get(I["Website-Id"]),r=o.headers.get(I["Website-No"]),n={};return t&&t.length>0&&(n.deviceId=t),e&&e.length>0&&(n.lang=e),s&&s.length>0&&(n.websiteId=s),r&&r.length>0&&(n.websiteNo=r),n},Y=o=>{const t={},e=o.get(g.IBOOT_DEVICE_ID)?.value,s=o.get(g.IBOOT_LOCALE)?.value,r=o.get(g.IBOOT_WEBSITE_ID)?.value,n=o.get(g.IBOOT_WEBSITE_NO)?.value;return e&&e.length>0&&(t.deviceId=e),s&&s.length>0&&(t.lang=s),r&&r.length>0&&(t.websiteId=r),n&&n.length>0&&(t.websiteNo=n),t};class J{baseUrl;helloMethod;apiKey;userType;userFrom;deviceId;version="1";lang;websiteId;websiteNo;isProduction;constructor(t){this.isProduction=process.env.APP_ENV==="production",this.isProduction||u.warn("This is currently a development environment!"),this.baseUrl=process.env.APP_BASEURL,this.apiKey=process.env.APP_APIKEY,this.userFrom=process.env.APP_USERFROM??"1",this.helloMethod=t.helloURL??"",this.userType=t.userType??A.TYPE_MGT,this.deviceId=t.deviceId??_(10),this.lang=t.lang??R,this.websiteId=t.websiteId,this.websiteNo=t.websiteNo}encrypt(t){const e=E.enc.Utf8.parse(this.apiKey);return E.AES.encrypt(t,e).toString()}decrypt(t){const e=E.enc.Utf8.parse(this.apiKey);return E.AES.decrypt(t,e,{mode:E.mode.ECB}).toString(E.enc.Utf8)}getDeviceId(){return this.deviceId}convertUrlParameter(t){const e=[];for(const s in t){let r=t[s];if(r&&typeof r=="string")r=r.trim();else if(r&&typeof r=="object"){if(b(r)){if(r.length===0)continue;for(let n=0;n<r.length;n++){const i=r[n];for(const c in i){const a=i[c]??void 0;if(a){if(typeof a=="object")b(a)&&a.forEach((l,h)=>{if(l)for(const f in l){const d=l[f];if(d&&d.toString().length>0){const p=`[${n.toString()}].${c+`[${h.toString()}].${f}`}=${y(d)}`;e.push(s+p)}}});else if(a.toString().length>0){const l=`[${n.toString()}].${c}=${y(a.toString())}`;e.push(s+l)}}}}}else for(const n in r){const i=r[n];if(i!=null&&i!=""&&i.length!=0){const c=`.${n}=${y(i)}`;e.push(s+c)}}continue}else r&&typeof r=="function"&&(r=null);r!=null&&r!=""&&r.length!=0&&e.push(s+"="+y(r))}return e&&e.join("&")}async sign(t){const e="&key="+await this.helloIboot(),s=[];for(const a in t)s.push(a);s.sort((a,l)=>a.toLowerCase().localeCompare(l.toLowerCase()));const r=[];s.forEach(a=>{let l=t[a];typeof l=="object"&&(l=JSON.stringify(l)),r.push(a+"="+l)});const n=r.join("&"),i=k.md5((n+e).toLocaleUpperCase()),c={params:n,md5:i};return u.debug(c,"API_SIGN"),i}assemblyParameter(t,e){const s=t??{};return s.timestamp=Date.now()+"",s.echostr=_(10),s.version=this.version,s.deviceId=this.deviceId,s.webid=this.websiteId,s.apiKey=this.apiKey,e&&e.length>0&&(s.username=e),s}async assemblyHeader({urlParams:t,token:e}){const s={"Content-Type":"application/x-www-form-urlencoded","Res-Type":"json","Device-Id":this.deviceId,"User-Type":e?.utype.toString()??this.userType.toString(),"User-From":this.userFrom,"Api-Key":this.apiKey};this.lang&&this.lang.length>0&&(s.Lang=this.lang),this.websiteId&&this.websiteId.length>0&&(s["Web-Id"]=this.websiteId),this.websiteNo&&this.websiteNo.length>0&&(s["Web-No"]=this.websiteNo),e&&(e.token&&e.token.length>0&&e.username&&e.username.length>0&&(s.Authorization=e.token,s.Username=e.username),e.xcsrf&&(s[e.xcsrf.csrfHeader]=e.xcsrf.csrfToken));const r=C(t),n=await this.sign(r);return s.Sign=n,s}async helloIboot(){if(this.helloMethod.length<=0)return this.apiKey;const t=this.getApiUrl(this.helloMethod),e={"Content-Type":"application/x-www-form-urlencoded","Res-Type":"json","Api-Key":this.apiKey},s={url:t,headers:e};u.debug(s,"HELLO");const r=await fetch(t,{method:"GET",headers:e,credentials:"include",cache:"force-cache"});if(r.ok){const n=await r.json();return u.debug(n.data,"HELLO RESULT"),n.data}throw new Error("hello iBoot error!")}getApiUrl(t){return`${this.baseUrl}/${t}`}async csrf(){const t=await this.get({url:"guest/csrf"});if(t.success){const e=t.data;if(e)return e.csrfToken.toString()}}async get({url:t,data:e,token:s,cache:r="default"}){const n=this.assemblyParameter(e),i=this.convertUrlParameter(n),c=await this.assemblyHeader({urlParams:i,token:s}),a=`${this.getApiUrl(t)}?${i.toString()}`,l={url:a,headers:c};u.info(l,"GET");const h=await fetch(a,{method:"GET",headers:c,credentials:"include",cache:r});if(h.ok){const d=await h.json();return u.debug(d,"GET_RESULT"),d}const f={code:h.status,success:!1,msg:h.statusText};return u.error(f),f}async post({url:t,data:e,token:s,cache:r="default"}){e=e?{...e}:{};const n=e.buffer??void 0,i=e.boundary??void 0;n&&delete e.buffer,i&&delete e.boundary;const c=this.assemblyParameter(e),a=this.convertUrlParameter(c),l=await this.assemblyHeader({urlParams:a,token:s});n&&i&&(l["Content-Type"]=`multipart/form-data; boundary=${i}`);const h=this.getApiUrl(t),f={url:h,headers:l,params:c};u.info(f,"POST");const d=n?`${h}?${a}`:h,p=await fetch(d,{method:"POST",headers:l,body:n||a.toString(),credentials:"include",cache:r});if(p.ok){const T=await p.json();return u.debug(T,"POST_RESULT"),T}const m={code:p.status,success:!1,msg:p.statusText};return u.error(m),m}async stream({url:t,data:e,token:s}){e=e?{...e}:{};const r=e.buffer??void 0,n=e.boundary??void 0;r&&delete e.buffer,n&&delete e.boundary;const i=this.assemblyParameter(e),c=this.convertUrlParameter(i),a=await this.assemblyHeader({urlParams:c,token:s});r&&n&&(a["Content-Type"]=`multipart/form-data; boundary=${n}`),a.Connection="keep-alive",a["X-Accel-Buffering"]="no",a["Cache-Control"]="no-cache";const l=this.getApiUrl(t);try{const h=r?`${l}?${c}`:l,f={url:l,headers:a,params:i};u.info(f,"STREAM");const d=await fetch(h,{method:"POST",headers:a,body:r||c.toString(),credentials:"include"});if(!d.ok)throw new Error(`HTTP error! status: ${d.status}`);if(!d.body)throw new Error("No response body");return new Response(d.body,{headers:{"Content-Type":"text/event-stream","Cache-Control":"no-cache",Connection:"keep-alive"}})}catch(h){throw h}}getUserInfo(t){const e=t.get(g.user)?.value;return e&&e.length>0?JSON.parse(e):null}getToken(t){const e=t.get("user")?.value;let s="",r=this.userType;if(e&&e.length>0){const n=JSON.parse(e);s=n.username,r=n.userType.toString()}return{username:s,utype:r,token:t.get(g.token)?.value??""}}getTokenByCookies=t=>{const e=t.get("user")?.value;let s="",r=this.userType;if(e&&e.length>0){const n=JSON.parse(e);s=n.username,r=n.userType.toString()}return{username:s,utype:r,token:t.get(g.token)?.value??""}};setToken(t,e){const{id:s,name:r,username:n,nickname:i,sex:c,headImg:a,token:l,lastLoginTime:h,tokenExpired:f,deviceId:d,userType:p,mustChangePwd:m}=t,T={id:s,name:r,username:n,nickname:i,sex:c,headImg:a,lastLoginTime:h,tokenExpired:f,deviceId:d,userType:p,status:0,accountType:0,enabled:!1,userFrom:1,needToReview:!1,socketOnline:!1,createTime:"",mustChangePwd:m};return e.cookies.set(g.token,l,{httpOnly:!0,secure:this.isProduction,sameSite:"strict",path:"/"}),e.cookies.set(g.user,JSON.stringify(T)),e}cleanToken(t){return t.cookies.set(g.token,"",{maxAge:-1,httpOnly:!0,path:"/"}),t.cookies.set(g.user,"",{maxAge:-1,httpOnly:!0,path:"/"}),t.cookies.delete(g.token),t.cookies.delete(g.user),t}}exports.ACCOUNT_TYPE_MAP=B;exports.CURRENT_WEBSITE_KEY=O;exports.DEFAULT_LOCALE=R;exports.DEVICE_ID_KEY=w;exports.HttpClient=J;exports.USER_FORM_MAP=D;exports.USER_SEX_MAP=x;exports.USER_TYPE_MAP=A;exports.dateFormat=W;exports.get=N;exports.getHttpClientOpts=G;exports.getHttpClientOptsByCookie=Y;exports.iGet=F;exports.iPost=j;exports.iPostSuccess=K;exports.isArray=b;exports.logger=u;exports.post=P;exports.randomString=_;exports.setDefaultRequestHeader=$;exports.urlEncode=y;exports.urlParamToJson=C;
|
|
1
|
+
"use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const B=require("pino"),D=require("pino-pretty"),x=require("js-md5"),E=require("crypto-js"),j={GENERAL:0,PHONE:1,EMAIL:2},U={TYPE_MGT:0,TYPE_C:1,TYPE_B:2},F={FROM_WEB:1,FROM_PC:2,FROM_WX_MINI_PRO:3,FROM_WX_PLU:4,FROM_WX_E:5,FROM_APP:6,FROM_DEVICE:7},H={unknown:0,male:1,female:2},W=D({colorize:!0,destination:process.stdout,translateTime:"SYS:yyyy-mm-dd HH:MM:ss",ignore:"pid,hostname",messageFormat:"{msg}",singleLine:!0}),d=B({name:process.env.APP_NAME??"iBoot",level:process.env.APP_ENV==="production"?"info":"debug",nestedKey:"payload"},W),_=r=>Array.isArray(r)||typeof r=="object"&&Object.prototype.toString.call(r)==="[object Array]",w=r=>{const t=r||32,e="ABCDEFGHJKMNPQRSTWXYZabcdefhijkmnprstwxyz2345678",n=e.length;let s="";for(let o=0;o<t;o++)s+=e.charAt(Math.floor(Math.random()*n));return s},M=(r,t)=>{const e={};return r.split("&").forEach(s=>{const o=s.split("=");o[1]&&(!t||t.indexOf(o[0])==-1)&&(e[o[0]]=o[1])}),e},y=r=>!r||r==""||r.length==0?"":encodeURIComponent(r).replace(/\(/g,"%28").replace(/\)/g,"%29").replace(/\'/g,"%27").replace(/\!/g,"%21").replace(/\~/g,"%7E"),$=(r,t)=>{const e={"M+":r.getMonth()+1,"d+":r.getDate(),"h+":r.getHours(),"m+":r.getMinutes(),"s+":r.getSeconds(),"q+":Math.floor((r.getMonth()+3)/3),S:r.getMilliseconds()};if(/(y+)/.test(t)){const s=new RegExp(/(y+)/).exec(t);let o=0;s&&s.length>0&&(o=s[0].length),t=t.replace(/(y+)/,r.getFullYear()+"").substring(4-o)}for(const n in e){const s="("+n+")",o=new RegExp(s);if(o.test(t)){const i=o.exec(t);let a=0;i&&i.length>0&&(a=i[0].length),t=t.replace(o,a==1?e[n].toString():("00"+e[n]).substring(e[n].toString().length))}}return t};function T(r){for(var t=1;t<arguments.length;t++){var e=arguments[t];for(var n in e)r[n]=e[n]}return r}var K={read:function(r){return r[0]==='"'&&(r=r.slice(1,-1)),r.replace(/(%[\dA-F]{2})+/gi,decodeURIComponent)},write:function(r){return encodeURIComponent(r).replace(/%(2[346BF]|3[AC-F]|40|5[BDE]|60|7[BCD])/g,decodeURIComponent)}};function S(r,t){function e(s,o,i){if(!(typeof document>"u")){i=T({},t,i),typeof i.expires=="number"&&(i.expires=new Date(Date.now()+i.expires*864e5)),i.expires&&(i.expires=i.expires.toUTCString()),s=encodeURIComponent(s).replace(/%(2[346B]|5E|60|7C)/g,decodeURIComponent).replace(/[()]/g,escape);var a="";for(var c in i)i[c]&&(a+="; "+c,i[c]!==!0&&(a+="="+i[c].split(";")[0]));return document.cookie=s+"="+r.write(o,s)+a}}function n(s){if(!(typeof document>"u"||arguments.length&&!s)){for(var o=document.cookie?document.cookie.split("; "):[],i={},a=0;a<o.length;a++){var c=o[a].split("="),l=c.slice(1).join("=");try{var u=decodeURIComponent(c[0]);if(i[u]=r.read(l,u),s===u)break}catch{}}return s?i[s]:i}}return Object.create({set:e,get:n,remove:function(s,o){e(s,"",T({},o,{expires:-1}))},withAttributes:function(s){return S(this.converter,T({},this.attributes,s))},withConverter:function(s){return S(T({},this.converter,s),this.attributes)}},{attributes:{value:Object.freeze(t)},converter:{value:Object.freeze(r)}})}var f=S(K,{path:"/"});const N="zh-CN",O="_device_id_key",P="_current_website_key",R="Get request error!",b="Post request error!",I={"Device-Id":"Device-Id",Lang:"Lang","Website-Id":"Website-Id","Website-No":"Website-No"},p={IBOOT_DEVICE_ID:"IBOOT_DEVICE_ID",IBOOT_LOCALE:"IBOOT_LOCALE",IBOOT_WEBSITE_ID:"IBOOT_WEBSITE_ID",IBOOT_WEBSITE_NO:"IBOOT_WEBSITE_NO",token:"token",user:"user"},L=()=>{const r=localStorage.getItem(O)??"",t=localStorage.getItem(P),e=t!=null?JSON.parse(t):null,n={};return r.length>0&&(n["Device-Id"]=r),e&&(n.Lang=e.language,e.websiteId&&(n["Website-Id"]=e.websiteId),e.websiteNo&&(n["Website-No"]=e.websiteNo)),n},G=(r,t)=>{localStorage.setItem(O,r),t&&localStorage.setItem(P,JSON.stringify(t))},k=async(r,t)=>{if(t?.data){const i=new URLSearchParams(t.data);r.indexOf("?")!=-1?r+=`&${i}`:r+=`?${i}`}const e=L(),n=t?.headers;if(n)for(const i in n){const a=n[i];a&&(e[i]=a)}d.debug({url:r,headers:e},"GET");const s=await fetch(r,{method:"GET",headers:e,cache:t?.useCache?"force-cache":"default"});if(s.ok){const i=await s.json();return d.debug(i,"GET_RESULT"),i}const o={code:s.status,success:!1,msg:s.statusText};return d.error(o),o},v=async(r,t)=>{const e=t?.data??{},n=t?.headers;let s;const o=new Headers(L());if(e instanceof FormData?s=e:(s=JSON.stringify(e),o.set("Content-Type","application/json")),n)for(let c in n){const l=n[c];l&&o.set(c,l)}d.debug({url:r,headers:o},"POST");const i=await fetch(r,{method:"POST",headers:o,body:s});if(i.ok){const c=await i.json();return d.debug(c,"POST_RESULT"),c}const a={code:i.status,success:!1,msg:i.statusText};return d.error(a),a},Y=async(r,t)=>{const e=await k(r,t);if(e.success)return e.data;if(t?.showError){t.showError(e.msg??R);return}throw Error(e.msg??R)},J=async(r,t)=>{const e=await v(r,t);if(e.success)return e.data;if(t?.showError){t.showError(e.msg??b);return}throw Error(e.msg??b)},V=async(r,t)=>{const e=await v(r,t);return e.success?(t?.showSuccess&&t.showSuccess(e.msg??"SUCCESS"),!0):(t?.showError&&t.showError(e.msg??b),!1)},q=r=>{const t=r.headers.get(I["Device-Id"]),e=r.headers.get(I.Lang),n=r.headers.get(I["Website-Id"]),s=r.headers.get(I["Website-No"]),o={};return t&&t.length>0&&(o.deviceId=t),e&&e.length>0&&(o.lang=e),n&&n.length>0&&(o.websiteId=n),s&&s.length>0&&(o.websiteNo=s),o},X=()=>{const r={},t=f.get(p.IBOOT_DEVICE_ID),e=f.get(p.IBOOT_LOCALE),n=f.get(p.IBOOT_WEBSITE_ID),s=f.get(p.IBOOT_WEBSITE_NO);return t&&t.length>0&&(r.deviceId=t),e&&e.length>0&&(r.lang=e),n&&n.length>0&&(r.websiteId=n),s&&s.length>0&&(r.websiteNo=s),r},z=()=>{const r=f.get(p.user);return r&&r.length>0?JSON.parse(r):null},Q=()=>{const r=f.get(p.user);if(r&&r.length>0){const t=JSON.parse(r),e=t.username,n=t.userType.toString(),s=f.get(p.token);if(s)return{username:e,utype:n,token:s}}},Z=r=>{const{id:t,name:e,username:n,nickname:s,headImg:o,token:i,lastLoginTime:a,deviceId:c,userType:l,mustChangePwd:u}=r,g={id:t,username:n,nickname:s||e,headImg:o,lastLoginTime:a,deviceId:c,userType:l,accountType:0,needToReview:!1,socketOnline:!1,mustChangePwd:u};f.set(p.token,i,{path:"/",secure:process.env.APP_ENV==="production",httpOnly:!0,sameSite:"strict"}),f.set(p.user,JSON.stringify(g))},ee=()=>{f.remove(p.token,{path:"/",secure:process.env.APP_ENV==="production",httpOnly:!0,sameSite:"strict"}),f.remove(p.user)};class te{baseUrl;helloMethod;apiKey;userType;userFrom;deviceId;version="1";lang;websiteId;websiteNo;constructor(t){process.env.APP_ENV==="production"||d.warn("This is currently a development environment!"),this.baseUrl=process.env.APP_BASEURL,this.apiKey=process.env.APP_APIKEY,this.userFrom=process.env.APP_USERFROM??"1",this.helloMethod=t.helloURL??"",this.userType=t.userType??U.TYPE_MGT,this.deviceId=t.deviceId??w(10),this.lang=t.lang??N,this.websiteId=t.websiteId,this.websiteNo=t.websiteNo}encrypt(t){const e=E.enc.Utf8.parse(this.apiKey);return E.AES.encrypt(t,e).toString()}decrypt(t){const e=E.enc.Utf8.parse(this.apiKey);return E.AES.decrypt(t,e,{mode:E.mode.ECB}).toString(E.enc.Utf8)}convertUrlParameter(t){const e=[];for(const n in t){let s=t[n];if(s&&typeof s=="string")s=s.trim();else if(s&&typeof s=="object"){if(_(s)){if(s.length===0)continue;for(let o=0;o<s.length;o++){const i=s[o];for(const a in i){const c=i[a]??void 0;if(c){if(typeof c=="object")_(c)&&c.forEach((l,u)=>{if(l)for(const g in l){const h=l[g];if(h&&h.toString().length>0){const m=`[${o.toString()}].${a+`[${u.toString()}].${g}`}=${y(h)}`;e.push(n+m)}}});else if(c.toString().length>0){const l=`[${o.toString()}].${a}=${y(c.toString())}`;e.push(n+l)}}}}}else for(const o in s){const i=s[o];if(i!=null&&i!=""&&i.length!=0){const a=`.${o}=${y(i)}`;e.push(n+a)}}continue}else s&&typeof s=="function"&&(s=null);s!=null&&s!=""&&s.length!=0&&e.push(n+"="+y(s))}return e&&e.join("&")}async sign(t){const e="&key="+await this.helloIboot(),n=[];for(const c in t)n.push(c);n.sort((c,l)=>c.toLowerCase().localeCompare(l.toLowerCase()));const s=[];n.forEach(c=>{let l=t[c];typeof l=="object"&&(l=JSON.stringify(l)),s.push(c+"="+l)});const o=s.join("&"),i=x.md5((o+e).toLocaleUpperCase()),a={params:o,md5:i};return d.debug(a,"API_SIGN"),i}assemblyParameter(t,e){const n=t??{};return n.timestamp=Date.now()+"",n.echostr=w(10),n.version=this.version,n.deviceId=this.deviceId,n.webid=this.websiteId,n.apiKey=this.apiKey,e&&e.length>0&&(n.username=e),n}async assemblyHeader({urlParams:t,token:e}){const n={"Content-Type":"application/x-www-form-urlencoded","Res-Type":"json","Device-Id":this.deviceId,"User-Type":e?.utype.toString()??this.userType.toString(),"User-From":this.userFrom,"Api-Key":this.apiKey};this.lang&&this.lang.length>0&&(n.Lang=this.lang),this.websiteId&&this.websiteId.length>0&&(n["Web-Id"]=this.websiteId),this.websiteNo&&this.websiteNo.length>0&&(n["Web-No"]=this.websiteNo),e&&(e.token&&e.token.length>0&&e.username&&e.username.length>0&&(n.Authorization=e.token,n.Username=e.username),e.xcsrf&&(n[e.xcsrf.csrfHeader]=e.xcsrf.csrfToken));const s=M(t),o=await this.sign(s);return n.Sign=o,n}async helloIboot(){if(this.helloMethod.length<=0)return this.apiKey;const t=this.getApiUrl(this.helloMethod),e={"Content-Type":"application/x-www-form-urlencoded","Res-Type":"json","Api-Key":this.apiKey},n={url:t,headers:e};d.debug(n,"HELLO");const s=await fetch(t,{method:"GET",headers:e,credentials:"include",cache:"force-cache"});if(s.ok){const o=await s.json();return d.debug(o.data,"HELLO RESULT"),o.data}throw new Error("hello iBoot error!")}getApiUrl(t){return`${this.baseUrl}/${t}`}async csrf(){const t=await this.get({url:"guest/csrf"});if(t.success){const e=t.data;if(e)return e.csrfToken.toString()}}async get({url:t,data:e,token:n,cache:s="default"}){const o=this.assemblyParameter(e),i=this.convertUrlParameter(o),a=await this.assemblyHeader({urlParams:i,token:n}),c=`${this.getApiUrl(t)}?${i.toString()}`,l={url:c,headers:a};d.info(l,"GET");const u=await fetch(c,{method:"GET",headers:a,credentials:"include",cache:s});if(u.ok){const h=await u.json();return d.debug(h,"GET_RESULT"),h}const g={code:u.status,success:!1,msg:u.statusText};return d.error(g),g}async post({url:t,data:e,token:n,cache:s="default"}){e=e?{...e}:{};const o=e.buffer??void 0,i=e.boundary??void 0;o&&delete e.buffer,i&&delete e.boundary;const a=this.assemblyParameter(e),c=this.convertUrlParameter(a),l=await this.assemblyHeader({urlParams:c,token:n});o&&i&&(l["Content-Type"]=`multipart/form-data; boundary=${i}`);const u=this.getApiUrl(t),g={url:u,headers:l,params:a};d.info(g,"POST");const h=o?`${u}?${c}`:u,m=await fetch(h,{method:"POST",headers:l,body:o||c.toString(),credentials:"include",cache:s});if(m.ok){const A=await m.json();return d.debug(A,"POST_RESULT"),A}const C={code:m.status,success:!1,msg:m.statusText};return d.error(C),C}async stream({url:t,data:e,token:n}){e=e?{...e}:{};const s=e.buffer??void 0,o=e.boundary??void 0;s&&delete e.buffer,o&&delete e.boundary;const i=this.assemblyParameter(e),a=this.convertUrlParameter(i),c=await this.assemblyHeader({urlParams:a,token:n});s&&o&&(c["Content-Type"]=`multipart/form-data; boundary=${o}`),c.Connection="keep-alive",c["X-Accel-Buffering"]="no",c["Cache-Control"]="no-cache";const l=this.getApiUrl(t);try{const u=s?`${l}?${a}`:l,g={url:l,headers:c,params:i};d.info(g,"STREAM");const h=await fetch(u,{method:"POST",headers:c,body:s||a.toString(),credentials:"include"});if(!h.ok)throw new Error(`HTTP error! status: ${h.status}`);if(!h.body)throw new Error("No response body");return new Response(h.body,{headers:{"Content-Type":"text/event-stream","Cache-Control":"no-cache",Connection:"keep-alive"}})}catch(u){throw u}}}exports.ACCOUNT_TYPE_MAP=j;exports.CURRENT_WEBSITE_KEY=P;exports.DEFAULT_LOCALE=N;exports.DEVICE_ID_KEY=O;exports.HttpClient=te;exports.USER_FORM_MAP=F;exports.USER_SEX_MAP=H;exports.USER_TYPE_MAP=U;exports.cleanToken=ee;exports.dateFormat=$;exports.get=k;exports.getHttpOpts=q;exports.getHttpOptsByCookie=X;exports.getLoginUser=z;exports.getToken=Q;exports.iGet=Y;exports.iPost=J;exports.iPostSuccess=V;exports.isArray=_;exports.logger=d;exports.post=v;exports.randomString=w;exports.setDefaultRequestHeader=G;exports.setToken=Z;exports.urlEncode=y;exports.urlParamToJson=M;
|
|
2
2
|
//# sourceMappingURL=iboot-http-client.cjs.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"iboot-http-client.cjs.js","sources":["../src/types/user.ts","../src/logger.ts","../src/utils.ts","../src/http-client.ts"],"sourcesContent":["export const ACCOUNT_TYPE_MAP = {\r\n GENERAL : 0,\r\n PHONE : 1,\r\n EMAIL : 2\r\n}\r\n\r\nexport const USER_TYPE_MAP = {\r\n TYPE_MGT : 0,\r\n TYPE_C : 1,\r\n TYPE_B : 2\r\n}\r\n\r\nexport const USER_FORM_MAP = {\r\n FROM_WEB : 1,\r\n FROM_PC : 2,\r\n FROM_WX_MINI_PRO : 3,\r\n FROM_WX_PLU : 4,\r\n FROM_WX_E : 5,\r\n FROM_APP : 6,\r\n FROM_DEVICE : 7\r\n}\r\n\r\nexport const USER_SEX_MAP = {\r\n unknown:0,\r\n male:1,\r\n female:2,\r\n}\r\n\r\nexport type ACCOUNT_TYPE = typeof ACCOUNT_TYPE_MAP[keyof typeof ACCOUNT_TYPE_MAP];\r\nexport type USER_TYPE = typeof USER_TYPE_MAP[keyof typeof USER_TYPE_MAP];\r\nexport type USER_FORM = typeof USER_FORM_MAP[keyof typeof USER_FORM_MAP];\r\nexport type USER_SEX = typeof USER_SEX_MAP[keyof typeof USER_SEX_MAP];\r\n\r\n// User interface\r\nexport interface User {\r\n id: string;\r\n name: string;\r\n username: string;\r\n nickname: string;\r\n sex: USER_SEX;\r\n headImg: string;\r\n lastLoginTime: string;\r\n tokenExpired: string;\r\n deviceId: string;\r\n userType: USER_TYPE;\r\n status: number;\r\n accountType: number;\r\n enabled: boolean;\r\n userFrom: USER_FORM;\r\n needToReview: boolean;\r\n socketOnline: boolean;\r\n createTime: string;\r\n mustChangePwd: boolean;\r\n}","import pino from \"pino\";\r\nimport PinoPretty from \"pino-pretty\";\r\n// import koffi from 'koffi';\r\n\r\n// if (typeof window === 'undefined' && process.platform === 'win32') {\r\n// try {\r\n// const CP_UTF8 = 65001;\r\n// const kernel32 = koffi.load('Kernel32');\r\n// const setConsoleOutputCP = kernel32.func('SetConsoleOutputCP', 'bool', ['int']);\r\n// const setConsoleCP = kernel32.func('SetConsoleCP', 'bool', ['int']);\r\n// setConsoleOutputCP(CP_UTF8);\r\n// setConsoleCP(CP_UTF8);\r\n// } catch (e) {\r\n// console.log(e);\r\n// }\r\n// }\r\n\r\n// export const logger = pino({\r\n// name: process.env.APP_NAME ?? 'iBoot',\r\n// level: process.env.NODE_ENV !== 'production' ? 'debug' : 'info',\r\n// nestedKey: 'payload',\r\n// transport: {\r\n// target: 'pino-pretty',\r\n// options: {\r\n// colorize: true,\r\n// translateTime: 'SYS:yyyy-mm-dd HH:MM:ss',\r\n// ignore: 'pid,hostname',\r\n// messageFormat: '{msg}',\r\n// singleLine: true,\r\n// },\r\n// }\r\n// });\r\n\r\nconst stream = PinoPretty({\r\n colorize: true,\r\n destination: process.stdout,\r\n translateTime: 'SYS:yyyy-mm-dd HH:MM:ss',\r\n ignore: 'pid,hostname',\r\n messageFormat: '{msg}',\r\n singleLine: true,\r\n});\r\n\r\nexport const logger = pino({\r\n name: process.env.APP_NAME ?? 'iBoot',\r\n level: process.env.APP_ENV === 'production' ? 'info' : 'debug',\r\n nestedKey: 'payload'\r\n}, stream);","export const isArray = (obj: unknown) => {\r\n return (\r\n Array.isArray(obj) ||\r\n (typeof obj === \"object\" &&\r\n Object.prototype.toString.call(obj) === \"[object Array]\")\r\n );\r\n};\r\n\r\n/**\r\n * 生成指定长度的随机串\r\n * @param {Object} len\r\n */\r\nexport const randomString = (len: number) => {\r\n const l = len || 32;\r\n const $chars = \"ABCDEFGHJKMNPQRSTWXYZabcdefhijkmnprstwxyz2345678\";\r\n const maxPos = $chars.length;\r\n let pwd = \"\";\r\n for (let i = 0; i < l; i++) {\r\n pwd += $chars.charAt(Math.floor(Math.random() * maxPos));\r\n }\r\n return pwd;\r\n};\r\n\r\nexport const urlParamToJson = (urlQueryParams: string, exclude?: string[]): Record<string, string> => {\r\n const result: Record<string, string> = {};\r\n const strs = urlQueryParams.split(\"&\");\r\n strs.forEach((item) => {\r\n const arr = item.split(\"=\");\r\n if (arr[1]) {\r\n if (!exclude || exclude.indexOf(arr[0]) == -1) {\r\n result[arr[0]] = arr[1];\r\n }\r\n }\r\n });\r\n return result;\r\n};\r\n\r\nexport const urlEncode = (value: string | null | undefined): string => {\r\n if (!value || value == \"\" || value.length == 0) {\r\n return \"\";\r\n }\r\n return (\r\n encodeURIComponent(value)\r\n // .replace(/%20/g, '+')\r\n // .replace(/%2B/g, '\\\\+')\r\n .replace(/\\(/g, \"%28\")\r\n .replace(/\\)/g, \"%29\")\r\n .replace(/\\'/g, \"%27\")\r\n .replace(/\\!/g, \"%21\")\r\n .replace(/\\~/g, \"%7E\")\r\n );\r\n};\r\n\r\nexport const dateFormat = (dt: Date, fmt: string) => {\r\n const o:Record<string, number> = {\r\n \"M+\": dt.getMonth() + 1, //月份\r\n \"d+\": dt.getDate(), //日\r\n \"h+\": dt.getHours(), //小时\r\n \"m+\": dt.getMinutes(), //分\r\n \"s+\": dt.getSeconds(), //秒\r\n \"q+\": Math.floor((dt.getMonth() + 3) / 3), //季度\r\n S: dt.getMilliseconds(), //毫秒\r\n };\r\n if (/(y+)/.test(fmt)){\r\n const regx = new RegExp(/(y+)/);\r\n const value = regx.exec(fmt);\r\n let len = 0;\r\n if(value && value.length > 0){\r\n len = value[0].length;\r\n }\r\n fmt = fmt.replace(/(y+)/, dt.getFullYear() + \"\").substring(4 - len);\r\n }\r\n\r\n for (const k in o){\r\n const s:string = \"(\" + k + \")\";\r\n const regx = new RegExp(s);\r\n if(regx.test(fmt)){\r\n const value = regx.exec(fmt);\r\n let len = 0;\r\n if(value && value.length > 0){\r\n len = value[0].length;\r\n }\r\n fmt = fmt.replace(regx, len == 1 ? o[k].toString() : (\"00\" + o[k]).substring(o[k].toString().length))\r\n }\r\n }\r\n return fmt;\r\n}","import md5 from \"js-md5\";\r\nimport CryptoJS from 'crypto-js';\r\nimport { isArray, randomString, urlEncode, urlParamToJson } from \"./utils\";\r\nimport {\r\n ClientGetParams,\r\n ClientPostParams,\r\n CookieNames,\r\n CSRFToken,\r\n CurWebsite,\r\n HttpClientOpts,\r\n HttpHeaderNames,\r\n HttpHeaders,\r\n HttpToken,\r\n IbootCookieStore,\r\n IbootReadonlyCookies,\r\n IbootWritableCookie,\r\n ResultModel,\r\n ServerRequestOptions\r\n} from \"./types/http\";\r\nimport { User, USER_TYPE, USER_TYPE_MAP } from \"./types/user\";\r\nimport { logger } from \"./logger\";\r\n\r\nexport const DEFAULT_LOCALE = 'zh-CN';\r\nexport const DEVICE_ID_KEY = \"_device_id_key\";\r\nexport const CURRENT_WEBSITE_KEY = \"_current_website_key\"\r\n\r\nconst _GET_ERROR = \"Get request error!\"\r\nconst _POST_ERROR = \"Post request error!\";\r\n\r\nconst HEADER_NAMES: Record<HttpHeaderNames, HttpHeaderNames> = {\r\n \"Device-Id\": \"Device-Id\",\r\n \"Lang\": \"Lang\",\r\n \"Website-Id\": \"Website-Id\",\r\n \"Website-No\": \"Website-No\",\r\n}\r\n\r\nconst COOKIE_NAMES: Record<CookieNames, CookieNames> = {\r\n \"IBOOT_DEVICE_ID\": \"IBOOT_DEVICE_ID\",\r\n \"IBOOT_LOCALE\": \"IBOOT_LOCALE\",\r\n \"IBOOT_WEBSITE_ID\": \"IBOOT_WEBSITE_ID\",\r\n \"IBOOT_WEBSITE_NO\": \"IBOOT_WEBSITE_NO\",\r\n \"token\": \"token\",\r\n \"user\": \"user\"\r\n}\r\n\r\nconst getDefaultRequestHeader = (): HttpHeaders => {\r\n const devideId = localStorage.getItem(DEVICE_ID_KEY) ?? ''\r\n const json = localStorage.getItem(CURRENT_WEBSITE_KEY);\r\n const curWebsite: CurWebsite | null = json != null ? JSON.parse(json) : null;\r\n const header: HttpHeaders = {}\r\n if (devideId.length > 0) {\r\n header['Device-Id'] = devideId\r\n }\r\n if (curWebsite) {\r\n header['Lang'] = curWebsite.language;\r\n if (curWebsite.websiteId) {\r\n header['Website-Id'] = curWebsite.websiteId;\r\n }\r\n if (curWebsite.websiteNo) {\r\n header['Website-No'] = curWebsite.websiteNo\r\n }\r\n }\r\n return header;\r\n};\r\n\r\n\r\nexport const setDefaultRequestHeader = (deviceId: string, website?: CurWebsite): void => {\r\n localStorage.setItem(DEVICE_ID_KEY, deviceId);\r\n if (website) {\r\n localStorage.setItem(CURRENT_WEBSITE_KEY, JSON.stringify(website));\r\n }\r\n}\r\n\r\nexport const get = async <T>(url: string, opts?: ClientGetParams): Promise<ResultModel<T>> => {\r\n if (opts?.data) {\r\n const params = new URLSearchParams(opts.data);\r\n if (url.indexOf(\"?\") != -1) {\r\n url += `&${params}`\r\n } else {\r\n url += `?${params}`\r\n }\r\n }\r\n const headers: Record<string, string> = getDefaultRequestHeader();\r\n const heads = opts?.headers;\r\n if (heads) {\r\n for (const field in heads) {\r\n const value = heads[field];\r\n if (value) {\r\n headers[field] = value;\r\n }\r\n }\r\n }\r\n logger.debug({\r\n \"url\": url,\r\n \"headers\": headers,\r\n }, \"GET\")\r\n const response = await fetch(url, {\r\n method: 'GET',\r\n headers: headers,\r\n cache: opts?.useCache ? 'force-cache' : 'default'\r\n });\r\n\r\n if (response.ok) {\r\n const result = await response.json();\r\n logger.debug(result, \"GET_RESULT\")\r\n return result\r\n }\r\n const msg = {\r\n code: response.status,\r\n success: false,\r\n msg: response.statusText\r\n }\r\n logger.error(msg);\r\n return msg;\r\n}\r\n\r\nexport const post = async <T>(url: string, opts?: ClientPostParams): Promise<ResultModel<T>> => {\r\n const data = opts?.data ?? {};\r\n const heads = opts?.headers;\r\n let body: string | FormData;\r\n const proxyHeaders = new Headers(getDefaultRequestHeader());\r\n if (!(data instanceof FormData)) {\r\n body = JSON.stringify(data);\r\n proxyHeaders.set('Content-Type', 'application/json');\r\n } else {\r\n body = data;\r\n }\r\n if (heads) {\r\n for (let field in heads) {\r\n const value = heads[field];\r\n if (value) {\r\n proxyHeaders.set(field, value);\r\n }\r\n }\r\n }\r\n logger.debug({\r\n \"url\": url,\r\n \"headers\": proxyHeaders,\r\n }, \"POST\")\r\n const response = await fetch(url, {\r\n method: 'POST',\r\n headers: proxyHeaders,\r\n body: body,\r\n });\r\n if (response.ok) {\r\n const result = await response.json();\r\n logger.debug(result, \"POST_RESULT\")\r\n return result\r\n }\r\n const msg = {\r\n code: response.status,\r\n success: false,\r\n msg: response.statusText,\r\n }\r\n logger.error(msg);\r\n return msg;\r\n}\r\n\r\nexport const iGet = async <T>(url: string, opts?: ClientGetParams): Promise<T | undefined> => {\r\n const res = await get<T>(url, opts);\r\n if (res.success) {\r\n return res.data as T;\r\n }\r\n if (opts?.showError) {\r\n opts.showError(res.msg ?? _GET_ERROR);\r\n return;\r\n }\r\n throw Error(res.msg ?? _GET_ERROR)\r\n}\r\n\r\nexport const iPost = async <T>(url: string, opts?: ClientPostParams): Promise<T | undefined> => {\r\n const res = await post<T>(url, opts);\r\n if (res.success) {\r\n return res.data;\r\n }\r\n if (opts?.showError) {\r\n opts.showError(res.msg ?? _POST_ERROR);\r\n return;\r\n }\r\n throw Error(res.msg ?? _POST_ERROR)\r\n}\r\n\r\nexport const iPostSuccess = async (url: string, opts?: ClientPostParams): Promise<boolean> => {\r\n const res = await post(url, opts);\r\n if (res.success) {\r\n if (opts?.showSuccess) {\r\n opts.showSuccess(res.msg ?? 'SUCCESS')\r\n }\r\n return true;\r\n }\r\n if (opts?.showError) {\r\n opts.showError(res.msg ?? _POST_ERROR);\r\n }\r\n return false;\r\n}\r\n\r\n\r\nexport const getHttpClientOpts = (request: Request): HttpClientOpts => {\r\n const deviceId = request.headers.get(HEADER_NAMES[\"Device-Id\"]);\r\n const lang = request.headers.get(HEADER_NAMES.Lang);\r\n const websiteId = request.headers.get(HEADER_NAMES[\"Website-Id\"]);\r\n const websiteNo = request.headers.get(HEADER_NAMES[\"Website-No\"]);\r\n const result: HttpClientOpts = {}\r\n if (deviceId && deviceId.length > 0) {\r\n result.deviceId = deviceId;\r\n }\r\n if (lang && lang.length > 0) {\r\n result.lang = lang\r\n }\r\n if (websiteId && websiteId.length > 0) {\r\n result.websiteId = websiteId\r\n }\r\n if (websiteNo && websiteNo.length > 0) {\r\n result.websiteNo = websiteNo\r\n }\r\n return result;\r\n}\r\n\r\nexport const getHttpClientOptsByCookie = (cookie: IbootReadonlyCookies): HttpClientOpts => {\r\n const result: HttpClientOpts = {}\r\n const deviceId = cookie.get(COOKIE_NAMES.IBOOT_DEVICE_ID)?.value;\r\n const lang = cookie.get(COOKIE_NAMES.IBOOT_LOCALE)?.value;\r\n const websiteId = cookie.get(COOKIE_NAMES.IBOOT_WEBSITE_ID)?.value;\r\n const websiteNo = cookie.get(COOKIE_NAMES.IBOOT_WEBSITE_NO)?.value;\r\n if (deviceId && deviceId.length > 0) {\r\n result.deviceId = deviceId;\r\n }\r\n if (lang && lang.length > 0) {\r\n result.lang = lang\r\n }\r\n if (websiteId && websiteId.length > 0) {\r\n result.websiteId = websiteId\r\n }\r\n if (websiteNo && websiteNo.length > 0) {\r\n result.websiteNo = websiteNo\r\n }\r\n return result;\r\n}\r\n\r\nexport class HttpClient {\r\n private readonly baseUrl: string;\r\n private readonly helloMethod: string;\r\n private readonly apiKey: string;\r\n private readonly userType: USER_TYPE;\r\n private readonly userFrom: string;\r\n private readonly deviceId: string;\r\n private readonly version: string = '1';\r\n private readonly lang: string;\r\n private readonly websiteId?: string;\r\n private readonly websiteNo?: string;\r\n private readonly isProduction: boolean;\r\n constructor(opts: Readonly<HttpClientOpts>) {\r\n this.isProduction = process.env.APP_ENV === 'production';\r\n if (!this.isProduction) {\r\n logger.warn(\"This is currently a development environment!\")\r\n }\r\n this.baseUrl = process.env.APP_BASEURL!;\r\n this.apiKey = process.env.APP_APIKEY!;\r\n this.userFrom = process.env.APP_USERFROM ?? '1';\r\n\r\n this.helloMethod = opts.helloURL ?? ''\r\n this.userType = opts.userType ?? USER_TYPE_MAP.TYPE_MGT;\r\n this.deviceId = opts.deviceId ?? randomString(10);\r\n this.lang = opts.lang ?? DEFAULT_LOCALE;\r\n this.websiteId = opts.websiteId\r\n this.websiteNo = opts.websiteNo\r\n }\r\n\r\n encrypt(data: string) {\r\n const key = CryptoJS.enc.Utf8.parse(this.apiKey);\r\n return CryptoJS.AES.encrypt(data, key).toString();\r\n }\r\n\r\n decrypt(data: string) {\r\n const key = CryptoJS.enc.Utf8.parse(this.apiKey);\r\n return CryptoJS.AES.decrypt(data, key, {\r\n mode: CryptoJS.mode.ECB,\r\n }).toString(CryptoJS.enc.Utf8);\r\n }\r\n\r\n getDeviceId() {\r\n return this.deviceId;\r\n }\r\n\r\n //eslint-disable-next-line @typescript-eslint/no-explicit-any\r\n private convertUrlParameter(data: Record<string, any>): string {\r\n const p: string[] = [];\r\n for (const o in data) {\r\n let v = data[o];\r\n if (v && typeof v === \"string\") {\r\n v = v.trim();\r\n } else if (v && typeof v === \"object\") {\r\n if (isArray(v)) {\r\n if (v.length === 0) {\r\n continue;\r\n }\r\n for (let i = 0; i < v.length; i++) {\r\n const arr_item = v[i];\r\n for (const item_field in arr_item) {\r\n const item_field_value = arr_item[item_field] ?? undefined;\r\n if (item_field_value) {\r\n if (typeof item_field_value === \"object\") {\r\n if (isArray(item_field_value)) {\r\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\r\n item_field_value.forEach((item1: any, index: number) => {\r\n if (item1) {\r\n for (const f in item1) {\r\n const v = item1[f];\r\n if (v && v.toString().length > 0) {\r\n const s = `[${i.toString()}].${item_field + `[${index.toString()}].${f}`}=${urlEncode(v)}`;\r\n p.push(o + s);\r\n }\r\n }\r\n }\r\n });\r\n }\r\n } else if (item_field_value.toString().length > 0) {\r\n const s = `[${i.toString()}].${item_field}=${urlEncode(item_field_value.toString())}`;\r\n p.push(o + s);\r\n }\r\n }\r\n }\r\n }\r\n } else {\r\n for (const obj in v) {\r\n const oValue = v[obj];\r\n if (oValue != null && oValue != \"\" && oValue.length != 0) {\r\n const s = `.${obj}=${urlEncode(oValue)}`;\r\n p.push(o + s);\r\n }\r\n }\r\n }\r\n continue;\r\n } else if (v && typeof v === \"function\") {\r\n v = null;\r\n }\r\n\r\n if (v != null && v != \"\" && v.length != 0) {\r\n p.push(o + \"=\" + urlEncode(v));\r\n }\r\n }\r\n return (p && p.join(\"&\"));\r\n }\r\n\r\n private async sign(data: Record<string, object | string>): Promise<string> {\r\n //由后台分配\r\n const _apiKey = \"&key=\" + (await this.helloIboot());\r\n const arr: string[] = [];\r\n for (const o in data) {\r\n arr.push(o);\r\n }\r\n arr.sort((a: string, b: string) => {\r\n return a.toLowerCase().localeCompare(b.toLowerCase());\r\n });\r\n const res: string[] = [];\r\n arr.forEach((v: string) => {\r\n let value: object | string = data[v];\r\n if (typeof value === \"object\") {\r\n value = JSON.stringify(value);\r\n }\r\n res.push(v + \"=\" + value);\r\n });\r\n const paramsStr: string = res.join(\"&\");\r\n const str = md5.md5((paramsStr + _apiKey).toLocaleUpperCase());\r\n const logInfo = {\r\n params: paramsStr,\r\n md5: str\r\n }\r\n logger.debug(logInfo, \"API_SIGN\")\r\n return str;\r\n }\r\n\r\n //eslint-disable-next-line @typescript-eslint/no-explicit-any\r\n private assemblyParameter(data?: Record<string, any>, username?: string): Record<string, any> {\r\n const params = data ?? {};\r\n params['timestamp'] = Date.now() + '';\r\n params['echostr'] = randomString(10);\r\n params['version'] = this.version;\r\n params['deviceId'] = this.deviceId;\r\n params['webid'] = this.websiteId;\r\n params['apiKey'] = this.apiKey;\r\n if (username && username.length > 0) {\r\n params['username'] = username;\r\n }\r\n return params;\r\n }\r\n\r\n private async assemblyHeader({ urlParams, token }: Readonly<{ urlParams: string, token?: Readonly<HttpToken> }>) {\r\n const h: Record<string, string> = {\r\n 'Content-Type': 'application/x-www-form-urlencoded',\r\n 'Res-Type': 'json',\r\n 'Device-Id': this.deviceId,\r\n 'User-Type': token?.utype.toString() ?? this.userType.toString(),\r\n 'User-From': this.userFrom,\r\n 'Api-Key': this.apiKey\r\n }\r\n if (this.lang && this.lang.length > 0) {\r\n h['Lang'] = this.lang\r\n }\r\n if (this.websiteId && this.websiteId.length > 0) {\r\n h['Web-Id'] = this.websiteId\r\n }\r\n if (this.websiteNo && this.websiteNo.length > 0) {\r\n h['Web-No'] = this.websiteNo\r\n }\r\n if (token) {\r\n if (token.token && token.token.length > 0 && token.username && token.username.length > 0) {\r\n h['Authorization'] = token.token;\r\n h['Username'] = token.username;\r\n }\r\n if (token.xcsrf) {\r\n h[token.xcsrf.csrfHeader] = token.xcsrf.csrfToken;\r\n }\r\n }\r\n const data = urlParamToJson(urlParams);\r\n const sign = await this.sign(data);\r\n h['Sign'] = sign\r\n return h;\r\n }\r\n\r\n private async helloIboot(): Promise<string> {\r\n if (this.helloMethod.length <= 0) {\r\n return this.apiKey;\r\n }\r\n const api = this.getApiUrl(this.helloMethod);\r\n const h = {\r\n 'Content-Type': 'application/x-www-form-urlencoded',\r\n 'Res-Type': 'json',\r\n 'Api-Key': this.apiKey\r\n }\r\n const logInfo = {\r\n \"url\": api,\r\n \"headers\": h\r\n }\r\n logger.debug(logInfo, \"HELLO\");\r\n const res = await fetch(api, {\r\n method: 'GET',\r\n headers: h,\r\n credentials: 'include',\r\n cache: 'force-cache'\r\n });\r\n if (res.ok) {\r\n const result = await res.json();\r\n logger.debug(result.data, 'HELLO RESULT')\r\n return result.data;\r\n }\r\n throw new Error('hello iBoot error!')\r\n }\r\n\r\n private getApiUrl(url: string): string {\r\n return `${this.baseUrl}/${url}`;\r\n }\r\n\r\n async csrf(): Promise<string | undefined> {\r\n const res = await this.get<CSRFToken>({\r\n url: 'guest/csrf'\r\n });\r\n if (res.success) {\r\n const data = res.data;\r\n if (data) {\r\n return data[\"csrfToken\"].toString();\r\n }\r\n }\r\n return undefined;\r\n }\r\n\r\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\r\n async get<T>({ url, data, token, cache = 'default' }: Readonly<ServerRequestOptions>): Promise<ResultModel<T>> {\r\n const params = this.assemblyParameter(data);\r\n const urlParams = this.convertUrlParameter(params);\r\n const headers = await this.assemblyHeader({ \"urlParams\": urlParams, \"token\": token });\r\n const api = `${this.getApiUrl(url)}?${urlParams.toString()}`\r\n const logInfo = {\r\n \"url\": api,\r\n \"headers\": headers\r\n }\r\n logger.info(logInfo, \"GET\");\r\n const res = await fetch(api, {\r\n \"method\": 'GET',\r\n \"headers\": headers,\r\n \"credentials\": 'include',\r\n \"cache\": cache\r\n });\r\n if (res.ok) {\r\n const result = await res.json();\r\n logger.debug(result, \"GET_RESULT\")\r\n return result;\r\n }\r\n const msg = {\r\n code: res.status,\r\n success: false,\r\n msg: res.statusText\r\n }\r\n logger.error(msg);\r\n return msg;\r\n }\r\n\r\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\r\n async post<T>({ url, data, token, cache = \"default\" }: Readonly<ServerRequestOptions>): Promise<ResultModel<T>> {\r\n data = data ? { ...data } : {}\r\n const buffer = data.buffer ?? undefined;\r\n const boundary = data.boundary ?? undefined;\r\n if (buffer) {\r\n delete data.buffer;\r\n }\r\n if (boundary) {\r\n delete data.boundary;\r\n }\r\n const params = this.assemblyParameter(data);\r\n const urlParams = this.convertUrlParameter(params);\r\n const headers = await this.assemblyHeader({ \"urlParams\": urlParams, \"token\": token });\r\n if (buffer && boundary) {\r\n headers['Content-Type'] = `multipart/form-data; boundary=${boundary}`;\r\n }\r\n const api = this.getApiUrl(url);\r\n const logInfo = {\r\n \"url\": api,\r\n \"headers\": headers,\r\n \"params\": params\r\n }\r\n logger.info(logInfo, \"POST\");\r\n const urls = buffer ? `${api}?${urlParams}` : api;\r\n const res = await fetch(urls, {\r\n \"method\": 'POST',\r\n \"headers\": headers,\r\n \"body\": buffer ? buffer : urlParams.toString(),\r\n \"credentials\": 'include',\r\n \"cache\": cache\r\n });\r\n if (res.ok) {\r\n const result = await res.json();\r\n logger.debug(result, \"POST_RESULT\")\r\n return result;\r\n }\r\n const msg = {\r\n code: res.status,\r\n success: false,\r\n msg: res.statusText\r\n }\r\n logger.error(msg);\r\n return msg;\r\n }\r\n\r\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\r\n async stream({ url, data, token }: Readonly<Omit<ServerRequestOptions, 'cache'>>): Promise<Response> {\r\n data = data ? { ...data } : {}\r\n const buffer = data.buffer ?? undefined;\r\n const boundary = data.boundary ?? undefined;\r\n if (buffer) {\r\n delete data.buffer;\r\n }\r\n if (boundary) {\r\n delete data.boundary;\r\n }\r\n const params = this.assemblyParameter(data);\r\n const urlParams = this.convertUrlParameter(params); //new URLSearchParams(params);\r\n\r\n const headers = await this.assemblyHeader({ \"urlParams\": urlParams, \"token\": token });\r\n if (buffer && boundary) {\r\n headers['Content-Type'] = `multipart/form-data; boundary=${boundary}`;\r\n }\r\n headers['Connection'] = 'keep-alive';\r\n headers['X-Accel-Buffering'] = 'no'; //防止 Nginx 缓存\r\n headers['Cache-Control'] = 'no-cache';\r\n const api = this.getApiUrl(url)\r\n try {\r\n const url = buffer ? `${api}?${urlParams}` : api;\r\n const logInfo = {\r\n \"url\": api,\r\n \"headers\": headers,\r\n \"params\": params\r\n }\r\n logger.info(logInfo, \"STREAM\");\r\n const res = await fetch(url, {\r\n method: 'POST',\r\n headers: headers,\r\n body: buffer ? buffer : urlParams.toString(),\r\n credentials: 'include'\r\n });\r\n\r\n if (!res.ok) {\r\n throw new Error(`HTTP error! status: ${res.status}`);\r\n }\r\n if (!res.body) {\r\n throw new Error('No response body');\r\n }\r\n return new Response(res.body, {\r\n headers: {\r\n 'Content-Type': 'text/event-stream',\r\n 'Cache-Control': 'no-cache',\r\n 'Connection': 'keep-alive'\r\n }\r\n });\r\n } catch (e) {\r\n throw e;\r\n }\r\n }\r\n\r\n getUserInfo(cookies: IbootCookieStore): User | null {\r\n const userjson = cookies.get(COOKIE_NAMES.user)?.value;\r\n if (userjson && userjson.length > 0) {\r\n return JSON.parse(userjson);\r\n }\r\n return null;\r\n }\r\n\r\n getToken(cookies: IbootCookieStore): HttpToken {\r\n const userjson = cookies.get('user')?.value;\r\n let username = '';\r\n let utype = this.userType;\r\n if (userjson && userjson.length > 0) {\r\n const user = JSON.parse(userjson);\r\n username = user.username;\r\n utype = user.userType.toString();\r\n }\r\n return {\r\n username: username,\r\n utype: utype,\r\n token: cookies.get(COOKIE_NAMES.token)?.value ?? ''\r\n }\r\n }\r\n\r\n getTokenByCookies = (cookies: IbootReadonlyCookies): HttpToken => {\r\n const userjson = cookies.get('user')?.value;\r\n let username = '';\r\n let utype = this.userType;\r\n if (userjson && userjson.length > 0) {\r\n const user = JSON.parse(userjson);\r\n username = user.username;\r\n utype = user.userType.toString();\r\n }\r\n return {\r\n username: username,\r\n utype: utype,\r\n token: cookies.get(COOKIE_NAMES.token)?.value ?? ''\r\n }\r\n }\r\n\r\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\r\n setToken(data: User & { token: string, extFields: Record<string, any> }, response: { cookies: IbootWritableCookie }) {\r\n const { id, name, username, nickname, sex, headImg, token, lastLoginTime, tokenExpired, deviceId, userType, mustChangePwd } = data;\r\n const loginUser: User = {\r\n id: id,\r\n name: name,\r\n username: username,\r\n nickname: nickname,\r\n sex: sex,\r\n headImg: headImg,\r\n lastLoginTime: lastLoginTime,\r\n tokenExpired: tokenExpired,\r\n deviceId: deviceId,\r\n userType: userType,\r\n status: 0,\r\n accountType: 0,\r\n enabled: false,\r\n userFrom: 1,\r\n needToReview: false,\r\n socketOnline: false,\r\n createTime: \"\",\r\n mustChangePwd: mustChangePwd\r\n };\r\n response.cookies.set(COOKIE_NAMES.token, token, {\r\n httpOnly: true,\r\n secure: this.isProduction, // 在生产环境中启用Secure\r\n sameSite: 'strict', // 防止 CSRF 攻击\r\n path: '/', // Cookie 的路径\r\n });\r\n response.cookies.set(COOKIE_NAMES.user, JSON.stringify(loginUser));\r\n return response;\r\n }\r\n\r\n cleanToken(response: { cookies: IbootWritableCookie }) {\r\n response.cookies.set(COOKIE_NAMES.token, '', {\r\n maxAge: -1, //设置为 -1表示立即过期\r\n httpOnly: true,\r\n path: '/',\r\n });\r\n response.cookies.set(COOKIE_NAMES.user, '', {\r\n maxAge: -1, //设置为 -1表示立即过期\r\n httpOnly: true,\r\n path: '/',\r\n });\r\n response.cookies.delete(COOKIE_NAMES.token);\r\n response.cookies.delete(COOKIE_NAMES.user)\r\n return response;\r\n }\r\n}"],"names":["ACCOUNT_TYPE_MAP","USER_TYPE_MAP","USER_FORM_MAP","USER_SEX_MAP","stream","PinoPretty","logger","pino","isArray","obj","randomString","len","l","$chars","maxPos","pwd","i","urlParamToJson","urlQueryParams","exclude","result","item","arr","urlEncode","value","dateFormat","dt","fmt","o","k","s","regx","DEFAULT_LOCALE","DEVICE_ID_KEY","CURRENT_WEBSITE_KEY","_GET_ERROR","_POST_ERROR","HEADER_NAMES","COOKIE_NAMES","getDefaultRequestHeader","devideId","json","curWebsite","header","setDefaultRequestHeader","deviceId","website","get","url","opts","params","headers","heads","field","response","msg","post","data","body","proxyHeaders","iGet","res","iPost","iPostSuccess","getHttpClientOpts","request","lang","websiteId","websiteNo","getHttpClientOptsByCookie","cookie","HttpClient","key","CryptoJS","p","v","arr_item","item_field","item_field_value","item1","index","oValue","_apiKey","b","paramsStr","str","md5","logInfo","username","urlParams","token","h","sign","api","cache","buffer","boundary","urls","e","cookies","userjson","utype","user","id","name","nickname","sex","headImg","lastLoginTime","tokenExpired","userType","mustChangePwd","loginUser"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"iboot-http-client.cjs.js","sources":["../src/types/user.ts","../src/logger.ts","../src/utils.ts","../node_modules/js-cookie/dist/js.cookie.mjs","../src/http-client.ts"],"sourcesContent":["export const ACCOUNT_TYPE_MAP = {\r\n GENERAL : 0,\r\n PHONE : 1,\r\n EMAIL : 2\r\n}\r\n\r\nexport const USER_TYPE_MAP = {\r\n TYPE_MGT : 0,\r\n TYPE_C : 1,\r\n TYPE_B : 2\r\n}\r\n\r\nexport const USER_FORM_MAP = {\r\n FROM_WEB : 1,\r\n FROM_PC : 2,\r\n FROM_WX_MINI_PRO : 3,\r\n FROM_WX_PLU : 4,\r\n FROM_WX_E : 5,\r\n FROM_APP : 6,\r\n FROM_DEVICE : 7\r\n}\r\n\r\nexport const USER_SEX_MAP = {\r\n unknown:0,\r\n male:1,\r\n female:2,\r\n}\r\n\r\nexport type ACCOUNT_TYPE = typeof ACCOUNT_TYPE_MAP[keyof typeof ACCOUNT_TYPE_MAP];\r\nexport type USER_TYPE = typeof USER_TYPE_MAP[keyof typeof USER_TYPE_MAP];\r\nexport type USER_FORM = typeof USER_FORM_MAP[keyof typeof USER_FORM_MAP];\r\nexport type USER_SEX = typeof USER_SEX_MAP[keyof typeof USER_SEX_MAP];\r\n\r\n// User interface\r\nexport interface User {\r\n id: string;\r\n name?: string;\r\n username: string;\r\n nickname?: string;\r\n sex?: USER_SEX;\r\n headImg: string;\r\n lastLoginTime: string;\r\n tokenExpired?: string;\r\n deviceId: string;\r\n userType: USER_TYPE;\r\n status?: number;\r\n accountType: number;\r\n enabled?: boolean;\r\n userFrom?: USER_FORM;\r\n needToReview: boolean;\r\n socketOnline: boolean;\r\n createTime?: string;\r\n mustChangePwd: boolean;\r\n}","import pino from \"pino\";\r\nimport PinoPretty from \"pino-pretty\";\r\n// import koffi from 'koffi';\r\n\r\n// if (typeof window === 'undefined' && process.platform === 'win32') {\r\n// try {\r\n// const CP_UTF8 = 65001;\r\n// const kernel32 = koffi.load('Kernel32');\r\n// const setConsoleOutputCP = kernel32.func('SetConsoleOutputCP', 'bool', ['int']);\r\n// const setConsoleCP = kernel32.func('SetConsoleCP', 'bool', ['int']);\r\n// setConsoleOutputCP(CP_UTF8);\r\n// setConsoleCP(CP_UTF8);\r\n// } catch (e) {\r\n// console.log(e);\r\n// }\r\n// }\r\n\r\n// export const logger = pino({\r\n// name: process.env.APP_NAME ?? 'iBoot',\r\n// level: process.env.NODE_ENV !== 'production' ? 'debug' : 'info',\r\n// nestedKey: 'payload',\r\n// transport: {\r\n// target: 'pino-pretty',\r\n// options: {\r\n// colorize: true,\r\n// translateTime: 'SYS:yyyy-mm-dd HH:MM:ss',\r\n// ignore: 'pid,hostname',\r\n// messageFormat: '{msg}',\r\n// singleLine: true,\r\n// },\r\n// }\r\n// });\r\n\r\nconst stream = PinoPretty({\r\n colorize: true,\r\n destination: process.stdout,\r\n translateTime: 'SYS:yyyy-mm-dd HH:MM:ss',\r\n ignore: 'pid,hostname',\r\n messageFormat: '{msg}',\r\n singleLine: true,\r\n});\r\n\r\nexport const logger = pino({\r\n name: process.env.APP_NAME ?? 'iBoot',\r\n level: process.env.APP_ENV === 'production' ? 'info' : 'debug',\r\n nestedKey: 'payload'\r\n}, stream);","export const isArray = (obj: unknown) => {\r\n return (\r\n Array.isArray(obj) ||\r\n (typeof obj === \"object\" &&\r\n Object.prototype.toString.call(obj) === \"[object Array]\")\r\n );\r\n};\r\n\r\n/**\r\n * 生成指定长度的随机串\r\n * @param {Object} len\r\n */\r\nexport const randomString = (len: number) => {\r\n const l = len || 32;\r\n const $chars = \"ABCDEFGHJKMNPQRSTWXYZabcdefhijkmnprstwxyz2345678\";\r\n const maxPos = $chars.length;\r\n let pwd = \"\";\r\n for (let i = 0; i < l; i++) {\r\n pwd += $chars.charAt(Math.floor(Math.random() * maxPos));\r\n }\r\n return pwd;\r\n};\r\n\r\nexport const urlParamToJson = (urlQueryParams: string, exclude?: string[]): Record<string, string> => {\r\n const result: Record<string, string> = {};\r\n const strs = urlQueryParams.split(\"&\");\r\n strs.forEach((item) => {\r\n const arr = item.split(\"=\");\r\n if (arr[1]) {\r\n if (!exclude || exclude.indexOf(arr[0]) == -1) {\r\n result[arr[0]] = arr[1];\r\n }\r\n }\r\n });\r\n return result;\r\n};\r\n\r\nexport const urlEncode = (value: string | null | undefined): string => {\r\n if (!value || value == \"\" || value.length == 0) {\r\n return \"\";\r\n }\r\n return (\r\n encodeURIComponent(value)\r\n // .replace(/%20/g, '+')\r\n // .replace(/%2B/g, '\\\\+')\r\n .replace(/\\(/g, \"%28\")\r\n .replace(/\\)/g, \"%29\")\r\n .replace(/\\'/g, \"%27\")\r\n .replace(/\\!/g, \"%21\")\r\n .replace(/\\~/g, \"%7E\")\r\n );\r\n};\r\n\r\nexport const dateFormat = (dt: Date, fmt: string) => {\r\n const o:Record<string, number> = {\r\n \"M+\": dt.getMonth() + 1, //月份\r\n \"d+\": dt.getDate(), //日\r\n \"h+\": dt.getHours(), //小时\r\n \"m+\": dt.getMinutes(), //分\r\n \"s+\": dt.getSeconds(), //秒\r\n \"q+\": Math.floor((dt.getMonth() + 3) / 3), //季度\r\n S: dt.getMilliseconds(), //毫秒\r\n };\r\n if (/(y+)/.test(fmt)){\r\n const regx = new RegExp(/(y+)/);\r\n const value = regx.exec(fmt);\r\n let len = 0;\r\n if(value && value.length > 0){\r\n len = value[0].length;\r\n }\r\n fmt = fmt.replace(/(y+)/, dt.getFullYear() + \"\").substring(4 - len);\r\n }\r\n\r\n for (const k in o){\r\n const s:string = \"(\" + k + \")\";\r\n const regx = new RegExp(s);\r\n if(regx.test(fmt)){\r\n const value = regx.exec(fmt);\r\n let len = 0;\r\n if(value && value.length > 0){\r\n len = value[0].length;\r\n }\r\n fmt = fmt.replace(regx, len == 1 ? o[k].toString() : (\"00\" + o[k]).substring(o[k].toString().length))\r\n }\r\n }\r\n return fmt;\r\n}","/*! js-cookie v3.0.5 | MIT */\n/* eslint-disable no-var */\nfunction assign (target) {\n for (var i = 1; i < arguments.length; i++) {\n var source = arguments[i];\n for (var key in source) {\n target[key] = source[key];\n }\n }\n return target\n}\n/* eslint-enable no-var */\n\n/* eslint-disable no-var */\nvar defaultConverter = {\n read: function (value) {\n if (value[0] === '\"') {\n value = value.slice(1, -1);\n }\n return value.replace(/(%[\\dA-F]{2})+/gi, decodeURIComponent)\n },\n write: function (value) {\n return encodeURIComponent(value).replace(\n /%(2[346BF]|3[AC-F]|40|5[BDE]|60|7[BCD])/g,\n decodeURIComponent\n )\n }\n};\n/* eslint-enable no-var */\n\n/* eslint-disable no-var */\n\nfunction init (converter, defaultAttributes) {\n function set (name, value, attributes) {\n if (typeof document === 'undefined') {\n return\n }\n\n attributes = assign({}, defaultAttributes, attributes);\n\n if (typeof attributes.expires === 'number') {\n attributes.expires = new Date(Date.now() + attributes.expires * 864e5);\n }\n if (attributes.expires) {\n attributes.expires = attributes.expires.toUTCString();\n }\n\n name = encodeURIComponent(name)\n .replace(/%(2[346B]|5E|60|7C)/g, decodeURIComponent)\n .replace(/[()]/g, escape);\n\n var stringifiedAttributes = '';\n for (var attributeName in attributes) {\n if (!attributes[attributeName]) {\n continue\n }\n\n stringifiedAttributes += '; ' + attributeName;\n\n if (attributes[attributeName] === true) {\n continue\n }\n\n // Considers RFC 6265 section 5.2:\n // ...\n // 3. If the remaining unparsed-attributes contains a %x3B (\";\")\n // character:\n // Consume the characters of the unparsed-attributes up to,\n // not including, the first %x3B (\";\") character.\n // ...\n stringifiedAttributes += '=' + attributes[attributeName].split(';')[0];\n }\n\n return (document.cookie =\n name + '=' + converter.write(value, name) + stringifiedAttributes)\n }\n\n function get (name) {\n if (typeof document === 'undefined' || (arguments.length && !name)) {\n return\n }\n\n // To prevent the for loop in the first place assign an empty array\n // in case there are no cookies at all.\n var cookies = document.cookie ? document.cookie.split('; ') : [];\n var jar = {};\n for (var i = 0; i < cookies.length; i++) {\n var parts = cookies[i].split('=');\n var value = parts.slice(1).join('=');\n\n try {\n var found = decodeURIComponent(parts[0]);\n jar[found] = converter.read(value, found);\n\n if (name === found) {\n break\n }\n } catch (e) {}\n }\n\n return name ? jar[name] : jar\n }\n\n return Object.create(\n {\n set,\n get,\n remove: function (name, attributes) {\n set(\n name,\n '',\n assign({}, attributes, {\n expires: -1\n })\n );\n },\n withAttributes: function (attributes) {\n return init(this.converter, assign({}, this.attributes, attributes))\n },\n withConverter: function (converter) {\n return init(assign({}, this.converter, converter), this.attributes)\n }\n },\n {\n attributes: { value: Object.freeze(defaultAttributes) },\n converter: { value: Object.freeze(converter) }\n }\n )\n}\n\nvar api = init(defaultConverter, { path: '/' });\n/* eslint-enable no-var */\n\nexport { api as default };\n","import md5 from \"js-md5\";\r\nimport CryptoJS from 'crypto-js';\r\nimport { isArray, randomString, urlEncode, urlParamToJson } from \"./utils\";\r\nimport {\r\n ClientGetParams,\r\n ClientPostParams,\r\n CookieNames,\r\n CSRFToken,\r\n CurWebsite,\r\n HttpClientOpts,\r\n HttpHeaderNames,\r\n HttpHeaders,\r\n HttpToken,\r\n ResultModel,\r\n ServerRequestOptions\r\n} from \"./types/http\";\r\nimport { User, USER_TYPE, USER_TYPE_MAP } from \"./types/user\";\r\nimport { logger } from \"./logger\";\r\nimport Cookies from 'js-cookie';\r\n\r\nexport const DEFAULT_LOCALE = 'zh-CN';\r\nexport const DEVICE_ID_KEY = \"_device_id_key\";\r\nexport const CURRENT_WEBSITE_KEY = \"_current_website_key\"\r\n\r\nconst _GET_ERROR = \"Get request error!\"\r\nconst _POST_ERROR = \"Post request error!\";\r\n\r\nconst HEADER_NAMES: Record<HttpHeaderNames, HttpHeaderNames> = {\r\n \"Device-Id\": \"Device-Id\",\r\n \"Lang\": \"Lang\",\r\n \"Website-Id\": \"Website-Id\",\r\n \"Website-No\": \"Website-No\",\r\n}\r\n\r\nconst COOKIE_NAMES: Record<CookieNames, CookieNames> = {\r\n \"IBOOT_DEVICE_ID\": \"IBOOT_DEVICE_ID\",\r\n \"IBOOT_LOCALE\": \"IBOOT_LOCALE\",\r\n \"IBOOT_WEBSITE_ID\": \"IBOOT_WEBSITE_ID\",\r\n \"IBOOT_WEBSITE_NO\": \"IBOOT_WEBSITE_NO\",\r\n \"token\": \"token\",\r\n \"user\": \"user\"\r\n}\r\n\r\nconst getDefaultRequestHeader = (): HttpHeaders => {\r\n const devideId = localStorage.getItem(DEVICE_ID_KEY) ?? ''\r\n const json = localStorage.getItem(CURRENT_WEBSITE_KEY);\r\n const curWebsite: CurWebsite | null = json != null ? JSON.parse(json) : null;\r\n const header: HttpHeaders = {}\r\n if (devideId.length > 0) {\r\n header['Device-Id'] = devideId\r\n }\r\n if (curWebsite) {\r\n header['Lang'] = curWebsite.language;\r\n if (curWebsite.websiteId) {\r\n header['Website-Id'] = curWebsite.websiteId;\r\n }\r\n if (curWebsite.websiteNo) {\r\n header['Website-No'] = curWebsite.websiteNo\r\n }\r\n }\r\n return header;\r\n};\r\n\r\n\r\nexport const setDefaultRequestHeader = (deviceId: string, website?: CurWebsite): void => {\r\n localStorage.setItem(DEVICE_ID_KEY, deviceId);\r\n if (website) {\r\n localStorage.setItem(CURRENT_WEBSITE_KEY, JSON.stringify(website));\r\n }\r\n}\r\n\r\nexport const get = async <T>(url: string, opts?: ClientGetParams): Promise<ResultModel<T>> => {\r\n if (opts?.data) {\r\n const params = new URLSearchParams(opts.data);\r\n if (url.indexOf(\"?\") != -1) {\r\n url += `&${params}`\r\n } else {\r\n url += `?${params}`\r\n }\r\n }\r\n const headers: Record<string, string> = getDefaultRequestHeader();\r\n const heads = opts?.headers;\r\n if (heads) {\r\n for (const field in heads) {\r\n const value = heads[field];\r\n if (value) {\r\n headers[field] = value;\r\n }\r\n }\r\n }\r\n logger.debug({\r\n \"url\": url,\r\n \"headers\": headers,\r\n }, \"GET\")\r\n const response = await fetch(url, {\r\n method: 'GET',\r\n headers: headers,\r\n cache: opts?.useCache ? 'force-cache' : 'default'\r\n });\r\n\r\n if (response.ok) {\r\n const result = await response.json();\r\n logger.debug(result, \"GET_RESULT\")\r\n return result\r\n }\r\n const msg = {\r\n code: response.status,\r\n success: false,\r\n msg: response.statusText\r\n }\r\n logger.error(msg);\r\n return msg;\r\n}\r\n\r\nexport const post = async <T>(url: string, opts?: ClientPostParams): Promise<ResultModel<T>> => {\r\n const data = opts?.data ?? {};\r\n const heads = opts?.headers;\r\n let body: string | FormData;\r\n const proxyHeaders = new Headers(getDefaultRequestHeader());\r\n if (!(data instanceof FormData)) {\r\n body = JSON.stringify(data);\r\n proxyHeaders.set('Content-Type', 'application/json');\r\n } else {\r\n body = data;\r\n }\r\n if (heads) {\r\n for (let field in heads) {\r\n const value = heads[field];\r\n if (value) {\r\n proxyHeaders.set(field, value);\r\n }\r\n }\r\n }\r\n logger.debug({\r\n \"url\": url,\r\n \"headers\": proxyHeaders,\r\n }, \"POST\")\r\n const response = await fetch(url, {\r\n method: 'POST',\r\n headers: proxyHeaders,\r\n body: body,\r\n });\r\n if (response.ok) {\r\n const result = await response.json();\r\n logger.debug(result, \"POST_RESULT\")\r\n return result\r\n }\r\n const msg = {\r\n code: response.status,\r\n success: false,\r\n msg: response.statusText,\r\n }\r\n logger.error(msg);\r\n return msg;\r\n}\r\n\r\nexport const iGet = async <T>(url: string, opts?: ClientGetParams): Promise<T | undefined> => {\r\n const res = await get<T>(url, opts);\r\n if (res.success) {\r\n return res.data as T;\r\n }\r\n if (opts?.showError) {\r\n opts.showError(res.msg ?? _GET_ERROR);\r\n return;\r\n }\r\n throw Error(res.msg ?? _GET_ERROR)\r\n}\r\n\r\nexport const iPost = async <T>(url: string, opts?: ClientPostParams): Promise<T | undefined> => {\r\n const res = await post<T>(url, opts);\r\n if (res.success) {\r\n return res.data;\r\n }\r\n if (opts?.showError) {\r\n opts.showError(res.msg ?? _POST_ERROR);\r\n return;\r\n }\r\n throw Error(res.msg ?? _POST_ERROR)\r\n}\r\n\r\nexport const iPostSuccess = async (url: string, opts?: ClientPostParams): Promise<boolean> => {\r\n const res = await post(url, opts);\r\n if (res.success) {\r\n if (opts?.showSuccess) {\r\n opts.showSuccess(res.msg ?? 'SUCCESS')\r\n }\r\n return true;\r\n }\r\n if (opts?.showError) {\r\n opts.showError(res.msg ?? _POST_ERROR);\r\n }\r\n return false;\r\n}\r\n\r\n\r\nexport const getHttpOpts = (request: Request): HttpClientOpts => {\r\n const deviceId = request.headers.get(HEADER_NAMES[\"Device-Id\"]);\r\n const lang = request.headers.get(HEADER_NAMES.Lang);\r\n const websiteId = request.headers.get(HEADER_NAMES[\"Website-Id\"]);\r\n const websiteNo = request.headers.get(HEADER_NAMES[\"Website-No\"]);\r\n const result: HttpClientOpts = {}\r\n if (deviceId && deviceId.length > 0) {\r\n result.deviceId = deviceId;\r\n }\r\n if (lang && lang.length > 0) {\r\n result.lang = lang\r\n }\r\n if (websiteId && websiteId.length > 0) {\r\n result.websiteId = websiteId\r\n }\r\n if (websiteNo && websiteNo.length > 0) {\r\n result.websiteNo = websiteNo\r\n }\r\n return result;\r\n}\r\n\r\nexport const getHttpOptsByCookie = (): HttpClientOpts => {\r\n const result: HttpClientOpts = {}\r\n const deviceId = Cookies.get(COOKIE_NAMES.IBOOT_DEVICE_ID);\r\n const lang = Cookies.get(COOKIE_NAMES.IBOOT_LOCALE);\r\n const websiteId = Cookies.get(COOKIE_NAMES.IBOOT_WEBSITE_ID);\r\n const websiteNo = Cookies.get(COOKIE_NAMES.IBOOT_WEBSITE_NO);\r\n if (deviceId && deviceId.length > 0) {\r\n result.deviceId = deviceId;\r\n }\r\n if (lang && lang.length > 0) {\r\n result.lang = lang\r\n }\r\n if (websiteId && websiteId.length > 0) {\r\n result.websiteId = websiteId\r\n }\r\n if (websiteNo && websiteNo.length > 0) {\r\n result.websiteNo = websiteNo\r\n }\r\n return result;\r\n}\r\n\r\n\r\nexport const getLoginUser = (): User | null => {\r\n const userjson = Cookies.get(COOKIE_NAMES.user);\r\n if (userjson && userjson.length > 0) {\r\n return JSON.parse(userjson);\r\n }\r\n return null;\r\n}\r\n\r\nexport const getToken = (): HttpToken | undefined => {\r\n const userjson = Cookies.get(COOKIE_NAMES.user);\r\n if (userjson && userjson.length > 0) {\r\n const user = JSON.parse(userjson);\r\n const username = user.username;\r\n const utype = user.userType.toString();\r\n const token = Cookies.get(COOKIE_NAMES.token);\r\n if (token) {\r\n return {\r\n \"username\": username,\r\n \"utype\": utype,\r\n \"token\": token\r\n }\r\n }\r\n }\r\n return undefined\r\n}\r\n\r\nexport const setToken = (data: User & { token: string }): void => {\r\n const { id, name, username, nickname, headImg, token, lastLoginTime, deviceId, userType, mustChangePwd } = data;\r\n const loginUser: User = {\r\n id: id,\r\n username: username,\r\n nickname: nickname || name,\r\n headImg: headImg,\r\n lastLoginTime: lastLoginTime,\r\n deviceId: deviceId,\r\n userType: userType,\r\n accountType: 0,\r\n needToReview: false,\r\n socketOnline: false,\r\n mustChangePwd: mustChangePwd\r\n };\r\n\r\n Cookies.set(COOKIE_NAMES.token, token, {\r\n path: '/',\r\n secure: process.env.APP_ENV === 'production',\r\n httpOnly: true,\r\n sameSite: 'strict'\r\n });\r\n Cookies.set(COOKIE_NAMES.user, JSON.stringify(loginUser))\r\n}\r\n\r\nexport const cleanToken = (): void => {\r\n Cookies.remove(COOKIE_NAMES.token, {\r\n path: '/',\r\n secure: process.env.APP_ENV === 'production',\r\n httpOnly: true,\r\n sameSite: 'strict'\r\n });\r\n Cookies.remove(COOKIE_NAMES.user);\r\n}\r\n\r\nexport class HttpClient {\r\n private readonly baseUrl: string;\r\n private readonly helloMethod: string;\r\n private readonly apiKey: string;\r\n private readonly userType: USER_TYPE;\r\n private readonly userFrom: string;\r\n private readonly deviceId: string;\r\n private readonly version: string = '1';\r\n private readonly lang: string;\r\n private readonly websiteId?: string;\r\n private readonly websiteNo?: string;\r\n constructor(opts: Readonly<HttpClientOpts>) {\r\n const isProduction = process.env.APP_ENV === 'production';\r\n if (!isProduction) {\r\n logger.warn(\"This is currently a development environment!\")\r\n }\r\n this.baseUrl = process.env.APP_BASEURL!;\r\n this.apiKey = process.env.APP_APIKEY!;\r\n this.userFrom = process.env.APP_USERFROM ?? '1';\r\n\r\n this.helloMethod = opts.helloURL ?? ''\r\n this.userType = opts.userType ?? USER_TYPE_MAP.TYPE_MGT;\r\n this.deviceId = opts.deviceId ?? randomString(10);\r\n this.lang = opts.lang ?? DEFAULT_LOCALE;\r\n this.websiteId = opts.websiteId\r\n this.websiteNo = opts.websiteNo\r\n }\r\n\r\n encrypt(data: string) {\r\n const key = CryptoJS.enc.Utf8.parse(this.apiKey);\r\n return CryptoJS.AES.encrypt(data, key).toString();\r\n }\r\n\r\n decrypt(data: string) {\r\n const key = CryptoJS.enc.Utf8.parse(this.apiKey);\r\n return CryptoJS.AES.decrypt(data, key, {\r\n mode: CryptoJS.mode.ECB,\r\n }).toString(CryptoJS.enc.Utf8);\r\n }\r\n\r\n //eslint-disable-next-line @typescript-eslint/no-explicit-any\r\n private convertUrlParameter(data: Record<string, any>): string {\r\n const p: string[] = [];\r\n for (const o in data) {\r\n let v = data[o];\r\n if (v && typeof v === \"string\") {\r\n v = v.trim();\r\n } else if (v && typeof v === \"object\") {\r\n if (isArray(v)) {\r\n if (v.length === 0) {\r\n continue;\r\n }\r\n for (let i = 0; i < v.length; i++) {\r\n const arr_item = v[i];\r\n for (const item_field in arr_item) {\r\n const item_field_value = arr_item[item_field] ?? undefined;\r\n if (item_field_value) {\r\n if (typeof item_field_value === \"object\") {\r\n if (isArray(item_field_value)) {\r\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\r\n item_field_value.forEach((item1: any, index: number) => {\r\n if (item1) {\r\n for (const f in item1) {\r\n const v = item1[f];\r\n if (v && v.toString().length > 0) {\r\n const s = `[${i.toString()}].${item_field + `[${index.toString()}].${f}`}=${urlEncode(v)}`;\r\n p.push(o + s);\r\n }\r\n }\r\n }\r\n });\r\n }\r\n } else if (item_field_value.toString().length > 0) {\r\n const s = `[${i.toString()}].${item_field}=${urlEncode(item_field_value.toString())}`;\r\n p.push(o + s);\r\n }\r\n }\r\n }\r\n }\r\n } else {\r\n for (const obj in v) {\r\n const oValue = v[obj];\r\n if (oValue != null && oValue != \"\" && oValue.length != 0) {\r\n const s = `.${obj}=${urlEncode(oValue)}`;\r\n p.push(o + s);\r\n }\r\n }\r\n }\r\n continue;\r\n } else if (v && typeof v === \"function\") {\r\n v = null;\r\n }\r\n\r\n if (v != null && v != \"\" && v.length != 0) {\r\n p.push(o + \"=\" + urlEncode(v));\r\n }\r\n }\r\n return (p && p.join(\"&\"));\r\n }\r\n\r\n private async sign(data: Record<string, object | string>): Promise<string> {\r\n //由后台分配\r\n const _apiKey = \"&key=\" + (await this.helloIboot());\r\n const arr: string[] = [];\r\n for (const o in data) {\r\n arr.push(o);\r\n }\r\n arr.sort((a: string, b: string) => {\r\n return a.toLowerCase().localeCompare(b.toLowerCase());\r\n });\r\n const res: string[] = [];\r\n arr.forEach((v: string) => {\r\n let value: object | string = data[v];\r\n if (typeof value === \"object\") {\r\n value = JSON.stringify(value);\r\n }\r\n res.push(v + \"=\" + value);\r\n });\r\n const paramsStr: string = res.join(\"&\");\r\n const str = md5.md5((paramsStr + _apiKey).toLocaleUpperCase());\r\n const logInfo = {\r\n params: paramsStr,\r\n md5: str\r\n }\r\n logger.debug(logInfo, \"API_SIGN\")\r\n return str;\r\n }\r\n\r\n //eslint-disable-next-line @typescript-eslint/no-explicit-any\r\n private assemblyParameter(data?: Record<string, any>, username?: string): Record<string, any> {\r\n const params = data ?? {};\r\n params['timestamp'] = Date.now() + '';\r\n params['echostr'] = randomString(10);\r\n params['version'] = this.version;\r\n params['deviceId'] = this.deviceId;\r\n params['webid'] = this.websiteId;\r\n params['apiKey'] = this.apiKey;\r\n if (username && username.length > 0) {\r\n params['username'] = username;\r\n }\r\n return params;\r\n }\r\n\r\n private async assemblyHeader({ urlParams, token }: Readonly<{ urlParams: string, token?: Readonly<HttpToken> }>) {\r\n const h: Record<string, string> = {\r\n 'Content-Type': 'application/x-www-form-urlencoded',\r\n 'Res-Type': 'json',\r\n 'Device-Id': this.deviceId,\r\n 'User-Type': token?.utype.toString() ?? this.userType.toString(),\r\n 'User-From': this.userFrom,\r\n 'Api-Key': this.apiKey\r\n }\r\n if (this.lang && this.lang.length > 0) {\r\n h['Lang'] = this.lang\r\n }\r\n if (this.websiteId && this.websiteId.length > 0) {\r\n h['Web-Id'] = this.websiteId\r\n }\r\n if (this.websiteNo && this.websiteNo.length > 0) {\r\n h['Web-No'] = this.websiteNo\r\n }\r\n if (token) {\r\n if (token.token && token.token.length > 0 && token.username && token.username.length > 0) {\r\n h['Authorization'] = token.token;\r\n h['Username'] = token.username;\r\n }\r\n if (token.xcsrf) {\r\n h[token.xcsrf.csrfHeader] = token.xcsrf.csrfToken;\r\n }\r\n }\r\n const data = urlParamToJson(urlParams);\r\n const sign = await this.sign(data);\r\n h['Sign'] = sign\r\n return h;\r\n }\r\n\r\n private async helloIboot(): Promise<string> {\r\n if (this.helloMethod.length <= 0) {\r\n return this.apiKey;\r\n }\r\n const api = this.getApiUrl(this.helloMethod);\r\n const h = {\r\n 'Content-Type': 'application/x-www-form-urlencoded',\r\n 'Res-Type': 'json',\r\n 'Api-Key': this.apiKey\r\n }\r\n const logInfo = {\r\n \"url\": api,\r\n \"headers\": h\r\n }\r\n logger.debug(logInfo, \"HELLO\");\r\n const res = await fetch(api, {\r\n method: 'GET',\r\n headers: h,\r\n credentials: 'include',\r\n cache: 'force-cache'\r\n });\r\n if (res.ok) {\r\n const result = await res.json();\r\n logger.debug(result.data, 'HELLO RESULT')\r\n return result.data;\r\n }\r\n throw new Error('hello iBoot error!')\r\n }\r\n\r\n private getApiUrl(url: string): string {\r\n return `${this.baseUrl}/${url}`;\r\n }\r\n\r\n async csrf(): Promise<string | undefined> {\r\n const res = await this.get<CSRFToken>({\r\n url: 'guest/csrf'\r\n });\r\n if (res.success) {\r\n const data = res.data;\r\n if (data) {\r\n return data[\"csrfToken\"].toString();\r\n }\r\n }\r\n return undefined;\r\n }\r\n\r\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\r\n async get<T>({ url, data, token, cache = 'default' }: Readonly<ServerRequestOptions>): Promise<ResultModel<T>> {\r\n const params = this.assemblyParameter(data);\r\n const urlParams = this.convertUrlParameter(params);\r\n const headers = await this.assemblyHeader({ \"urlParams\": urlParams, \"token\": token });\r\n const api = `${this.getApiUrl(url)}?${urlParams.toString()}`\r\n const logInfo = {\r\n \"url\": api,\r\n \"headers\": headers\r\n }\r\n logger.info(logInfo, \"GET\");\r\n const res = await fetch(api, {\r\n \"method\": 'GET',\r\n \"headers\": headers,\r\n \"credentials\": 'include',\r\n \"cache\": cache\r\n });\r\n if (res.ok) {\r\n const result = await res.json();\r\n logger.debug(result, \"GET_RESULT\")\r\n return result;\r\n }\r\n const msg = {\r\n code: res.status,\r\n success: false,\r\n msg: res.statusText\r\n }\r\n logger.error(msg);\r\n return msg;\r\n }\r\n\r\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\r\n async post<T>({ url, data, token, cache = \"default\" }: Readonly<ServerRequestOptions>): Promise<ResultModel<T>> {\r\n data = data ? { ...data } : {}\r\n const buffer = data.buffer ?? undefined;\r\n const boundary = data.boundary ?? undefined;\r\n if (buffer) {\r\n delete data.buffer;\r\n }\r\n if (boundary) {\r\n delete data.boundary;\r\n }\r\n const params = this.assemblyParameter(data);\r\n const urlParams = this.convertUrlParameter(params);\r\n const headers = await this.assemblyHeader({ \"urlParams\": urlParams, \"token\": token });\r\n if (buffer && boundary) {\r\n headers['Content-Type'] = `multipart/form-data; boundary=${boundary}`;\r\n }\r\n const api = this.getApiUrl(url);\r\n const logInfo = {\r\n \"url\": api,\r\n \"headers\": headers,\r\n \"params\": params\r\n }\r\n logger.info(logInfo, \"POST\");\r\n const urls = buffer ? `${api}?${urlParams}` : api;\r\n const res = await fetch(urls, {\r\n \"method\": 'POST',\r\n \"headers\": headers,\r\n \"body\": buffer ? buffer : urlParams.toString(),\r\n \"credentials\": 'include',\r\n \"cache\": cache\r\n });\r\n if (res.ok) {\r\n const result = await res.json();\r\n logger.debug(result, \"POST_RESULT\")\r\n return result;\r\n }\r\n const msg = {\r\n code: res.status,\r\n success: false,\r\n msg: res.statusText\r\n }\r\n logger.error(msg);\r\n return msg;\r\n }\r\n\r\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\r\n async stream({ url, data, token }: Readonly<Omit<ServerRequestOptions, 'cache'>>): Promise<Response> {\r\n data = data ? { ...data } : {}\r\n const buffer = data.buffer ?? undefined;\r\n const boundary = data.boundary ?? undefined;\r\n if (buffer) {\r\n delete data.buffer;\r\n }\r\n if (boundary) {\r\n delete data.boundary;\r\n }\r\n const params = this.assemblyParameter(data);\r\n const urlParams = this.convertUrlParameter(params); //new URLSearchParams(params);\r\n\r\n const headers = await this.assemblyHeader({ \"urlParams\": urlParams, \"token\": token });\r\n if (buffer && boundary) {\r\n headers['Content-Type'] = `multipart/form-data; boundary=${boundary}`;\r\n }\r\n headers['Connection'] = 'keep-alive';\r\n headers['X-Accel-Buffering'] = 'no'; //防止 Nginx 缓存\r\n headers['Cache-Control'] = 'no-cache';\r\n const api = this.getApiUrl(url)\r\n try {\r\n const url = buffer ? `${api}?${urlParams}` : api;\r\n const logInfo = {\r\n \"url\": api,\r\n \"headers\": headers,\r\n \"params\": params\r\n }\r\n logger.info(logInfo, \"STREAM\");\r\n const res = await fetch(url, {\r\n method: 'POST',\r\n headers: headers,\r\n body: buffer ? buffer : urlParams.toString(),\r\n credentials: 'include'\r\n });\r\n\r\n if (!res.ok) {\r\n throw new Error(`HTTP error! status: ${res.status}`);\r\n }\r\n if (!res.body) {\r\n throw new Error('No response body');\r\n }\r\n return new Response(res.body, {\r\n headers: {\r\n 'Content-Type': 'text/event-stream',\r\n 'Cache-Control': 'no-cache',\r\n 'Connection': 'keep-alive'\r\n }\r\n });\r\n } catch (e) {\r\n throw e;\r\n }\r\n }\r\n}"],"names":["ACCOUNT_TYPE_MAP","USER_TYPE_MAP","USER_FORM_MAP","USER_SEX_MAP","stream","PinoPretty","logger","pino","isArray","obj","randomString","len","l","$chars","maxPos","pwd","i","urlParamToJson","urlQueryParams","exclude","result","item","arr","urlEncode","value","dateFormat","dt","fmt","o","k","regx","assign","target","source","key","defaultConverter","init","converter","defaultAttributes","set","name","attributes","stringifiedAttributes","attributeName","get","cookies","jar","parts","found","api","DEFAULT_LOCALE","DEVICE_ID_KEY","CURRENT_WEBSITE_KEY","_GET_ERROR","_POST_ERROR","HEADER_NAMES","COOKIE_NAMES","getDefaultRequestHeader","devideId","json","curWebsite","header","setDefaultRequestHeader","deviceId","website","url","opts","params","headers","heads","field","response","msg","post","data","body","proxyHeaders","iGet","res","iPost","iPostSuccess","getHttpOpts","request","lang","websiteId","websiteNo","getHttpOptsByCookie","Cookies","getLoginUser","userjson","getToken","user","username","utype","token","setToken","id","nickname","headImg","lastLoginTime","userType","mustChangePwd","loginUser","cleanToken","HttpClient","CryptoJS","p","v","arr_item","item_field","item_field_value","item1","index","f","s","oValue","_apiKey","a","b","paramsStr","str","md5","logInfo","urlParams","h","sign","cache","buffer","boundary","urls","e"],"mappings":"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","x_google_ignoreList":[3]}
|