Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(match2): always populate game and match dates #5221

Merged
merged 5 commits into from
Jan 3, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
83 changes: 65 additions & 18 deletions components/match2/commons/match_group_input_util.lua
Original file line number Diff line number Diff line change
Expand Up @@ -808,9 +808,10 @@ function MatchGroupInputUtil.placementFromWinner(status, winner, opponentIndex)
end

---@param match table
---@param maps table[]
---@param opponents {score: integer?}[]
---@return boolean
function MatchGroupInputUtil.matchIsFinished(match, opponents)
function MatchGroupInputUtil.matchIsFinished(match, maps, opponents)
if MatchGroupInputUtil.isNotPlayed(match.winner, match.finished) then
return true
end
Expand All @@ -835,8 +836,14 @@ function MatchGroupInputUtil.matchIsFinished(match, opponents)
end

-- If enough time has passed since match started, it should be marked as finished
local threshold = match.dateexact and ASSUME_FINISHED_AFTER.EXACT or ASSUME_FINISHED_AFTER.ESTIMATE
if match.timestamp ~= DateExt.defaultTimestamp and (match.timestamp + threshold) < NOW then
local function recordLiveLongEnough(record)
if not record.timestamp or record.timestamp == DateExt.defaultTimestamp then
return false
end
local longLiveTime = record.dateexact and ASSUME_FINISHED_AFTER.EXACT or ASSUME_FINISHED_AFTER.ESTIMATE
return NOW > (record.timestamp + longLiveTime)
end
if (#maps > 0 and Array.all(maps, recordLiveLongEnough)) or (#maps == 0 and recordLiveLongEnough(match)) then
return true
end

Expand Down Expand Up @@ -1046,6 +1053,14 @@ function MatchGroupInputUtil.mergeStandaloneIntoMatch(match, standaloneMatch)
return match
end

---@alias readDateFunction fun(match: table): {
---date: string,
---dateexact: boolean,
---timestamp: integer,
---timezoneId: string?,
---timezoneOffset:string?,
---}

---@class MatchParserInterface
---@field extractMaps fun(match: table, opponents: table[], mapProps: any?): table[]
---@field getBestOf fun(bestOfInput: string|integer|nil, maps: table[]): integer?
Expand All @@ -1055,13 +1070,7 @@ end
---@field adjustOpponent? fun(opponent: MGIParsedOpponent, opponentIndex: integer)
---@field getLinks? fun(match: table, games: table[]): table
---@field getHeadToHeadLink? fun(match: table, opponents: table[]): string?
---@field readDate? fun(match: table): {
---date: string,
---dateexact: boolean,
---timestamp: integer,
---timezoneId: string?,
---timezoneOffset:string?,
---}
---@field readDate? readDateFunction
---@field getMode? fun(opponents: table[]): string
---@field DEFAULT_MODE? string
---@field DATE_FALLBACKS? string[]
Expand Down Expand Up @@ -1096,8 +1105,7 @@ function MatchGroupInputUtil.standardProcessMatch(match, Parser, FfaParser, mapP
Parser = Parser or {}
local matchInput = Table.deepCopy(match)

local dateProps = Parser.readDate and Parser.readDate(match)
or MatchGroupInputUtil.readDate(match.date, Parser.DATE_FALLBACKS)
local dateProps = MatchGroupInputUtil.getMatchDate(Parser, matchInput)
Table.mergeInto(match, dateProps)

local opponents = Array.mapIndexes(function(opponentIndex)
Expand Down Expand Up @@ -1133,7 +1141,7 @@ function MatchGroupInputUtil.standardProcessMatch(match, Parser, FfaParser, mapP
}, autoScoreFunction)
end)

match.finished = MatchGroupInputUtil.matchIsFinished(match, opponents)
match.finished = MatchGroupInputUtil.matchIsFinished(match, games, opponents)

if match.finished then
match.status = MatchGroupInputUtil.getMatchStatus(matchInput.winner, matchInput.finished)
Expand Down Expand Up @@ -1203,6 +1211,9 @@ function MatchGroupInputUtil.standardProcessMaps(match, opponents, Parser)
local finishedInput = map.finished --[[@as string?]]
local winnerInput = map.winner --[[@as string?]]

local dateToUse = map.date or match.date
Table.mergeInto(map, MatchGroupInputUtil.readDate(dateToUse))

if Parser.ADD_SUB_GROUP then
subGroup = tonumber(map.subgroup) or (subGroup + 1)
map.subgroup = subGroup
Expand Down Expand Up @@ -1278,7 +1289,7 @@ end
---@field calculateMatchScore? fun(maps: table[], opponents: table[]): fun(opponentIndex: integer): integer?
---@field getExtraData? fun(match: table, games: table[], opponents: table[], settings: table): table?
---@field getMode? fun(opponents: table[]): string
---@field readDate? fun(match: table): table
---@field readDate? readDateFunction
---@field adjustOpponent? fun(opponent: table[], opponentIndex: integer, match: table)
---@field matchIsFinished? fun(match: table, opponents: table[]): boolean
---@field getMatchWinner? fun(status: string, winnerInput: integer|string|nil, opponents: table[]): integer?
Expand Down Expand Up @@ -1313,8 +1324,7 @@ function MatchGroupInputUtil.standardProcessFfaMatch(match, Parser, mapProps)
local finishedInput = match.finished --[[@as string?]]
local winnerInput = match.winner --[[@as string?]]

local dateProps = Parser.readDate and Parser.readDate(match)
or MatchGroupInputUtil.readDate(match.date, Parser.DATE_FALLBACKS)
local dateProps = MatchGroupInputUtil.getMatchDate(Parser, match)
Table.mergeInto(match, dateProps)

local opponents = Array.mapIndexes(function(opponentIndex)
Expand Down Expand Up @@ -1345,7 +1355,7 @@ function MatchGroupInputUtil.standardProcessFfaMatch(match, Parser, mapProps)
end)

match.finished = Parser.matchIsFinished and Parser.matchIsFinished(match, opponents)
or MatchGroupInputUtil.matchIsFinished(match, opponents)
or MatchGroupInputUtil.matchIsFinished(match, games, opponents)

if match.finished then
match.status = MatchGroupInputUtil.getMatchStatus(winnerInput, finishedInput)
Expand Down Expand Up @@ -1390,7 +1400,8 @@ function MatchGroupInputUtil.standardProcessFfaMaps(match, opponents, scoreSetti
local finishedInput = map.finished --[[@as string?]]
local winnerInput = map.winner --[[@as string?]]

Table.mergeInto(map, MatchGroupInputUtil.readDate(map.date))
local dateToUse = map.date or match.date
Table.mergeInto(map, MatchGroupInputUtil.readDate(dateToUse))
map.finished = MatchGroupInputUtil.mapIsFinished(map)

map.opponents = Array.map(opponents, function(matchOpponent)
Expand Down Expand Up @@ -1566,4 +1577,40 @@ function MatchGroupInputUtil.makeBattleRoyaleMapOpponentDetails(scoreDataInput,
return opponent
end

---@param matchParser {readDate?: readDateFunction, DATE_FALLBACKS?: string[]}
---@param matchInput table
---@return {date: string, dateexact: boolean, timestamp: integer, timezoneId: string?, timezoneOffset: string?}
function MatchGroupInputUtil.getMatchDate(matchParser, matchInput)
local defaultDateParser = function(record)
return MatchGroupInputUtil.readDate(record.date, matchParser.DATE_FALLBACKS)
end
local dateParsingFunction = matchParser.readDate or defaultDateParser

if matchInput.date then
-- If there's a match date in the input, use it
return dateParsingFunction(matchInput)
end

-- Otherwise, use the date from the earliest game in the match
local easlierGameTimestamp, earliestGameDateStruct = DateExt.maxTimestamp, nil

-- We have to loop through the maps unparsed as we haven't parsed the maps at this point yet
for _, map in Table.iter.pairsByPrefix(matchInput, 'map', {requireIndex = true}) do
if map.date then
local gameDateStruct = dateParsingFunction(map)
if gameDateStruct.timestamp < easlierGameTimestamp then
earliestGameDateStruct = gameDateStruct
easlierGameTimestamp = gameDateStruct.timestamp
end
end
end

-- We couldn't find game date neither, let's use the defaults for the match
if not earliestGameDateStruct then
return dateParsingFunction(matchInput)
end

return earliestGameDateStruct
end

return MatchGroupInputUtil
Original file line number Diff line number Diff line change
Expand Up @@ -40,7 +40,7 @@ function CustomMatchGroupInput.processMatch(match, options)
Table.mergeInto(match, MatchGroupInputUtil.getTournamentContext(match))
match.game, match.mapsInfo = CustomMatchGroupInput._getMapsAndGame(match)

Table.mergeInto(match, MatchGroupInputUtil.readDate(match.date))
Table.mergeInto(match, MatchGroupInputUtil.getMatchDate({}, match))

local opponents = Array.mapIndexes(function(opponentIndex)
return CustomMatchGroupInput.readOpponent(match, opponentIndex, OPPONENT_CONFIG)
Expand All @@ -66,7 +66,7 @@ function CustomMatchGroupInput.processMatch(match, options)

local winnerInput = match.winner --[[@as string?]]
local finishedInput = match.finished --[[@as string?]]
match.finished = MatchGroupInputUtil.matchIsFinished(match, opponents)
match.finished = MatchGroupInputUtil.matchIsFinished(match, games, opponents)

if match.finished then
match.status = MatchGroupInputUtil.getMatchStatus(winnerInput, finishedInput)
Expand Down
3 changes: 3 additions & 0 deletions components/match2/wikis/dota2/match_group_input_custom.lua
Original file line number Diff line number Diff line change
Expand Up @@ -88,6 +88,9 @@ function MatchFunctions.extractMaps(match, opponents, MapParser)
local finishedInput = map.finished --[[@as string?]]
local winnerInput = map.winner --[[@as string?]]

local dateToUse = map.date or match.date
Table.mergeInto(map, MatchGroupInputUtil.readDate(dateToUse))

map.map = MapFunctions.getMapName(map)
map.length = MapParser.getLength(map)
map.vod = map.vod or String.nilIfEmpty(match['vodgame' .. mapIndex])
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -82,6 +82,9 @@ function MatchFunctions.extractMaps(match, opponents, MapParser)
local finishedInput = map.finished --[[@as string?]]
local winnerInput = map.winner --[[@as string?]]

local dateToUse = map.date or match.date
Table.mergeInto(map, MatchGroupInputUtil.readDate(dateToUse))

map.length = MapParser.getLength(map)
map.vod = map.vod or String.nilIfEmpty(match['vodgame' .. mapIndex])
map.extradata = MapFunctions.getExtraData(MapParser, map, #opponents)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -55,6 +55,9 @@ function MatchFunctions.extractMaps(match, opponents)
local finishedInput = map.finished --[[@as string?]]
local winnerInput = map.winner --[[@as string?]]

local dateToUse = map.date or match.date
Table.mergeInto(map, MatchGroupInputUtil.readDate(dateToUse))

map.extradata = MapFunctions.getExtraData(map)
map.finished = MatchGroupInputUtil.mapIsFinished(map)

Expand Down
Loading