Guess retriever

Retrieves guesses under a certain score threshold from your duels

您需要先安裝使用者腳本管理器擴展,如 TampermonkeyGreasemonkeyViolentmonkey 之後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyUserscripts 後才能安裝該腳本。

你需要先安裝一款使用者腳本管理器擴展,比如 Tampermonkey,才能安裝此腳本

您需要先安裝使用者腳本管理器擴充功能後才能安裝該腳本。

(我已經安裝了使用者腳本管理器,讓我安裝!)

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

(我已經安裝了使用者樣式管理器,讓我安裝!)

// ==UserScript==
// @name         Guess retriever
// @version      0.2
// @description  Retrieves guesses under a certain score threshold from your duels
// @author       You
// @license      MIT
// @match        https://www.geoguessr.com/*
// @icon         
// @grant        unsafeWindow
// @require      https://unpkg.com/@popperjs/[email protected]/dist/umd/popper.min.js
// @namespace    https://greasyfork.org/users/1011193
// @grant        GM_addStyle
// @grant        GM_setValue
// @grant        GM_getValue
// @require      https://greasyfork.org/scripts/460322-geoguessr-styles-scan/code/Geoguessr%20Styles%20Scan.js?version=1151668
// ==/UserScript==


let hide = false;
let styles = GM_getValue("guessFinderStyles");
if (!styles) {
    hide = true;
    styles = {};
}
let css =`
#guessFinderPopupWrapper, #guessFinderSearchWrapper, #guessFinderSlantedRoot, #guessFinderSlantedStart, #guessFinderInputWrapper, #guessFinderPopup, #guessFinderToggle, #guessFinderTogglePicture, #runButton, .buttonContainer {
    box-sizing: border-box;
  }

  #guessFinderPopup {
    background: rgba(26, 26, 46, 0.9);
    padding: 15px;
    border-radius: 10px;
    max-height: 80vh;
    overflow-y: auto;
    width: 28em;
  }

  #guessFinderPopup div {
    display: flex;
    justify-content: space-between;
    align-items: center;
    margin-top: 10px;
  }

  #guessFinderPopup input {
    background: rgba(255,255,255,0.1);
    color: white;
    border: none;
    border-radius: 5px;
  }

  .buttonContainer {
    display: flex;
    justify-content: center;
    align-items: center;
    margin-top: 20px;
  }

  #runButton {
    background: rgba(255, 255, 255, 0.1);
    color: white;
    border: none;
    border-radius: 5px;
    padding: 10px 20px;
  }

  #runButton:hover {
    background: rgba(255, 255, 255, 0.25);
    cursor: pointer;
  }

  #guessFinderToggle {
    width: 59.19px;
  }

  #guessFinderTogglePicture {
    justify-content: center;
  }

  #guessFinderTogglePicture img {
    width: 20px;
    filter: brightness(0) invert(1);
    opacity: 60%;
  }

.inputContainer {
  display: flex;
  justify-content: space-between;
  align-items: center;
  margin-top: 10px;
}

.inputLabel {
  margin: 0;
  padding-right: 6px;
  color: white; /* Adjust if necessary */
}

#dropdownInput {
  background: rgba(255,255,255,0.1);
  color: white;
  border: none;
  border-radius: 5px;
  padding: 5px; /* Adjust padding as needed */
}
  `

GM_addStyle(css);

const guiHTMLHeader = `
<div id="guessFinderPopupWrapper">
  <div id="guessFinderSearchWrapper">
    <div id="guessFinderSlantedRoot">
      <div id="guessFinderSlantedStart"></div>
      <div id="guessFinderInputWrapper">
        <div id="guessFinderPopup" style="background: rgba(26, 26, 46, 0.9); padding: 15px; border-radius: 10px; max-height: 80vh; overflow-y: auto; width: 28em">
          <div style="display: flex; justify-content: space-between; align-items: center; margin-top: 10px;">
            <span id="mmaAPIkey" style="margin: 0; padding-right: 6px;"> Map Making App API</span>
            <input id="mmaAPIkeyInput" name="mmaAPIkey" style="background: rgba(255,255,255,0.1); color: white; border: none; border-radius: 5px">
          </div>
          <div style="display: flex; justify-content: space-between; align-items: center; margin-top: 10px;">
            <span id="scoreThresholdLabel" style="margin: 0; padding-right: 6px;">Score Threshold</span>
            <input type="number" id="scoreThresholdInput" name="scoreThreshold" style="background: rgba(255,255,255,0.1); color: white; border: none; border-radius: 5px">
          </div>

          <div style="display: flex; justify-content: space-between; align-items: center; margin-top: 10px;">
            <span id="monthsOfDuelsLabel" style="margin: 0; padding-right: 6px;">Months of Duels</span>
            <input type="number" id="monthsOfDuelsInput" name="monthsOfDuels" style="background: rgba(255,255,255,0.1); color: white; border: none; border-radius: 5px">
          </div>
          <div class="inputContainer">
              <span class="inputLabel">Map Making App Map:</span>
              <select id="dropdownInput" name="dropdownInput">
                   <option value="option1">Option 1</option>
                   <option value="option2">Option 2</option>
                   <option value="option3">Option 3</option>
                   <option value="option4">Option 4</option>
              </select>
          </div>
          <div style="display: flex; justify-content: center; align-items: center; margin-top: 20px;">
            <button id="runButton" >Run</button>
          </div>
          <div style="display: flex; justify-content: center; align-items: center; margin-top: 20px;">
            <p id="guessFinderStatus"></p>
          </div>
        </div>
        <button style="width: 59.19px" id="guessFinderToggle"><picture id="guessFinderTogglePicture" style="justify-content: center"><img src="https://www.svgrepo.com/show/532540/location-pin-alt-1.svg" style="width: 20px; filter: brightness(0) invert(1); opacity: 60%;"></picture></button>
      </div>
      <div id="guessFinderSlantedEnd"></div>
    </div>
  </div>
</div>

`

const showPopup = (showButton, popup) => {
    popup.style.display = 'block';
    Popper.createPopper(showButton, popup, {
        placement: 'bottom',
        modifiers: [
            {
                name: 'offset',
                options: {
                    offset: [0, 10],
                },
            },
        ],
    });
}

const delay = ms => new Promise(resolve => setTimeout(resolve, ms));

const iterativeSetTimeout = async (func, initDelay, cond) => {
    while (!cond()) {
        await delay(initDelay);
        await func();
        initDelay *= 2;
    }
};


const stylesUsed = [
    "header_item__",
    "quick-search_wrapper__",
    "slanted-wrapper_root__",
    "slanted-wrapper_variantGrayTransparent__",
    "slanted-wrapper_start__",
    "quick-search_searchInputWrapper__",
    "slanted-wrapper_end__",
    "slanted-wrapper_right__",
    "quick-search_searchInputButton__",
    "quick-search_iconSection__",
];

const uploadDownloadStyles = async () => {
    stylesUsed.forEach(style => {
    });
    await iterativeSetTimeout(scanStyles, 0.1, () => checkAllStylesFound(stylesUsed) !== undefined);
    if (hide) {
        document.querySelector("#guessFinderPopupWrapper").hidden = "";
    }
    stylesUsed.forEach(style => {
        styles[style] = cn(style);
    });
    setStyles();
    GM_setValue("guessFinderStyles", styles);
}

const getStyle = style => {
    return styles[style];
}

const setStyles = () => {
    try {
        document.querySelector("#guessFinderPopupWrapper").className = getStyle("header_item__");
        document.querySelector("#guessFinderSearchWrapper").className = getStyle("quick-search_wrapper__");
        document.querySelector("#guessFinderSlantedRoot").className = getStyle("slanted-wrapper_root__") + " " + getStyle("slanted-wrapper_variantGrayTransparent__");
        document.querySelector("#guessFinderSlantedStart").className = getStyle("slanted-wrapper_start__")+ " " + getStyle("slanted-wrapper_right__");
        document.querySelector("#guessFinderInputWrapper").className = getStyle("quick-search_searchInputWrapper__");
        document.querySelector("#guessFinderSlantedEnd").className = getStyle("slanted-wrapper_end__")+ " " + getStyle("slanted-wrapper_right__");
        document.querySelector("#guessFinderToggle").className = getStyle("quick-search_searchInputButton__");
        document.querySelector("#guessFinderLabel1").className = getStyle("label_sizeXSmall__") + getStyle("label_variantWhite__");
        document.querySelector("#guessFinderLabel2").className = getStyle("label_sizeXSmall__") + getStyle("label_variantWhite__");
        document.querySelector("#guessFinderTogglePicture").className = getStyle("quick-search_iconSection__");
        document.querySelectorAll(".deleteButton").forEach(el => el.className = el.className + " " + getStyle("quick-search_searchInputButton__"));
    } catch (err) {
        console.error(err);
    }
}


const insertHeaderGui = async (header, gui) => {

    header.insertAdjacentHTML('afterbegin', gui);

    // Resolve class names
    if (hide) {
        document.querySelector("#guessFinderPopupWrapper").hidden = "true"
    }

    scanStyles().then(() => uploadDownloadStyles());
    setStyles();

    const showButton = document.querySelector('#guessFinderToggle');
    const popup = document.querySelector('#guessFinderPopup');
    popup.style.display = 'none';

    document.addEventListener('click', (e) => {
        const target = e.target;
        if (target == popup || popup.contains(target) || !document.contains(target)) return;
        if (target.matches('#guessFinderToggle, #guessFinderToggle *')) {
            e.preventDefault();
            showPopup(showButton, popup);
        } else {
            popup.style.display = 'none';
        }
    });
}

let MAP_MAKING_API_KEY = localStorage.getItem("guessFinderMMAApiKey");

async function mmaFetch(url, options = {}) {
	const response = await fetch(new URL(url, 'https://map-making.app'), {
		...options,
		headers: {
			accept: 'application/json',
			authorization: `API ${MAP_MAKING_API_KEY.trim()}`,
			...options.headers
		}
	});
	if (!response.ok) {
		let message = 'Unknown error';
		try {
			const res = await response.json();
			if (res.message) {
				message = res.message;
			}
		} catch {}
		alert(`An error occurred while trying to connect to Map Making App. ${message}`);
		throw Object.assign(new Error(message), { response });
	}
	return response;
}

async function getMaps(suppress = false) {
    if (MAP_MAKING_API_KEY === null) {
        if (!suppress) alert("Please input an API key for Map Making App");
        return;
    }
	const response = await mmaFetch(`/api/maps`);
	const maps = await response.json();
	return maps;
}

async function importLocations(mapId, locations) {
	const response = await mmaFetch(`/api/maps/${mapId}/locations`, {
		method: 'post',
		headers: {
			'content-type': 'application/json'
		},
		body: JSON.stringify({
			edits: [{
				action: { type: 4 },
				create: locations,
				remove: []
			}]
		})
	});
	await response.json();
}

const API_Key = 'bdc_e4a84278a5684f4786dd8277e4948ac4';

const ERROR_RESP = -1000000;

let count = 0;

async function getCountryCode(coords) {
    if (coords[0] <= -85.05) return 'AQ';
    count++
    if (API_Key.toLowerCase().match("^(bdc_)?[a-f0-9]{32}$") != null) {
        const api = "https://api.bigdatacloud.net/data/reverse-geocode?latitude="+coords.lat+"&longitude="+coords.lng+"&localityLanguage=en&key="+API_Key;
        return await fetch(api)
            .then(res => (res.status !== 200) ? ERROR_RESP : res.json())
            .then(out => (out === ERROR_RESP) ? ERROR_RESP : CountryDict[out.countryCode]);
    } else {
        const api = `https://nominatim.openstreetmap.org/reverse.php?lat=${coords.lat}&lon=${coords.lng}&zoom=21&format=jsonv2&accept-language=en`;
        return await fetch(api)
            .then(res => (res.status !== 200) ? ERROR_RESP : res.json())
            .then(out => (out === ERROR_RESP) ? ERROR_RESP : CountryDict[out?.address?.country_code?.toUpperCase()]);
    }
};

const fetchGeoGuessrData = async (id, cutoffDate) => {
    let after = null;
    let paginationToken = "", fetchedGames = [];
    let page = 0;
    let duelsFound = 0;
    while (true) {
        page++
        let url = "https://www.geoguessr.com/api/v4/feed/private";
        if (paginationToken !== "") {
            url += "?paginationToken=" + paginationToken;
        }
        let response = await fetch(url),
            n = await response.text(),
            jsonData = JSON.parse(n);
        if (jsonData.entries.length === 0) {
            console.log("All data fetched.");
            break;
        }
        const filteredEntries = jsonData.entries.filter(entry => {
            const entryTime = new Date(entry.time);
            const cutoffTime = new Date(cutoffDate);
            return entryTime >= cutoffTime;
        });
        if (filteredEntries.length === 0) {
            console.log("All data fetched.");
            break;
        }

        // Extract game IDs from filtered entries
        for (const entry of filteredEntries) {
            // Parse the payload as JSON
            const payloadData = JSON.parse(entry.payload);
            if (Array.isArray(payloadData)) {
                // If payloadData is an array, process each item
                for (const payloadItem of payloadData) {
                    if (payloadItem.payload && payloadItem.payload.gameId && payloadItem.payload.gameMode === "Duels") {
                        fetchedGames.push(payloadItem.payload.gameId);
                        duelsFound++;
                    }
                }
            } else {
                // If payloadData is an object, check and process directly
                if (payloadData.gameId && payloadData.gameMode === "Duels") {
                    fetchedGames.push(payloadData.gameId);
                }
            }
        }

        statusUpdater.updateStatus("fetchingDuels", duelsFound, new Date(filteredEntries[filteredEntries.length-1].time).toISOString());
        paginationToken = jsonData.paginationToken;
        await new Promise(resolve => {
            setTimeout(() => {
                resolve();
            }, 500);
        });
    }
    statusUpdater.updateStatus("filteringDuplicates");
    fetchedGames = fetchedGames.filter((game, index, array) => array.indexOf(game) === index); // removes duplicates
//    return fetchedGames.includes(after) ? fetchedGames.slice(0, fetchedGames.indexOf(after)) : fetchedGames;
    return fetchedGames;
};

const findIncorrectGuesses = async (gameData, playerId, status, score = null) => {
    let incorrectRounds = [];

    const rounds = gameData.rounds;

    // Finding the team and player by playerId
    let foundPlayer = null;
    let t = null;
    gameData.teams.forEach(team => {
        team.players.forEach(player => {
            if (player.playerId == playerId) {
                foundPlayer = player;
                t = team;
            }
        });
    });

    if (!foundPlayer) {
        return []; // Player not found, returning empty array
    }

     for (const round of t.roundResults) {
         let guessedCountryCode = null, actualCountryCode = null
         const roundLocation = rounds.find(r => r.roundNumber === round.roundNumber).panorama;
         status.guesses++;
         console.log(round);
         if (score === null) {
             return;
             guessedCountryCode = await getCountryCode({ lat: round.bestGuess.lat, lng: round.bestGuess.lng });
             actualCountryCode = await getCountryCode({ lat: roundLocation.lat, lng: roundLocation.lng });
             if (guessedCountryCode !== actualCountryCode) {
                 status.matching++;
                 incorrectRounds.push({
                     roundNumber: round.roundNumber,
                     score: round.score,
                     guessedCountryCode,
                     actualCountryCode,
                     panorama: roundLocation
                 });
             }
         } else if (round.score <= score) {
             status.matching++;
             incorrectRounds.push({
                 roundNumber: round.roundNumber,
                 score: round.score,
                 guessedCountryCode,
                 actualCountryCode,
                 panorama: roundLocation
             });
         }
         statusUpdater.updateStatus('findingGuesses', status.guesses, score === null ? 'wrongCountry' : "belowThreshold", status.matching);
    }
    return incorrectRounds;
};

const decodePanoId = (hexString) => {
    let decodedString = "";
    for (let i = 0; i < hexString.length; i += 2) {
        const hexPair = hexString.substr(i, 2);
        const char = String.fromCharCode(parseInt(hexPair, 16));
        decodedString += char;
    }
    return decodedString;
};

function extractDate(timestamp) {
    var match = timestamp.match(/^(\d{4}-\d{2})/);
    return match ? match[1] : null;
}

const formatScore = score => {
    const base = Math.floor(score / 1000);
    const decimal = Math.floor((score % 1000) / 100) * 0.1;
    const formattedScore = base + decimal;

    return formattedScore.toFixed(1) + 'k';
};

const subtractMonths = (months) => {
    if (months < 0) {
        throw new Error('Number of months must be positive');
    }

    const date = new Date();
    date.setMonth(date.getMonth() - months);

    return date.toISOString().split('T')[0]; // Returns the date in 'YYYY-MM-DD' format
};

function StatusUpdater(elementId) {
  this.element = document.getElementById(elementId);
  this.states = {
    fetchingDuels: (duelsCount, currentDate) =>
      `Fetching duel ids. <br/> Duels found: ${duelsCount} <br/> Checked through date: ${currentDate}`,
    filteringDuplicates: () => `Filtering out duplicate ids`,
    findingGuesses: (guessesCount, additionalState, additionalCount) => {
        let additionalText = '';
        if (additionalState === 'wrongCountry') {
            additionalText = ` Finding wrong country guesses. Guesses found: ${additionalCount}`;
        } else if (additionalState === 'belowThreshold') {
            additionalText = ` Finding guesses below score threshold. Guesses found: ${additionalCount}`;
        }
        return `Finding guesses. <br/> Guesses checked: ${guessesCount} <br/> ${additionalText}`;
    },
    pushingLocations: (mapName, locationsCount) =>
      `Pushing locations to ${mapName} <br/> locations pushed: ${locationsCount}`,
    done: () => `Done!`
  };
}

StatusUpdater.prototype.updateStatus = function(state, ...args) {
  if (this.states[state]) {
    this.element.innerHTML = this.states[state](...args);
  } else {
    console.error("Invalid state");
  }
};

let statusUpdater;

let userId = localStorage.getItem("guessFinderUserId")
if (userId == null) {
    fetch('https://geoguessr.com/api/v3/profiles', {method: "GET", "credentials": "include"})
        .then(response => response.json())
        .then(data => {
        if(data && data.user.id) {
            userId = data.user.id;
            localStorage.setItem("guessFinderUserId", userId);
        } else {
            console.log('ID not found in the response');
        }
    })
        .catch(error => console.error('Error:', error));
}

const run = async () => {
    MAP_MAKING_API_KEY = document.getElementById('mmaAPIkeyInput').value
    localStorage.setItem('guessFinderMMAApiKey', MAP_MAKING_API_KEY);
    const maps = await getMaps();
    const mapId = document.getElementById('dropdownInput').value;
    if (mapId == "") return;
    const months = document.getElementById("monthsOfDuelsInput").value;
    if (months <= 0) return;
    const scoreThreshold = document.getElementById("scoreThresholdInput").value;
    if (scoreThreshold == "" || scoreThreshold < 0) return;
    const map = maps.find(map => map.id == mapId);
    const duelIds = await fetchGeoGuessrData(userId, subtractMonths(months));
    let locsPushed = 0;
    let status = {matching: 0, guesses: 0}
    for (const id of duelIds) {
        let api_url = `https://game-server.geoguessr.com/api/duels/${id}`;
        let res = await fetch(api_url, {method: "GET", "credentials": "include"})
        let json = await res.json();
        const incorrectGuesses = await findIncorrectGuesses(json, userId, status, scoreThreshold);
        if (incorrectGuesses.length) {
//            console.log(`https://www.geoguessr.com/duels/${res}/summary`);
  //          console.log(incorrectGuesses);
            for (const guess of incorrectGuesses) {
                let loc = guess.panorama;
                if (loc.panoId) loc.panoId = decodePanoId(loc.panoId);
                let tags = [];
                if (guess.guessedCountryCode) tags.push(`guessed ${guess.guessedCountryCode}`)
                if (guess.actualCountryCode) tags.push(`actual ${guess.actualCountryCode}`)
                tags.push(`date: ${extractDate(json.rounds[0].startTime)}`)
                tags.push(`score: ${formatScore(guess.score)}`)
                await importLocations(map.id, [{
                    id: -1,
                    location: loc,
                    panoId: loc.panoId ?? null,
                    heading: loc.heading ?? 0,
                    pitch: loc.pitch ?? 0,
                    zoom: loc.zoom === 0 ? null : loc.zoom,
                    tags,
                    flags: loc.panoId ? 1 : 0
                }]);
                locsPushed++;
            }
        }
    }
    statusUpdater.updateStatus("done");
}

const populateMaps = maps => {
    const dropdown = document.querySelector("#dropdownInput");
    maps.forEach( item => {
        let option = document.createElement('option');
        option.value = item.id;
        option.textContent = item.name;
        dropdown.appendChild(option);
    });
};

const addPopup = async (refresh=false) => {
    if (refresh || (document.querySelector('[class^=header_header__]') && document.querySelector('#guessFinderPopupWrapper') === null)) {
        if (!refresh) {
            insertHeaderGui(document.querySelector('[class^=header_context__]'), guiHTMLHeader)
            const dropdown = document.querySelector("#dropdownInput");
            const maps = await getMaps(true);

            // Clear existing options
            dropdown.innerHTML = '';
            // Append new options
            let defaultOption = document.createElement('option');
            defaultOption.value = '';
            defaultOption.textContent = ''; // Empty text for default option
            dropdown.appendChild(defaultOption);

            if (maps) {
                populateMaps(maps);
            }

            let apiKey = document.getElementById("mmaAPIkeyInput");
            apiKey.value = MAP_MAKING_API_KEY;
//            console.log(MAP_MAKING_API_KEY);

            apiKey.addEventListener("input", async () => {
                MAP_MAKING_API_KEY = apiKey.value
                const maps = await getMaps();
                populateMaps(maps);
            });

            const runButton = document.getElementById('runButton');
            if (runButton) {
                runButton.addEventListener('click', run);
            }
            statusUpdater = new StatusUpdater("guessFinderStatus");
        }
    }
}

const updateImage = (refresh=false) => {
    // Don't do anything while the page is loading
    if (document.querySelector("[class^=page-loading_loading__]")) return;
    addPopup();
}


new MutationObserver(async (mutations) => {
    updateImage()
}).observe(document.body, { subtree: true, childList: true });

const CountryDict = {
    AF: 'AF',
    AX: 'FI', // Aland Islands
    AL: 'AL',
    DZ: 'DZ',
    AS: 'US', // American Samoa
    AD: 'AD',
    AO: 'AO',
    AI: 'GB', // Anguilla
    AQ: 'AQ', // Antarctica
    AG: 'AG',
    AR: 'AR',
    AM: 'AM',
    AW: 'NL', // Aruba
    AU: 'AU',
    AT: 'AT',
    AZ: 'AZ',
    BS: 'BS',
    BH: 'BH',
    BD: 'BD',
    BB: 'BB',
    BY: 'BY',
    BE: 'BE',
    BZ: 'BZ',
    BJ: 'BJ',
    BM: 'GB', // Bermuda
    BT: 'BT',
    BO: 'BO',
    BQ: 'NL', // Bonaire, Sint Eustatius, Saba
    BA: 'BA',
    BW: 'BW',
    BV: 'NO', // Bouvet Island
    BR: 'BR',
    IO: 'GB', // British Indian Ocean Territory
    BN: 'BN',
    BG: 'BG',
    BF: 'BF',
    BI: 'BI',
    KH: 'KH',
    CM: 'CM',
    CA: 'CA',
    CV: 'CV',
    KY: 'UK', // Cayman Islands
    CF: 'CF',
    TD: 'TD',
    CL: 'CL',
    CN: 'CN',
    CX: 'AU', // Christmas Islands
    CC: 'AU', // Cocos (Keeling) Islands
    CO: 'CO',
    KM: 'KM',
    CG: 'CG',
    CD: 'CD',
    CK: 'NZ', // Cook Islands
    CR: 'CR',
    CI: 'CI',
    HR: 'HR',
    CU: 'CU',
    CW: 'NL', // Curacao
    CY: 'CY',
    CZ: 'CZ',
    DK: 'DK',
    DJ: 'DJ',
    DM: 'DM',
    DO: 'DO',
    EC: 'EC',
    EG: 'EG',
    SV: 'SV',
    GQ: 'GQ',
    ER: 'ER',
    EE: 'EE',
    ET: 'ET',
    FK: 'GB', // Falkland Islands
    FO: 'DK', // Faroe Islands
    FJ: 'FJ',
    FI: 'FI',
    FR: 'FR',
    GF: 'FR', // French Guiana
    PF: 'FR', // French Polynesia
    TF: 'FR', // French Southern Territories
    GA: 'GA',
    GM: 'GM',
    GE: 'GE',
    DE: 'DE',
    GH: 'GH',
    GI: 'UK', // Gibraltar
    GR: 'GR',
    GL: 'DK', // Greenland
    GD: 'GD',
    GP: 'FR', // Guadeloupe
    GU: 'US', // Guam
    GT: 'GT',
    GG: 'GB', // Guernsey
    GN: 'GN',
    GW: 'GW',
    GY: 'GY',
    HT: 'HT',
    HM: 'AU', // Heard Island and McDonald Islands
    VA: 'VA',
    HN: 'HN',
    HK: 'CN', // Hong Kong
    HU: 'HU',
    IS: 'IS',
    IN: 'IN',
    ID: 'ID',
    IR: 'IR',
    IQ: 'IQ',
    IE: 'IE',
    IM: 'GB', // Isle of Man
    IL: 'IL',
    IT: 'IT',
    JM: 'JM',
    JP: 'JP',
    JE: 'GB', // Jersey
    JO: 'JO',
    KZ: 'KZ',
    KE: 'KE',
    KI: 'KI',
    KR: 'KR',
    KW: 'KW',
    KG: 'KG',
    LA: 'LA',
    LV: 'LV',
    LB: 'LB',
    LS: 'LS',
    LR: 'LR',
    LY: 'LY',
    LI: 'LI',
    LT: 'LT',
    LU: 'LU',
    MO: 'CN', // Macao
    MK: 'MK',
    MG: 'MG',
    MW: 'MW',
    MY: 'MY',
    MV: 'MV',
    ML: 'ML',
    MT: 'MT',
    MH: 'MH',
    MQ: 'FR', // Martinique
    MR: 'MR',
    MU: 'MU',
    YT: 'FR', // Mayotte
    MX: 'MX',
    FM: 'FM',
    MD: 'MD',
    MC: 'MC',
    MN: 'MN',
    ME: 'ME',
    MS: 'GB', // Montserrat
    MA: 'MA',
    MZ: 'MZ',
    MM: 'MM',
    NA: 'NA',
    NR: 'NR',
    NP: 'NP',
    NL: 'NL',
    AN: 'NL', // Netherlands Antilles
    NC: 'FR', // New Caledonia
    NZ: 'NZ',
    NI: 'NI',
    NE: 'NE',
    NG: 'NG',
    NU: 'NZ', // Niue
    NF: 'AU', // Norfolk Island
    MP: 'US', // Northern Mariana Islands
    NO: 'NO',
    OM: 'OM',
    PK: 'PK',
    PW: 'PW',
    PS: 'IL', // Palestine
    PA: 'PA',
    PG: 'PG',
    PY: 'PY',
    PE: 'PE',
    PH: 'PH',
    PN: 'GB', // Pitcairn
    PL: 'PL',
    PT: 'PT',
    PR: 'US', // Puerto Rico
    QA: 'QA',
    RE: 'FR', // Reunion
    RO: 'RO',
    RU: 'RU',
    RW: 'RW',
    BL: 'FR', // Saint Barthelemy
    SH: 'GB', // Saint Helena
    KN: 'KN',
    LC: 'LC',
    MF: 'FR', // Saint Martin
    PM: 'FR', // Saint Pierre and Miquelon
    VC: 'VC',
    WS: 'WS',
    SM: 'SM',
    ST: 'ST',
    SA: 'SA',
    SN: 'SN',
    RS: 'RS',
    SC: 'SC',
    SL: 'SL',
    SG: 'SG',
    SX: 'NL', // Sint Maarten
    SK: 'SK',
    SI: 'SI',
    SB: 'SB',
    SO: 'SO',
    ZA: 'ZA',
    GS: 'GB', // South Georgia and the South Sandwich Islands
    ES: 'ES',
    LK: 'LK',
    SD: 'SD',
    SR: 'SR',
    SJ: 'NO', // Svalbard and Jan Mayen
    SZ: 'SZ',
    SE: 'SE',
    CH: 'CH',
    SY: 'SY',
    TW: 'TW', // Taiwan
    TJ: 'TJ',
    TZ: 'TZ',
    TH: 'TH',
    TL: 'TL',
    TG: 'TG',
    TK: 'NZ', // Tokelau
    TO: 'TO',
    TT: 'TT',
    TN: 'TN',
    TR: 'TR',
    TM: 'TM',
    TC: 'GB', // Turcs and Caicos Islands
    TV: 'TV',
    UG: 'UG',
    UA: 'UA',
    AE: 'AE',
    GB: 'GB',
    US: 'US',
    UM: 'US', // US Minor Outlying Islands
    UY: 'UY',
    UZ: 'UZ',
    VU: 'VU',
    VE: 'VE',
    VN: 'VN',
    VG: 'GB', // British Virgin Islands
    VI: 'US', // US Virgin Islands
    WF: 'FR', // Wallis and Futuna
    EH: 'MA', // Western Sahara
    YE: 'YE',
    ZM: 'ZM',
    ZW: 'ZW'
};