From 51bc045b97d579853688a4f5607a1469250daa2d Mon Sep 17 00:00:00 2001 From: Rikard Blixt Date: Wed, 9 Oct 2024 16:52:40 +0200 Subject: [PATCH] refactor(match2): MLBB MGI (#4827) * refactor(match2): MLBB MGI * fixes * fix extradata --- .../match_group_input_custom.lua | 573 ++++-------------- 1 file changed, 130 insertions(+), 443 deletions(-) diff --git a/components/match2/wikis/mobilelegends/match_group_input_custom.lua b/components/match2/wikis/mobilelegends/match_group_input_custom.lua index e33b189a729..dd2145f8ddd 100644 --- a/components/match2/wikis/mobilelegends/match_group_input_custom.lua +++ b/components/match2/wikis/mobilelegends/match_group_input_custom.lua @@ -7,523 +7,210 @@ -- local Array = require('Module:Array') -local DateExt = require('Module:Date/Ext') local FnUtil = require('Module:FnUtil') local Logic = require('Module:Logic') local Lua = require('Module:Lua') +local Operator = require('Module:Operator') local String = require('Module:StringUtils') local Table = require('Module:Table') local Variables = require('Module:Variables') -local ChampionNames = mw.loadData('Module:HeroNames') local Streams = require('Module:Links/Stream') +local ChampionNames = mw.loadData('Module:HeroNames') -local MatchGroupInput = Lua.import('Module:MatchGroup/Input/Util') -local Opponent = Lua.import('Module:Opponent') - -local STATUS_SCORE = 'S' -local STATUS_DRAW = 'D' -local STATUS_DEFAULT_WIN = 'W' -local STATUS_FORFEIT = 'FF' -local STATUS_DISQUALIFIED = 'DQ' -local STATUS_DEFAULT_LOSS = 'L' -local ALLOWED_STATUSES = { - STATUS_DRAW, - STATUS_DEFAULT_WIN, - STATUS_FORFEIT, - STATUS_DISQUALIFIED, - STATUS_DEFAULT_LOSS, -} -local MAX_NUM_OPPONENTS = 2 -local MAX_NUM_PLAYERS = 5 -local DEFAULT_BESTOF = 3 -local DEFAULT_MODE = 'team' -local NO_SCORE = -99 -local DUMMY_MAP = 'default' -local NP_STATUSES = {'skip', 'np', 'canceled', 'cancelled'} -local DEFAULT_RESULT_TYPE = 'default' -local NOT_PLAYED_SCORE = -1 -local NOW = os.time(os.date('!*t') --[[@as osdateparam]]) +local MatchGroupInputUtil = Lua.import('Module:MatchGroup/Input/Util') + +local DEFAULT_BESTOF_MATCH = 3 -- containers for process helper functions -local matchFunctions = {} -local mapFunctions = {} +local MatchFunctions = {} +local MapFunctions = {} local CustomMatchGroupInput = {} --- called from Module:MatchGroup+ ---@param match table ---@param options table? ---@return table function CustomMatchGroupInput.processMatch(match, options) - -- Count number of maps, check for empty maps to remove, and automatically count score - match = matchFunctions.getBestOf(match) - match = matchFunctions.getScoreFromMapWinners(match) - - -- process match - Table.mergeInto( - match, - matchFunctions.readDate(match) - ) - match = matchFunctions.getOpponents(match) - match = matchFunctions.getTournamentVars(match) - match = matchFunctions.getVodStuff(match) - match = matchFunctions.getExtraData(match) - - -- Adjust map data, especially set participants data - match = matchFunctions.adjustMapData(match) + local finishedInput = match.finished --[[@as string?]] + local winnerInput = match.winner --[[@as string?]] - return match -end - ----@param match table ----@return table -function matchFunctions.adjustMapData(match) - local opponents = {} - for opponentIndex = 1, MAX_NUM_OPPONENTS do - opponents[opponentIndex] = match['opponent' .. opponentIndex] - end - local mapIndex = 1 - while match['map'..mapIndex] do - match['map'..mapIndex] = mapFunctions.getParticipants(match['map'..mapIndex], opponents) - mapIndex = mapIndex + 1 - end - - return match -end + Table.mergeInto(match, MatchGroupInputUtil.readDate(match.date, {'tournament_enddate'})) --- called from Module:Match/Subobjects ----@param map table ----@return table -function CustomMatchGroupInput.processMap(map) - if map.map == DUMMY_MAP then - map.map = nil - end - map = mapFunctions.getScoresAndWinner(map) + local opponents = Array.mapIndexes(function(opponentIndex) + return MatchGroupInputUtil.readOpponent(match, opponentIndex, {}) + end) + local games = CustomMatchGroupInput.extractMaps(match, opponents) + match.bestof = MatchFunctions.getBestOf(match) - return map -end + local autoScoreFunction = MatchGroupInputUtil.canUseAutoScore(match, games) + and MatchFunctions.calculateMatchScore(games, match.bestof) + or nil + Array.forEach(opponents, function(opponent, opponentIndex) + opponent.score, opponent.status = MatchGroupInputUtil.computeOpponentScore({ + walkover = match.walkover, + winner = match.winner, + opponentIndex = opponentIndex, + score = opponent.score, + }, autoScoreFunction) + end) ----@param record table ----@param timestamp integer -function CustomMatchGroupInput.processOpponent(record, timestamp) - local opponent = Opponent.readOpponentArgs(record) - or Opponent.blank() + match.finished = MatchGroupInputUtil.matchIsFinished(match, opponents) - -- Convert byes to literals - if Opponent.isBye(opponent) then - opponent = {type = Opponent.literal, name = 'BYE'} + if match.finished then + match.resulttype = MatchGroupInputUtil.getResultType(winnerInput, finishedInput, opponents) + match.walkover = MatchGroupInputUtil.getWalkover(match.resulttype, opponents) + match.winner = MatchGroupInputUtil.getWinner(match.resulttype, winnerInput, opponents) + MatchGroupInputUtil.setPlacement(opponents, match.winner, 1, 2, match.resulttype) end - ---@type number|string - local teamTemplateDate = timestamp - -- If date is default date, resolve using tournament dates instead - -- default date indicates that the match is missing a date - -- In order to get correct child team template, we will use an approximately date and not the default date - if teamTemplateDate == DateExt.defaultTimestamp then - teamTemplateDate = Variables.varDefaultMulti('tournament_enddate', 'tournament_startdate', NOW) - end + MatchFunctions.getTournamentVars(match) - Opponent.resolve(opponent, teamTemplateDate, {syncPlayer=true}) - MatchGroupInput.mergeRecordWithOpponent(record, opponent) -end + match.stream = Streams.processStreams(match) ----@param data table ----@param indexedScores table[] ----@return table ----@return table[] -function CustomMatchGroupInput.getResultTypeAndWinner(data, indexedScores) - -- Map or Match wasn't played, set not played - if - Table.includes(NP_STATUSES, data.finished) or - Table.includes(NP_STATUSES, data.winner) - then - data.resulttype = 'np' - data.finished = true - -- Map or Match is marked as finished. - -- Calculate and set winner, resulttype, placements and walkover (if applicable for the outcome) - elseif Logic.readBool(data.finished) then - if MatchGroupInput.isDraw(indexedScores) then - data.winner = 0 - data.resulttype = 'draw' - indexedScores = CustomMatchGroupInput.setPlacement(indexedScores, data.winner, 'draw') - elseif CustomMatchGroupInput.placementCheckSpecialStatus(indexedScores) then - data.winner = MatchGroupInput.getDefaultWinner(indexedScores) - data.resulttype = DEFAULT_RESULT_TYPE - if MatchGroupInput.hasForfeit(indexedScores) then - data.walkover = 'ff' - elseif MatchGroupInput.hasDisqualified(indexedScores) then - data.walkover = 'dq' - elseif MatchGroupInput.hasDefaultWinLoss(indexedScores) then - data.walkover = 'l' - end - indexedScores = CustomMatchGroupInput.setPlacement(indexedScores, data.winner, DEFAULT_RESULT_TYPE) - else - local winner - indexedScores, winner = CustomMatchGroupInput.setPlacement(indexedScores, data.winner, nil, data.finished) - data.winner = data.winner or winner - end - end + match.games = games + match.opponents = opponents - --set it as finished if we have a winner - if not Logic.isEmpty(data.winner) then - data.finished = true - end + match.extradata = MatchFunctions.getExtraData(match) - return data, indexedScores + return match end +---@param match table ---@param opponents table[] ----@param winner integer? ----@param specialType string? ----@param finished boolean|string? ---@return table[] ----@return integer? -function CustomMatchGroupInput.setPlacement(opponents, winner, specialType, finished) - if specialType == 'draw' then - for key, _ in pairs(opponents) do - opponents[key].placement = 1 - end - elseif specialType == DEFAULT_RESULT_TYPE then - for key, _ in pairs(opponents) do - if key == winner then - opponents[key].placement = 1 - else - opponents[key].placement = 2 - end - end - else - local lastScore = NO_SCORE - local lastPlacement = NO_SCORE - local counter = 0 - for scoreIndex, opp in Table.iter.spairs(opponents, CustomMatchGroupInput.placementSortFunction) do - local score = tonumber(opp.score) - counter = counter + 1 - if counter == 1 and (winner or '') == '' then - if finished then - winner = scoreIndex - end - end - if lastScore == score then - opponents[scoreIndex].placement = tonumber(opponents[scoreIndex].placement or '') or lastPlacement - else - opponents[scoreIndex].placement = tonumber(opponents[scoreIndex].placement or '') or counter - lastPlacement = counter - lastScore = score or NO_SCORE - end +function CustomMatchGroupInput.extractMaps(match, opponents) + local maps = {} + for key, map, mapIndex in Table.iter.pairsByPrefix(match, 'map', {requireIndex = true}) do + local finishedInput = map.finished --[[@as string?]] + local winnerInput = map.winner --[[@as string?]] + + map.vod = map.vod or String.nilIfEmpty(match['vodgame' .. mapIndex]) + map.participants = MapFunctions.getParticipants(map, opponents) + map.extradata = MapFunctions.getExtraData(map, #opponents) + + local opponentInfo = Array.map(opponents, function(_, opponentIndex) + local score, status = MatchGroupInputUtil.computeOpponentScore({ + walkover = map.walkover, + winner = map.winner, + opponentIndex = opponentIndex, + score = map['score' .. opponentIndex], + }) + return {score = score, status = status} + end) + + map.finished = MatchGroupInputUtil.mapIsFinished(map, opponentInfo) + + map.scores = Array.map(opponentInfo, Operator.property('score')) + if map.finished then + map.resulttype = MatchGroupInputUtil.getResultType(winnerInput, finishedInput, opponentInfo) + map.walkover = MatchGroupInputUtil.getWalkover(map.resulttype, opponentInfo) + map.winner = MatchGroupInputUtil.getWinner(map.resulttype, winnerInput, opponentInfo) end - end - return opponents, winner -end + table.insert(maps, map) + match[key] = nil + end ----@param tbl table[] ----@param key1 integer ----@param key2 integer ----@return boolean -function CustomMatchGroupInput.placementSortFunction(tbl, key1, key2) - local value1 = tonumber(tbl[key1].score or NO_SCORE) or NO_SCORE - local value2 = tonumber(tbl[key2].score or NO_SCORE) or NO_SCORE - return value1 > value2 + return maps end --- Check if any opponent has a none-standard status ----@param tbl table ----@return boolean -function CustomMatchGroupInput.placementCheckSpecialStatus(tbl) - return Table.any(tbl, - function (_, scoreinfo) - return scoreinfo.status ~= STATUS_SCORE and String.isNotEmpty(scoreinfo.status) - end - ) -end +CustomMatchGroupInput.processMap = FnUtil.identity -- -- match related functions -- ----@param match table ----@return table -function matchFunctions.getBestOf(match) - match.bestof = Logic.emptyOr(match.bestof, Variables.varDefault('bestof', DEFAULT_BESTOF)) - Variables.varDefine('bestof', match.bestof) - return match -end - --- Calculate the match scores based on the map results (counting map wins) --- Only update an opponents result if it's --- 1) Not manually added --- 2) At least one map has a winner ----@param match table ----@return table -function matchFunctions.getScoreFromMapWinners(match) - local newScores = {} - local foundScores = false - - local mapIndex = 1 - while match['map'..mapIndex] do - local winner = tonumber(match['map'..mapIndex].winner) - foundScores = true - if winner and winner > 0 and winner <= MAX_NUM_OPPONENTS then - newScores[winner] = (newScores[winner] or 0) + 1 - end - mapIndex = mapIndex + 1 - end - - for index = 1, MAX_NUM_OPPONENTS do - if not match['opponent' .. index].score and foundScores then - match['opponent' .. index].score = newScores[index] or 0 - end +---@param maps table[] +---@param bestOf integer +---@return fun(opponentIndex: integer): integer? +function MatchFunctions.calculateMatchScore(maps, bestOf) + return function(opponentIndex) + return MatchGroupInputUtil.computeMatchScoreFromMapWinners(maps, opponentIndex) end - - return match -end - ----@param matchArgs table ----@return {date: string, dateexact: boolean, timestamp: integer, timezoneId: string?, timezoneOffset: string?} -function matchFunctions.readDate(matchArgs) - return MatchGroupInput.readDate(matchArgs.date, { - 'tournament_enddate', - 'tournament_startdate', - }) end ---@param match table ----@return table -function matchFunctions.getTournamentVars(match) - match.mode = Logic.emptyOr(match.mode, Variables.varDefault('tournament_mode', DEFAULT_MODE)) - match.publishertier = Logic.emptyOr(match.publishertier, Variables.varDefault('tournament_publishertier')) - return MatchGroupInput.getCommonTournamentVars(match) +---@return integer +function MatchFunctions.getBestOf(match) + local bestof = tonumber(Logic.emptyOr(match.bestof, Variables.varDefault('bestof'))) + Variables.varDefine('bestof', bestof) + return bestof or DEFAULT_BESTOF_MATCH end ---@param match table ---@return table -function matchFunctions.getVodStuff(match) - match.stream = Streams.processStreams(match) - match.vod = Logic.emptyOr(match.vod, Variables.varDefault('vod')) - - match.links = {} - local links = match.links - if match.reddit then links.reddit = match.reddit end - - return match +function MatchFunctions.getTournamentVars(match) + match.mode = Logic.emptyOr(match.mode, Variables.varDefault('tournament_mode', 'team')) + return MatchGroupInputUtil.getCommonTournamentVars(match) end ---@param match table ---@return table -function matchFunctions.getExtraData(match) - match.extradata = { - mvp = MatchGroupInput.readMvp(match), - mvpteam = match.mvpteam or match.winner, - casters = MatchGroupInput.readCasters(match, {noSort = true}), +function MatchFunctions.getExtraData(match) + return { + mvp = MatchGroupInputUtil.readMvp(match), + casters = MatchGroupInputUtil.readCasters(match, {noSort = true}), } - return match -end - ----@param match table ----@return table -function matchFunctions.getOpponents(match) - -- read opponents and ignore empty ones - local opponents = {} - local isScoreSet = false - for opponentIndex = 1, MAX_NUM_OPPONENTS do - -- read opponent - local opponent = match['opponent' .. opponentIndex] - if not Logic.isEmpty(opponent) then - CustomMatchGroupInput.processOpponent(opponent, match.timestamp) - - -- apply status - opponent.score = string.upper(opponent.score or '') - if Logic.isNumeric(opponent.score) then - opponent.score = tonumber(opponent.score) - opponent.status = STATUS_SCORE - isScoreSet = true - elseif Table.includes(ALLOWED_STATUSES, opponent.score) then - opponent.status = opponent.score - opponent.score = NOT_PLAYED_SCORE - end - - -- get players from vars for teams - assert(Opponent.isType(opponent.type), 'Unsupported Opponent Type "' .. (opponent.type or '') .. '"') - if opponent.type == Opponent.team then - if not Logic.isEmpty(opponent.name) then - match = MatchGroupInput.readPlayersOfTeam(match, opponentIndex, opponent.name, { - resolveRedirect = true, - applyUnderScores = true, - maxNumPlayers = MAX_NUM_PLAYERS, - }) - end - end - - opponents[opponentIndex] = opponent - end - end - - --apply walkover input - match.walkover = string.upper(match.walkover or '') - if Logic.isNumeric(match.walkover) then - local winnerIndex = tonumber(match.walkover) - opponents = matchFunctions._makeAllOpponentsLoseByWalkover(opponents, STATUS_DEFAULT_LOSS) - opponents[winnerIndex].status = STATUS_DEFAULT_WIN - match.finished = true - elseif Logic.isNumeric(match.winner) and Table.includes(ALLOWED_STATUSES, match.walkover) then - local winnerIndex = tonumber(match.winner) - opponents = matchFunctions._makeAllOpponentsLoseByWalkover(opponents, match.walkover) - opponents[winnerIndex].status = STATUS_DEFAULT_WIN - match.finished = true - end - - -- see if match should actually be finished if bestof limit was reached - match.finished = Logic.readBool(match.finished) - or isScoreSet and ( - Array.any(opponents, function(opponent) return tonumber(opponent.score or 0) > match.bestof/2 end) - or Array.all(opponents, function(opponent) return tonumber(opponent.score or 0) == match.bestof/2 end) - ) - - -- see if match should actually be finished if score is set - if isScoreSet and not Logic.readBool(match.finished) and match.timestamp ~= DateExt.defaultTimestamp then - local threshold = match.dateexact and 30800 or 86400 - if match.timestamp + threshold < NOW then - match.finished = true - end - end - - -- apply placements and winner if finshed - if - not Logic.isEmpty(match.winner) or - Logic.readBool(match.finished) or - CustomMatchGroupInput.placementCheckSpecialStatus(opponents) - then - match.finished = true - match, opponents = CustomMatchGroupInput.getResultTypeAndWinner(match, opponents) - end - - -- Update all opponents with new values - for opponentIndex, opponent in pairs(opponents) do - match['opponent' .. opponentIndex] = opponent - end - return match -end - ----@param opponents table[] ----@param walkoverType string? ----@return any -function matchFunctions._makeAllOpponentsLoseByWalkover(opponents, walkoverType) - for index, _ in pairs(opponents) do - opponents[index].score = NOT_PLAYED_SCORE - opponents[index].status = walkoverType - end - return opponents end -- -- map related functions -- --- Parse extradata information ---@param map table +---@param opponentCount integer ---@return table -function mapFunctions.getAdditionalExtraData(map) - map.extradata.comment = map.comment - map.extradata.team1side = string.lower(map.team1side or '') - map.extradata.team2side = string.lower(map.team2side or '') - - return map -end +function MapFunctions.getExtraData(map, opponentCount) + local extradata = { + comment = map.comment, + team1side = string.lower(map.team1side or ''), + team2side = string.lower(map.team2side or ''), + } --- Parse participant information ----@param map table ----@param opponents table[] ----@return table -function mapFunctions.getParticipants(map, opponents) - local participants = {} - local championData = {} - local getCharacterName = FnUtil.curry(MatchGroupInput.getCharacterName, ChampionNames) - for opponentIndex = 1, MAX_NUM_OPPONENTS do - for playerIndex = 1, MAX_NUM_PLAYERS do - local champ = map['t' .. opponentIndex .. 'h' .. playerIndex] - championData['team' .. opponentIndex .. 'champion' .. playerIndex] = getCharacterName(champ) - - championData['t' .. opponentIndex .. 'kda' .. playerIndex] = - map['t' .. opponentIndex .. 'kda' .. playerIndex] - - local player = map['t' .. opponentIndex .. 'p' .. playerIndex] - if String.isNotEmpty(player) then - participants = mapFunctions.attachToParticipant( - player, - opponentIndex, - opponents[opponentIndex].match2players, - participants, - championData['team' .. opponentIndex .. 'champion' .. playerIndex], - championData['team' .. opponentIndex .. 'kda' .. playerIndex] - ) - end - end - local banIndex = 1 - local currentBan = map['t' .. opponentIndex .. 'b' .. banIndex] - while currentBan do - championData['team' .. opponentIndex .. 'ban' .. banIndex] = getCharacterName(currentBan) - banIndex = banIndex + 1 - currentBan = map['t' .. opponentIndex .. 'b' .. banIndex] + local getCharacterName = FnUtil.curry(MatchGroupInputUtil.getCharacterName, ChampionNames) + for opponentIndex = 1, opponentCount do + for _, ban, idx in Table.iter.pairsByPrefix(map, 't' .. opponentIndex .. 'b') do + extradata['team' .. opponentIndex .. 'ban' .. idx] = getCharacterName(ban) end - end - - map.extradata = championData - map.participants = participants - return mapFunctions.getAdditionalExtraData(map) -end - ----@param player string ----@param opponentIndex integer ----@param players table[] ----@param participants table ----@param champion string? ----@param kda string? ----@return table -function mapFunctions.attachToParticipant(player, opponentIndex, players, participants, champion, kda) - player = mw.ext.TeamLiquidIntegration.resolve_redirect(player):gsub(' ', '_') - for playerIndex, item in pairs(players or {}) do - if player == item.name then - participants[opponentIndex .. '_' .. playerIndex] = { - champion = champion, - kda = kda - } - break + for _, pick, idx in Table.iter.pairsByPrefix(map, 't' .. opponentIndex .. 'h') do + extradata['team' .. opponentIndex .. 'champion' .. idx] = getCharacterName(pick) end end - return participants + return extradata end --- Calculate Score and Winner of the map ---@param map table +---@param opponents table[] ---@return table -function mapFunctions.getScoresAndWinner(map) - map.scores = {} - local indexedScores = {} - for scoreIndex = 1, MAX_NUM_OPPONENTS do - -- read scores - local score = map['score' .. scoreIndex] or map['t' .. scoreIndex .. 'score'] - local obj = {} - if not Logic.isEmpty(score) then - if Logic.isNumeric(score) then - obj.status = STATUS_SCORE - score = tonumber(score) - map['score' .. scoreIndex] = score - obj.score = score - elseif Table.includes(ALLOWED_STATUSES, score) then - obj.status = score - obj.score = NOT_PLAYED_SCORE +function MapFunctions.getParticipants(map, opponents) + local allParticipants = {} + local getCharacterName = FnUtil.curry(MatchGroupInputUtil.getCharacterName, ChampionNames) + Array.forEach(opponents, function(opponent, opponentIndex) + local players = Array.mapIndexes(function(playerIndex) + return opponent.match2players[playerIndex] or Logic.nilIfEmpty(map['t' .. opponentIndex .. 'h' .. playerIndex]) + end) + local participants, unattachedParticipants = MatchGroupInputUtil.parseParticipants( + opponent.match2players, + players, + function(playerIndex) + local player = map['t' .. opponentIndex .. 'p' .. playerIndex] + return player and {name = player} or nil + end, + function(playerIndex, playerIdData) + local character = map['t' .. opponentIndex .. 'h' .. playerIndex] + return { + champion = getCharacterName(character), + } end - table.insert(map.scores, score) - indexedScores[scoreIndex] = obj - else - break - end - end - - map = CustomMatchGroupInput.getResultTypeAndWinner(map, indexedScores) + ) + Array.forEach(unattachedParticipants, function(participant) + table.insert(participants, participant) + end) + Table.mergeInto(allParticipants, Table.map(participants, MatchGroupInputUtil.prefixPartcipants(opponentIndex))) + end) - return map + return allParticipants end return CustomMatchGroupInput