From fc79fb6d15796cb189d43403b08cf6db13d83da9 Mon Sep 17 00:00:00 2001 From: James Berry Date: Wed, 17 Nov 2021 12:39:17 +0000 Subject: [PATCH] Performance improvements (#612) * Monster tracking performance improvements * Tidy * Push DTS generation out of main loop * Push DTS generation out of main loop Co-authored-by: James Berry --- src/controllerWorker.js | 6 + src/controllers/gym.js | 135 ++-- src/controllers/monster.js | 692 +++++++++--------- src/controllers/pokestop.js | 275 +++---- src/controllers/pokestop_lure.js | 136 ++-- src/controllers/raid.js | 369 +++++----- .../migrations/v14_better2_monster_index.js | 15 + 7 files changed, 870 insertions(+), 758 deletions(-) create mode 100644 src/lib/db/migrations/v14_better2_monster_index.js diff --git a/src/controllerWorker.js b/src/controllerWorker.js index 5e94e7666..9975335cd 100644 --- a/src/controllerWorker.js +++ b/src/controllerWorker.js @@ -297,5 +297,11 @@ if (!isMainThread) { controllerWeatherManager.on('weatherForecastRequested', (data) => notifyWeatherController('weatherForecastRequested', data)) monsterController.on('userCares', (data) => notifyWeatherController('userCares', data)) + monsterController.on('postMessage', (jobs) => queuePort.postMessage({ queue: jobs })) + raidController.on('postMessage', (jobs) => queuePort.postMessage({ queue: jobs })) + pokestopController.on('postMessage', (jobs) => queuePort.postMessage({ queue: jobs })) + pokestopLureController.on('postMessage', (jobs) => queuePort.postMessage({ queue: jobs })) + gymController.on('postMessage', (jobs) => queuePort.postMessage({ queue: jobs })) + setInterval(currentStatus, 60000) } diff --git a/src/controllers/gym.js b/src/controllers/gym.js index ad5ef8c43..209b3b124 100644 --- a/src/controllers/gym.js +++ b/src/controllers/gym.js @@ -147,72 +147,79 @@ class Gym extends Controller { return [] } - data.imgUrl = await this.imgUicons.gymIcon(data.teamId, data.slotsAvailable, false, data.ex) - data.stickerUrl = await this.stickerUicons.gymIcon(data.teamId, data.slotsAvailable, false, data.ex) - - const geoResult = await this.getAddress({ lat: data.latitude, lon: data.longitude }) - const jobs = [] - - await this.getStaticMapUrl(logReference, data, 'gym', ['teamId', 'latitude', 'longitude', 'imgUrl']) - data.staticmap = data.staticMap // deprecated - - for (const cares of whoCares) { - this.log.debug(`${logReference}: Creating gym alert for ${cares.id} ${cares.name} ${cares.type} ${cares.language} ${cares.template}`, cares) - - const rateLimitTtr = this.getRateLimitTimeToRelease(cares.id) - if (rateLimitTtr) { - this.log.verbose(`${logReference}: Not creating gym alert (Rate limit) for ${cares.type} ${cares.id} ${cares.name} Time to release: ${rateLimitTtr}`) - // eslint-disable-next-line no-continue - continue - } - this.log.verbose(`${logReference}: Creating gym alert for ${cares.type} ${cares.id} ${cares.name} ${cares.language} ${cares.template}`) - - const language = cares.language || this.config.general.locale - const translator = this.translatorFactory.Translator(language) - let [platform] = cares.type.split(':') - if (platform === 'webhook') platform = 'discord' - - data.teamName = translator.translate(data.teamNameEng) - data.oldTeamName = translator.translate(data.oldTeamNameEng) - data.previousControlName = translator.translate(data.previousControlNameEng) - data.teamEmojiEng = data.teamId >= 0 ? this.emojiLookup.lookup(this.GameData.utilData.teams[data.teamId].emoji, platform) : '' - data.teamEmoji = translator.translate(data.teamEmojiEng) - - // full build - - const view = { - ...geoResult, - ...data, - time: data.distime, - tthh: data.tth.hours, - tthm: data.tth.minutes, - tths: data.tth.seconds, - now: new Date(), - nowISO: new Date().toISOString(), - areas: data.matchedAreas.filter((area) => area.displayInMatches).map((area) => area.name).join(', '), + setImmediate(async () => { + try { + data.imgUrl = await this.imgUicons.gymIcon(data.teamId, data.slotsAvailable, false, data.ex) + data.stickerUrl = await this.stickerUicons.gymIcon(data.teamId, data.slotsAvailable, false, data.ex) + + const geoResult = await this.getAddress({ lat: data.latitude, lon: data.longitude }) + const jobs = [] + + await this.getStaticMapUrl(logReference, data, 'gym', ['teamId', 'latitude', 'longitude', 'imgUrl']) + data.staticmap = data.staticMap // deprecated + + for (const cares of whoCares) { + this.log.debug(`${logReference}: Creating gym alert for ${cares.id} ${cares.name} ${cares.type} ${cares.language} ${cares.template}`, cares) + + const rateLimitTtr = this.getRateLimitTimeToRelease(cares.id) + if (rateLimitTtr) { + this.log.verbose(`${logReference}: Not creating gym alert (Rate limit) for ${cares.type} ${cares.id} ${cares.name} Time to release: ${rateLimitTtr}`) + // eslint-disable-next-line no-continue + continue + } + this.log.verbose(`${logReference}: Creating gym alert for ${cares.type} ${cares.id} ${cares.name} ${cares.language} ${cares.template}`) + + const language = cares.language || this.config.general.locale + const translator = this.translatorFactory.Translator(language) + let [platform] = cares.type.split(':') + if (platform === 'webhook') platform = 'discord' + + data.teamName = translator.translate(data.teamNameEng) + data.oldTeamName = translator.translate(data.oldTeamNameEng) + data.previousControlName = translator.translate(data.previousControlNameEng) + data.teamEmojiEng = data.teamId >= 0 ? this.emojiLookup.lookup(this.GameData.utilData.teams[data.teamId].emoji, platform) : '' + data.teamEmoji = translator.translate(data.teamEmojiEng) + + // full build + + const view = { + ...geoResult, + ...data, + time: data.distime, + tthh: data.tth.hours, + tthm: data.tth.minutes, + tths: data.tth.seconds, + now: new Date(), + nowISO: new Date().toISOString(), + areas: data.matchedAreas.filter((area) => area.displayInMatches).map((area) => area.name).join(', '), + } + + const templateType = 'gym' + const message = await this.createMessage(logReference, templateType, platform, cares.template, language, cares.ping, view) + + const work = { + lat: data.latitude.toString().substring(0, 8), + lon: data.longitude.toString().substring(0, 8), + message, + target: cares.id, + type: cares.type, + name: cares.name, + tth: data.tth, + clean: cares.clean, + emoji: data.emoji, + logReference, + language, + } + + jobs.push(work) + } + this.emit('postMessage', jobs) + } catch (e) { + this.log.error(`${data.id || data.gym_id}: Can't seem to handle gym (user cares): `, e, data) } + }) - const templateType = 'gym' - const message = await this.createMessage(logReference, templateType, platform, cares.template, language, cares.ping, view) - - const work = { - lat: data.latitude.toString().substring(0, 8), - lon: data.longitude.toString().substring(0, 8), - message, - target: cares.id, - type: cares.type, - name: cares.name, - tth: data.tth, - clean: cares.clean, - emoji: data.emoji, - logReference, - language, - } - - jobs.push(work) - } - - return jobs + return [] } catch (e) { this.log.error(`${data.id || data.gym_id}: Can't seem to handle gym: `, e, data) } diff --git a/src/controllers/monster.js b/src/controllers/monster.js index 8ce9b208a..ceb0a9b11 100644 --- a/src/controllers/monster.js +++ b/src/controllers/monster.js @@ -11,38 +11,85 @@ class Monster extends Controller { return boostingWeathers } + async performQuery(query) { + let result = await this.db.raw(query) + result = this.returnByDatabaseType(result) + + return result + } + async monsterWhoCares(data) { const { areastring, strictareastring } = this.buildAreaString(data.matched) - let pokemonQueryString = `(pokemon_id=${data.pokemon_id} or pokemon_id=0) and (form = 0 or form = ${data.form})` - if (data.pvpEvoLookup) { - pokemonQueryString = `(pokemon_id=${data.pvpPokemonId} and pvp_ranking_league > 0)` // form checked per league later + const pvpQueryLimit = this.config.pvp.pvpQueryMaxRank || this.config.pvp.pvpFilterMaxRank || 100 + + let result = [] + + // Basic Pokemon + result.push(...await this.performQuery(this.buildQuery('Basic', data, strictareastring, areastring, { pokemon_id: data.pokemon_id, form: data.form, includeEverything: true }, 0))) + + // PVP Pokemon + + for (const [league, leagueData] of Object.entries(data.pvpBestRank)) { + // if rank is good enough + if (leagueData.rank <= pvpQueryLimit) { + result.push(...await this.performQuery(this.buildQuery(`PVP ${league}`, data, strictareastring, areastring, + { + pokemon_id: data.pokemon_id, + form: data.form, + includeEverything: true, + }, league, leagueData))) + } } + // PVP Evolution mon + + if (this.config.pvp.pvpEvolutionDirectTracking) { + if (Object.keys(data.pvpEvolutionData).length !== 0) { + for (const [pokemonId, pvpMon] of Object.entries(data.pvpEvolutionData)) { + for (const [league, leagueData] of Object.entries(pvpMon)) { + // if rank is good enough + if (leagueData.rank <= pvpQueryLimit) { + result.push(...await this.performQuery(this.buildQuery(`PVPEvo ${league}`, data, strictareastring, areastring, + { + pokemon_id: pokemonId, + form: leagueData.form, + includeEverything: false, + }, + league, leagueData))) + } + } + } + } + } + + // remove any duplicates + const alertIds = [] + result = result.filter((alert) => { + if (!alertIds.includes(alert.id)) { + alertIds.push(alert.id) + return alert + } + }) + return result + } + + buildQuery(queryName, data, strictareastring, areastring, pokemonTarget, league, leagueData) { + const pokemonQueryString = `(pokemon_id=${pokemonTarget.pokemon_id}${pokemonTarget.includeEverything ? ' or pokemon_id=0' : ''}) and (form = 0 or form = ${pokemonTarget.form})` + let pvpQueryString = '' const pvpSecurityEnabled = this.config.discord.commandSecurity && this.config.discord.commandSecurity.pvp - if (!data.pvpEvoLookup) { + + if (!league) { pvpQueryString = 'pvp_ranking_league = 0' - if (pvpSecurityEnabled) { - pvpQueryString = pvpQueryString.concat(' or ((humans.blocked_alerts IS NULL OR humans.blocked_alerts NOT LIKE \'%pvp%\') and (1 = 0') - } - for (const [league, leagueData] of Object.entries(data.pvpBestRank)) { - pvpQueryString = pvpQueryString.concat(` or (pvp_ranking_league = ${league} and pvp_ranking_worst >= ${leagueData.rank} and pvp_ranking_best <= ${leagueData.rank} and pvp_ranking_min_cp <= ${leagueData.cp})`) - } - if (pvpSecurityEnabled) { - pvpQueryString = pvpQueryString.concat('))') - } } else { if (pvpSecurityEnabled) { - pvpQueryString = '(humans.blocked_alerts IS NULL OR humans.blocked_alerts NOT LIKE \'%pvp%\') and (1 = 0' - } else { - pvpQueryString = '1 = 0' - } - for (const [league, leagueData] of Object.entries(data.pvpEvoLookup)) { - pvpQueryString = pvpQueryString.concat(` or ((form = 0 or form = ${leagueData.form}) and pvp_ranking_league = ${league} and pvp_ranking_worst >= ${leagueData.rank} and pvp_ranking_best <= ${leagueData.rank} and pvp_ranking_min_cp <= ${leagueData.cp})`) + pvpQueryString = pvpQueryString.concat('((humans.blocked_alerts IS NULL OR humans.blocked_alerts NOT LIKE \'%pvp%\') and (') } + pvpQueryString = pvpQueryString.concat(`pvp_ranking_league = ${league} and pvp_ranking_worst >= ${leagueData.rank} and pvp_ranking_best <= ${leagueData.rank} and pvp_ranking_min_cp <= ${leagueData.cp}`) + if (pvpSecurityEnabled) { - pvpQueryString = pvpQueryString.concat(')') + pvpQueryString = pvpQueryString.concat('))') } } @@ -100,26 +147,8 @@ class Monster extends Controller { and ((monsters.distance = 0 and (${areastring})) or monsters.distance > 0) `) } - this.log.silly(`${data.encounter_id}: Query ${query}`) - - let result = await this.db.raw(query) - - if (!['pg', 'mysql'].includes(this.config.database.client)) { - result = result.filter((res) => +res.distance === 0 || +res.distance > 0 && +res.distance > this.getDistance({ - lat: res.latitude, - lon: res.longitude, - }, { lat: data.latitude, lon: data.longitude })) - } - result = this.returnByDatabaseType(result) - // remove any duplicates - const alertIds = [] - result = result.filter((alert) => { - if (!alertIds.includes(alert.id)) { - alertIds.push(alert.id) - return alert - } - }) - return result + this.log.silly(`${data.encounter_id}: Query[${queryName}] ${query}`) + return query } async handle(obj) { @@ -313,23 +342,8 @@ class Monster extends Controller { data.matchedAreas = this.pointInArea([data.latitude, data.longitude]) data.matched = data.matchedAreas.map((x) => x.name.toLowerCase()) - data.pvpEvoLookup = 0 const whoCares = await this.monsterWhoCares(data) - if (this.config.pvp.pvpEvolutionDirectTracking) { - const pvpEvoData = data - if (Object.keys(data.pvpEvolutionData).length !== 0) { - for (const [key, pvpMon] of Object.entries(data.pvpEvolutionData)) { - pvpEvoData.pvpPokemonId = key - pvpEvoData.pvpEvoLookup = pvpMon - const pvpWhoCares = await this.monsterWhoCares(pvpEvoData) - if (pvpWhoCares[0]) { - whoCares.push(...pvpWhoCares) - } - } - } - } - let hrend = process.hrtime(hrstart) const hrendms = hrend[1] / 1000000 if (whoCares.length) { @@ -340,302 +354,322 @@ class Monster extends Controller { if (!whoCares.length) return [] - if (whoCares.length > 1 && this.config.pvp.pvpEvolutionDirectTracking) { - const whoCaresNoDuplicates = whoCares.filter((v, i, a) => a.findIndex((t) => (t.id === v.id)) === i) - whoCares.length = 0 - whoCares.push(...whoCaresNoDuplicates) - } - - hrstart = process.hrtime() - let discordCacheBad = true // assume the worst - whoCares.forEach((cares) => { - if (!this.isRateLimited(cares.id)) discordCacheBad = false - }) + setImmediate(async () => { + try { + hrstart = process.hrtime() + let discordCacheBad = true // assume the worst + whoCares.forEach((cares) => { + if (!this.isRateLimited(cares.id)) discordCacheBad = false + }) - if (discordCacheBad) { - whoCares.forEach((cares) => { - this.log.verbose(`${logReference}: Not creating monster alert (Rate limit) for ${cares.type} ${cares.id} ${cares.name} Time to release: ${this.getRateLimitTimeToRelease(cares.id)}`) - }) + if (discordCacheBad) { + whoCares.forEach((cares) => { + this.log.verbose(`${logReference}: Not creating monster alert (Rate limit) for ${cares.type} ${cares.id} ${cares.name} Time to release: ${this.getRateLimitTimeToRelease(cares.id)}`) + }) - return [] - } - - if (data.display_pokemon_id && data.display_pokemon_id !== data.pokemon_id) { - if (data.display_form === undefined || data.display_form === null) data.display_form = 0 - const displayMonster = this.GameData.monsters[`${data.display_pokemon_id}_${data.display_form}`] || this.GameData.monsters[`${data.display_pokemon_id}_0`] + return [] + } - if (displayMonster) { - data.disguisePokemonNameEng = displayMonster.name - if (displayMonster.form) data.disguideFormNameEng = displayMonster.form.name - } - } + if (data.display_pokemon_id && data.display_pokemon_id !== data.pokemon_id) { + if (data.display_form === undefined || data.display_form === null) data.display_form = 0 + const displayMonster = this.GameData.monsters[`${data.display_pokemon_id}_${data.display_form}`] || this.GameData.monsters[`${data.display_pokemon_id}_0`] - data.imgUrl = await this.imgUicons.pokemonIcon(data.pokemon_id, data.form, 0, data.gender, data.costume, data.shinyPossible && this.config.general.requestShinyImages) - data.stickerUrl = await this.stickerUicons.pokemonIcon(data.pokemon_id, data.form, 0, data.gender, data.costume, data.shinyPossible && this.config.general.requestShinyImages) - // data.imgUrl = `${this.config.general.imgUrl}pokemon_icon_${data.pokemon_id.toString().padStart(3, '0')}_${data.form ? data.form.toString() : '00'}.png` - // data.stickerUrl = `${this.config.general.stickerUrl}pokemon_icon_${data.pokemon_id.toString().padStart(3, '0')}_${data.form ? data.form.toString() : '00'}.webp` - - const geoResult = await this.getAddress({ lat: data.latitude, lon: data.longitude }) - const jobs = [] - - await this.getStaticMapUrl(logReference, data, 'monster', ['pokemon_id', 'latitude', 'longitude', 'form', 'costume', 'imgUrl']) - data.staticmap = data.staticMap // deprecated - - // get Weather Forecast information - - const { nextHourTimestamp } = this.weatherData.getWeatherTimes() - if (this.config.weather.enableWeatherForecast && data.disappear_time > nextHourTimestamp) { - const weatherForecast = await this.weatherData.getWeatherForecast(weatherCellId) - - let pokemonShouldBeBoosted = false - let pokemonWillBeBoosted = false - const currentBoostedTypes = weatherForecast.current ? this.GameData.utilData.weatherTypeBoost[weatherForecast.current] : [] - const forecastBoostedTypes = weatherForecast.next ? this.GameData.utilData.weatherTypeBoost[weatherForecast.next] : [] - if (weatherForecast.current > 0 && currentBoostedTypes.filter((boostedType) => data.types.includes(boostedType)).length > 0) pokemonShouldBeBoosted = true - if (weatherForecast.next > 0 && ((data.weather > 0 && weatherForecast.next !== data.weather) || (weatherForecast.current > 0 && weatherForecast.next !== weatherForecast.current) || (pokemonShouldBeBoosted && data.weather === 0))) { - const weatherChangeTime = moment((data.disappear_time - (data.disappear_time % 3600)) * 1000).tz(geoTz(data.latitude, data.longitude).toString()).format(this.config.locale.time).slice(0, -3) - pokemonWillBeBoosted = forecastBoostedTypes.filter((boostedType) => data.types.includes(boostedType)).length > 0 ? 1 : 0 - if (data.weather > 0 && !pokemonWillBeBoosted || data.weather === 0 && pokemonWillBeBoosted) { - weatherForecast.current = data.weather > 0 ? data.weather : weatherForecast.current - if (pokemonShouldBeBoosted && data.weather === 0) { - data.weatherCurrent = 0 - } else { - data.weatherCurrent = weatherForecast.current + if (displayMonster) { + data.disguisePokemonNameEng = displayMonster.name + if (displayMonster.form) data.disguideFormNameEng = displayMonster.form.name } - data.weatherChangeTime = weatherChangeTime - data.weatherNext = weatherForecast.next } - } - this.log.debug(`${logReference}: Pokemon ${data.pokemon_id} cell: ${weatherCellId} types ${JSON.stringify(data.types)} weather ${data.weather} Forecast ${weatherForecast.current} [boosted ${pokemonShouldBeBoosted} ${JSON.stringify(currentBoostedTypes)}] next ${weatherForecast.next} [boosted ${pokemonWillBeBoosted} ${JSON.stringify(forecastBoostedTypes)}]`) - } - const event = this.eventParser.eventChangesSpawn(moment().unix(), data.disappear_time, data.latitude, data.longitude) - if (event) { - data.futureEvent = true - data.futureEventTime = event.time - data.futureEventName = event.name - data.futureEventTrigger = event.reason - } + data.imgUrl = await this.imgUicons.pokemonIcon(data.pokemon_id, data.form, 0, data.gender, data.costume, data.shinyPossible && this.config.general.requestShinyImages) + data.stickerUrl = await this.stickerUicons.pokemonIcon(data.pokemon_id, data.form, 0, data.gender, data.costume, data.shinyPossible && this.config.general.requestShinyImages) + // data.imgUrl = `${this.config.general.imgUrl}pokemon_icon_${data.pokemon_id.toString().padStart(3, '0')}_${data.form ? data.form.toString() : '00'}.png` + // data.stickerUrl = `${this.config.general.stickerUrl}pokemon_icon_${data.pokemon_id.toString().padStart(3, '0')}_${data.form ? data.form.toString() : '00'}.webp` - for (const cares of whoCares) { - this.log.debug(`${logReference}: Creating monster alert for ${cares.id} ${cares.name} ${cares.type} ${cares.language} ${cares.template}`, cares) - - const rateLimitTtr = this.getRateLimitTimeToRelease(cares.id) - if (rateLimitTtr) { - this.log.verbose(`${logReference}: Not creating monster alert (Rate limit) for ${cares.type} ${cares.id} ${cares.name} Time to release: ${rateLimitTtr}`) - // eslint-disable-next-line no-continue - continue - } - this.log.verbose(`${logReference}: Creating monster alert for ${cares.type} ${cares.id} ${cares.name} ${cares.language} ${cares.template}`) + const geoResult = await this.getAddress({ + lat: data.latitude, + lon: data.longitude, + }) + const jobs = [] + + await this.getStaticMapUrl(logReference, data, 'monster', ['pokemon_id', 'latitude', 'longitude', 'form', 'costume', 'imgUrl']) + data.staticmap = data.staticMap // deprecated + + // get Weather Forecast information + + const { nextHourTimestamp } = this.weatherData.getWeatherTimes() + if (this.config.weather.enableWeatherForecast && data.disappear_time > nextHourTimestamp) { + const weatherForecast = await this.weatherData.getWeatherForecast(weatherCellId) + + let pokemonShouldBeBoosted = false + let pokemonWillBeBoosted = false + const currentBoostedTypes = weatherForecast.current ? this.GameData.utilData.weatherTypeBoost[weatherForecast.current] : [] + const forecastBoostedTypes = weatherForecast.next ? this.GameData.utilData.weatherTypeBoost[weatherForecast.next] : [] + if (weatherForecast.current > 0 && currentBoostedTypes.filter((boostedType) => data.types.includes(boostedType)).length > 0) pokemonShouldBeBoosted = true + if (weatherForecast.next > 0 && ((data.weather > 0 && weatherForecast.next !== data.weather) || (weatherForecast.current > 0 && weatherForecast.next !== weatherForecast.current) || (pokemonShouldBeBoosted && data.weather === 0))) { + const weatherChangeTime = moment((data.disappear_time - (data.disappear_time % 3600)) * 1000) + .tz(geoTz(data.latitude, data.longitude) + .toString()) + .format(this.config.locale.time) + .slice(0, -3) + pokemonWillBeBoosted = forecastBoostedTypes.filter((boostedType) => data.types.includes(boostedType)).length > 0 ? 1 : 0 + if (data.weather > 0 && !pokemonWillBeBoosted || data.weather === 0 && pokemonWillBeBoosted) { + weatherForecast.current = data.weather > 0 ? data.weather : weatherForecast.current + if (pokemonShouldBeBoosted && data.weather === 0) { + data.weatherCurrent = 0 + } else { + data.weatherCurrent = weatherForecast.current + } + data.weatherChangeTime = weatherChangeTime + data.weatherNext = weatherForecast.next + } + } + this.log.debug(`${logReference}: Pokemon ${data.pokemon_id} cell: ${weatherCellId} types ${JSON.stringify(data.types)} weather ${data.weather} Forecast ${weatherForecast.current} [boosted ${pokemonShouldBeBoosted} ${JSON.stringify(currentBoostedTypes)}] next ${weatherForecast.next} [boosted ${pokemonWillBeBoosted} ${JSON.stringify(forecastBoostedTypes)}]`) + } - if (this.config.weather.weatherChangeAlert) { - // Emit event so we can tell weather controller (different worker) about the pokemon being monitored + const event = this.eventParser.eventChangesSpawn(moment() + .unix(), data.disappear_time, data.latitude, data.longitude) + if (event) { + data.futureEvent = true + data.futureEventTime = event.time + data.futureEventName = event.name + data.futureEventTrigger = event.reason + } - this.emit('userCares', { - target: { - id: cares.id, - name: cares.name, - type: cares.type, - clean: cares.clean, - ping: cares.ping, - template: cares.template, - language: cares.language, - }, - weatherCellId, - caresUntil: data.disappear_time, - pokemon: encountered - ? { - pokemon_id: data.pokemon_id, - form: data.form, - name: monster.name, - formName: monster.form.name, - iv: data.iv, - cp: data.cp, - latitude: data.latitude, - longitude: data.longitude, - disappear_time: data.disappear_time, - alteringWeathers: data.alteringWeathers, - } : null, + for (const cares of whoCares) { + this.log.debug(`${logReference}: Creating monster alert for ${cares.id} ${cares.name} ${cares.type} ${cares.language} ${cares.template}`, cares) - }) - } + const rateLimitTtr = this.getRateLimitTimeToRelease(cares.id) + if (rateLimitTtr) { + this.log.verbose(`${logReference}: Not creating monster alert (Rate limit) for ${cares.type} ${cares.id} ${cares.name} Time to release: ${rateLimitTtr}`) + // eslint-disable-next-line no-continue + continue + } + this.log.verbose(`${logReference}: Creating monster alert for ${cares.type} ${cares.id} ${cares.name} ${cares.language} ${cares.template}`) + + if (this.config.weather.weatherChangeAlert) { + // Emit event so we can tell weather controller (different worker) about the pokemon being monitored + + this.emit('userCares', { + target: { + id: cares.id, + name: cares.name, + type: cares.type, + clean: cares.clean, + ping: cares.ping, + template: cares.template, + language: cares.language, + }, + weatherCellId, + caresUntil: data.disappear_time, + pokemon: encountered + ? { + pokemon_id: data.pokemon_id, + form: data.form, + name: monster.name, + formName: monster.form.name, + iv: data.iv, + cp: data.cp, + latitude: data.latitude, + longitude: data.longitude, + disappear_time: data.disappear_time, + alteringWeathers: data.alteringWeathers, + } : null, + + }) + } - const language = cares.language || this.config.general.locale - const translator = this.translatorFactory.Translator(language) - let [platform] = cares.type.split(':') - if (platform === 'webhook') platform = 'discord' + const language = cares.language || this.config.general.locale + const translator = this.translatorFactory.Translator(language) + let [platform] = cares.type.split(':') + if (platform === 'webhook') platform = 'discord' - data.name = translator.translate(monster.name) - data.formName = translator.translate(monster.form.name) - if (data.disguisePokemonNameEng) data.disguisePokemonName = translator.translate(data.disguisePokemonNameEng) - if (data.disguiseFormNameEng) data.disguiseFormName = translator.translate(data.disguiseFormNameEng) + data.name = translator.translate(monster.name) + data.formName = translator.translate(monster.form.name) + if (data.disguisePokemonNameEng) data.disguisePokemonName = translator.translate(data.disguisePokemonNameEng) + if (data.disguiseFormNameEng) data.disguiseFormName = translator.translate(data.disguiseFormNameEng) - data.genderData = { - name: translator.translate(data.genderDataEng.name), - emoji: translator.translate(this.emojiLookup.lookup(data.genderDataEng.emoji, platform)), - } - data.genderName = data.genderData.name - data.genderEmoji = data.genderData.emoji - data.shinyPossibleEmoji = data.shinyPossible ? translator.translate(this.emojiLookup.lookup('shiny', platform)) : '' - data.rarityName = translator.translate(data.rarityNameEng) - data.quickMoveName = encountered && this.GameData.moves[data.quickMoveId] ? translator.translate(this.GameData.moves[data.quickMoveId].name) : '' - data.quickMoveEmoji = this.GameData.moves[data.quickMoveId] && this.GameData.utilData.types[this.GameData.moves[data.quickMoveId].type] ? translator.translate(this.emojiLookup.lookup(this.GameData.utilData.types[this.GameData.moves[data.quickMoveId].type].emoji, platform)) : '' - data.chargeMoveName = encountered && this.GameData.moves[data.chargeMoveId] ? translator.translate(this.GameData.moves[data.chargeMoveId].name) : '' - data.chargeMoveEmoji = this.GameData.moves[data.chargeMoveId] && this.GameData.utilData.types[this.GameData.moves[data.chargeMoveId].type] ? translator.translate(this.emojiLookup.lookup(this.GameData.utilData.types[this.GameData.moves[data.chargeMoveId].type].emoji, platform)) : '' - data.boosted = !!data.weather - data.generationName = translator.translate(data.generationNameEng) - data.boostWeatherId = data.weather ? data.weather : '' - data.boostWeatherName = data.weather ? translator.translate(this.GameData.utilData.weather[data.weather].name) : '' - data.boostWeatherEmoji = data.weather ? translator.translate(this.emojiLookup.lookup(this.GameData.utilData.weather[data.weather].emoji, platform)) : '' - data.gameWeatherId = this.GameData.utilData.weather[currentCellWeather] ? currentCellWeather : '' - data.gameWeatherName = this.GameData.utilData.weather[currentCellWeather] ? translator.translate(this.GameData.utilData.weather[currentCellWeather].name) : '' - data.gameWeatherEmoji = this.GameData.utilData.weather[currentCellWeather] ? translator.translate(this.emojiLookup.lookup(this.GameData.utilData.weather[currentCellWeather].emoji, platform)) : '' - data.formname = data.formName // deprecated - data.quickMove = data.quickMoveName // deprecated - data.chargeMove = data.chargeMoveName // deprecated - data.move1emoji = data.quickMoveEmoji // deprecated - data.move2emoji = data.chargeMoveEmoji // deprecated - data.boost = data.boostWeatherName // deprecated - data.boostemoji = data.boostWeatherEmoji // deprecated - data.gameweather = data.gameWeatherName // deprecated - data.gameweatheremoji = data.gameWeatherEmoji // deprecated - if (data.weatherNext) { - if (!data.weatherCurrent) { - data.weatherChange = `⚠️ ${translator.translate('Possible weather change at')} ${data.weatherChangeTime} : ➡️ ${translator.translate(this.GameData.utilData.weather[data.weatherNext].name)} ${translator.translate(this.emojiLookup.lookup(this.GameData.utilData.weather[data.weatherNext].emoji, platform))}` - data.weatherCurrentName = translator.translate('unknown') - data.weatherCurrentEmoji = '❓' - } else { - data.weatherChange = `⚠️ ${translator.translate('Possible weather change at')} ${data.weatherChangeTime} : ${translator.translate(this.GameData.utilData.weather[data.weatherCurrent].name)} ${translator.translate(this.emojiLookup.lookup(this.GameData.utilData.weather[data.weatherCurrent].emoji, platform))} ➡️ ${translator.translate(this.GameData.utilData.weather[data.weatherNext].name)} ${translator.translate(this.emojiLookup.lookup(this.GameData.utilData.weather[data.weatherNext].emoji, platform))}` - data.weatherCurrentName = translator.translate(this.GameData.utilData.weather[data.weatherCurrent].name) - data.weatherCurrentEmoji = translator.translate(this.emojiLookup.lookup(this.GameData.utilData.weather[data.weatherCurrent].emoji, platform)) - } - data.weatherNextName = translator.translate(this.GameData.utilData.weather[data.weatherNext].name) - data.weatherNextEmoji = translator.translate(this.emojiLookup.lookup(this.GameData.utilData.weather[data.weatherNext].emoji, platform)) - } + data.genderData = { + name: translator.translate(data.genderDataEng.name), + emoji: translator.translate(this.emojiLookup.lookup(data.genderDataEng.emoji, platform)), + } + data.genderName = data.genderData.name + data.genderEmoji = data.genderData.emoji + data.shinyPossibleEmoji = data.shinyPossible ? translator.translate(this.emojiLookup.lookup('shiny', platform)) : '' + data.rarityName = translator.translate(data.rarityNameEng) + data.quickMoveName = encountered && this.GameData.moves[data.quickMoveId] ? translator.translate(this.GameData.moves[data.quickMoveId].name) : '' + data.quickMoveEmoji = this.GameData.moves[data.quickMoveId] && this.GameData.utilData.types[this.GameData.moves[data.quickMoveId].type] ? translator.translate(this.emojiLookup.lookup(this.GameData.utilData.types[this.GameData.moves[data.quickMoveId].type].emoji, platform)) : '' + data.chargeMoveName = encountered && this.GameData.moves[data.chargeMoveId] ? translator.translate(this.GameData.moves[data.chargeMoveId].name) : '' + data.chargeMoveEmoji = this.GameData.moves[data.chargeMoveId] && this.GameData.utilData.types[this.GameData.moves[data.chargeMoveId].type] ? translator.translate(this.emojiLookup.lookup(this.GameData.utilData.types[this.GameData.moves[data.chargeMoveId].type].emoji, platform)) : '' + data.boosted = !!data.weather + data.generationName = translator.translate(data.generationNameEng) + data.boostWeatherId = data.weather ? data.weather : '' + data.boostWeatherName = data.weather ? translator.translate(this.GameData.utilData.weather[data.weather].name) : '' + data.boostWeatherEmoji = data.weather ? translator.translate(this.emojiLookup.lookup(this.GameData.utilData.weather[data.weather].emoji, platform)) : '' + data.gameWeatherId = this.GameData.utilData.weather[currentCellWeather] ? currentCellWeather : '' + data.gameWeatherName = this.GameData.utilData.weather[currentCellWeather] ? translator.translate(this.GameData.utilData.weather[currentCellWeather].name) : '' + data.gameWeatherEmoji = this.GameData.utilData.weather[currentCellWeather] ? translator.translate(this.emojiLookup.lookup(this.GameData.utilData.weather[currentCellWeather].emoji, platform)) : '' + data.formname = data.formName // deprecated + data.quickMove = data.quickMoveName // deprecated + data.chargeMove = data.chargeMoveName // deprecated + data.move1emoji = data.quickMoveEmoji // deprecated + data.move2emoji = data.chargeMoveEmoji // deprecated + data.boost = data.boostWeatherName // deprecated + data.boostemoji = data.boostWeatherEmoji // deprecated + data.gameweather = data.gameWeatherName // deprecated + data.gameweatheremoji = data.gameWeatherEmoji // deprecated + if (data.weatherNext) { + if (!data.weatherCurrent) { + data.weatherChange = `⚠️ ${translator.translate('Possible weather change at')} ${data.weatherChangeTime} : ➡️ ${translator.translate(this.GameData.utilData.weather[data.weatherNext].name)} ${translator.translate(this.emojiLookup.lookup(this.GameData.utilData.weather[data.weatherNext].emoji, platform))}` + data.weatherCurrentName = translator.translate('unknown') + data.weatherCurrentEmoji = '❓' + } else { + data.weatherChange = `⚠️ ${translator.translate('Possible weather change at')} ${data.weatherChangeTime} : ${translator.translate(this.GameData.utilData.weather[data.weatherCurrent].name)} ${translator.translate(this.emojiLookup.lookup(this.GameData.utilData.weather[data.weatherCurrent].emoji, platform))} ➡️ ${translator.translate(this.GameData.utilData.weather[data.weatherNext].name)} ${translator.translate(this.emojiLookup.lookup(this.GameData.utilData.weather[data.weatherNext].emoji, platform))}` + data.weatherCurrentName = translator.translate(this.GameData.utilData.weather[data.weatherCurrent].name) + data.weatherCurrentEmoji = translator.translate(this.emojiLookup.lookup(this.GameData.utilData.weather[data.weatherCurrent].emoji, platform)) + } + data.weatherNextName = translator.translate(this.GameData.utilData.weather[data.weatherNext].name) + data.weatherNextEmoji = translator.translate(this.emojiLookup.lookup(this.GameData.utilData.weather[data.weatherNext].emoji, platform)) + } - const e = [] - const n = [] - monster.types.forEach((type) => { - e.push(translator.translate(this.emojiLookup.lookup(this.GameData.utilData.types[type.name].emoji, platform))) - n.push(type.name) - }) - data.emoji = e - data.typeNameEng = n - data.typeName = data.typeNameEng.map((type) => translator.translate(type)).join(', ') - data.emojiString = e.join('') - data.typeEmoji = data.emojiString - - const createPvpDisplay = (leagueData, maxRank, minCp) => { - const displayList = [] - for (const rank of leagueData) { - if (rank.rank <= maxRank - && rank.cp >= minCp) { - const displayRank = {} - - displayRank.rank = +rank.rank - displayRank.formId = +rank.form || 0 - displayRank.evolution = rank.evolution - displayRank.level = +rank.level - displayRank.cap = rank.cap - displayRank.capped = rank.capped - displayRank.levelWithCap = displayRank.cap && !displayRank.capped ? `${displayRank.level}/${displayRank.cap}` : displayRank.level - displayRank.cp = rank.cp - displayRank.pokemonId = +rank.pokemon - displayRank.percentage = rank.percentage <= 1 ? (rank.percentage * 100).toFixed(2) : rank.percentage.toFixed(2) - - let monsterName - let formName - let stats - - let mon = this.GameData.monsters[`${displayRank.pokemonId}_${displayRank.formId}`] - if (!mon) { - mon = this.GameData.monsters[`${displayRank.pokemonId}_0`] - if (!mon) { - monsterName = `${translator.translate('Unknown monster')} ${displayRank.pokemonId}` - stats = { - baseAttack: 0, - baseDefense: 0, - baseStamina: 0, + const e = [] + const n = [] + monster.types.forEach((type) => { + e.push(translator.translate(this.emojiLookup.lookup(this.GameData.utilData.types[type.name].emoji, platform))) + n.push(type.name) + }) + data.emoji = e + data.typeNameEng = n + data.typeName = data.typeNameEng.map((type) => translator.translate(type)) + .join(', ') + data.emojiString = e.join('') + data.typeEmoji = data.emojiString + + const createPvpDisplay = (leagueData, maxRank, minCp) => { + const displayList = [] + for (const rank of leagueData) { + if (rank.rank <= maxRank + && rank.cp >= minCp) { + const displayRank = {} + + displayRank.rank = +rank.rank + displayRank.formId = +rank.form || 0 + displayRank.evolution = rank.evolution + displayRank.level = +rank.level + displayRank.cap = rank.cap + displayRank.capped = rank.capped + displayRank.levelWithCap = displayRank.cap && !displayRank.capped ? `${displayRank.level}/${displayRank.cap}` : displayRank.level + displayRank.cp = rank.cp + displayRank.pokemonId = +rank.pokemon + displayRank.percentage = rank.percentage <= 1 ? (rank.percentage * 100).toFixed(2) : rank.percentage.toFixed(2) + + let monsterName + let formName + let stats + + let mon = this.GameData.monsters[`${displayRank.pokemonId}_${displayRank.formId}`] + if (!mon) { + mon = this.GameData.monsters[`${displayRank.pokemonId}_0`] + if (!mon) { + monsterName = `${translator.translate('Unknown monster')} ${displayRank.pokemonId}` + stats = { + baseAttack: 0, + baseDefense: 0, + baseStamina: 0, + } + } else { + monsterName = mon.name + stats = monster.stats + } + formName = `${displayRank.formId}` + } else { + monsterName = mon.name + formName = mon.form.name + if (formName === undefined || formName === 'Normal') formName = '' + stats = monster.stats } - } else { - monsterName = mon.name - stats = monster.stats + displayRank.baseStats = stats + displayRank.nameEng = monsterName + displayRank.formEng = formName + displayRank.name = translator.translate(monsterName) + displayRank.form = translator.translate(formName) + if (displayRank.evolution) { + displayRank.fullNameEng = 'Mega '.concat(displayRank.nameEng, displayRank.formEng ? ' ' : '', displayRank.formEng) + displayRank.fullName = translator.translateFormat(this.GameData.utilData.megaName[1], + displayRank.name.concat(displayRank.form ? ' ' : '', displayRank.form)) + } else { + displayRank.fullNameEng = displayRank.nameEng.concat(displayRank.formEng ? ' ' : '', displayRank.formEng) + displayRank.fullName = displayRank.name.concat(displayRank.form ? ' ' : '', displayRank.form) + } + displayList.push(displayRank) } - formName = `${displayRank.formId}` - } else { - monsterName = mon.name - formName = mon.form.name - if (formName === undefined || formName === 'Normal') formName = '' - stats = monster.stats - } - displayRank.baseStats = stats - displayRank.nameEng = monsterName - displayRank.formEng = formName - displayRank.name = translator.translate(monsterName) - displayRank.form = translator.translate(formName) - if (displayRank.evolution) { - displayRank.fullNameEng = 'Mega '.concat(displayRank.nameEng, displayRank.formEng ? ' ' : '', displayRank.formEng) - displayRank.fullName = translator.translateFormat(this.GameData.utilData.megaName[1], - displayRank.name.concat(displayRank.form ? ' ' : '', displayRank.form)) - } else { - displayRank.fullNameEng = displayRank.nameEng.concat(displayRank.formEng ? ' ' : '', displayRank.formEng) - displayRank.fullName = displayRank.name.concat(displayRank.form ? ' ' : '', displayRank.form) } - displayList.push(displayRank) + return displayList.length ? displayList : null } - } - return displayList.length ? displayList : null - } - data.pvpGreat = data.pvp_rankings_great_league ? createPvpDisplay(data.pvp_rankings_great_league, this.config.pvp.pvpDisplayMaxRank, this.config.pvp.pvpDisplayGreatMinCP) : null - data.pvpUltra = data.pvp_rankings_ultra_league ? createPvpDisplay(data.pvp_rankings_ultra_league, this.config.pvp.pvpDisplayMaxRank, this.config.pvp.pvpDisplayUltraMinCP) : null - data.pvpLittle = data.pvp_rankings_little_league ? createPvpDisplay(data.pvp_rankings_little_league, this.config.pvp.pvpDisplayMaxRank, this.config.pvp.pvpDisplayLittleMinCP) : null - data.pvpAvailable = data.pvpGreat !== null || data.pvpUltra !== null || data.pvpLittle !== null - - data.distance = cares.longitude ? this.getDistance({ lat: cares.latitude, lon: cares.longitude }, { lat: data.latitude, lon: data.longitude }) : '' - - const view = { - ...geoResult, - ...data, - id: data.pokemon_id, - baseStats: monster.stats, - time: data.disappearTime, - tthh: data.tth.hours, - tthm: data.tth.minutes, - tths: data.tth.seconds, - now: new Date(), - nowISO: new Date().toISOString(), - pvpUserRanking: cares.pvp_ranking_worst === 4096 ? 0 : cares.pvp_ranking_worst, - areas: data.matchedAreas.filter((area) => area.displayInMatches).map((area) => area.name).join(', '), - pvpDisplayMaxRank: this.config.pvp.pvpDisplayMaxRank, - pvpDisplayGreatMinCP: this.config.pvp.pvpDisplayGreatMinCP, - pvpDisplayUltraMinCP: this.config.pvp.pvpDisplayUltraMinCP, - pvpDisplayLittleMinCP: this.config.pvp.pvpDisplayLittleMinCP, - } + data.pvpGreat = data.pvp_rankings_great_league ? createPvpDisplay(data.pvp_rankings_great_league, this.config.pvp.pvpDisplayMaxRank, this.config.pvp.pvpDisplayGreatMinCP) : null + data.pvpUltra = data.pvp_rankings_ultra_league ? createPvpDisplay(data.pvp_rankings_ultra_league, this.config.pvp.pvpDisplayMaxRank, this.config.pvp.pvpDisplayUltraMinCP) : null + data.pvpLittle = data.pvp_rankings_little_league ? createPvpDisplay(data.pvp_rankings_little_league, this.config.pvp.pvpDisplayMaxRank, this.config.pvp.pvpDisplayLittleMinCP) : null + data.pvpAvailable = data.pvpGreat !== null || data.pvpUltra !== null || data.pvpLittle !== null + + data.distance = cares.longitude ? this.getDistance({ + lat: cares.latitude, + lon: cares.longitude, + }, { + lat: data.latitude, + lon: data.longitude, + }) : '' + + const view = { + ...geoResult, + ...data, + id: data.pokemon_id, + baseStats: monster.stats, + time: data.disappearTime, + tthh: data.tth.hours, + tthm: data.tth.minutes, + tths: data.tth.seconds, + now: new Date(), + nowISO: new Date().toISOString(), + pvpUserRanking: cares.pvp_ranking_worst === 4096 ? 0 : cares.pvp_ranking_worst, + areas: data.matchedAreas.filter((area) => area.displayInMatches) + .map((area) => area.name) + .join(', '), + pvpDisplayMaxRank: this.config.pvp.pvpDisplayMaxRank, + pvpDisplayGreatMinCP: this.config.pvp.pvpDisplayGreatMinCP, + pvpDisplayUltraMinCP: this.config.pvp.pvpDisplayUltraMinCP, + pvpDisplayLittleMinCP: this.config.pvp.pvpDisplayLittleMinCP, + } - const templateType = (data.iv === -1) ? 'monsterNoIv' : 'monster' - const message = await this.createMessage(logReference, templateType, platform, cares.template, language, cares.ping, view) - - const work = { - lat: data.latitude.toString().substring(0, 8), - lon: data.longitude.toString().substring(0, 8), - message, - target: cares.id, - type: cares.type, - name: cares.name, - tth: data.tth, - clean: cares.clean, - emoji: data.emoji, - logReference, - language, - } - jobs.push(work) - } - hrend = process.hrtime(hrstart) - const hrendprocessing = hrend[1] / 1000000 - this.log.verbose(`${data.encounter_id}: ${monster.name} appeared and ${whoCares.length} humans cared [end]. (${hrendms} ms sql + ${hrendprocessing} ms processing dts)`) + const templateType = (data.iv === -1) ? 'monsterNoIv' : 'monster' + const message = await this.createMessage(logReference, templateType, platform, cares.template, language, cares.ping, view) + + const work = { + lat: data.latitude.toString() + .substring(0, 8), + lon: data.longitude.toString() + .substring(0, 8), + message, + target: cares.id, + type: cares.type, + name: cares.name, + tth: data.tth, + clean: cares.clean, + emoji: data.emoji, + logReference, + language, + } + jobs.push(work) + } + hrend = process.hrtime(hrstart) + const hrendprocessing = hrend[1] / 1000000 + this.log.verbose(`${data.encounter_id}: ${monster.name} appeared and ${whoCares.length} humans cared [end]. (${hrendms} ms sql + ${hrendprocessing} ms processing dts)`) - return jobs + this.emit('postMessage', jobs) + } catch (e) { + this.log.error(`${data.encounter_id}: Can't seem to handle monster (user cared): `, e, data) + } + }) + return [] } catch (e) { this.log.error(`${data.encounter_id}: Can't seem to handle monster: `, e, data) } diff --git a/src/controllers/pokestop.js b/src/controllers/pokestop.js index 6f76a332d..aac7da117 100644 --- a/src/controllers/pokestop.js +++ b/src/controllers/pokestop.js @@ -144,151 +144,158 @@ class Invasion extends Controller { return [] } - data.imgUrl = await this.imgUicons.invasionIcon(data.gruntTypeId) - data.stickerUrl = await this.stickerUicons.invasionIcon(data.gruntTypeId) + setImmediate(async () => { + try { + data.imgUrl = await this.imgUicons.invasionIcon(data.gruntTypeId) + data.stickerUrl = await this.stickerUicons.invasionIcon(data.gruntTypeId) - const geoResult = await this.getAddress({ lat: data.latitude, lon: data.longitude }) - const jobs = [] + const geoResult = await this.getAddress({ lat: data.latitude, lon: data.longitude }) + const jobs = [] - await this.getStaticMapUrl(logReference, data, 'pokestop', ['latitude', 'longitude', 'imgUrl', 'gruntTypeId']) - data.staticmap = data.staticMap // deprecated + await this.getStaticMapUrl(logReference, data, 'pokestop', ['latitude', 'longitude', 'imgUrl', 'gruntTypeId']) + data.staticmap = data.staticMap // deprecated - for (const cares of whoCares) { - this.log.debug(`${logReference}: Creating invasion alert for ${cares.id} ${cares.name} ${cares.type} ${cares.language} ${cares.template}`, cares) + for (const cares of whoCares) { + this.log.debug(`${logReference}: Creating invasion alert for ${cares.id} ${cares.name} ${cares.type} ${cares.language} ${cares.template}`, cares) - const rateLimitTtr = this.getRateLimitTimeToRelease(cares.id) - if (rateLimitTtr) { - this.log.verbose(`${logReference}: Not creating invasion alert (Rate limit) for ${cares.type} ${cares.id} ${cares.name} Time to release: ${rateLimitTtr}`) - // eslint-disable-next-line no-continue - continue - } - this.log.verbose(`${logReference}: Creating invasion alert for ${cares.type} ${cares.id} ${cares.name} ${cares.language} ${cares.template}`) - - const language = cares.language || this.config.general.locale - const translator = this.translatorFactory.Translator(language) - let [platform] = cares.type.split(':') - if (platform === 'webhook') platform = 'discord' - - data.gruntTypeEmoji = translator.translate(this.emojiLookup.lookup('grunt-unknown', platform)) - - // full build - if (data.gruntTypeId) { - data.gender = 0 - data.gruntName = translator.translate('Grunt') - data.gruntType = translator.translate('Mixed') - data.gruntRewards = '' - if (data.gruntTypeId in this.GameData.grunts) { - const gruntType = this.GameData.grunts[data.gruntTypeId] - data.gruntName = translator.translate(`${gruntType.type} ${gruntType.grunt}`) - data.gender = gruntType.gender - data.genderDataEng = this.GameData.utilData.genders[data.gender] - if (!data.genderDataEng) { - data.genderDataEng = { name: '', emoji: '' } - } - if (this.GameData.utilData.types[gruntType.type]) { - data.gruntTypeEmoji = translator.translate(this.emojiLookup.lookup(this.GameData.utilData.types[gruntType.type].emoji, platform)) - } - if (gruntType.type in this.GameData.utilData.types) { - data.gruntTypeColor = this.GameData.utilData.types[gruntType.type].color + const rateLimitTtr = this.getRateLimitTimeToRelease(cares.id) + if (rateLimitTtr) { + this.log.verbose(`${logReference}: Not creating invasion alert (Rate limit) for ${cares.type} ${cares.id} ${cares.name} Time to release: ${rateLimitTtr}`) + // eslint-disable-next-line no-continue + continue } - data.gruntType = translator.translate(gruntType.type) - - let gruntRewards = '' - const gruntRewardsList = {} - gruntRewardsList.first = { chance: 100, monsters: [] } - if (gruntType.encounters && gruntType.encounters.first) { - if (gruntType.secondReward && gruntType.encounters.second) { - // one out of two rewards - gruntRewards = '85%: ' - gruntRewardsList.first = { chance: 85, monsters: [] } - let first = true - gruntType.encounters.first.forEach((fr) => { - if (!first) gruntRewards += ', ' - else first = false - - const firstReward = +fr - const firstRewardMonster = Object.values(this.GameData.monsters).find((mon) => mon.id === firstReward && !mon.form.id) - gruntRewards += firstRewardMonster ? translator.translate(firstRewardMonster.name) : '' - gruntRewardsList.first.monsters.push({ - id: firstReward, - name: translator.translate(firstRewardMonster.name), - }) - }) - gruntRewards += '\\n15%: ' - gruntRewardsList.second = { chance: 15, monsters: [] } - first = true - gruntType.encounters.second.forEach((sr) => { - if (!first) gruntRewards += ', ' - else first = false - - const secondReward = +sr - const secondRewardMonster = Object.values(this.GameData.monsters).find((mon) => mon.id === secondReward && !mon.form.id) - - gruntRewards += secondRewardMonster ? translator.translate(secondRewardMonster.name) : '' - gruntRewardsList.second.monsters.push({ - id: secondReward, - name: translator.translate(secondRewardMonster.name), - }) - }) - } else { - // Single Reward 100% of encounter (might vary based on actual fight). - let first = true - gruntType.encounters.first.forEach((fr) => { - if (!first) gruntRewards += ', ' - else first = false - - const firstReward = +fr - const firstRewardMonster = Object.values(this.GameData.monsters).find((mon) => mon.id === firstReward && !mon.form.id) - gruntRewards += firstRewardMonster ? translator.translate(firstRewardMonster.name) : '' - gruntRewardsList.first.monsters.push({ - id: firstReward, - name: translator.translate(firstRewardMonster.name), - }) - }) + this.log.verbose(`${logReference}: Creating invasion alert for ${cares.type} ${cares.id} ${cares.name} ${cares.language} ${cares.template}`) + + const language = cares.language || this.config.general.locale + const translator = this.translatorFactory.Translator(language) + let [platform] = cares.type.split(':') + if (platform === 'webhook') platform = 'discord' + + data.gruntTypeEmoji = translator.translate(this.emojiLookup.lookup('grunt-unknown', platform)) + + // full build + if (data.gruntTypeId) { + data.gender = 0 + data.gruntName = translator.translate('Grunt') + data.gruntType = translator.translate('Mixed') + data.gruntRewards = '' + if (data.gruntTypeId in this.GameData.grunts) { + const gruntType = this.GameData.grunts[data.gruntTypeId] + data.gruntName = translator.translate(`${gruntType.type} ${gruntType.grunt}`) + data.gender = gruntType.gender + data.genderDataEng = this.GameData.utilData.genders[data.gender] + if (!data.genderDataEng) { + data.genderDataEng = { name: '', emoji: '' } + } + if (this.GameData.utilData.types[gruntType.type]) { + data.gruntTypeEmoji = translator.translate(this.emojiLookup.lookup(this.GameData.utilData.types[gruntType.type].emoji, platform)) + } + if (gruntType.type in this.GameData.utilData.types) { + data.gruntTypeColor = this.GameData.utilData.types[gruntType.type].color + } + data.gruntType = translator.translate(gruntType.type) + + let gruntRewards = '' + const gruntRewardsList = {} + gruntRewardsList.first = { chance: 100, monsters: [] } + if (gruntType.encounters && gruntType.encounters.first) { + if (gruntType.secondReward && gruntType.encounters.second) { + // one out of two rewards + gruntRewards = '85%: ' + gruntRewardsList.first = { chance: 85, monsters: [] } + let first = true + gruntType.encounters.first.forEach((fr) => { + if (!first) gruntRewards += ', ' + else first = false + + const firstReward = +fr + const firstRewardMonster = Object.values(this.GameData.monsters).find((mon) => mon.id === firstReward && !mon.form.id) + gruntRewards += firstRewardMonster ? translator.translate(firstRewardMonster.name) : '' + gruntRewardsList.first.monsters.push({ + id: firstReward, + name: translator.translate(firstRewardMonster.name), + }) + }) + gruntRewards += '\\n15%: ' + gruntRewardsList.second = { chance: 15, monsters: [] } + first = true + gruntType.encounters.second.forEach((sr) => { + if (!first) gruntRewards += ', ' + else first = false + + const secondReward = +sr + const secondRewardMonster = Object.values(this.GameData.monsters).find((mon) => mon.id === secondReward && !mon.form.id) + + gruntRewards += secondRewardMonster ? translator.translate(secondRewardMonster.name) : '' + gruntRewardsList.second.monsters.push({ + id: secondReward, + name: translator.translate(secondRewardMonster.name), + }) + }) + } else { + // Single Reward 100% of encounter (might vary based on actual fight). + let first = true + gruntType.encounters.first.forEach((fr) => { + if (!first) gruntRewards += ', ' + else first = false + + const firstReward = +fr + const firstRewardMonster = Object.values(this.GameData.monsters).find((mon) => mon.id === firstReward && !mon.form.id) + gruntRewards += firstRewardMonster ? translator.translate(firstRewardMonster.name) : '' + gruntRewardsList.first.monsters.push({ + id: firstReward, + name: translator.translate(firstRewardMonster.name), + }) + }) + } + data.gruntRewards = gruntRewards + data.gruntRewardsList = gruntRewardsList + } } - data.gruntRewards = gruntRewards - data.gruntRewardsList = gruntRewardsList } - } - } - const view = { - ...geoResult, - ...data, - time: data.distime, - tthh: data.tth.hours, - tthm: data.tth.minutes, - tths: data.tth.seconds, - confirmedTime: data.disappear_time_verified, - now: new Date(), - nowISO: new Date().toISOString(), - genderData: data.genderDataEng ? { - name: translator.translate(data.genderDataEng.name), - emoji: translator.translate(this.emojiLookup.lookup(data.genderDataEng.emoji, platform)), - } : { name: '', emoji: '' }, - areas: data.matchedAreas.filter((area) => area.displayInMatches).map((area) => area.name).join(', '), - } + const view = { + ...geoResult, + ...data, + time: data.distime, + tthh: data.tth.hours, + tthm: data.tth.minutes, + tths: data.tth.seconds, + confirmedTime: data.disappear_time_verified, + now: new Date(), + nowISO: new Date().toISOString(), + genderData: data.genderDataEng ? { + name: translator.translate(data.genderDataEng.name), + emoji: translator.translate(this.emojiLookup.lookup(data.genderDataEng.emoji, platform)), + } : { name: '', emoji: '' }, + areas: data.matchedAreas.filter((area) => area.displayInMatches).map((area) => area.name).join(', '), + } - const templateType = 'invasion' - const message = await this.createMessage(logReference, templateType, platform, cares.template, language, cares.ping, view) - - const work = { - lat: data.latitude.toString().substring(0, 8), - lon: data.longitude.toString().substring(0, 8), - message, - target: cares.id, - type: cares.type, - name: cares.name, - tth: data.tth, - clean: cares.clean, - emoji: data.emoji, - logReference, - language, + const templateType = 'invasion' + const message = await this.createMessage(logReference, templateType, platform, cares.template, language, cares.ping, view) + + const work = { + lat: data.latitude.toString().substring(0, 8), + lon: data.longitude.toString().substring(0, 8), + message, + target: cares.id, + type: cares.type, + name: cares.name, + tth: data.tth, + clean: cares.clean, + emoji: data.emoji, + logReference, + language, + } + jobs.push(work) + } + this.emit('postMessage', jobs) + } catch (e) { + this.log.error(`${data.pokestop_id}: Can't seem to handle pokestop (user cares): `, e, data) } - jobs.push(work) - } + }) - return jobs + return [] } catch (e) { this.log.error(`${data.pokestop_id}: Can't seem to handle pokestop: `, e, data) } diff --git a/src/controllers/pokestop_lure.js b/src/controllers/pokestop_lure.js index 8458a0a6e..fea1c806b 100644 --- a/src/controllers/pokestop_lure.js +++ b/src/controllers/pokestop_lure.js @@ -124,69 +124,83 @@ class Lure extends Controller { return [] } - data.imgUrl = await this.imgUicons.pokestopIcon(data.lureTypeId) - data.stickerUrl = await this.stickerUicons.pokestopIcon(data.lureTypeId) - - const geoResult = await this.getAddress({ lat: data.latitude, lon: data.longitude }) - const jobs = [] - - await this.getStaticMapUrl(logReference, data, 'pokestop', ['latitude', 'longitude', 'imgUrl', 'lureTypeId']) - data.staticmap = data.staticMap // deprecated - - for (const cares of whoCares) { - this.log.debug(`${logReference}: Creating lure alert for ${cares.id} ${cares.name} ${cares.type} ${cares.language} ${cares.template}`, cares) - - const rateLimitTtr = this.getRateLimitTimeToRelease(cares.id) - if (rateLimitTtr) { - this.log.verbose(`${logReference}: Not creating lure alert (Rate limit) for ${cares.type} ${cares.id} ${cares.name} Time to release: ${rateLimitTtr}`) - // eslint-disable-next-line no-continue - continue - } - this.log.verbose(`${logReference}: Creating lure alert for ${cares.type} ${cares.id} ${cares.name} ${cares.language} ${cares.template}`) - - const language = cares.language || this.config.general.locale - const translator = this.translatorFactory.Translator(language) - let [platform] = cares.type.split(':') - if (platform === 'webhook') platform = 'discord' - - // full build - data.lureTypeName = translator.translate(data.lureTypeNameEng) - data.lureType = data.lureTypeName - data.lureTypeEmoji = this.emojiLookup.lookup(this.GameData.utilData.lures[data.lure_id].emoji, platform) - - const view = { - ...geoResult, - ...data, - time: data.distime, - tthh: data.tth.hours, - tthm: data.tth.minutes, - tths: data.tth.seconds, - now: new Date(), - nowISO: new Date().toISOString(), - areas: data.matchedAreas.filter((area) => area.displayInMatches).map((area) => area.name).join(', '), + setImmediate(async () => { + try { + data.imgUrl = await this.imgUicons.pokestopIcon(data.lureTypeId) + data.stickerUrl = await this.stickerUicons.pokestopIcon(data.lureTypeId) + + const geoResult = await this.getAddress({ + lat: data.latitude, + lon: data.longitude, + }) + const jobs = [] + + await this.getStaticMapUrl(logReference, data, 'pokestop', ['latitude', 'longitude', 'imgUrl', 'lureTypeId']) + data.staticmap = data.staticMap // deprecated + + for (const cares of whoCares) { + this.log.debug(`${logReference}: Creating lure alert for ${cares.id} ${cares.name} ${cares.type} ${cares.language} ${cares.template}`, cares) + + const rateLimitTtr = this.getRateLimitTimeToRelease(cares.id) + if (rateLimitTtr) { + this.log.verbose(`${logReference}: Not creating lure alert (Rate limit) for ${cares.type} ${cares.id} ${cares.name} Time to release: ${rateLimitTtr}`) + // eslint-disable-next-line no-continue + continue + } + this.log.verbose(`${logReference}: Creating lure alert for ${cares.type} ${cares.id} ${cares.name} ${cares.language} ${cares.template}`) + + const language = cares.language || this.config.general.locale + const translator = this.translatorFactory.Translator(language) + let [platform] = cares.type.split(':') + if (platform === 'webhook') platform = 'discord' + + // full build + data.lureTypeName = translator.translate(data.lureTypeNameEng) + data.lureType = data.lureTypeName + data.lureTypeEmoji = this.emojiLookup.lookup(this.GameData.utilData.lures[data.lure_id].emoji, platform) + + const view = { + ...geoResult, + ...data, + time: data.distime, + tthh: data.tth.hours, + tthm: data.tth.minutes, + tths: data.tth.seconds, + now: new Date(), + nowISO: new Date().toISOString(), + areas: data.matchedAreas.filter((area) => area.displayInMatches) + .map((area) => area.name) + .join(', '), + } + + const templateType = 'lure' + const message = await this.createMessage(logReference, templateType, platform, cares.template, language, cares.ping, view) + + const work = { + lat: data.latitude.toString() + .substring(0, 8), + lon: data.longitude.toString() + .substring(0, 8), + message, + target: cares.id, + type: cares.type, + name: cares.name, + tth: data.tth, + clean: cares.clean, + emoji: data.emoji, + logReference, + language, + } + + jobs.push(work) + } + this.emit('postMessage', jobs) + } catch (e) { + this.log.error(`${data.pokestop_id}: Can't seem to handle pokestop(lure) (user cares): `, e, data) } + }) - const templateType = 'lure' - const message = await this.createMessage(logReference, templateType, platform, cares.template, language, cares.ping, view) - - const work = { - lat: data.latitude.toString().substring(0, 8), - lon: data.longitude.toString().substring(0, 8), - message, - target: cares.id, - type: cares.type, - name: cares.name, - tth: data.tth, - clean: cares.clean, - emoji: data.emoji, - logReference, - language, - } - - jobs.push(work) - } - - return jobs + return [] } catch (e) { this.log.error(`${data.pokestop_id}: Can't seem to handle pokestop(lure): `, e, data) } diff --git a/src/controllers/raid.js b/src/controllers/raid.js index bbd1b47f2..721c8445b 100644 --- a/src/controllers/raid.js +++ b/src/controllers/raid.js @@ -195,7 +195,8 @@ class Raid extends Controller { data.chargeMoveNameEng = this.GameData.moves[data.move_2] ? this.GameData.moves[data.move_2].name : '' data.shinyPossible = this.shinyPossible.isShinyPossible(data.pokemonId, data.formId) // eslint-disable-next-line prefer-destructuring - data.generation = this.GameData.utilData.genException[`${data.pokemon_id}_${data.form}`] || Object.entries(this.GameData.utilData.genData).find(([, genData]) => data.pokemonId >= genData.min && data.pokemonId <= genData.max)[0] + data.generation = this.GameData.utilData.genException[`${data.pokemon_id}_${data.form}`] || Object.entries(this.GameData.utilData.genData) + .find(([, genData]) => data.pokemonId >= genData.min && data.pokemonId <= genData.max)[0] data.generationNameEng = this.GameData.utilData.genData[data.generation].name data.generationRoman = this.GameData.utilData.genData[data.generation].roman @@ -228,114 +229,134 @@ class Raid extends Controller { return [] } - data.imgUrl = await this.imgUicons.pokemonIcon(data.pokemon_id, data.form, data.evolution, data.gender, data.costume, data.shinyPossible && this.config.general.requestShinyImages) - data.stickerUrl = await this.stickerUicons.pokemonIcon(data.pokemon_id, data.form, data.evolution, data.gender, data.costume, data.shinyPossible && this.config.general.requestShinyImages) - // data.imgUrl = `${this.config.general.imgUrl}pokemon_icon_${data.pokemon_id.toString().padStart(3, '0')}_${data.form ? data.form.toString() : '00'}${data.evolution > 0 ? `_${data.evolution.toString()}` : ''}.png` - // data.stickerUrl = `${this.config.general.stickerUrl}pokemon_icon_${data.pokemon_id.toString().padStart(3, '0')}_${data.form ? data.form.toString() : '00'}${data.evolution > 0 ? `_${data.evolution.toString()}` : ''}.webp` - - const geoResult = await this.getAddress({ lat: data.latitude, lon: data.longitude }) - const jobs = [] - - await this.getStaticMapUrl(logReference, data, 'raid', ['pokemon_id', 'latitude', 'longitude', 'form', 'level', 'imgUrl']) - data.staticmap = data.staticMap // deprecated - - for (const cares of whoCares) { - this.log.debug(`${logReference}: Creating raid alert for ${cares.id} ${cares.name} ${cares.type} ${cares.language} ${cares.template}`, cares) - - const rateLimitTtr = this.getRateLimitTimeToRelease(cares.id) - if (rateLimitTtr) { - this.log.verbose(`${logReference}: Not creating raid alert (Rate limit) for ${cares.type} ${cares.id} ${cares.name} Time to release: ${rateLimitTtr}`) - // eslint-disable-next-line no-continue - continue - } - this.log.verbose(`${logReference}: Creating raid alert for ${cares.type} ${cares.id} ${cares.name} ${cares.language} ${cares.template}`) - - const language = cares.language || this.config.general.locale - const translator = this.translatorFactory.Translator(language) - let [platform] = cares.type.split(':') - if (platform === 'webhook') platform = 'discord' - - data.name = translator.translate(data.nameEng) - data.formName = translator.translate(data.formNameEng) - data.evolutionName = translator.translate(data.evolutionNameEng) - data.megaName = data.evolution ? translator.translateFormat(this.GameData.utilData.megaName[data.evolution], data.name) : data.name - data.teamNameEng = data.team_id ? this.GameData.utilData.teams[data.team_id].name : 'Harmony' - data.teamName = translator.translate(data.teamNameEng) - data.teamEmoji = data.team_id !== undefined ? this.emojiLookup.lookup(this.GameData.utilData.teams[data.team_id].emoji, platform) : '' - data.quickMoveName = this.GameData.moves[data.move_1] ? translator.translate(this.GameData.moves[data.move_1].name) : '' - data.quickMoveEmoji = this.GameData.moves[data.move_1] && this.GameData.moves[data.move_1].type ? translator.translate(this.emojiLookup.lookup(this.GameData.utilData.types[this.GameData.moves[data.move_1].type].emoji, platform)) : '' - data.chargeMoveName = this.GameData.moves[data.move_2] ? translator.translate(this.GameData.moves[data.move_2].name) : '' - data.chargeMoveEmoji = this.GameData.moves[data.move_2] && this.GameData.moves[data.move_2].type ? translator.translate(this.emojiLookup.lookup(this.GameData.utilData.types[this.GameData.moves[data.move_2].type].emoji, platform)) : '' - data.gameWeatherName = data.weather ? translator.translate(data.gameWeatherNameEng) : '' - data.gameWeatherEmoji = data.weather ? translator.translate(this.emojiLookup.lookup(this.GameData.utilData.weather[data.weather].emoji, platform)) : '' - data.shinyPossibleEmoji = data.shinyPossible ? translator.translate(this.emojiLookup.lookup('shiny', platform)) : '' - data.generationName = translator.translate(data.generationNameEng) - - data.quickMove = data.quickMoveName // deprecated - data.chargeMove = data.chargeMoveName // deprecated - data.move1 = data.quickMoveName // deprecated - data.move2 = data.chargeMoveName // deprecated - data.move1emoji = data.quickMoveEmoji // deprecated - data.move2emoji = data.chargeMoveEmoji // deprecated - - const e = [] - const t = [] - const n = [] - monster.types.forEach((type) => { - e.push(this.emojiLookup.lookup(this.GameData.utilData.types[type.name].emoji, platform)) - t.push(type.id) - n.push(type.name) - }) - data.types = t - data.typeNameEng = n - data.emoji = e - - data.typeName = data.typeNameEng.map((type) => translator.translate(type)).join(', ') - data.typeEmoji = data.emoji.map((emoji) => translator.translate(emoji)).join('') - - data.boostingWeathers = data.types.map((type) => parseInt(Object.keys(this.GameData.utilData.weatherTypeBoost).find((key) => this.GameData.utilData.weatherTypeBoost[key].includes(type)), 10)) - data.boosted = !!data.boostingWeathers.includes(data.weather) - data.boostWeatherNameEng = data.boosted ? this.GameData.utilData.weather[data.weather].name : '' - data.boostWeatherId = data.boosted ? data.weather : '' - data.boostWeatherName = data.boosted ? translator.translate(this.GameData.utilData.weather[data.weather].name) : '' - data.boostWeatherEmoji = data.boosted ? translator.translate(this.emojiLookup.lookup(this.GameData.utilData.weather[data.weather].emoji, platform)) : '' - - const view = { - ...geoResult, - ...data, - pokemonName: data.pokemonName, - id: data.pokemon_id, - baseStats: monster.stats, - time: data.disappearTime, - tthh: data.tth.hours, - tthm: data.tth.minutes, - tths: data.tth.seconds, - confirmedTime: data.disappear_time_verified, - now: new Date(), - nowISO: new Date().toISOString(), - genderData: { name: translator.translate(data.genderDataEng.name), emoji: translator.translate(this.emojiLookup.lookup(data.genderDataEng.emoji, platform)) }, - areas: data.matchedAreas.filter((area) => area.displayInMatches).map((area) => area.name).join(', '), - } - - const templateType = 'raid' - const message = await this.createMessage(logReference, templateType, platform, cares.template, language, cares.ping, view) - - const work = { - lat: data.latitude.toString().substring(0, 8), - lon: data.longitude.toString().substring(0, 8), - message, - target: cares.id, - type: cares.type, - name: cares.name, - tth: data.tth, - clean: cares.clean, - emoji: data.emoji, - logReference, - language, + setImmediate(async () => { + try { + data.imgUrl = await this.imgUicons.pokemonIcon(data.pokemon_id, data.form, data.evolution, data.gender, data.costume, data.shinyPossible && this.config.general.requestShinyImages) + data.stickerUrl = await this.stickerUicons.pokemonIcon(data.pokemon_id, data.form, data.evolution, data.gender, data.costume, data.shinyPossible && this.config.general.requestShinyImages) + // data.imgUrl = `${this.config.general.imgUrl}pokemon_icon_${data.pokemon_id.toString().padStart(3, '0')}_${data.form ? data.form.toString() : '00'}${data.evolution > 0 ? `_${data.evolution.toString()}` : ''}.png` + // data.stickerUrl = `${this.config.general.stickerUrl}pokemon_icon_${data.pokemon_id.toString().padStart(3, '0')}_${data.form ? data.form.toString() : '00'}${data.evolution > 0 ? `_${data.evolution.toString()}` : ''}.webp` + + const geoResult = await this.getAddress({ + lat: data.latitude, + lon: data.longitude, + }) + const jobs = [] + + await this.getStaticMapUrl(logReference, data, 'raid', ['pokemon_id', 'latitude', 'longitude', 'form', 'level', 'imgUrl']) + data.staticmap = data.staticMap // deprecated + + for (const cares of whoCares) { + this.log.debug(`${logReference}: Creating raid alert for ${cares.id} ${cares.name} ${cares.type} ${cares.language} ${cares.template}`, cares) + + const rateLimitTtr = this.getRateLimitTimeToRelease(cares.id) + if (rateLimitTtr) { + this.log.verbose(`${logReference}: Not creating raid alert (Rate limit) for ${cares.type} ${cares.id} ${cares.name} Time to release: ${rateLimitTtr}`) + // eslint-disable-next-line no-continue + continue + } + this.log.verbose(`${logReference}: Creating raid alert for ${cares.type} ${cares.id} ${cares.name} ${cares.language} ${cares.template}`) + + const language = cares.language || this.config.general.locale + const translator = this.translatorFactory.Translator(language) + let [platform] = cares.type.split(':') + if (platform === 'webhook') platform = 'discord' + + data.name = translator.translate(data.nameEng) + data.formName = translator.translate(data.formNameEng) + data.evolutionName = translator.translate(data.evolutionNameEng) + data.megaName = data.evolution ? translator.translateFormat(this.GameData.utilData.megaName[data.evolution], data.name) : data.name + data.teamNameEng = data.team_id ? this.GameData.utilData.teams[data.team_id].name : 'Harmony' + data.teamName = translator.translate(data.teamNameEng) + data.teamEmoji = data.team_id !== undefined ? this.emojiLookup.lookup(this.GameData.utilData.teams[data.team_id].emoji, platform) : '' + data.quickMoveName = this.GameData.moves[data.move_1] ? translator.translate(this.GameData.moves[data.move_1].name) : '' + data.quickMoveEmoji = this.GameData.moves[data.move_1] && this.GameData.moves[data.move_1].type ? translator.translate(this.emojiLookup.lookup(this.GameData.utilData.types[this.GameData.moves[data.move_1].type].emoji, platform)) : '' + data.chargeMoveName = this.GameData.moves[data.move_2] ? translator.translate(this.GameData.moves[data.move_2].name) : '' + data.chargeMoveEmoji = this.GameData.moves[data.move_2] && this.GameData.moves[data.move_2].type ? translator.translate(this.emojiLookup.lookup(this.GameData.utilData.types[this.GameData.moves[data.move_2].type].emoji, platform)) : '' + data.gameWeatherName = data.weather ? translator.translate(data.gameWeatherNameEng) : '' + data.gameWeatherEmoji = data.weather ? translator.translate(this.emojiLookup.lookup(this.GameData.utilData.weather[data.weather].emoji, platform)) : '' + data.shinyPossibleEmoji = data.shinyPossible ? translator.translate(this.emojiLookup.lookup('shiny', platform)) : '' + data.generationName = translator.translate(data.generationNameEng) + + data.quickMove = data.quickMoveName // deprecated + data.chargeMove = data.chargeMoveName // deprecated + data.move1 = data.quickMoveName // deprecated + data.move2 = data.chargeMoveName // deprecated + data.move1emoji = data.quickMoveEmoji // deprecated + data.move2emoji = data.chargeMoveEmoji // deprecated + + const e = [] + const t = [] + const n = [] + monster.types.forEach((type) => { + e.push(this.emojiLookup.lookup(this.GameData.utilData.types[type.name].emoji, platform)) + t.push(type.id) + n.push(type.name) + }) + data.types = t + data.typeNameEng = n + data.emoji = e + + data.typeName = data.typeNameEng.map((type) => translator.translate(type)) + .join(', ') + data.typeEmoji = data.emoji.map((emoji) => translator.translate(emoji)) + .join('') + + data.boostingWeathers = data.types.map((type) => parseInt(Object.keys(this.GameData.utilData.weatherTypeBoost) + .find((key) => this.GameData.utilData.weatherTypeBoost[key].includes(type)), 10)) + data.boosted = !!data.boostingWeathers.includes(data.weather) + data.boostWeatherNameEng = data.boosted ? this.GameData.utilData.weather[data.weather].name : '' + data.boostWeatherId = data.boosted ? data.weather : '' + data.boostWeatherName = data.boosted ? translator.translate(this.GameData.utilData.weather[data.weather].name) : '' + data.boostWeatherEmoji = data.boosted ? translator.translate(this.emojiLookup.lookup(this.GameData.utilData.weather[data.weather].emoji, platform)) : '' + + const view = { + ...geoResult, + ...data, + pokemonName: data.pokemonName, + id: data.pokemon_id, + baseStats: monster.stats, + time: data.disappearTime, + tthh: data.tth.hours, + tthm: data.tth.minutes, + tths: data.tth.seconds, + confirmedTime: data.disappear_time_verified, + now: new Date(), + nowISO: new Date().toISOString(), + genderData: { + name: translator.translate(data.genderDataEng.name), + emoji: translator.translate(this.emojiLookup.lookup(data.genderDataEng.emoji, platform)), + }, + areas: data.matchedAreas.filter((area) => area.displayInMatches) + .map((area) => area.name) + .join(', '), + } + + const templateType = 'raid' + const message = await this.createMessage(logReference, templateType, platform, cares.template, language, cares.ping, view) + + const work = { + lat: data.latitude.toString() + .substring(0, 8), + lon: data.longitude.toString() + .substring(0, 8), + message, + target: cares.id, + type: cares.type, + name: cares.name, + tth: data.tth, + clean: cares.clean, + emoji: data.emoji, + logReference, + language, + } + jobs.push(work) + } + this.emit('postMessage', jobs) + } catch (e) { + this.log.error(`${data.gym_id}: Can't seem to handle raid (user cared): `, e, data) } - jobs.push(work) - } - return jobs + }) + return [] } data.tth = moment.preciseDiff(Date.now(), data.start * 1000, true) @@ -370,70 +391,78 @@ class Raid extends Controller { return [] } - data.imgUrl = await this.imgUicons.eggIcon(data.level) - data.stickerUrl = await this.stickerUicons.eggIcon(data.level) - // data.imgUrl = `${this.config.general.imgUrl}egg${data.level}.png` - // data.stickerUrl = `${this.config.general.stickerUrl}egg${data.level}.webp` - - const geoResult = await this.getAddress({ lat: data.latitude, lon: data.longitude }) - const jobs = [] - - await this.getStaticMapUrl(logReference, data, 'raid', ['latitude', 'longitude', 'level', 'imgUrl']) - data.staticmap = data.staticMap // deprecated - - for (const cares of whoCares) { - this.log.debug(`${logReference}: Creating egg alert for ${cares.id} ${cares.name} ${cares.type} ${cares.language} ${cares.template}`, cares) - const rateLimitTtr = this.getRateLimitTimeToRelease(cares.id) - if (rateLimitTtr) { - this.log.verbose(`${logReference}: Not creating egg alert (Rate limit) for ${cares.type} ${cares.id} ${cares.name} Time to release: ${rateLimitTtr}`) - // eslint-disable-next-line no-continue - continue - } - this.log.verbose(`${logReference}: Creating egg alert for ${cares.type} ${cares.id} ${cares.name} ${cares.language} ${cares.template}`) - - const language = cares.language || this.config.general.locale - // eslint-disable-next-line no-unused-vars - const translator = this.translatorFactory.Translator(language) - let [platform] = cares.type.split(':') - if (platform === 'webhook') platform = 'discord' - - data.teamNameEng = data.team_id ? this.GameData.utilData.teams[data.team_id].name : 'Harmony' - data.teamName = translator.translate(data.teamNameEng) - data.teamEmoji = data.team_id !== undefined ? this.emojiLookup.lookup(this.GameData.utilData.teams[data.team_id].emoji, platform) : '' - - const view = { - ...geoResult, - ...data, - id: data.pokemon_id, - time: data.hatchtime, - tthh: data.tth.hours, - tthm: data.tth.minutes, - tths: data.tth.seconds, - confirmedTime: data.disappear_time_verified, - now: new Date(), - nowISO: new Date().toISOString(), - areas: data.matchedAreas.filter((area) => area.displayInMatches).map((area) => area.name).join(', '), + setImmediate(async () => { + try { + data.imgUrl = await this.imgUicons.eggIcon(data.level) + data.stickerUrl = await this.stickerUicons.eggIcon(data.level) + // data.imgUrl = `${this.config.general.imgUrl}egg${data.level}.png` + // data.stickerUrl = `${this.config.general.stickerUrl}egg${data.level}.webp` + + const geoResult = await this.getAddress({ lat: data.latitude, lon: data.longitude }) + const jobs = [] + + await this.getStaticMapUrl(logReference, data, 'raid', ['latitude', 'longitude', 'level', 'imgUrl']) + data.staticmap = data.staticMap // deprecated + + for (const cares of whoCares) { + this.log.debug(`${logReference}: Creating egg alert for ${cares.id} ${cares.name} ${cares.type} ${cares.language} ${cares.template}`, cares) + const rateLimitTtr = this.getRateLimitTimeToRelease(cares.id) + if (rateLimitTtr) { + this.log.verbose(`${logReference}: Not creating egg alert (Rate limit) for ${cares.type} ${cares.id} ${cares.name} Time to release: ${rateLimitTtr}`) + // eslint-disable-next-line no-continue + continue + } + this.log.verbose(`${logReference}: Creating egg alert for ${cares.type} ${cares.id} ${cares.name} ${cares.language} ${cares.template}`) + + const language = cares.language || this.config.general.locale + // eslint-disable-next-line no-unused-vars + const translator = this.translatorFactory.Translator(language) + let [platform] = cares.type.split(':') + if (platform === 'webhook') platform = 'discord' + + data.teamNameEng = data.team_id ? this.GameData.utilData.teams[data.team_id].name : 'Harmony' + data.teamName = translator.translate(data.teamNameEng) + data.teamEmoji = data.team_id !== undefined ? this.emojiLookup.lookup(this.GameData.utilData.teams[data.team_id].emoji, platform) : '' + + const view = { + ...geoResult, + ...data, + id: data.pokemon_id, + time: data.hatchtime, + tthh: data.tth.hours, + tthm: data.tth.minutes, + tths: data.tth.seconds, + confirmedTime: data.disappear_time_verified, + now: new Date(), + nowISO: new Date().toISOString(), + areas: data.matchedAreas.filter((area) => area.displayInMatches).map((area) => area.name).join(', '), + } + + const templateType = 'egg' + const message = await this.createMessage(logReference, templateType, platform, cares.template, language, cares.ping, view) + + const work = { + lat: data.latitude.toString().substring(0, 8), + lon: data.longitude.toString().substring(0, 8), + message, + target: cares.id, + type: cares.type, + name: cares.name, + tth: data.tth, + clean: cares.clean, + emoji: data.emoji, + logReference, + language, + } + jobs.push(work) + } + this.emit('postMessage', jobs) + } catch (e) { + this.log.error(`${data.gym_id}: Can't seem to handle raid (user cared): `, e, data) } + }) - const templateType = 'egg' - const message = await this.createMessage(logReference, templateType, platform, cares.template, language, cares.ping, view) - - const work = { - lat: data.latitude.toString().substring(0, 8), - lon: data.longitude.toString().substring(0, 8), - message, - target: cares.id, - type: cares.type, - name: cares.name, - tth: data.tth, - clean: cares.clean, - emoji: data.emoji, - logReference, - language, - } - jobs.push(work) - } - return jobs + return [] } catch (e) { this.log.error(`${data.gym_id}: Can't seem to handle raid: `, e, data) } diff --git a/src/lib/db/migrations/v14_better2_monster_index.js b/src/lib/db/migrations/v14_better2_monster_index.js new file mode 100644 index 000000000..0fcc12530 --- /dev/null +++ b/src/lib/db/migrations/v14_better2_monster_index.js @@ -0,0 +1,15 @@ +const { log } = require('../../logger') + +exports.up = async function migrationUp(knex) { + // add monster index to speed up mon lookup + await knex.schema.alterTable('monsters', (table) => { + table.dropIndex(['pokemon_id', 'min_iv']) + table.index(['pvp_ranking_league', 'pokemon_id', 'min_iv']) + table.index(['pvp_ranking_league', 'pokemon_id', 'pvp_ranking_worst']) + }) + log.info('Better (2) monster index migration applied') +} + +exports.down = async function migrationDown(knex) { + log.info(knex) +}