trangleCalc.js 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248
  1. const Logs = require('../libs/logs');
  2. const IOR_KEYS_MAP = require('./iorKeys');
  3. const GOLD_BASE = 1000;
  4. const WIN_MIN = process.env.NODE_ENV == 'development' ? -50 : -10;
  5. const JC_REBATE_RATIO = 0.07;
  6. /**
  7. * 筛选最优赔率
  8. */
  9. function getOptimalSelections(data, combos) {
  10. const results = [];
  11. combos.forEach((rule, index) => {
  12. let validOptions = [];
  13. for (let i = 0; i < 3; i++) {
  14. const jcIndex = i;
  15. const selection = [];
  16. let isValid = true;
  17. for (let j = 0; j < 3; j++) {
  18. const key = rule[j];
  19. const item = data[key];
  20. if (!item) {
  21. isValid = false;
  22. break;
  23. }
  24. if (j === jcIndex) {
  25. if (!('jc' in item)) {
  26. isValid = false;
  27. break;
  28. }
  29. selection.push({
  30. k: key,
  31. p: 'jc',
  32. v: item.jc,
  33. o: item,
  34. });
  35. }
  36. else {
  37. const candidates = ['ps', 'ob'].filter((k) => k in item);
  38. if (candidates.length === 0) {
  39. isValid = false;
  40. break;
  41. }
  42. const best = candidates.reduce((a, b) => item[a] > item[b] ? a : b);
  43. selection.push({
  44. k: key,
  45. p: best,
  46. v: item[best],
  47. o: item,
  48. });
  49. }
  50. }
  51. if (isValid) {
  52. validOptions.push(selection);
  53. }
  54. }
  55. if (validOptions.length > 0) {
  56. const iors = validOptions.reduce((a, b) => {
  57. const sumA = a.reduce((sum, x) => sum + x.v, 0);
  58. const sumB = b.reduce((sum, x) => sum + x.v, 0);
  59. return sumA > sumB ? a : b;
  60. });
  61. results.push({ rule, iors, index });
  62. }
  63. });
  64. return results;
  65. }
  66. /**
  67. * 精确浮点数字
  68. * @param {number} number
  69. * @param {number} x
  70. * @returns {number}
  71. */
  72. const fixFloat = (number, x=2) => {
  73. return parseFloat(number.toFixed(x));
  74. }
  75. /**
  76. * 计算盈利
  77. */
  78. const triangleProfitCalc = (goldsInfo, oddsOption) => {
  79. const {
  80. gold_home: x,
  81. gold_away: y,
  82. gold_special: z,
  83. odds_home: a,
  84. odds_away: b,
  85. odds_special: c
  86. } = goldsInfo;
  87. const { crossType, jcIndex } = oddsOption;
  88. /**
  89. * crossType:
  90. * la: 全输
  91. * wa: 全赢
  92. * lh: 半输
  93. * wh: 半赢
  94. * dr: 和局
  95. * la_wh_wa, la_dr_wa, la_lh_wa, lh_dr_wa, lh_lh_wa, la_la_wa
  96. */
  97. let jc_rebate = 0;
  98. if (jcIndex == 0) {
  99. jc_rebate = x * JC_REBATE_RATIO;
  100. }
  101. else if (jcIndex == 1) {
  102. jc_rebate = y * JC_REBATE_RATIO;
  103. }
  104. else if (jcIndex == 2) {
  105. jc_rebate = z * JC_REBATE_RATIO;
  106. }
  107. let win_home = 0, win_away = 0, win_special = 0;
  108. win_home = fixFloat(a * x - y - z + jc_rebate);
  109. win_away = fixFloat(b * y - x - z + jc_rebate);
  110. switch (crossType) {
  111. case 'la_wh_wa': // 全输 半赢 全赢
  112. win_special = c*z - x + b*y/2;
  113. break;
  114. case 'la_dr_wa': // 全输 和局 全赢
  115. win_special = c*z - x;
  116. break;
  117. case 'la_lh_wa': // 全输 半输 全赢
  118. win_special = c*z - x - y/2;
  119. break;
  120. case 'lh_dr_wa': // 半输 和局 全赢
  121. win_special = c*z - x/2;
  122. break;
  123. case 'lh_lh_wa': // 半输 半输 全赢
  124. win_special = c*z - x/2 - y/2;
  125. break;
  126. case 'la_la_wa': // 全输 全输 全赢
  127. win_special = c*z - x - y;
  128. break;
  129. }
  130. win_special = fixFloat(win_special + jc_rebate);
  131. const win_average = fixFloat((win_home + win_away + win_special) / 3);
  132. return { win_home, win_away, win_special, win_average }
  133. }
  134. const triangleGoldCalc = (oddsInfo, oddsOption) => {
  135. const { odds_home: a, odds_away: b, odds_special: c } = oddsInfo;
  136. if (!a || !b || !c) {
  137. return;
  138. }
  139. const { crossType, jcIndex } = oddsOption;
  140. const x = GOLD_BASE;
  141. const y = (a + 1) * x / (b + 1);
  142. let z;
  143. switch (crossType) {
  144. case 'la_wh_wa': // 全输 半赢 全赢
  145. z = b * y / 2 / (c + 1);
  146. break;
  147. case 'la_dr_wa': // 全输 和局 全赢
  148. z = b * y / (c + 1);
  149. break;
  150. case 'la_lh_wa': // 全输 半输 全赢
  151. z = (2 * b + 1) * y / 2 / (c + 1);
  152. break;
  153. case 'lh_dr_wa': // 半输 和局 全赢
  154. z = (b * y - x / 2) / (c + 1);
  155. break;
  156. case 'lh_lh_wa': // 半输 半输 全赢
  157. z = (b * y + y / 2 - x / 2) / (c + 1);
  158. break;
  159. case 'la_la_wa': // 全输 全输 全赢
  160. z = (b + 1) * y / (c + 1);
  161. break;
  162. default:
  163. z = 0;
  164. }
  165. return {
  166. gold_home: x,
  167. gold_away: fixFloat(y),
  168. gold_special: fixFloat(z),
  169. odds_home: a,
  170. odds_away: b,
  171. odds_special: c
  172. };
  173. }
  174. const eventSolutions = (oddsInfo, oddsOption) => {
  175. const { reverse } = oddsOption;
  176. const goldsInfo = triangleGoldCalc(oddsInfo, oddsOption);
  177. if (!goldsInfo) {
  178. return;
  179. }
  180. const profitInfo = triangleProfitCalc(goldsInfo, oddsOption);
  181. return {
  182. ...goldsInfo,
  183. ...profitInfo,
  184. reverse,
  185. }
  186. }
  187. const eventsCombination = (passableEvents) => {
  188. const solutions = [];
  189. passableEvents.forEach(events => {
  190. const { odds, info } = events;
  191. Object.keys(IOR_KEYS_MAP).forEach(group => {
  192. const rules = IOR_KEYS_MAP[group];
  193. const optimalSelections = getOptimalSelections(odds, rules);
  194. optimalSelections.forEach(selection => {
  195. const { rule, iors, index } = selection;
  196. const [, , , crossType, reverse] = rule;
  197. const oddsHome = iors[0];
  198. const oddsAway = iors[1];
  199. const oddsSpecial = iors[2];
  200. const jcIndex = iors.findIndex(item => item.p == 'jc');
  201. if (!oddsHome || !oddsAway || !oddsSpecial) {
  202. return;
  203. }
  204. const cpr = [ oddsHome, oddsAway, oddsSpecial ];
  205. const oddsInfo = {
  206. odds_home: fixFloat(oddsHome.v - 1),
  207. odds_away: fixFloat(oddsAway.v - 1),
  208. odds_special: fixFloat(oddsSpecial.v - 1)
  209. };
  210. const oddsOption = { crossType, reverse, jcIndex };
  211. const sol = eventSolutions(oddsInfo, oddsOption);
  212. if (sol?.win_average > WIN_MIN) {
  213. const id = info.id;
  214. const keys = cpr.map(item => item.k).join('_');
  215. const sid = `${id}_${keys}`;
  216. const timestamp = Date.now();
  217. solutions.push({sid, sol, cpr, info, rule: `${group}:${index}`, timestamp});
  218. }
  219. });
  220. });
  221. });
  222. return solutions;
  223. }
  224. module.exports = eventsCombination;