@pisell/pisellos 0.0.505 → 0.0.507

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.
@@ -86,7 +86,7 @@ export declare class OrderModule extends BaseModule implements Module, OrderModu
86
86
  enhancePayload?: SubmitPayloadEnhancer;
87
87
  }): Promise<T>;
88
88
  createOrder(params: CommitOrderParams['query']): {
89
- type: "virtual" | "appointment_booking";
89
+ type: "appointment_booking" | "virtual";
90
90
  platform: string;
91
91
  sales_channel: string;
92
92
  order_sales_channel: string;
@@ -2,94 +2,202 @@ function _typeof(o) { "@babel/helpers - typeof"; return _typeof = "function" ==
2
2
  function _regeneratorRuntime() { "use strict"; /*! regenerator-runtime -- Copyright (c) 2014-present, Facebook, Inc. -- license (MIT): https://github.com/facebook/regenerator/blob/main/LICENSE */ _regeneratorRuntime = function _regeneratorRuntime() { return e; }; var t, e = {}, r = Object.prototype, n = r.hasOwnProperty, o = Object.defineProperty || function (t, e, r) { t[e] = r.value; }, i = "function" == typeof Symbol ? Symbol : {}, a = i.iterator || "@@iterator", c = i.asyncIterator || "@@asyncIterator", u = i.toStringTag || "@@toStringTag"; function define(t, e, r) { return Object.defineProperty(t, e, { value: r, enumerable: !0, configurable: !0, writable: !0 }), t[e]; } try { define({}, ""); } catch (t) { define = function define(t, e, r) { return t[e] = r; }; } function wrap(t, e, r, n) { var i = e && e.prototype instanceof Generator ? e : Generator, a = Object.create(i.prototype), c = new Context(n || []); return o(a, "_invoke", { value: makeInvokeMethod(t, r, c) }), a; } function tryCatch(t, e, r) { try { return { type: "normal", arg: t.call(e, r) }; } catch (t) { return { type: "throw", arg: t }; } } e.wrap = wrap; var h = "suspendedStart", l = "suspendedYield", f = "executing", s = "completed", y = {}; function Generator() {} function GeneratorFunction() {} function GeneratorFunctionPrototype() {} var p = {}; define(p, a, function () { return this; }); var d = Object.getPrototypeOf, v = d && d(d(values([]))); v && v !== r && n.call(v, a) && (p = v); var g = GeneratorFunctionPrototype.prototype = Generator.prototype = Object.create(p); function defineIteratorMethods(t) { ["next", "throw", "return"].forEach(function (e) { define(t, e, function (t) { return this._invoke(e, t); }); }); } function AsyncIterator(t, e) { function invoke(r, o, i, a) { var c = tryCatch(t[r], t, o); if ("throw" !== c.type) { var u = c.arg, h = u.value; return h && "object" == _typeof(h) && n.call(h, "__await") ? e.resolve(h.__await).then(function (t) { invoke("next", t, i, a); }, function (t) { invoke("throw", t, i, a); }) : e.resolve(h).then(function (t) { u.value = t, i(u); }, function (t) { return invoke("throw", t, i, a); }); } a(c.arg); } var r; o(this, "_invoke", { value: function value(t, n) { function callInvokeWithMethodAndArg() { return new e(function (e, r) { invoke(t, n, e, r); }); } return r = r ? r.then(callInvokeWithMethodAndArg, callInvokeWithMethodAndArg) : callInvokeWithMethodAndArg(); } }); } function makeInvokeMethod(e, r, n) { var o = h; return function (i, a) { if (o === f) throw new Error("Generator is already running"); if (o === s) { if ("throw" === i) throw a; return { value: t, done: !0 }; } for (n.method = i, n.arg = a;;) { var c = n.delegate; if (c) { var u = maybeInvokeDelegate(c, n); if (u) { if (u === y) continue; return u; } } if ("next" === n.method) n.sent = n._sent = n.arg;else if ("throw" === n.method) { if (o === h) throw o = s, n.arg; n.dispatchException(n.arg); } else "return" === n.method && n.abrupt("return", n.arg); o = f; var p = tryCatch(e, r, n); if ("normal" === p.type) { if (o = n.done ? s : l, p.arg === y) continue; return { value: p.arg, done: n.done }; } "throw" === p.type && (o = s, n.method = "throw", n.arg = p.arg); } }; } function maybeInvokeDelegate(e, r) { var n = r.method, o = e.iterator[n]; if (o === t) return r.delegate = null, "throw" === n && e.iterator.return && (r.method = "return", r.arg = t, maybeInvokeDelegate(e, r), "throw" === r.method) || "return" !== n && (r.method = "throw", r.arg = new TypeError("The iterator does not provide a '" + n + "' method")), y; var i = tryCatch(o, e.iterator, r.arg); if ("throw" === i.type) return r.method = "throw", r.arg = i.arg, r.delegate = null, y; var a = i.arg; return a ? a.done ? (r[e.resultName] = a.value, r.next = e.nextLoc, "return" !== r.method && (r.method = "next", r.arg = t), r.delegate = null, y) : a : (r.method = "throw", r.arg = new TypeError("iterator result is not an object"), r.delegate = null, y); } function pushTryEntry(t) { var e = { tryLoc: t[0] }; 1 in t && (e.catchLoc = t[1]), 2 in t && (e.finallyLoc = t[2], e.afterLoc = t[3]), this.tryEntries.push(e); } function resetTryEntry(t) { var e = t.completion || {}; e.type = "normal", delete e.arg, t.completion = e; } function Context(t) { this.tryEntries = [{ tryLoc: "root" }], t.forEach(pushTryEntry, this), this.reset(!0); } function values(e) { if (e || "" === e) { var r = e[a]; if (r) return r.call(e); if ("function" == typeof e.next) return e; if (!isNaN(e.length)) { var o = -1, i = function next() { for (; ++o < e.length;) if (n.call(e, o)) return next.value = e[o], next.done = !1, next; return next.value = t, next.done = !0, next; }; return i.next = i; } } throw new TypeError(_typeof(e) + " is not iterable"); } return GeneratorFunction.prototype = GeneratorFunctionPrototype, o(g, "constructor", { value: GeneratorFunctionPrototype, configurable: !0 }), o(GeneratorFunctionPrototype, "constructor", { value: GeneratorFunction, configurable: !0 }), GeneratorFunction.displayName = define(GeneratorFunctionPrototype, u, "GeneratorFunction"), e.isGeneratorFunction = function (t) { var e = "function" == typeof t && t.constructor; return !!e && (e === GeneratorFunction || "GeneratorFunction" === (e.displayName || e.name)); }, e.mark = function (t) { return Object.setPrototypeOf ? Object.setPrototypeOf(t, GeneratorFunctionPrototype) : (t.__proto__ = GeneratorFunctionPrototype, define(t, u, "GeneratorFunction")), t.prototype = Object.create(g), t; }, e.awrap = function (t) { return { __await: t }; }, defineIteratorMethods(AsyncIterator.prototype), define(AsyncIterator.prototype, c, function () { return this; }), e.AsyncIterator = AsyncIterator, e.async = function (t, r, n, o, i) { void 0 === i && (i = Promise); var a = new AsyncIterator(wrap(t, r, n, o), i); return e.isGeneratorFunction(r) ? a : a.next().then(function (t) { return t.done ? t.value : a.next(); }); }, defineIteratorMethods(g), define(g, u, "Generator"), define(g, a, function () { return this; }), define(g, "toString", function () { return "[object Generator]"; }), e.keys = function (t) { var e = Object(t), r = []; for (var n in e) r.push(n); return r.reverse(), function next() { for (; r.length;) { var t = r.pop(); if (t in e) return next.value = t, next.done = !1, next; } return next.done = !0, next; }; }, e.values = values, Context.prototype = { constructor: Context, reset: function reset(e) { if (this.prev = 0, this.next = 0, this.sent = this._sent = t, this.done = !1, this.delegate = null, this.method = "next", this.arg = t, this.tryEntries.forEach(resetTryEntry), !e) for (var r in this) "t" === r.charAt(0) && n.call(this, r) && !isNaN(+r.slice(1)) && (this[r] = t); }, stop: function stop() { this.done = !0; var t = this.tryEntries[0].completion; if ("throw" === t.type) throw t.arg; return this.rval; }, dispatchException: function dispatchException(e) { if (this.done) throw e; var r = this; function handle(n, o) { return a.type = "throw", a.arg = e, r.next = n, o && (r.method = "next", r.arg = t), !!o; } for (var o = this.tryEntries.length - 1; o >= 0; --o) { var i = this.tryEntries[o], a = i.completion; if ("root" === i.tryLoc) return handle("end"); if (i.tryLoc <= this.prev) { var c = n.call(i, "catchLoc"), u = n.call(i, "finallyLoc"); if (c && u) { if (this.prev < i.catchLoc) return handle(i.catchLoc, !0); if (this.prev < i.finallyLoc) return handle(i.finallyLoc); } else if (c) { if (this.prev < i.catchLoc) return handle(i.catchLoc, !0); } else { if (!u) throw new Error("try statement without catch or finally"); if (this.prev < i.finallyLoc) return handle(i.finallyLoc); } } } }, abrupt: function abrupt(t, e) { for (var r = this.tryEntries.length - 1; r >= 0; --r) { var o = this.tryEntries[r]; if (o.tryLoc <= this.prev && n.call(o, "finallyLoc") && this.prev < o.finallyLoc) { var i = o; break; } } i && ("break" === t || "continue" === t) && i.tryLoc <= e && e <= i.finallyLoc && (i = null); var a = i ? i.completion : {}; return a.type = t, a.arg = e, i ? (this.method = "next", this.next = i.finallyLoc, y) : this.complete(a); }, complete: function complete(t, e) { if ("throw" === t.type) throw t.arg; return "break" === t.type || "continue" === t.type ? this.next = t.arg : "return" === t.type ? (this.rval = this.arg = t.arg, this.method = "return", this.next = "end") : "normal" === t.type && e && (this.next = e), y; }, finish: function finish(t) { for (var e = this.tryEntries.length - 1; e >= 0; --e) { var r = this.tryEntries[e]; if (r.finallyLoc === t) return this.complete(r.completion, r.afterLoc), resetTryEntry(r), y; } }, catch: function _catch(t) { for (var e = this.tryEntries.length - 1; e >= 0; --e) { var r = this.tryEntries[e]; if (r.tryLoc === t) { var n = r.completion; if ("throw" === n.type) { var o = n.arg; resetTryEntry(r); } return o; } } throw new Error("illegal catch attempt"); }, delegateYield: function delegateYield(e, r, n) { return this.delegate = { iterator: values(e), resultName: r, nextLoc: n }, "next" === this.method && (this.arg = t), y; } }, e; }
3
3
  function asyncGeneratorStep(gen, resolve, reject, _next, _throw, key, arg) { try { var info = gen[key](arg); var value = info.value; } catch (error) { reject(error); return; } if (info.done) { resolve(value); } else { Promise.resolve(value).then(_next, _throw); } }
4
4
  function _asyncToGenerator(fn) { return function () { var self = this, args = arguments; return new Promise(function (resolve, reject) { var gen = fn.apply(self, args); function _next(value) { asyncGeneratorStep(gen, resolve, reject, _next, _throw, "next", value); } function _throw(err) { asyncGeneratorStep(gen, resolve, reject, _next, _throw, "throw", err); } _next(undefined); }); }; }
5
- function safeStringify(payload) {
6
- if (payload === undefined || payload === null) return '';
5
+ var DEFAULT_THROTTLE_MS = 3000;
6
+ var pendingQueue = [];
7
+ var flushTimer = null;
8
+ var cachedProviderConfig;
9
+ function safeStringify(value) {
10
+ if (value === undefined || value === null) return '';
11
+ if (typeof value === 'string') return value;
12
+ if (typeof value === 'number' || typeof value === 'boolean' || typeof value === 'bigint') {
13
+ return String(value);
14
+ }
7
15
  try {
8
- return JSON.stringify(payload);
16
+ return JSON.stringify(value);
9
17
  } catch (error) {
10
18
  return "[unserializable: ".concat(String(error), "]");
11
19
  }
12
20
  }
13
- function createFeishuMessageContent(contentArr) {
14
- return JSON.stringify(contentArr.map(function (item) {
15
- return [{
16
- tag: 'text',
17
- text: "".concat(item.key, ": ")
18
- }, {
19
- tag: 'text',
20
- text: item.value
21
- }];
21
+ function isWebhookUsable(webhook) {
22
+ if (!webhook) return false;
23
+ if (webhook.includes('REPLACE_ME')) return false;
24
+ return true;
25
+ }
26
+ function buildRecordSegments(record) {
27
+ var context = record.context || {};
28
+ return [[{
29
+ tag: 'text',
30
+ text: "[".concat(record.level, "] ").concat(record.timestamp, " ").concat(record.title, "\n")
31
+ }], [{
32
+ tag: 'text',
33
+ text: "\u7F13\u5B58\u6807\u8BC6: ".concat(safeStringify(context.cacheId), "\n")
34
+ }], [{
35
+ tag: 'text',
36
+ text: "\u65E5\u5FD7\u6765\u6E90: ".concat(safeStringify(context), "\n")
37
+ }], [{
38
+ tag: 'text',
39
+ text: "\u65E5\u5FD7\u5185\u5BB9: ".concat(safeStringify(record.payload), "\n")
40
+ }], [{
41
+ tag: 'text',
42
+ text: "\u6269\u5C55\u4FE1\u606F: ".concat(safeStringify(record.extra), "\n")
43
+ }], [{
44
+ tag: 'text',
45
+ text: '------\n'
46
+ }]];
47
+ }
48
+ function buildFeishuBody(records) {
49
+ var title = records.length === 1 ? records[0].title : "ScanOrder \u65E5\u5FD7\u6279\u91CF (".concat(records.length, " \u6761)");
50
+ var content = [];
51
+ records.forEach(function (record) {
52
+ buildRecordSegments(record).forEach(function (segment) {
53
+ return content.push(segment);
54
+ });
55
+ });
56
+ return JSON.stringify({
57
+ msg_type: 'post',
58
+ content: JSON.stringify({
59
+ post: {
60
+ zh_cn: {
61
+ title: title,
62
+ content: content
63
+ }
64
+ }
65
+ })
66
+ });
67
+ }
68
+ function postToFeishu(_x, _x2) {
69
+ return _postToFeishu.apply(this, arguments);
70
+ }
71
+ function _postToFeishu() {
72
+ _postToFeishu = _asyncToGenerator( /*#__PURE__*/_regeneratorRuntime().mark(function _callee2(webhook, body) {
73
+ return _regeneratorRuntime().wrap(function _callee2$(_context2) {
74
+ while (1) switch (_context2.prev = _context2.next) {
75
+ case 0:
76
+ _context2.next = 2;
77
+ return fetch(webhook, {
78
+ method: 'POST',
79
+ headers: {
80
+ 'Content-Type': 'application/json'
81
+ },
82
+ body: body
83
+ });
84
+ case 2:
85
+ case "end":
86
+ return _context2.stop();
87
+ }
88
+ }, _callee2);
89
+ }));
90
+ return _postToFeishu.apply(this, arguments);
91
+ }
92
+ function flushQueue() {
93
+ return _flushQueue.apply(this, arguments);
94
+ }
95
+ function _flushQueue() {
96
+ _flushQueue = _asyncToGenerator( /*#__PURE__*/_regeneratorRuntime().mark(function _callee3() {
97
+ var _cachedProviderConfig;
98
+ var records, webhook;
99
+ return _regeneratorRuntime().wrap(function _callee3$(_context3) {
100
+ while (1) switch (_context3.prev = _context3.next) {
101
+ case 0:
102
+ flushTimer = null;
103
+ if (pendingQueue.length) {
104
+ _context3.next = 3;
105
+ break;
106
+ }
107
+ return _context3.abrupt("return");
108
+ case 3:
109
+ records = pendingQueue.splice(0, pendingQueue.length);
110
+ webhook = (_cachedProviderConfig = cachedProviderConfig) === null || _cachedProviderConfig === void 0 || (_cachedProviderConfig = _cachedProviderConfig.feishu) === null || _cachedProviderConfig === void 0 ? void 0 : _cachedProviderConfig.webhook;
111
+ if (isWebhookUsable(webhook)) {
112
+ _context3.next = 7;
113
+ break;
114
+ }
115
+ return _context3.abrupt("return");
116
+ case 7:
117
+ if (!(typeof fetch !== 'function')) {
118
+ _context3.next = 10;
119
+ break;
120
+ }
121
+ console.warn('[ScanOrderLogger] 当前环境不支持 fetch,跳过 Feishu 日志批量上报');
122
+ return _context3.abrupt("return");
123
+ case 10:
124
+ _context3.prev = 10;
125
+ _context3.next = 13;
126
+ return postToFeishu(webhook, buildFeishuBody(records));
127
+ case 13:
128
+ _context3.next = 18;
129
+ break;
130
+ case 15:
131
+ _context3.prev = 15;
132
+ _context3.t0 = _context3["catch"](10);
133
+ console.warn('[ScanOrderLogger] Feishu 批量上报失败', _context3.t0);
134
+ case 18:
135
+ case "end":
136
+ return _context3.stop();
137
+ }
138
+ }, _callee3, null, [[10, 15]]);
22
139
  }));
140
+ return _flushQueue.apply(this, arguments);
23
141
  }
24
142
  export var feishuLoggerProvider = {
25
143
  send: function send(payload) {
26
144
  return _asyncToGenerator( /*#__PURE__*/_regeneratorRuntime().mark(function _callee() {
27
- var _payload$providerConf;
28
- var webhook, record, contentArr, contentStr;
145
+ var _payload$providerConf, _payload$providerConf2, _payload$providerConf3;
146
+ var webhook, throttleMs;
29
147
  return _regeneratorRuntime().wrap(function _callee$(_context) {
30
148
  while (1) switch (_context.prev = _context.next) {
31
149
  case 0:
150
+ cachedProviderConfig = payload.providerConfig;
32
151
  webhook = (_payload$providerConf = payload.providerConfig) === null || _payload$providerConf === void 0 || (_payload$providerConf = _payload$providerConf.feishu) === null || _payload$providerConf === void 0 ? void 0 : _payload$providerConf.webhook;
33
- if (!(webhook && webhook.includes('REPLACE_ME'))) {
34
- _context.next = 3;
152
+ if (webhook) {
153
+ _context.next = 5;
35
154
  break;
36
155
  }
156
+ console.warn('[ScanOrderLogger] Feishu webhook 未配置,跳过日志上报');
37
157
  return _context.abrupt("return");
38
- case 3:
39
- if (webhook) {
40
- _context.next = 6;
158
+ case 5:
159
+ if (isWebhookUsable(webhook)) {
160
+ _context.next = 7;
41
161
  break;
42
162
  }
43
- console.warn('[ScanOrderLogger] Feishu webhook 未配置,跳过日志上报');
44
163
  return _context.abrupt("return");
45
- case 6:
164
+ case 7:
46
165
  if (!(typeof fetch !== 'function')) {
47
- _context.next = 9;
166
+ _context.next = 10;
48
167
  break;
49
168
  }
50
169
  console.warn('[ScanOrderLogger] 当前环境不支持 fetch,跳过 Feishu 日志上报');
51
170
  return _context.abrupt("return");
52
- case 9:
53
- record = payload.record;
54
- contentArr = [{
55
- key: '日志级别',
56
- value: record.level
57
- }, {
58
- key: '日志时间',
59
- value: record.timestamp
60
- }, {
61
- key: '日志标题',
62
- value: record.title
63
- }, {
64
- key: '缓存标识',
65
- value: record.context.cacheId || ''
66
- }, {
67
- key: '日志来源',
68
- value: safeStringify(record.context)
69
- }, {
70
- key: '日志内容',
71
- value: safeStringify(record.payload)
72
- }, {
73
- key: '扩展信息',
74
- value: safeStringify(record.extra)
75
- }];
76
- contentStr = createFeishuMessageContent(contentArr);
77
- _context.next = 14;
78
- return fetch(webhook, {
79
- headers: {
80
- 'Content-Type': 'application/json'
81
- },
82
- method: 'POST',
83
- body: JSON.stringify({
84
- msg_type: 'post',
85
- content: "{\"post\":{\"zh_cn\":{\"title\":\"".concat(record.title, "\",\"content\":").concat(contentStr, "}}}")
86
- })
87
- });
88
- case 14:
171
+ case 10:
172
+ throttleMs = (_payload$providerConf2 = (_payload$providerConf3 = payload.providerConfig) === null || _payload$providerConf3 === void 0 || (_payload$providerConf3 = _payload$providerConf3.feishu) === null || _payload$providerConf3 === void 0 ? void 0 : _payload$providerConf3.throttleMs) !== null && _payload$providerConf2 !== void 0 ? _payload$providerConf2 : DEFAULT_THROTTLE_MS;
173
+ if (!(throttleMs <= 0)) {
174
+ _context.next = 21;
175
+ break;
176
+ }
177
+ _context.prev = 12;
178
+ _context.next = 15;
179
+ return postToFeishu(webhook, buildFeishuBody([payload.record]));
180
+ case 15:
181
+ _context.next = 20;
182
+ break;
183
+ case 17:
184
+ _context.prev = 17;
185
+ _context.t0 = _context["catch"](12);
186
+ console.warn('[ScanOrderLogger] Feishu 日志上报失败', _context.t0);
187
+ case 20:
188
+ return _context.abrupt("return");
189
+ case 21:
190
+ pendingQueue.push(payload.record);
191
+ if (!flushTimer) {
192
+ flushTimer = setTimeout(function () {
193
+ void flushQueue();
194
+ }, throttleMs);
195
+ }
196
+ case 23:
89
197
  case "end":
90
198
  return _context.stop();
91
199
  }
92
- }, _callee);
200
+ }, _callee, null, [[12, 17]]);
93
201
  }))();
94
202
  }
95
203
  };
@@ -2,6 +2,12 @@ export type ScanOrderLogLevel = 'info' | 'warning' | 'error' | 'debug';
2
2
  export type ScanOrderLoggerProviderType = 'feishu' | 'grafana';
3
3
  export interface ScanOrderLoggerProviderFeishuConfig {
4
4
  webhook?: string;
5
+ /**
6
+ * 节流窗口毫秒数,默认 3000ms
7
+ * - >0:首条日志进入队列并启动定时器,窗口结束后合并成一条 Feishu post 统一发送
8
+ * - <=0:关闭节流,保持每条日志立即单独发送
9
+ */
10
+ throttleMs?: number;
5
11
  }
6
12
  export interface ScanOrderLoggerProviderGrafanaConfig {
7
13
  endpoint?: string;
@@ -311,7 +311,7 @@ export declare class BookingByStepImpl extends BaseModule implements Module {
311
311
  date: string;
312
312
  status: string;
313
313
  week: string;
314
- weekNum: 0 | 1 | 2 | 3 | 4 | 5 | 6;
314
+ weekNum: 0 | 2 | 1 | 3 | 5 | 4 | 6;
315
315
  }[]>;
316
316
  submitTimeSlot(timeSlots: TimeSliceItem): void;
317
317
  private getScheduleDataByIds;
@@ -131,7 +131,7 @@ export declare class BookingTicketImpl extends BaseModule implements Module {
131
131
  * 获取当前的客户搜索条件
132
132
  * @returns 当前搜索条件
133
133
  */
134
- getCurrentCustomerSearchParams(): Omit<import("../../modules").ShopGetCustomerListParams, "num" | "skip">;
134
+ getCurrentCustomerSearchParams(): Omit<import("../../modules").ShopGetCustomerListParams, "skip" | "num">;
135
135
  /**
136
136
  * 获取客户列表状态(包含滚动加载相关状态)
137
137
  * @returns 客户状态
@@ -87,11 +87,11 @@ export var ScanOrderImpl = /*#__PURE__*/function (_BaseModule) {
87
87
  key: "serializeError",
88
88
  value: function serializeError(error) {
89
89
  if (error instanceof Error) {
90
- return {
90
+ return JSON.stringify({
91
91
  name: error.name,
92
92
  message: error.message,
93
93
  stack: error.stack
94
- };
94
+ });
95
95
  }
96
96
  return {
97
97
  message: String(error)
@@ -761,8 +761,20 @@ export var ScanOrderImpl = /*#__PURE__*/function (_BaseModule) {
761
761
  var ruleProduct = this.enabledReservationRuleProducts[0];
762
762
  var resourceState = this.store.resource;
763
763
  var resourceSelectType = resourceState === null || resourceState === void 0 ? void 0 : resourceState.resourceSelectType;
764
+
765
+ // single 预约独占整张桌台,capacity 取 form_record.capacity;其他情况(multiple / capacity / undefined)占 1 个容量位
766
+ var resolveResourceCapacity = function resolveResourceCapacity() {
767
+ if (resourceSelectType === 'single') {
768
+ var _resourceState$table_;
769
+ var raw = resourceState === null || resourceState === void 0 || (_resourceState$table_ = resourceState.table_form_record) === null || _resourceState$table_ === void 0 ? void 0 : _resourceState$table_.capacity;
770
+ var num = Number(raw);
771
+ if (Number.isFinite(num) && num > 0) return num;
772
+ return 1;
773
+ }
774
+ return 1;
775
+ };
764
776
  return function (payload, _ref) {
765
- var _tempOrder$resource_i, _ref2, _pickOriginal, _resourceState$table_, _resourceState$relati;
777
+ var _tempOrder$resource_i, _ref2, _pickOriginal, _resourceState$table_2, _resourceState$relati;
766
778
  var bookingUuid = _ref.bookingUuid,
767
779
  tempOrder = _ref.tempOrder;
768
780
  var resourceId = (_tempOrder$resource_i = tempOrder.resource_id) !== null && _tempOrder$resource_i !== void 0 ? _tempOrder$resource_i : resourceState === null || resourceState === void 0 ? void 0 : resourceState.relationId;
@@ -773,7 +785,7 @@ export var ScanOrderImpl = /*#__PURE__*/function (_BaseModule) {
773
785
  }
774
786
  return undefined;
775
787
  };
776
- var mainField = (_ref2 = (_pickOriginal = pickOriginal(tempOrder.table_number)) !== null && _pickOriginal !== void 0 ? _pickOriginal : pickOriginal(resourceState === null || resourceState === void 0 || (_resourceState$table_ = resourceState.table_form_record) === null || _resourceState$table_ === void 0 ? void 0 : _resourceState$table_.name)) !== null && _ref2 !== void 0 ? _ref2 : '';
788
+ var mainField = (_ref2 = (_pickOriginal = pickOriginal(tempOrder.table_number)) !== null && _pickOriginal !== void 0 ? _pickOriginal : pickOriginal(resourceState === null || resourceState === void 0 || (_resourceState$table_2 = resourceState.table_form_record) === null || _resourceState$table_2 === void 0 ? void 0 : _resourceState$table_2.name)) !== null && _ref2 !== void 0 ? _ref2 : '';
777
789
  var resourceEntry = resourceState && resourceId ? {
778
790
  relation_type: 'form',
779
791
  like_status: 'common',
@@ -781,15 +793,22 @@ export var ScanOrderImpl = /*#__PURE__*/function (_BaseModule) {
781
793
  main_field: mainField,
782
794
  form_id: resourceState.tableFormId,
783
795
  relation_id: (_resourceState$relati = resourceState.relationId) !== null && _resourceState$relati !== void 0 ? _resourceState$relati : resourceId,
784
- capacity: 1,
796
+ capacity: resolveResourceCapacity(),
785
797
  metadata: {}
786
798
  } : undefined;
787
799
  var ruleProductUid = ruleProduct ? createUuidV4() : undefined;
800
+ var bookingCapacityValue = resolveResourceCapacity();
788
801
  var nextBookings = (payload.bookings || []).map(function (booking, idx) {
789
802
  return _objectSpread(_objectSpread(_objectSpread({}, booking), {}, {
790
803
  appointment_status: 'started',
791
- metadata: _objectSpread(_objectSpread({}, booking.metadata || {}), resourceSelectType ? {
804
+ metadata: _objectSpread(_objectSpread(_objectSpread({}, booking.metadata || {}), resourceSelectType ? {
792
805
  resource_select_type: resourceSelectType
806
+ } : {}), resourceSelectType ? {
807
+ capacity: [{
808
+ id: 0,
809
+ value: bookingCapacityValue,
810
+ name: ''
811
+ }]
793
812
  } : {})
794
813
  }, idx === 0 && resourceEntry ? {
795
814
  resources: [resourceEntry]
@@ -1789,20 +1808,17 @@ export var ScanOrderImpl = /*#__PURE__*/function (_BaseModule) {
1789
1808
  }
1790
1809
  } else if (lastOrderId) {
1791
1810
  // 如果有其他人点的单
1792
- // TODO: 是否独占?如果非独占,其他人点的单应该不影响我下单
1793
- availabilityInfo = allowSnack ? {
1794
- mode: deskmateValid ? 'additional_order_with_code' : 'additional_order',
1795
- order_id: lastOrderId,
1796
- relation_id: relationId,
1797
- table_form_id: tableFormId,
1798
- deskmate_valid: deskmateValid
1799
- } : {
1800
- mode: 'resource_busy',
1801
- order_id: lastOrderId,
1802
- relation_id: relationId,
1803
- table_form_id: tableFormId,
1804
- deskmate_valid: deskmateValid
1805
- };
1811
+ // multiple 预约且容量仍有剩余时,其他人的单不影响当前用户下单,保持 idle 走正常下单流程
1812
+ var canFallthroughToIdle = resourceSelectType === 'multiple' && !isFull;
1813
+ if (!canFallthroughToIdle) {
1814
+ availabilityInfo = {
1815
+ mode: 'resource_busy',
1816
+ order_id: '0',
1817
+ relation_id: relationId,
1818
+ table_form_id: tableFormId,
1819
+ deskmate_valid: deskmateValid
1820
+ };
1821
+ }
1806
1822
  }
1807
1823
  return _objectSpread(_objectSpread({}, availabilityInfo), {}, {
1808
1824
  currentOrderId: currentOrderId,
@@ -2146,7 +2162,7 @@ export var ScanOrderImpl = /*#__PURE__*/function (_BaseModule) {
2146
2162
  case 78:
2147
2163
  _context28.prev = 78;
2148
2164
  _context28.t0 = _context28["catch"](1);
2149
- this.logMethodError('checkResourceAvailable', _context28.t0, {
2165
+ this.logMethodError('checkResourceAvailable', _context28.t0.message, {
2150
2166
  resourceId: resourceId
2151
2167
  });
2152
2168
  throw _context28.t0;
@@ -2225,7 +2241,7 @@ export var ScanOrderImpl = /*#__PURE__*/function (_BaseModule) {
2225
2241
  case 11:
2226
2242
  _context30.prev = 11;
2227
2243
  _context30.t0 = _context30["catch"](2);
2228
- this.logMethodError('getProductList', _context30.t0);
2244
+ this.logMethodError('getProductList', _context30.t0.message);
2229
2245
  throw _context30.t0;
2230
2246
  case 15:
2231
2247
  case "end":
@@ -86,7 +86,7 @@ export declare class OrderModule extends BaseModule implements Module, OrderModu
86
86
  enhancePayload?: SubmitPayloadEnhancer;
87
87
  }): Promise<T>;
88
88
  createOrder(params: CommitOrderParams['query']): {
89
- type: "virtual" | "appointment_booking";
89
+ type: "appointment_booking" | "virtual";
90
90
  platform: string;
91
91
  sales_channel: string;
92
92
  order_sales_channel: string;
@@ -22,58 +22,124 @@ __export(feishu_exports, {
22
22
  feishuLoggerProvider: () => feishuLoggerProvider
23
23
  });
24
24
  module.exports = __toCommonJS(feishu_exports);
25
- function safeStringify(payload) {
26
- if (payload === void 0 || payload === null)
25
+ var DEFAULT_THROTTLE_MS = 3e3;
26
+ var pendingQueue = [];
27
+ var flushTimer = null;
28
+ var cachedProviderConfig;
29
+ function safeStringify(value) {
30
+ if (value === void 0 || value === null)
27
31
  return "";
32
+ if (typeof value === "string")
33
+ return value;
34
+ if (typeof value === "number" || typeof value === "boolean" || typeof value === "bigint") {
35
+ return String(value);
36
+ }
28
37
  try {
29
- return JSON.stringify(payload);
38
+ return JSON.stringify(value);
30
39
  } catch (error) {
31
40
  return `[unserializable: ${String(error)}]`;
32
41
  }
33
42
  }
34
- function createFeishuMessageContent(contentArr) {
35
- return JSON.stringify(
36
- contentArr.map((item) => [
37
- { tag: "text", text: `${item.key}: ` },
38
- { tag: "text", text: item.value }
39
- ])
40
- );
43
+ function isWebhookUsable(webhook) {
44
+ if (!webhook)
45
+ return false;
46
+ if (webhook.includes("REPLACE_ME"))
47
+ return false;
48
+ return true;
49
+ }
50
+ function buildRecordSegments(record) {
51
+ const context = record.context || {};
52
+ return [
53
+ [{ tag: "text", text: `[${record.level}] ${record.timestamp} ${record.title}
54
+ ` }],
55
+ [{ tag: "text", text: `缓存标识: ${safeStringify(context.cacheId)}
56
+ ` }],
57
+ [{ tag: "text", text: `日志来源: ${safeStringify(context)}
58
+ ` }],
59
+ [{ tag: "text", text: `日志内容: ${safeStringify(record.payload)}
60
+ ` }],
61
+ [{ tag: "text", text: `扩展信息: ${safeStringify(record.extra)}
62
+ ` }],
63
+ [{ tag: "text", text: "------\n" }]
64
+ ];
65
+ }
66
+ function buildFeishuBody(records) {
67
+ const title = records.length === 1 ? records[0].title : `ScanOrder 日志批量 (${records.length} 条)`;
68
+ const content = [];
69
+ records.forEach((record) => {
70
+ buildRecordSegments(record).forEach((segment) => content.push(segment));
71
+ });
72
+ return JSON.stringify({
73
+ msg_type: "post",
74
+ content: JSON.stringify({
75
+ post: {
76
+ zh_cn: {
77
+ title,
78
+ content
79
+ }
80
+ }
81
+ })
82
+ });
83
+ }
84
+ async function postToFeishu(webhook, body) {
85
+ await fetch(webhook, {
86
+ method: "POST",
87
+ headers: {
88
+ "Content-Type": "application/json"
89
+ },
90
+ body
91
+ });
92
+ }
93
+ async function flushQueue() {
94
+ var _a;
95
+ flushTimer = null;
96
+ if (!pendingQueue.length)
97
+ return;
98
+ const records = pendingQueue.splice(0, pendingQueue.length);
99
+ const webhook = (_a = cachedProviderConfig == null ? void 0 : cachedProviderConfig.feishu) == null ? void 0 : _a.webhook;
100
+ if (!isWebhookUsable(webhook)) {
101
+ return;
102
+ }
103
+ if (typeof fetch !== "function") {
104
+ console.warn("[ScanOrderLogger] 当前环境不支持 fetch,跳过 Feishu 日志批量上报");
105
+ return;
106
+ }
107
+ try {
108
+ await postToFeishu(webhook, buildFeishuBody(records));
109
+ } catch (error) {
110
+ console.warn("[ScanOrderLogger] Feishu 批量上报失败", error);
111
+ }
41
112
  }
42
113
  var feishuLoggerProvider = {
43
114
  async send(payload) {
44
- var _a, _b;
115
+ var _a, _b, _c, _d;
116
+ cachedProviderConfig = payload.providerConfig;
45
117
  const webhook = (_b = (_a = payload.providerConfig) == null ? void 0 : _a.feishu) == null ? void 0 : _b.webhook;
46
- if (webhook && webhook.includes("REPLACE_ME"))
47
- return;
48
118
  if (!webhook) {
49
119
  console.warn("[ScanOrderLogger] Feishu webhook 未配置,跳过日志上报");
50
120
  return;
51
121
  }
122
+ if (!isWebhookUsable(webhook))
123
+ return;
52
124
  if (typeof fetch !== "function") {
53
125
  console.warn("[ScanOrderLogger] 当前环境不支持 fetch,跳过 Feishu 日志上报");
54
126
  return;
55
127
  }
56
- const { record } = payload;
57
- const contentArr = [
58
- { key: "日志级别", value: record.level },
59
- { key: "日志时间", value: record.timestamp },
60
- { key: "日志标题", value: record.title },
61
- { key: "缓存标识", value: record.context.cacheId || "" },
62
- { key: "日志来源", value: safeStringify(record.context) },
63
- { key: "日志内容", value: safeStringify(record.payload) },
64
- { key: "扩展信息", value: safeStringify(record.extra) }
65
- ];
66
- const contentStr = createFeishuMessageContent(contentArr);
67
- await fetch(webhook, {
68
- headers: {
69
- "Content-Type": "application/json"
70
- },
71
- method: "POST",
72
- body: JSON.stringify({
73
- msg_type: "post",
74
- content: `{"post":{"zh_cn":{"title":"${record.title}","content":${contentStr}}}}`
75
- })
76
- });
128
+ const throttleMs = ((_d = (_c = payload.providerConfig) == null ? void 0 : _c.feishu) == null ? void 0 : _d.throttleMs) ?? DEFAULT_THROTTLE_MS;
129
+ if (throttleMs <= 0) {
130
+ try {
131
+ await postToFeishu(webhook, buildFeishuBody([payload.record]));
132
+ } catch (error) {
133
+ console.warn("[ScanOrderLogger] Feishu 日志上报失败", error);
134
+ }
135
+ return;
136
+ }
137
+ pendingQueue.push(payload.record);
138
+ if (!flushTimer) {
139
+ flushTimer = setTimeout(() => {
140
+ void flushQueue();
141
+ }, throttleMs);
142
+ }
77
143
  }
78
144
  };
79
145
  // Annotate the CommonJS export names for ESM import in node:
@@ -2,6 +2,12 @@ export type ScanOrderLogLevel = 'info' | 'warning' | 'error' | 'debug';
2
2
  export type ScanOrderLoggerProviderType = 'feishu' | 'grafana';
3
3
  export interface ScanOrderLoggerProviderFeishuConfig {
4
4
  webhook?: string;
5
+ /**
6
+ * 节流窗口毫秒数,默认 3000ms
7
+ * - >0:首条日志进入队列并启动定时器,窗口结束后合并成一条 Feishu post 统一发送
8
+ * - <=0:关闭节流,保持每条日志立即单独发送
9
+ */
10
+ throttleMs?: number;
5
11
  }
6
12
  export interface ScanOrderLoggerProviderGrafanaConfig {
7
13
  endpoint?: string;
@@ -311,7 +311,7 @@ export declare class BookingByStepImpl extends BaseModule implements Module {
311
311
  date: string;
312
312
  status: string;
313
313
  week: string;
314
- weekNum: 0 | 1 | 2 | 3 | 4 | 5 | 6;
314
+ weekNum: 0 | 2 | 1 | 3 | 5 | 4 | 6;
315
315
  }[]>;
316
316
  submitTimeSlot(timeSlots: TimeSliceItem): void;
317
317
  private getScheduleDataByIds;
@@ -131,7 +131,7 @@ export declare class BookingTicketImpl extends BaseModule implements Module {
131
131
  * 获取当前的客户搜索条件
132
132
  * @returns 当前搜索条件
133
133
  */
134
- getCurrentCustomerSearchParams(): Omit<import("../../modules").ShopGetCustomerListParams, "num" | "skip">;
134
+ getCurrentCustomerSearchParams(): Omit<import("../../modules").ShopGetCustomerListParams, "skip" | "num">;
135
135
  /**
136
136
  * 获取客户列表状态(包含滚动加载相关状态)
137
137
  * @returns 客户状态
@@ -84,11 +84,11 @@ var _ScanOrderImpl = class extends import_BaseModule.BaseModule {
84
84
  }
85
85
  serializeError(error) {
86
86
  if (error instanceof Error) {
87
- return {
87
+ return JSON.stringify({
88
88
  name: error.name,
89
89
  message: error.message,
90
90
  stack: error.stack
91
- };
91
+ });
92
92
  }
93
93
  return {
94
94
  message: String(error)
@@ -503,6 +503,17 @@ var _ScanOrderImpl = class extends import_BaseModule.BaseModule {
503
503
  const ruleProduct = this.enabledReservationRuleProducts[0];
504
504
  const resourceState = this.store.resource;
505
505
  const resourceSelectType = resourceState == null ? void 0 : resourceState.resourceSelectType;
506
+ const resolveResourceCapacity = () => {
507
+ var _a;
508
+ if (resourceSelectType === "single") {
509
+ const raw = (_a = resourceState == null ? void 0 : resourceState.table_form_record) == null ? void 0 : _a.capacity;
510
+ const num = Number(raw);
511
+ if (Number.isFinite(num) && num > 0)
512
+ return num;
513
+ return 1;
514
+ }
515
+ return 1;
516
+ };
506
517
  return (payload, { bookingUuid, tempOrder }) => {
507
518
  var _a;
508
519
  const resourceId = tempOrder.resource_id ?? (resourceState == null ? void 0 : resourceState.relationId);
@@ -521,16 +532,18 @@ var _ScanOrderImpl = class extends import_BaseModule.BaseModule {
521
532
  main_field: mainField,
522
533
  form_id: resourceState.tableFormId,
523
534
  relation_id: resourceState.relationId ?? resourceId,
524
- capacity: 1,
535
+ capacity: resolveResourceCapacity(),
525
536
  metadata: {}
526
537
  } : void 0;
527
538
  const ruleProductUid = ruleProduct ? (0, import_utils2.createUuidV4)() : void 0;
539
+ const bookingCapacityValue = resolveResourceCapacity();
528
540
  const nextBookings = (payload.bookings || []).map((booking, idx) => ({
529
541
  ...booking,
530
542
  appointment_status: "started",
531
543
  metadata: {
532
544
  ...booking.metadata || {},
533
- ...resourceSelectType ? { resource_select_type: resourceSelectType } : {}
545
+ ...resourceSelectType ? { resource_select_type: resourceSelectType } : {},
546
+ ...resourceSelectType ? { capacity: [{ id: 0, value: bookingCapacityValue, name: "" }] } : {}
534
547
  },
535
548
  ...idx === 0 && resourceEntry ? { resources: [resourceEntry] } : {},
536
549
  ...idx === 0 && ruleProductUid ? { product_uid: ruleProductUid } : {}
@@ -1054,19 +1067,16 @@ var _ScanOrderImpl = class extends import_BaseModule.BaseModule {
1054
1067
  };
1055
1068
  }
1056
1069
  } else if (lastOrderId) {
1057
- availabilityInfo = allowSnack ? {
1058
- mode: deskmateValid ? "additional_order_with_code" : "additional_order",
1059
- order_id: lastOrderId,
1060
- relation_id: relationId,
1061
- table_form_id: tableFormId,
1062
- deskmate_valid: deskmateValid
1063
- } : {
1064
- mode: "resource_busy",
1065
- order_id: lastOrderId,
1066
- relation_id: relationId,
1067
- table_form_id: tableFormId,
1068
- deskmate_valid: deskmateValid
1069
- };
1070
+ const canFallthroughToIdle = resourceSelectType === "multiple" && !isFull;
1071
+ if (!canFallthroughToIdle) {
1072
+ availabilityInfo = {
1073
+ mode: "resource_busy",
1074
+ order_id: "0",
1075
+ relation_id: relationId,
1076
+ table_form_id: tableFormId,
1077
+ deskmate_valid: deskmateValid
1078
+ };
1079
+ }
1070
1080
  }
1071
1081
  return {
1072
1082
  ...availabilityInfo,
@@ -1319,7 +1329,7 @@ var _ScanOrderImpl = class extends import_BaseModule.BaseModule {
1319
1329
  });
1320
1330
  return availabilityInfo;
1321
1331
  } catch (error) {
1322
- this.logMethodError("checkResourceAvailable", error, {
1332
+ this.logMethodError("checkResourceAvailable", error.message, {
1323
1333
  resourceId
1324
1334
  });
1325
1335
  throw error;
@@ -1362,7 +1372,7 @@ var _ScanOrderImpl = class extends import_BaseModule.BaseModule {
1362
1372
  });
1363
1373
  return formattedRes;
1364
1374
  } catch (error) {
1365
- this.logMethodError("getProductList", error);
1375
+ this.logMethodError("getProductList", error.message);
1366
1376
  throw error;
1367
1377
  }
1368
1378
  }
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "private": false,
3
3
  "name": "@pisell/pisellos",
4
- "version": "0.0.505",
4
+ "version": "0.0.507",
5
5
  "description": "一个可扩展的前端模块化SDK框架,支持插件系统",
6
6
  "main": "dist/index.js",
7
7
  "types": "dist/index.d.ts",