trangleCalc.js 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204
  1. const crypto = require('crypto');
  2. const Logs = require('../libs/logs');
  3. const IOR_KEYS_MAP = require('./iorKeys');
  4. const { getSetting } = require('./settings');
  5. const { eventSolutions } = require('./eventSolutions');
  6. /**
  7. * 筛选最优赔率
  8. */
  9. const oddRebateValue = (odds, platform, key) => {
  10. const setting = getSetting();
  11. let rebateRatio = setting[`${platform}RebateRatio`] ?? 0;
  12. let rebateType = setting[`${platform}RebateType`] ?? 0;
  13. if (platform == 'hg' && key.startsWith('ior_m')) {
  14. rebateRatio -= 0.75;
  15. }
  16. if (rebateType == 0) {
  17. return odds * (1 + rebateRatio / 100);
  18. }
  19. return odds + rebateRatio / 100;
  20. }
  21. const getOptimalSelections = (odds, rules) => {
  22. const results = [];
  23. rules.forEach((rule, ruleIndex) => {
  24. let validOptions = [];
  25. for (let i = 0; i < 3; i++) {
  26. const innerIndex = i;
  27. const selection = [];
  28. let isValid = true;
  29. for (let j = 0; j < 3; j++) {
  30. const key = rule[j];
  31. let item = odds[key];
  32. if (key == '-') {
  33. item = { no: { v: 1 } };
  34. }
  35. if (!item) {
  36. isValid = false;
  37. break;
  38. }
  39. if (j === innerIndex) {
  40. if (!('ps' in item)) {
  41. isValid = false;
  42. break;
  43. }
  44. selection.push({
  45. k: key,
  46. p: 'ps',
  47. v: item.ps.v,
  48. r: item.ps.r,
  49. s: item.ps.s,
  50. o: item
  51. });
  52. }
  53. else {
  54. const candidates = ['ob', 'hg', 'no'].filter((k) => k in item);
  55. if (candidates.length === 0) {
  56. isValid = false;
  57. break;
  58. }
  59. // Logs.out('candidates', candidates)
  60. const best = candidates.reduce((a, b) => {
  61. const aValue = oddRebateValue(item[a].v-1, a, key);
  62. const bValue = oddRebateValue(item[b].v-1, b, key);
  63. const seletcted = aValue > bValue ? a : b;
  64. return seletcted;
  65. });
  66. // Logs.out('best', item, best)
  67. selection.push({
  68. k: key,
  69. p: best,
  70. v: item[best].v,
  71. r: item[best].r,
  72. s: item[best].s,
  73. o: item
  74. });
  75. }
  76. }
  77. if (isValid) {
  78. validOptions.push(selection);
  79. }
  80. }
  81. validOptions.forEach(iors => {
  82. results.push({ rule, iors, ruleIndex });
  83. });
  84. // if (validOptions.length > 0) {
  85. // const iors = validOptions.reduce((a, b) => {
  86. // const sumA = a.reduce((sum, x) => sum + x.v, 0);
  87. // const sumB = b.reduce((sum, x) => sum + x.v, 0);
  88. // return sumA > sumB ? a : b;
  89. // });
  90. // results.push({ rule, iors, ruleIndex });
  91. // }
  92. });
  93. return results;
  94. }
  95. /**
  96. * 精确浮点数字
  97. * @param {number} number
  98. * @param {number} x
  99. * @returns {number}
  100. */
  101. const fixFloat = (number, x=2) => {
  102. return parseFloat(number.toFixed(x));
  103. }
  104. /**
  105. * 盘口排序
  106. */
  107. const priority = { ps: 1, ob: 2, hg: 3 };
  108. const sortCpr = (cpr) => {
  109. const temp = [...cpr];
  110. temp.sort((a, b) => priority[a.p] - priority[b.p]);
  111. return temp;
  112. }
  113. /**
  114. * 添加返佣
  115. */
  116. const attachRebate = (ior) => {
  117. const { obRebateRatio, obRebateType, hgRebateRatio, hgRebateType } = getSetting();
  118. const { p, k } = ior;
  119. let rebate = 0, rebateType = 0;
  120. if (p == 'ps') {
  121. rebate = 0;
  122. rebateType = -1;
  123. }
  124. else if (p == 'ob') {
  125. rebate = obRebateRatio;
  126. rebateType = obRebateType;
  127. }
  128. else if (p == 'hg') {
  129. rebate = hgRebateRatio;
  130. rebateType = hgRebateType;
  131. if (k.startsWith('ior_m')) {
  132. rebate -= 0.75;
  133. }
  134. }
  135. return { ...ior, b: rebate, t: rebateType };
  136. }
  137. const eventsCombination = (passableEvents) => {
  138. const { innerDefaultAmount, innerRebateRatio } = getSetting();
  139. const solutions = [];
  140. passableEvents.forEach(events => {
  141. const { odds, info } = events;
  142. Object.keys(IOR_KEYS_MAP).forEach(iorGroup => {
  143. const rules = IOR_KEYS_MAP[iorGroup];
  144. const optimalSelections = getOptimalSelections(odds, rules);
  145. optimalSelections.forEach(selection => {
  146. const { rule, iors, ruleIndex } = selection;
  147. const [, , , crossType] = rule;
  148. const oddsSideA = attachRebate(iors[0]);
  149. const oddsSideB = attachRebate(iors[1]);
  150. const oddsSideC = attachRebate(iors[2]);
  151. const innerIndex = iors.findIndex(item => item.p == 'ps');
  152. if (!oddsSideA || !oddsSideB || !oddsSideC) {
  153. return;
  154. }
  155. const cpr = [ oddsSideA, oddsSideB, oddsSideC ];
  156. const betInfo = {
  157. cross_type: crossType,
  158. inner_index: innerIndex,
  159. inner_base: innerDefaultAmount,
  160. inner_rebate: fixFloat(innerRebateRatio / 100, 3),
  161. odds_side_a: fixFloat(oddsSideA.v - 1),
  162. odds_side_b: fixFloat(oddsSideB.v - 1),
  163. odds_side_c: fixFloat(oddsSideC.v - 1),
  164. rebate_side_a: parseFloat((oddsSideA.b / 100).toFixed(4)),
  165. rebate_type_side_a: oddsSideA.t,
  166. rebate_side_b: parseFloat((oddsSideB.b / 100).toFixed(4)),
  167. rebate_type_side_b: oddsSideB.t,
  168. rebate_side_c: parseFloat((oddsSideC.b / 100).toFixed(4)),
  169. rebate_type_side_c: oddsSideC.t,
  170. };
  171. const sol = eventSolutions(betInfo, true);
  172. if (cpr[2].k == '-') {
  173. cpr.pop();
  174. }
  175. if (!isNaN(sol?.win_average)) {
  176. const id = info.id;
  177. const sortedCpr = sortCpr(cpr);
  178. const keys = sortedCpr.map(item => `${item.k}`).join('_');
  179. const sid = crypto.createHash('sha1').update(`${id}_${keys}`).digest('hex');
  180. const crpGroup = `${id}_${sortedCpr[0].k}`;
  181. const timestamp = Date.now();
  182. solutions.push({sid, sol, cpr, info, group: crpGroup, rule: `${iorGroup}:${ruleIndex}`, timestamp});
  183. }
  184. });
  185. });
  186. });
  187. return solutions.sort((a, b) => {
  188. return b.sol.win_average - a.sol.win_average;
  189. });
  190. }
  191. module.exports = { eventsCombination };