[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 ; don't forget to enable this in /js/Profiler.js as well 1361 => 5100, 1362 => 5101, 1363 => 5102, 1365 => 5104, 1366 => 5108, 1364 => 5110, 1369 => 5112, 1370 => 5113, 1371 => 5114, 1372 => 5117, 1373 => 5119, 1374 => 5120, 1375 => 7805, 1376 => 5122, 1377 => 5123, // Naxxramas 10 1367 => 5103, 1368 => 5111, 1378 => 5124, 1379 => 5125, 1380 => 5126, 1381 => 5127, 1382 => 5128, 1383 => 7806, 1384 => 5130, 1385 => 5131, 1386 => 5132, 1387 => 5133, 1388 => 5134, 1389 => 5135, 1390 => 5136, // Naxxramas 25 2856 => 9938, 2857 => 9939, 2858 => 9940, 2859 => 9941, 2861 => 9943, 2865 => 9947, 2866 => 9948, 2868 => 9950, 2869 => 9951, 2870 => 9952, 2863 => 10558, 2864 => 10559, 2862 => 10560, 2867 => 10565, 2860 => 10580, // Ulduar 10 2872 => 9954, 2873 => 9955, 2874 => 9956, 2884 => 9957, 2875 => 9959, 2879 => 9963, 2880 => 9964, 2882 => 9966, 2883 => 9967, 3236 => 10542, 3257 => 10561, 3256 => 10562, 3258 => 10563, 2881 => 10566, 2885 => 10581, // Ulduar 25 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); 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`, CASE WHEN timezone IN (2, 3, 4) THEN "us" WHEN timezone IN (8, 9, 10, 11, 12) THEN "eu" WHEN timezone = 6 THEN "kr" WHEN timezone = 14 THEN "tw" WHEN timezone = 16 THEN "cn" END 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_ENABLE ? 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, errorCode 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]['errorCode']]; 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; if (!$char['name']) { trigger_error('char #'.$charGuid.' on realm #'.$realmId.' has empty name. skipping...', E_USER_WARNING); 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 $profile = DB::Aowow()->selectRow('SELECT id, lastupdated FROM ?_profiler_profiles WHERE realm = ?d AND realmGUID = ?d', $realmId, $char['guid']); if (!$profile) return false; // well ... it failed $profileId = $profile['id']; CLI::write('fetching char '.$char['name'].' (#'.$charGuid.') from realm #'.$realmId); if (!$char['online'] && $char['logout_time'] <= $profile['lastupdated']) { DB::Aowow()->query('UPDATE ?_profiler_profiles SET lastupdated = ?d WHERE id = ?d', time(), $profileId); CLI::write('char did not log in since last update. skipping...'); return true; } CLI::write('writing...'); $ra = (1 << ($char['race'] - 1)); $cl = (1 << ($char['class'] - 1)); /*************/ /* 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'], 'renameItr' => 0, 'race' => $char['race'], 'class' => $char['class'], 'level' => $char['level'], 'gender' => $char['gender'], 'skincolor' => $char['skin'], 'facetype' => $char['face'], // maybe features 'hairstyle' => $char['hairStyle'], 'haircolor' => $char['hairColor'], 'features' => $char['facialStyle'], // 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['activeTalentGroup'], 'guild' => null, 'guildRank' => null, 'gearscore' => 0, 'achievementpoints' => 0 ); // char is flagged for rename if ($char['at_login'] & 0x1) { $ri = DB::Aowow()->selectCell('SELECT MAX(renameItr) FROM ?_profiler_profiles WHERE realm = ?d AND realmGUID IS NOT NULL AND name = ?', $realmId, $char['name']); $data['renameItr'] = $ri ? ++$ri : 1; } /********************/ /* talents + glyphs */ /********************/ $t = DB::Characters($realmId)->selectCol('SELECT talentGroup AS ARRAY_KEY, spell AS ARRAY_KEY2, spell FROM character_talent WHERE guid = ?d', $char['guid']); $g = DB::Characters($realmId)->select('SELECT talentGroup 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 ($data['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 [$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 ($cl == 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); $racials = DB::Aowow()->select('SELECT `effect1MiscValue` AS ARRAY_KEY, `effect1DieSides` + `effect1BasePoints` AS qty, `reqRaceMask`, `reqClassMask` FROM ?_spell WHERE `typeCat` = -4 AND `effect1Id` = 6 AND `effect1AuraId` = 98'); // apply racial profession bonuses foreach ($skills as &$sk) { if (!isset($racials[$sk['typeId']])) continue; $r = $racials[$sk['typeId']]; if ((!$r['reqRaceMask'] || $r['reqRaceMask'] & (1 << ($char['race'] - 1))) && (!$r['reqClassMask'] || $r['reqClassMask'] & (1 << ($char['class'] - 1)))) { $sk['cur'] += $r['qty']; $sk['max'] += $r['qty']; } } unset($sk); if ($skills) { // apply auto-learned trade skills DB::Aowow()->query(' INSERT INTO ?_profiler_completion SELECT ?d, ?d, spellId, NULL, NULL FROM dbc_skilllineability WHERE skillLineId IN (?a) AND acquireMethod = 1 AND (reqRaceMask = 0 OR reqRaceMask & ?d) AND (reqClassMask = 0 OR reqClassMask & ?d)', $profileId, Type::SPELL, array_column($skills, 'typeId'), 1 << ($char['race'] - 1), 1 << ($char['class'] - 1) ); foreach (Util::createSqlBatchInsert($skills) as $sk) DB::Aowow()->query('INSERT INTO ?_profiler_completion (?#) VALUES '.$sk, array_keys($skills[0])); } CLI::write(' ..professions'); // reputation // get base values for this race/class $reputation = []; $baseRep = DB::Aowow()->selectCol(' SELECT id AS ARRAY_KEY, baseRepValue1 FROM aowow_factions WHERE baseRepValue1 && (baseRepRaceMask1 & ?d || (!baseRepRaceMask1 AND baseRepClassMask1)) && ((baseRepClassMask1 & ?d) || !baseRepClassMask1) UNION SELECT id AS ARRAY_KEY, baseRepValue2 FROM aowow_factions WHERE baseRepValue2 && (baseRepRaceMask2 & ?d || (!baseRepRaceMask2 AND baseRepClassMask2)) && ((baseRepClassMask2 & ?d) || !baseRepClassMask2) UNION SELECT id AS ARRAY_KEY, baseRepValue3 FROM aowow_factions WHERE baseRepValue3 && (baseRepRaceMask3 & ?d || (!baseRepRaceMask3 AND baseRepClassMask3)) && ((baseRepClassMask3 & ?d) || !baseRepClassMask3) UNION SELECT id AS ARRAY_KEY, baseRepValue4 FROM aowow_factions WHERE baseRepValue4 && (baseRepRaceMask4 & ?d || (!baseRepRaceMask4 AND baseRepClassMask4)) && ((baseRepClassMask4 & ?d) || !baseRepClassMask4) ', $ra, $cl, $ra, $cl, $ra, $cl, $ra, $cl); 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 & 0x4) = 0', $profileId, Type::FACTION, $char['guid'])) { // merge back base values for encountered factions foreach ($reputation as &$set) { if (empty($baseRep[$set['typeId']])) continue; $set['cur'] += $baseRep[$set['typeId']]; unset($baseRep[$set['typeId']]); } } // insert base values for not yet encountered factions foreach ($baseRep as $id => $val) $reputation[] = array( 'id' => $profileId, 'type' => Type::FACTION, 'typeId' => $id, 'cur' => $val ); 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) AND (flags & ?d) = 0', array_column($achievements, 'typeId'), ACHIEVEMENT_FLAG_COUNTER); } 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']; } CLI::write(' ..basic guild data'); // 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; if (!$guild['name']) { trigger_error('guild #'.$guildGuid.' on realm #'.$realmId.' has empty name. skipping...', E_USER_WARNING); 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; if (!$team['name']) { trigger_error('arena team #'.$teamGuid.' on realm #'.$realmId.' has empty name. skipping...', E_USER_WARNING); 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; } } ?>