[INVTYPE_HEAD], // head 2 => [INVTYPE_NECK], // neck 3 => [INVTYPE_SHOULDERS], // shoulder 4 => [INVTYPE_BODY], // shirt 5 => [INVTYPE_CHEST, INVTYPE_ROBE], // chest 6 => [INVTYPE_WAIST], // waist 7 => [INVTYPE_LEGS], // legs 8 => [INVTYPE_FEET], // feet 9 => [INVTYPE_WRISTS], // wrists 10 => [INVTYPE_HANDS], // hands 11 => [INVTYPE_FINGER], // finger1 12 => [INVTYPE_FINGER], // finger2 13 => [INVTYPE_TRINKET], // trinket1 14 => [INVTYPE_TRINKET], // trinket2 15 => [INVTYPE_CLOAK], // chest 16 => [INVTYPE_WEAPONMAINHAND, INVTYPE_WEAPON, INVTYPE_2HWEAPON], // mainhand 17 => [INVTYPE_WEAPONOFFHAND, INVTYPE_WEAPON, INVTYPE_HOLDABLE, INVTYPE_SHIELD], // offhand 18 => [INVTYPE_RANGED, INVTYPE_THROWN, INVTYPE_RELIC], // ranged + relic 19 => [INVTYPE_TABARD], // tabard ); public static $raidProgression = array( // statisticAchievement => relevantCriterium 1098 => 3271, // Onyxia's Lair 10 1756 => 13345, // Onyxia's Lair 25 4031 => 12230, 4034 => 12234, 4038 => 12238, 4042 => 12242, 4046 => 12246, // Trial of the Crusader 25 nh 4029 => 12231, 4035 => 12235, 4039 => 12239, 4043 => 12243, 4047 => 12247, // Trial of the Crusader 25 hc 4030 => 12229, 4033 => 12233, 4037 => 12237, 4041 => 12241, 4045 => 12245, // Trial of the Crusader 10 hc 4028 => 12228, 4032 => 12232, 4036 => 12236, 4040 => 12240, 4044 => 12244, // Trial of the Crusader 10 nh 4642 => 13091, 4656 => 13106, 4661 => 13111, 4664 => 13114, 4667 => 13117, 4670 => 13120, 4673 => 13123, 4676 => 13126, 4679 => 13129, 4682 => 13132, 4685 => 13135, 4688 => 13138, // Icecrown Citadel 25 hc 4641 => 13092, 4655 => 13105, 4660 => 13109, 4663 => 13112, 4666 => 13115, 4669 => 13118, 4672 => 13121, 4675 => 13124, 4678 => 13127, 4681 => 13130, 4683 => 13133, 4687 => 13136, // Icecrown Citadel 25 nh 4640 => 13090, 4654 => 13104, 4659 => 13110, 4662 => 13113, 4665 => 13116, 4668 => 13119, 4671 => 13122, 4674 => 13125, 4677 => 13128, 4680 => 13131, 4684 => 13134, 4686 => 13137, // Icecrown Citadel 10 hc 4639 => 13089, 4643 => 13093, 4644 => 13094, 4645 => 13095, 4646 => 13096, 4647 => 13097, 4648 => 13098, 4649 => 13099, 4650 => 13100, 4651 => 13101, 4652 => 13102, 4653 => 13103, // Icecrown Citadel 10 nh // 4823 => 13467, // Ruby Sanctum 25 hc // 4820 => 13465, // Ruby Sanctum 25 nh // 4822 => 13468, // Ruby Sanctum 10 hc // 4821 => 13466, // Ruby Sanctum 10 nh ); public static function getBuyoutForItem($itemId) { if (!$itemId) return 0; // try, when having filled char-DB at hand // return DB::Characters()->selectCell('SELECT SUM(a.buyoutprice) / SUM(ii.count) FROM auctionhouse a JOIN item_instance ii ON ii.guid = a.itemguid WHERE ii.itemEntry = ?d', $itemId); return 0; } public static function queueStart(&$msg = '') { $queuePID = self::queueStatus(); if ($queuePID) { $msg = 'queue already running'; return true; } if (OS_WIN) // here be gremlins! .. suggested was "start /B php prQueue" as background process. but that closes itself pclose(popen('start php prQueue --log=cache/profiling.log', 'r')); else exec('php prQueue --log=cache/profiling.log > /dev/null 2>/dev/null &'); usleep(500000); if (self::queueStatus()) return true; else { $msg = 'failed to start queue'; return false; } } public static function queueStatus() { if (!file_exists(self::PID_FILE)) return 0; $pid = file_get_contents(self::PID_FILE); $cmd = OS_WIN ? 'tasklist /NH /FO CSV /FI "PID eq %d"' : 'ps --no-headers p %d'; exec(sprintf($cmd, $pid), $out); if ($out && stripos($out[0], $pid) !== false) return $pid; // have pidFile but no process with this pid self::queueFree(); return 0; } public static function queueLock($pid) { $queuePID = self::queueStatus(); if ($queuePID && $queuePID != $pid) { trigger_error('pSync - another queue with PID #'.$queuePID.' is already running', E_USER_ERROR); CLI::write('Profiler::queueLock() - another queue with PID #'.$queuePID.' is already runnung', CLI::LOG_ERROR); return false; } // no queue running; create or overwrite pidFile $ok = false; if ($fh = fopen(self::PID_FILE, 'w')) { if (fwrite($fh, $pid)) $ok = true; fclose($fh); } return $ok; } public static function queueFree() { unlink(self::PID_FILE); } public static function urlize($str, $allowLocales = false, $profile = false) { $search = ['<', '>', ' / ', "'"]; $replace = ['<', '>', '-', '' ]; $str = str_replace($search, $replace, $str); if ($profile) { $str = str_replace(['(', ')'], ['', ''], $str); $accents = array( "ß" => "ss", "á" => "a", "ä" => "a", "à" => "a", "â" => "a", "è" => "e", "ê" => "e", "é" => "e", "ë" => "e", "í" => "i", "î" => "i", "ì" => "i", "ï" => "i", "ñ" => "n", "ò" => "o", "ó" => "o", "ö" => "o", "ô" => "o", "ú" => "u", "ü" => "u", "û" => "u", "ù" => "u", "œ" => "oe", "Á" => "A", "Ä" => "A", "À" => "A", "Â" => "A", "È" => "E", "Ê" => "E", "É" => "E", "Ë" => "E", "Í" => "I", "Î" => "I", "Ì" => "I", "Ï" => "I", "Ñ" => "N", "Ò" => "O", "Ó" => "O", "Ö" => "O", "Ô" => "O", "Ú" => "U", "Ü" => "U", "Û" => "U", "Ù" => "U", "Œ" => "Oe" ); $str = strtr($str, $accents); } $str = trim($str); if ($allowLocales) $str = str_replace(' ', '-', $str); else $str = preg_replace('/[^a-z0-9]/i', '-', $str); $str = str_replace('--', '-', $str); $str = str_replace('--', '-', $str); $str = rtrim($str, '-'); $str = strtolower($str); return $str; } public static function getRealms() { if (DB::isConnectable(DB_AUTH) && !self::$realms) { self::$realms = DB::Auth()->select('SELECT id AS ARRAY_KEY, name, IF(timezone IN (8, 9, 10, 11, 12), "eu", "us") AS region FROM realmlist WHERE allowedSecurityLevel = 0 AND gamebuild = ?d', WOW_BUILD); foreach (self::$realms as $rId => $rData) { if (DB::isConnectable(DB_CHARACTERS . $rId)) continue; // realm in db but no connection info set unset(self::$realms[$rId]); } } return self::$realms; } private static function queueInsert($realmId, $guid, $type, $localId) { if ($rData = DB::Aowow()->selectRow('SELECT requestTime AS time, status FROM ?_profiler_sync WHERE realm = ?d AND realmGUID = ?d AND `type` = ?d AND typeId = ?d AND status <> ?d', $realmId, $guid, $type, $localId, PR_QUEUE_STATUS_WORKING)) { // not on already scheduled - recalc time and set status to PR_QUEUE_STATUS_WAITING if ($rData['status'] != PR_QUEUE_STATUS_WAITING) { $newTime = CFG_DEBUG ? time() : max($rData['time'] + CFG_PROFILER_RESYNC_DELAY, time()); DB::Aowow()->query('UPDATE ?_profiler_sync SET requestTime = ?d, status = ?d, errorCode = 0 WHERE realm = ?d AND realmGUID = ?d AND `type` = ?d AND typeId = ?d', $newTime, PR_QUEUE_STATUS_WAITING, $realmId, $guid, $type, $localId); } } else DB::Aowow()->query('REPLACE INTO ?_profiler_sync (realm, realmGUID, `type`, typeId, requestTime, status, errorCode) VALUES (?d, ?d, ?d, ?d, UNIX_TIMESTAMP(), ?d, 0)', $realmId, $guid, $type, $localId, PR_QUEUE_STATUS_WAITING); } public static function scheduleResync($type, $realmId, $guid) { $newId = 0; switch ($type) { case TYPE_PROFILE: if ($newId = DB::Aowow()->selectCell('SELECT id FROM ?_profiler_profiles WHERE realm = ?d AND realmGUID = ?d', $realmId, $guid)) self::queueInsert($realmId, $guid, TYPE_PROFILE, $newId); break; case TYPE_GUILD: if ($newId = DB::Aowow()->selectCell('SELECT id FROM ?_profiler_guild WHERE realm = ?d AND realmGUID = ?d', $realmId, $guid)) self::queueInsert($realmId, $guid, TYPE_GUILD, $newId); break; case TYPE_ARENA_TEAM: if ($newId = DB::Aowow()->selectCell('SELECT id FROM ?_profiler_arena_team WHERE realm = ?d AND realmGUID = ?d', $realmId, $guid)) self::queueInsert($realmId, $guid, TYPE_ARENA_TEAM, $newId); break; default: trigger_error('scheduling resync for unknown type #'.$type.' omiting..', E_USER_WARNING); return 0; } if (!$newId) trigger_error('Profiler::scheduleResync() - tried to resync type #'.$type.' guid #'.$guid.' from realm #'.$realmId.' without preloaded data', E_USER_ERROR); else if (!self::queueStart($msg)) trigger_error('Profiler::scheduleResync() - '.$msg, E_USER_ERROR); return $newId; } public static function resyncStatus($type, array $subjectGUIDs) { $response = [CFG_PROFILER_QUEUE ? 2 : 0]; // in theory you could have multiple queues; used as divisor for: (15 / x) + 2 if (!$subjectGUIDs) $response[] = [PR_QUEUE_STATUS_ENDED, 0, 0, PR_QUEUE_ERROR_CHAR]; else { // error out all profiles with status WORKING, that are older than 60sec DB::Aowow()->query('UPDATE ?_profiler_sync SET status = ?d, errorCode = ?d WHERE status = ?d AND requestTime < ?d', PR_QUEUE_STATUS_ERROR, PR_QUEUE_ERROR_UNK, PR_QUEUE_STATUS_WORKING, time() - MINUTE); $subjectStatus = DB::Aowow()->select('SELECT typeId AS ARRAY_KEY, status, realm FROM ?_profiler_sync WHERE `type` = ?d AND typeId IN (?a)', $type, $subjectGUIDs); $queue = DB::Aowow()->selectCol('SELECT CONCAT(type, ":", typeId) FROM ?_profiler_sync WHERE status = ?d AND requestTime < UNIX_TIMESTAMP() ORDER BY requestTime ASC', PR_QUEUE_STATUS_WAITING); foreach ($subjectGUIDs as $guid) { if (empty($subjectStatus[$guid])) // whelp, thats some error.. $response[] = [PR_QUEUE_STATUS_ERROR, 0, 0, PR_QUEUE_ERROR_UNK]; else if ($subjectStatus[$guid]['status'] == PR_QUEUE_STATUS_ERROR) $response[] = [PR_QUEUE_STATUS_ERROR, 0, 0, $subjectStatus[$guid]['errCode']]; else $response[] = array( $subjectStatus[$guid]['status'], $subjectStatus[$guid]['status'] != PR_QUEUE_STATUS_READY ? CFG_PROFILER_RESYNC_PING : 0, array_search($type.':'.$guid, $queue) + 1, 0, 1 // nResycTries - unsure about this one ); } } return $response; } public static function getCharFromRealm($realmId, $charGuid) { $char = DB::Characters($realmId)->selectRow('SELECT c.* FROM characters c WHERE c.guid = ?d', $charGuid); if (!$char) return false; // reminder: this query should not fail: a placeholder entry is created as soon as a char listview is created or profile detail page is called $profileId = DB::Aowow()->selectCell('SELECT id FROM ?_profiler_profiles WHERE realm = ?d AND realmGUID = ?d', $realmId, $char['guid']); CLI::write('fetching char #'.$charGuid.' from realm #'.$realmId); CLI::write('writing...'); /*************/ /* equipment */ /*************/ /* enchantment-Indizes * 0: permEnchant * 3: tempEnchant * 6: gem1 * 9: gem2 * 12: gem3 * 15: socketBonus [not used] * 18: extraSocket [only check existance] * 21 - 30: randomProp enchantments */ DB::Aowow()->query('DELETE FROM ?_profiler_items WHERE id = ?d', $profileId); $items = DB::Characters($realmId)->select('SELECT ci.slot AS ARRAY_KEY, ii.itemEntry, ii.enchantments, ii.randomPropertyId FROM character_inventory ci JOIN item_instance ii ON ci.item = ii.guid WHERE ci.guid = ?d AND bag = 0 AND slot BETWEEN 0 AND 18', $char['guid']); $gemItems = []; $permEnch = []; $mhItem = 0; $ohItem = 0; foreach ($items as $slot => $item) { $ench = explode(' ', $item['enchantments']); $gEnch = []; foreach ([6, 9, 12] as $idx) if ($ench[$idx]) $gEnch[$idx] = $ench[$idx]; if ($gEnch) { $gi = DB::Aowow()->selectCol('SELECT gemEnchantmentId AS ARRAY_KEY, id FROM ?_items WHERE class = 3 AND gemEnchantmentId IN (?a)', $gEnch); foreach ($gEnch as $eId) { if (isset($gemItems[$eId])) $gemItems[$eId][1]++; else $gemItems[$eId] = [$gi[$eId], 1]; } } if ($slot + 1 == 16) $mhItem = $item['itemEntry']; if ($slot + 1 == 17) $ohItem = $item['itemEntry']; if ($ench[0]) $permEnch[$slot] = $ench[0]; $data = array( 'id' => $profileId, 'slot' => $slot + 1, 'item' => $item['itemEntry'], 'subItem' => $item['randomPropertyId'], 'permEnchant' => $ench[0], 'tempEnchant' => $ench[3], 'extraSocket' => (int)!!$ench[18], 'gem1' => isset($gemItems[$ench[6]]) ? $gemItems[$ench[6]][0] : 0, 'gem2' => isset($gemItems[$ench[9]]) ? $gemItems[$ench[9]][0] : 0, 'gem3' => isset($gemItems[$ench[12]]) ? $gemItems[$ench[12]][0] : 0, 'gem4' => 0 // serverside items cant have more than 3 sockets. (custom profile thing) ); DB::Aowow()->query('INSERT INTO ?_profiler_items (?#) VALUES (?a)', array_keys($data), array_values($data)); } CLI::write(' ..inventory'); /**************/ /* basic info */ /**************/ $data = array( 'realm' => $realmId, 'realmGUID' => $charGuid, 'name' => $char['name'], 'race' => $char['race'], 'class' => $char['class'], 'level' => $char['level'], 'gender' => $char['gender'], 'skincolor' => $char['playerBytes'] & 0xFF, 'facetype' => ($char['playerBytes'] >> 8) & 0xFF, // maybe features 'hairstyle' => ($char['playerBytes'] >> 16) & 0xFF, 'haircolor' => ($char['playerBytes'] >> 24) & 0xFF, 'features' => $char['playerBytes2'] & 0xFF, // maybe facetype 'title' => $char['chosenTitle'] ? DB::Aowow()->selectCell('SELECT id FROM ?_titles WHERE bitIdx = ?d', $char['chosenTitle']) : 0, 'playedtime' => $char['totaltime'], 'nomodelMask' => ($char['playerFlags'] & 0x400 ? (1 << SLOT_HEAD) : 0) | ($char['playerFlags'] & 0x800 ? (1 << SLOT_BACK) : 0), 'talenttree1' => 0, 'talenttree2' => 0, 'talenttree3' => 0, 'talentbuild1' => '', 'talentbuild2' => '', 'glyphs1' => '', 'glyphs2' => '', 'activespec' => $char['activespec'], 'guild' => null, 'guildRank' => null, 'gearscore' => 0, 'achievementpoints' => 0 ); /********************/ /* talents + glyphs */ /********************/ $t = DB::Characters($realmId)->selectCol('SELECT spec AS ARRAY_KEY, spell AS ARRAY_KEY2, spell FROM character_talent WHERE guid = ?d', $char['guid']); $g = DB::Characters($realmId)->select('SELECT spec AS ARRAY_KEY, glyph1 AS g1, glyph2 AS g4, glyph3 AS g5, glyph4 AS g2, glyph5 AS g3, glyph6 AS g6 FROM character_glyphs WHERE guid = ?d', $char['guid']); for ($i = 0; $i < 2; $i++) { // talents for ($j = 0; $j < 3; $j++) { $_ = DB::Aowow()->selectCol('SELECT spell AS ARRAY_KEY, MAX(IF(spell in (?a), rank, 0)) FROM ?_talents WHERE class = ?d AND tab = ?d GROUP BY id ORDER BY row, col ASC', !empty($t[$i]) ? $t[$i] : [0], $char['class'], $j); $data['talentbuild'.($i + 1)] .= implode('', $_); if ($char['activespec'] == $i) $data['talenttree'.($j + 1)] = array_sum($_); } // glyphs if (isset($g[$i])) { $gProps = []; for ($j = 1; $j <= 6; $j++) if ($g[$i]['g'.$j]) $gProps[$j] = $g[$i]['g'.$j]; if ($gProps) if ($gItems = DB::Aowow()->selectCol('SELECT i.id FROM ?_glyphproperties gp JOIN ?_spell s ON s.effect1MiscValue = gp.id AND s.effect1Id = 74 JOIN ?_items i ON i.class = 16 AND i.spellId1 = s.id WHERE gp.id IN (?a)', $gProps)) $data['glyphs'.($i + 1)] = implode(':', $gItems); } } $t = array( 'spent' => [$data['talenttree1'], $data['talenttree2'], $data['talenttree3']], 'spec' => 0 ); if ($t['spent'][0] > $t['spent'][1] && $t['spent'][0] > $t['spent'][2]) $t['spec'] = 1; else if ($t['spent'][1] > $t['spent'][0] && $t['spent'][1] > $t['spent'][2]) $t['spec'] = 2; else if ($t['spent'][2] > $t['spent'][1] && $t['spent'][2] > $t['spent'][0]) $t['spec'] = 3; // calc gearscore if ($items) $data['gearscore'] += (new ItemList(array(['id', array_column($items, 'itemEntry')])))->getScoreTotal($data['class'], $t, $mhItem, $ohItem); if ($gemItems) { $gemScores = new ItemList(array(['id', array_column($gemItems, 0)])); foreach ($gemItems as list($itemId, $mult)) if (isset($gemScores->json[$itemId]['gearscore'])) $data['gearscore'] += $gemScores->json[$itemId]['gearscore'] * $mult; } if ($permEnch) // fuck this shit .. we are guestimating this! { // enchantId => multiple spells => multiple items with varying itemlevels, quality, whatevs // cant reasonably get to the castItem from enchantId and slot $profSpec = DB::Aowow()->selectCol('SELECT id AS ARRAY_KEY, skillLevel AS "1", skillLine AS "0" FROM ?_itemenchantment WHERE id IN (?a)', $permEnch); foreach ($permEnch as $eId) { if ($x = Util::getEnchantmentScore(0, 0, !!$profSpec[$eId][1], $eId)) $data['gearscore'] += $x; else if ($profSpec[$eId][0] != 776) // not runeforging $data['gearscore'] += 17; // assume high quality enchantment for unknown cases } } $data['lastupdated'] = time(); CLI::write(' ..basic info'); /***************/ /* hunter pets */ /***************/ if ((1 << ($char['class'] - 1)) == CLASS_HUNTER) { DB::Aowow()->query('DELETE FROM ?_profiler_pets WHERE owner = ?d', $profileId); $pets = DB::Characters($realmId)->select('SELECT id AS ARRAY_KEY, id, entry, modelId, name FROM character_pet WHERE owner = ?d', $charGuid); foreach ($pets as $petGuid => $petData) { $morePet = DB::Aowow()->selectRow('SELECT p.`type`, c.family FROM ?_pet p JOIN ?_creature c ON c.family = p.id WHERE c.id = ?d', $petData['entry']); $petSpells = DB::Characters($realmId)->selectCol('SELECT spell FROM pet_spell WHERE guid = ?d', $petGuid); $_ = DB::Aowow()->selectCol('SELECT spell AS ARRAY_KEY, MAX(IF(spell in (?a), rank, 0)) FROM ?_talents WHERE class = 0 AND petTypeMask = ?d GROUP BY id ORDER BY row, col ASC', $petSpells ?: [0], 1 << $morePet['type']); $pet = array( 'id' => $petGuid, 'owner' => $profileId, 'name' => $petData['name'], 'family' => $morePet['family'], 'npc' => $petData['entry'], 'displayId' => $petData['modelId'], 'talents' => implode('', $_) ); DB::Aowow()->query('INSERT INTO ?_profiler_pets (?#) VALUES (?a)', array_keys($pet), array_values($pet)); } CLI::write(' ..hunter pets'); } /*******************/ /* completion data */ /*******************/ DB::Aowow()->query('DELETE FROM ?_profiler_completion WHERE id = ?d', $profileId); // done quests if ($quests = DB::Characters($realmId)->select('SELECT ?d AS id, ?d AS `type`, quest AS typeId FROM character_queststatus_rewarded WHERE guid = ?d', $profileId, TYPE_QUEST, $char['guid'])) foreach (Util::createSqlBatchInsert($quests) as $q) DB::Aowow()->query('INSERT INTO ?_profiler_completion (?#) VALUES '.$q, array_keys($quests[0])); CLI::write(' ..quests'); // known skills (professions only) $skAllowed = DB::Aowow()->selectCol('SELECT id FROM ?_skillline WHERE typeCat IN (9, 11) AND (cuFlags & ?d) = 0', CUSTOM_EXCLUDE_FOR_LISTVIEW); $skills = DB::Characters($realmId)->select('SELECT ?d AS id, ?d AS `type`, skill AS typeId, `value` AS cur, max FROM character_skills WHERE guid = ?d AND skill IN (?a)', $profileId, TYPE_SKILL, $char['guid'], $skAllowed); // manually apply racial profession bonuses foreach ($skills as &$sk) { // Blood Elves - Arcane Affinity if ($sk['typeId'] == 333 && $char['race'] == 10) { $sk['cur'] += 10; $sk['max'] += 10; } // Draenei - Gemcutting if ($sk['typeId'] == 755 && $char['race'] == 11) { $sk['cur'] += 5; $sk['max'] += 5; } // Tauren - Cultivation // Gnomes - Engineering Specialization if (($sk['typeId'] == 182 && $char['race'] == 6) || ($sk['typeId'] == 202 && $char['race'] == 7)) { $sk['cur'] += 15; $sk['max'] += 15; } } unset($sk); if ($skills) foreach (Util::createSqlBatchInsert($skills) as $sk) DB::Aowow()->query('INSERT INTO ?_profiler_completion (?#) VALUES '.$sk, array_keys($skills[0])); CLI::write(' ..professions'); // reputation if ($reputation = DB::Characters($realmId)->select('SELECT ?d AS id, ?d AS `type`, faction AS typeId, standing AS cur FROM character_reputation WHERE guid = ?d AND (flags & 0xC) = 0', $profileId, TYPE_FACTION, $char['guid'])) foreach (Util::createSqlBatchInsert($reputation) as $rep) DB::Aowow()->query('INSERT INTO ?_profiler_completion (?#) VALUES '.$rep, array_keys($reputation[0])); CLI::write(' ..reputation'); // known titles $tBlocks = explode(' ', $char['knownTitles']); $indizes = []; for ($i = 0; $i < 6; $i++) for ($j = 0; $j < 32; $j++) if ($tBlocks[$i] & (1 << $j)) $indizes[] = $j + ($i * 32); if ($indizes) DB::Aowow()->query('INSERT INTO ?_profiler_completion SELECT ?d, ?d, id, NULL, NULL FROM ?_titles WHERE bitIdx IN (?a)', $profileId, TYPE_TITLE, $indizes); CLI::write(' ..titles'); // achievements if ($achievements = DB::Characters($realmId)->select('SELECT ?d AS id, ?d AS `type`, achievement AS typeId, date AS cur FROM character_achievement WHERE guid = ?d', $profileId, TYPE_ACHIEVEMENT, $char['guid'])) { foreach (Util::createSqlBatchInsert($achievements) as $a) DB::Aowow()->query('INSERT INTO ?_profiler_completion (?#) VALUES '.$a, array_keys($achievements[0])); $data['achievementpoints'] = DB::Aowow()->selectCell('SELECT SUM(points) FROM ?_achievement WHERE id IN (?a)', array_column($achievements, 'typeId')); } CLI::write(' ..achievements'); // raid progression if ($progress = DB::Characters($realmId)->select('SELECT ?d AS id, ?d AS `type`, criteria AS typeId, date AS cur, counter AS `max` FROM character_achievement_progress WHERE guid = ?d AND criteria IN (?a)', $profileId, TYPE_ACHIEVEMENT, $char['guid'], self::$raidProgression)) { array_walk($progress, function (&$val) { $val['typeId'] = array_search($val['typeId'], self::$raidProgression); }); foreach (Util::createSqlBatchInsert($progress) as $p) DB::Aowow()->query('INSERT INTO ?_profiler_completion (?#) VALUES '.$p, array_keys($progress[0])); } CLI::write(' ..raid progression'); // known spells if ($spells = DB::Characters($realmId)->select('SELECT ?d AS id, ?d AS `type`, spell AS typeId FROM character_spell WHERE guid = ?d AND disabled = 0', $profileId, TYPE_SPELL, $char['guid'])) foreach (Util::createSqlBatchInsert($spells) as $s) DB::Aowow()->query('INSERT INTO ?_profiler_completion (?#) VALUES '.$s, array_keys($spells[0])); CLI::write(' ..known spells (vanity pets & mounts)'); /****************/ /* related data */ /****************/ // guilds if ($guild = DB::Characters($realmId)->selectRow('SELECT g.name AS name, g.guildid AS id, gm.rank FROM guild_member gm JOIN guild g ON g.guildid = gm.guildid WHERE gm.guid = ?d', $char['guid'])) { $guildId = 0; if (!($guildId = DB::Aowow()->selectCell('SELECT id FROM ?_profiler_guild WHERE realm = ?d AND realmGUID = ?d', $realmId, $guild['id']))) { $gData = array( // only most basic data 'realm' => $realmId, 'realmGUID' => $guild['id'], 'name' => $guild['name'], 'nameUrl' => self::urlize($guild['name']), 'cuFlags' => PROFILER_CU_NEEDS_RESYNC ); $guildId = DB::Aowow()->query('INSERT IGNORE INTO ?_profiler_guild (?#) VALUES (?a)', array_keys($gData), array_values($gData)); } $data['guild'] = $guildId; $data['guildRank'] = $guild['rank']; } // arena teams $teams = DB::Characters($realmId)->select('SELECT at.arenaTeamId AS ARRAY_KEY, at.name, at.type, IF(at.captainGuid = atm.guid, 1, 0) AS captain, atm.* FROM arena_team at JOIN arena_team_member atm ON atm.arenaTeamId = at.arenaTeamId WHERE atm.guid = ?d', $char['guid']); foreach ($teams as $rGuid => $t) { $teamId = 0; if (!($teamId = DB::Aowow()->selectCell('SELECT id FROM ?_profiler_arena_team WHERE realm = ?d AND realmGUID = ?d', $realmId, $rGuid))) { $team = array( // only most basic data 'realm' => $realmId, 'realmGUID' => $rGuid, 'name' => $t['name'], 'nameUrl' => self::urlize($t['name']), 'type' => $t['type'], 'cuFlags' => PROFILER_CU_NEEDS_RESYNC ); $teamId = DB::Aowow()->query('INSERT IGNORE INTO ?_profiler_arena_team (?#) VALUES (?a)', array_keys($team), array_values($team)); } $member = array( 'arenaTeamId' => $teamId, 'profileId' => $profileId, 'captain' => $t['captain'], 'weekGames' => $t['weekGames'], 'weekWins' => $t['weekWins'], 'seasonGames' => $t['seasonGames'], 'seasonWins' => $t['seasonWins'], 'personalRating' => $t['personalRating'] ); DB::Aowow()->query('INSERT INTO ?_profiler_arena_team_member (?#) VALUES (?a) ON DUPLICATE KEY UPDATE ?a', array_keys($member), array_values($member), array_slice($member, 2)); } CLI::write(' ..associated arena teams'); /*********************/ /* mark char as done */ /*********************/ if (DB::Aowow()->query('UPDATE ?_profiler_profiles SET ?a WHERE realm = ?d AND realmGUID = ?d', $data, $realmId, $charGuid) !== null) DB::Aowow()->query('UPDATE ?_profiler_profiles SET cuFlags = cuFlags & ?d WHERE id = ?d', ~PROFILER_CU_NEEDS_RESYNC, $profileId); return true; } public static function getGuildFromRealm($realmId, $guildGuid) { $guild = DB::Characters($realmId)->selectRow('SELECT guildId, name, createDate, info, backgroundColor, emblemStyle, emblemColor, borderStyle, borderColor FROM guild WHERE guildId = ?d', $guildGuid); if (!$guild) return false; // reminder: this query should not fail: a placeholder entry is created as soon as a team listview is created or team detail page is called $guildId = DB::Aowow()->selectCell('SELECT id FROM ?_profiler_guild WHERE realm = ?d AND realmGUID = ?d', $realmId, $guild['guildId']); CLI::write('fetching guild #'.$guildGuid.' from realm #'.$realmId); CLI::write('writing...'); /**************/ /* Guild Data */ /**************/ unset($guild['guildId']); $guild['nameUrl'] = self::urlize($guild['name']); DB::Aowow()->query('UPDATE ?_profiler_guild SET ?a WHERE realm = ?d AND realmGUID = ?d', $guild, $realmId, $guildGuid); // ranks DB::Aowow()->query('DELETE FROM ?_profiler_guild_rank WHERE guildId = ?d', $guildId); if ($ranks = DB::Characters($realmId)->select('SELECT ?d AS guildId, rid AS rank, rname AS name FROM guild_rank WHERE guildid = ?d', $guildId, $guildGuid)) foreach (Util::createSqlBatchInsert($ranks) as $r) DB::Aowow()->query('INSERT INTO ?_profiler_guild_rank (?#) VALUES '.$r, array_keys(reset($ranks))); CLI::write(' ..guild data'); /***************/ /* Member Data */ /***************/ $conditions = array( ['g.guildid', $guildGuid], ['deleteInfos_Account', null], ['level', MAX_LEVEL, '<='], // prevents JS errors [['extra_flags', self::CHAR_GMFLAGS, '&'], 0] // not a staff char ); // this here should all happen within ProfileList $members = new RemoteProfileList($conditions, ['sv' => $realmId]); if (!$members->error) $members->initializeLocalEntries(); else return false; CLI::write(' ..guild members'); /*********************/ /* mark guild as done */ /*********************/ DB::Aowow()->query('UPDATE ?_profiler_guild SET cuFlags = cuFlags & ?d WHERE id = ?d', ~PROFILER_CU_NEEDS_RESYNC, $guildId); return true; } public static function getArenaTeamFromRealm($realmId, $teamGuid) { $team = DB::Characters($realmId)->selectRow('SELECT arenaTeamId, name, type, captainGuid, rating, seasonGames, seasonWins, weekGames, weekWins, rank, backgroundColor, emblemStyle, emblemColor, borderStyle, borderColor FROM arena_team WHERE arenaTeamId = ?d', $teamGuid); if (!$team) return false; // reminder: this query should not fail: a placeholder entry is created as soon as a team listview is created or team detail page is called $teamId = DB::Aowow()->selectCell('SELECT id FROM ?_profiler_arena_team WHERE realm = ?d AND realmGUID = ?d', $realmId, $team['arenaTeamId']); CLI::write('fetching arena team #'.$teamGuid.' from realm #'.$realmId); CLI::write('writing...'); /*************/ /* Team Data */ /*************/ $captain = $team['captainGuid']; unset($team['captainGuid']); unset($team['arenaTeamId']); $team['nameUrl'] = self::urlize($team['name']); DB::Aowow()->query('UPDATE ?_profiler_arena_team SET ?a WHERE realm = ?d AND realmGUID = ?d', $team, $realmId, $teamGuid); CLI::write(' ..team data'); /***************/ /* Member Data */ /***************/ $members = DB::Characters($realmId)->select(' SELECT atm.guid AS ARRAY_KEY, atm.arenaTeamId, atm.weekGames, atm.weekWins, atm.seasonGames, atm.seasonWins, atm.personalrating FROM arena_team_member atm JOIN characters c ON c.guid = atm.guid AND c.deleteInfos_Account IS NULL AND c.level <= ?d AND (c.extra_flags & ?d) = 0 WHERE arenaTeamId = ?d', MAX_LEVEL, self::CHAR_GMFLAGS, $teamGuid ); $conditions = array( ['c.guid', array_keys($members)], ['deleteInfos_Account', null], ['level', MAX_LEVEL, '<='], // prevents JS errors [['extra_flags', self::CHAR_GMFLAGS, '&'], 0] // not a staff char ); $mProfiles = new RemoteProfileList($conditions, ['sv' => $realmId]); if (!$mProfiles->error) { $mProfiles->initializeLocalEntries(); foreach ($mProfiles->iterate() as $__) { $mGuid = $mProfiles->getField('guid'); $members[$mGuid]['arenaTeamId'] = $teamId; $members[$mGuid]['captain'] = (int)($mGuid == $captain); $members[$mGuid]['profileId'] = $mProfiles->getField('id'); } DB::Aowow()->query('DELETE FROM ?_profiler_arena_team_member WHERE arenaTeamId = ?d', $teamId); foreach (Util::createSqlBatchInsert($members) as $m) DB::Aowow()->query('INSERT INTO ?_profiler_arena_team_member (?#) VALUES '.$m, array_keys(reset($members))); } else return false; CLI::write(' ..team members'); /*********************/ /* mark team as done */ /*********************/ DB::Aowow()->query('UPDATE ?_profiler_arena_team SET cuFlags = cuFlags & ?d WHERE id = ?d', ~PROFILER_CU_NEEDS_RESYNC, $teamId); return true; } } ?>