2023-12-25 04:59:23 -05:00
__version__ = " 2.0.1 "
# Imports
import asyncio
import enum
import json
import uuid
import os
from abc import abstractmethod
from datetime import datetime
from urllib . parse import quote
import aiohttp
import requests
from aiohttp import ClientResponseError
# Enums
class platforms ( enum . Enum ) :
All = ' all '
Activision = ' acti '
Battlenet = ' battle '
PSN = ' psn '
Steam = ' steam '
Uno = ' uno '
XBOX = ' xbl '
class games ( enum . Enum ) :
ColdWar = ' cw '
ModernWarfare = ' mw '
ModernWarfare2 = ' mw2 '
Vanguard = ' vg '
Warzone = ' wz '
Warzone2 = ' wz2 '
class friendActions ( enum . Enum ) :
Invite = " invite "
Uninvite = " uninvite "
Remove = " remove "
Block = " block "
Unblock = " unblock "
class API :
"""
Call Of Duty API Wrapper
Developed by Todo Lodo & Engineer152
Contributors
- Werseter
Source Code : https : / / github . com / TodoLodo / cod - python - api
"""
def __init__ ( self ) :
# sub classes
self . Warzone = self . __WZ ( )
self . ModernWarfare = self . __MW ( )
self . Warzone2 = self . __WZ2 ( )
self . ModernWarfare2 = self . __MW2 ( )
self . ColdWar = self . __CW ( )
self . Vanguard = self . __VG ( )
self . Shop = self . __SHOP ( )
self . Me = self . __USER ( )
self . Misc = self . __ALT ( )
async def loginAsync ( self , sso_token : str ) - > None :
await API . _Common . loginAsync ( sso_token )
# Login
def login ( self , ssoToken : str ) :
API . _Common . login ( ssoToken )
class _Common :
requestHeaders = {
" content-type " : " application/json " ,
" user-agent " : " Mozilla/5.0 (Windows NT 10.0; Win64; x64) "
" AppleWebKit/537.36 (KHTML, like Gecko) "
" Chrome/74.0.3729.169 "
" Safari/537.36 " ,
" Accept " : " application/json " ,
" Connection " : " Keep-Alive "
}
cookies = { " new_SiteId " : " cod " , " ACT_SSO_LOCALE " : " en_US " , " country " : " US " ,
" ACT_SSO_COOKIE_EXPIRY " : " 1645556143194 " }
cachedMappings = None
fakeXSRF = str ( uuid . uuid4 ( ) )
2025-01-04 04:44:30 -05:00
baseUrl : str = " https://profile.callofduty.com/api/papi-client "
2023-12-25 04:59:23 -05:00
loggedIn : bool = False
# endPoints
# game platform lookupType gamertag type
fullDataUrl = " /stats/cod/v1/title/ %s /platform/ %s / %s / %s /profile/type/ %s "
# game platform lookupType gamertag type start end [?limit=n or '']
combatHistoryUrl = " /crm/cod/v2/title/ %s /platform/ %s / %s / %s /matches/ %s /start/ %d /end/ %d /details "
# game platform lookupType gamertag type start end
breakdownUrl = " /crm/cod/v2/title/ %s /platform/ %s / %s / %s /matches/ %s /start/ %d /end/ %d "
# game platform lookupType gamertag
seasonLootUrl = " /loot/title/ %s /platform/ %s / %s / %s /status/en "
# game platform
mapListUrl = " /ce/v1/title/ %s /platform/ %s /gameType/mp/communityMapData/availability "
# game platform type matchId
matchInfoUrl = " /crm/cod/v2/title/ %s /platform/ %s /fullMatch/ %s / %d /en "
@staticmethod
async def loginAsync ( sso_token : str ) - > None :
API . _Common . cookies [ " ACT_SSO_COOKIE " ] = sso_token
API . _Common . baseSsoToken = sso_token
r = await API . _Common . __Request ( f " { API . _Common . baseUrl } /crm/cod/v2/identities/ { sso_token } " )
if r [ ' status ' ] == ' success ' :
API . _Common . loggedIn = True
else :
raise InvalidToken ( sso_token )
@staticmethod
def login ( sso_token : str ) - > None :
API . _Common . cookies [ " ACT_SSO_COOKIE " ] = sso_token
API . _Common . baseSsoToken = sso_token
r = requests . get ( f " { API . _Common . baseUrl } /crm/cod/v2/identities/ { sso_token } " ,
headers = API . _Common . requestHeaders , cookies = API . _Common . cookies )
if r . json ( ) [ ' status ' ] == ' success ' :
API . _Common . loggedIn = True
API . _Common . cookies . update ( r . cookies )
else :
raise InvalidToken ( sso_token )
@staticmethod
def sso_token ( ) - > str :
return API . _Common . cookies [ " ACT_SSO_COOKIE " ]
# Requests
@staticmethod
async def __Request ( url ) :
async with aiohttp . client . ClientSession ( connector = aiohttp . TCPConnector ( verify_ssl = True ) ,
timeout = aiohttp . ClientTimeout ( total = 30 ) ) as session :
try :
async with session . get ( url , cookies = API . _Common . cookies ,
headers = API . _Common . requestHeaders ) as resp :
try :
resp . raise_for_status ( )
except ClientResponseError as err :
return { ' status ' : ' error ' , ' data ' : { ' type ' : type ( err ) , ' message ' : err . message } }
else :
API . _Common . cookies . update ( { c . key : c . value for c in session . cookie_jar } )
return await resp . json ( )
except asyncio . TimeoutError as err :
return { ' status ' : ' error ' , ' data ' : { ' type ' : type ( err ) , ' message ' : str ( err ) } }
async def __sendRequest ( self , url : str ) :
if self . loggedIn :
response = await API . _Common . __Request ( f " { self . baseUrl } { url } " )
if response [ ' status ' ] == ' success ' :
response [ ' data ' ] = await self . __perform_mapping ( response [ ' data ' ] )
return response
else :
raise NotLoggedIn
# client name url formatter
def __cleanClientName ( self , gamertag ) :
return quote ( gamertag . encode ( " utf-8 " ) )
# helper
def __helper ( self , platform , gamertag ) :
lookUpType = " gamer "
if platform == platforms . Uno :
lookUpType = " id "
if platform == platforms . Activision :
platform = platforms . Uno
if platform not in [ platforms . Activision , platforms . Battlenet , platforms . Uno , platforms . All , platforms . PSN ,
platforms . XBOX ] :
raise InvalidPlatform ( platform )
else :
gamertag = self . __cleanClientName ( gamertag )
return lookUpType , gamertag , platform
async def __get_mappings ( self ) :
if API . _Common . cachedMappings is None :
API . _Common . cachedMappings = (
await API . _Common . __Request ( ' https://engineer152.github.io/wz-data/weapon-ids.json ' ) ,
await API . _Common . __Request ( ' https://engineer152.github.io/wz-data/game-modes.json ' ) ,
await API . _Common . __Request ( ' https://engineer152.github.io/wz-data/perks.json ' ) )
return API . _Common . cachedMappings
# mapping
async def __perform_mapping ( self , data ) :
guns , modes , perks = await self . __get_mappings ( )
if not isinstance ( data , list ) or ' matches ' not in data :
return data
try :
for match in data [ ' matches ' ] :
# time stamps
try :
match [ ' utcStartDateTime ' ] = datetime . fromtimestamp (
match [ ' utcStartSeconds ' ] ) . strftime ( " % A, % B %d , % Y, % I: % M: % S " )
match [ ' utcEndDateTime ' ] = datetime . fromtimestamp (
match [ ' utcEndSeconds ' ] ) . strftime ( " % A, % B %d , % Y, % I: % M: % S " )
except KeyError :
pass
# loadouts list
for loadout in match [ ' player ' ] [ ' loadouts ' ] :
# weapons
if loadout [ ' primaryWeapon ' ] [ ' label ' ] is None :
try :
loadout [ ' primaryWeapon ' ] [ ' label ' ] = guns [ loadout [ ' primaryWeapon ' ] [ ' name ' ] ]
except KeyError :
pass
if loadout [ ' secondaryWeapon ' ] [ ' label ' ] is None :
try :
loadout [ ' secondaryWeapon ' ] [ ' label ' ] = guns [ loadout [ ' secondaryWeapon ' ] [ ' name ' ] ]
except KeyError :
pass
# perks list
for perk in loadout [ ' perks ' ] :
if perk [ ' label ' ] is None :
try :
perk [ ' label ' ] = perks [ perk [ ' name ' ] ]
except KeyError :
pass
# extra perks list
for perk in loadout [ ' extraPerks ' ] :
if perk [ ' label ' ] is None :
try :
perk [ ' label ' ] = perks [ perk [ ' name ' ] ]
except KeyError :
pass
# loadout list
for loadout in match [ ' player ' ] [ ' loadout ' ] :
if loadout [ ' primaryWeapon ' ] [ ' label ' ] is None :
try :
loadout [ ' primaryWeapon ' ] [ ' label ' ] = guns [ loadout [ ' primaryWeapon ' ] [ ' name ' ] ]
except KeyError :
pass
if loadout [ ' secondaryWeapon ' ] [ ' label ' ] is None :
try :
loadout [ ' secondaryWeapon ' ] [ ' label ' ] = guns [ loadout [ ' secondaryWeapon ' ] [ ' name ' ] ]
except KeyError :
pass
# perks list
for perk in loadout [ ' perks ' ] :
if perk [ ' label ' ] is None :
try :
perk [ ' label ' ] = perks [ perk [ ' name ' ] ]
except KeyError :
pass
# extra perks list
for perk in loadout [ ' extraPerks ' ] :
if perk [ ' label ' ] is None :
try :
perk [ ' label ' ] = perks [ perk [ ' name ' ] ]
except KeyError :
pass
except KeyError :
pass
# return mapped or unmapped data
return data
# API Requests
async def _fullDataReq ( self , game , platform , gamertag , type ) :
lookUpType , gamertag , platform = self . __helper ( platform , gamertag )
return await self . __sendRequest ( self . fullDataUrl % ( game , platform . value , lookUpType , gamertag , type ) )
async def _combatHistoryReq ( self , game , platform , gamertag , type , start , end ) :
lookUpType , gamertag , platform = self . __helper ( platform , gamertag )
return await self . __sendRequest (
self . combatHistoryUrl % ( game , platform . value , lookUpType , gamertag , type , start , end ) )
async def _breakdownReq ( self , game , platform , gamertag , type , start , end ) :
lookUpType , gamertag , platform = self . __helper ( platform , gamertag )
return await self . __sendRequest (
self . breakdownUrl % ( game , platform . value , lookUpType , gamertag , type , start , end ) )
async def _seasonLootReq ( self , game , platform , gamertag ) :
lookUpType , gamertag , platform = self . __helper ( platform , gamertag )
return await self . __sendRequest ( self . seasonLootUrl % ( game , platform . value , lookUpType , gamertag ) )
async def _mapListReq ( self , game , platform ) :
return await self . __sendRequest ( self . mapListUrl % ( game , platform . value ) )
async def _matchInfoReq ( self , game , platform , type , matchId ) :
return await self . __sendRequest ( self . matchInfoUrl % ( game , platform . value , type , matchId ) )
class __GameDataCommons ( _Common ) :
"""
Methods
== == == =
Sync
- - - -
fullData ( platform : platforms , gamertag : str )
returns player ' s game data of type dict
combatHistory ( platform : platforms , gamertag : str )
returns player ' s combat history of type dict
combatHistoryWithDate ( platform : platforms , gamertag : str , start : int , end : int )
returns player ' s combat history within the specified timeline of type dict
breakdown ( platform : platforms , gamertag : str )
returns player ' s combat history breakdown of type dict
breakdownWithDate ( platform : platforms , gamertag : str , start : int , end : int )
returns player ' s combat history breakdown within the specified timeline of type dict
seasonLoot ( platform : platforms , gamertag : str )
returns player ' s season loot
mapList ( platform : platforms )
returns available maps and available modes for each
matchInfo ( platform : platforms , matchId : int )
returns details match details of type dict
Async
- - - -
fullDataAsync ( platform : platforms , gamertag : str )
returns player ' s game data of type dict
combatHistoryAsync ( platform : platforms , gamertag : str )
returns player ' s combat history of type dict
combatHistoryWithDateAsync ( platform : platforms , gamertag : str , start : int , end : int )
returns player ' s combat history within the specified timeline of type dict
breakdownAsync ( platform : platforms , gamertag : str )
returns player ' s combat history breakdown of type dict
breakdownWithDateAsync ( platform : platforms , gamertag : str , start : int , end : int )
returns player ' s combat history breakdown within the specified timeline of type dict
seasonLootAsync ( platform : platforms , gamertag : str )
returns player ' s season loot
mapListAsync ( platform : platforms )
returns available maps and available modes for each
matchInfoAsync ( platform : platforms , matchId : int )
returns details match details of type dict
"""
def __init_subclass__ ( cls , * * kwargs ) :
cls . __doc__ = cls . __doc__ + super ( cls , cls ) . __doc__
@property
@abstractmethod
def _game ( self ) - > str :
raise NotImplementedError
@property
@abstractmethod
def _type ( self ) - > str :
raise NotImplementedError
async def fullDataAsync ( self , platform : platforms , gamertag : str ) :
data = await self . _fullDataReq ( self . _game , platform , gamertag , self . _type )
return data
def fullData ( self , platform : platforms , gamertag : str ) :
return asyncio . run ( self . fullDataAsync ( platform , gamertag ) )
async def combatHistoryAsync ( self , platform : platforms , gamertag : str ) :
data = await self . _combatHistoryReq ( self . _game , platform , gamertag , self . _type , 0 , 0 )
return data
def combatHistory ( self , platform : platforms , gamertag : str ) :
return asyncio . run ( self . combatHistoryAsync ( platform , gamertag ) )
async def combatHistoryWithDateAsync ( self , platform , gamertag : str , start : int , end : int ) :
data = await self . _combatHistoryReq ( self . _game , platform , gamertag , self . _type , start , end )
return data
def combatHistoryWithDate ( self , platform , gamertag : str , start : int , end : int ) :
return asyncio . run ( self . combatHistoryWithDateAsync ( platform , gamertag , start , end ) )
async def breakdownAsync ( self , platform , gamertag : str ) :
data = await self . _breakdownReq ( self . _game , platform , gamertag , self . _type , 0 , 0 )
return data
def breakdown ( self , platform , gamertag : str ) :
return asyncio . run ( self . breakdownAsync ( platform , gamertag ) )
async def breakdownWithDateAsync ( self , platform , gamertag : str , start : int , end : int ) :
data = await self . _breakdownReq ( self . _game , platform , gamertag , self . _type , start , end )
return data
def breakdownWithDate ( self , platform , gamertag : str , start : int , end : int ) :
return asyncio . run ( self . breakdownWithDateAsync ( platform , gamertag , start , end ) )
async def matchInfoAsync ( self , platform , matchId : int ) :
data = await self . _matchInfoReq ( self . _game , platform , self . _type , matchId )
return data
def matchInfo ( self , platform , matchId : int ) :
return asyncio . run ( self . matchInfoAsync ( platform , matchId ) )
async def seasonLootAsync ( self , platform , gamertag ) :
data = await self . _seasonLootReq ( self . _game , platform , gamertag )
return data
def seasonLoot ( self , platform , gamertag ) :
return asyncio . run ( self . seasonLootAsync ( platform , gamertag ) )
async def mapListAsync ( self , platform ) :
data = await self . _mapListReq ( self . _game , platform )
return data
def mapList ( self , platform ) :
return asyncio . run ( self . mapListAsync ( platform ) )
# WZ
class __WZ ( __GameDataCommons ) :
"""
Warzone class : A class to get players warzone stats , warzone combat history and specific warzone match details
classCategory : game
gameId / gameTitle : mw or wz
gameType : wz
"""
@property
def _game ( self ) - > str :
return " mw "
@property
def _type ( self ) - > str :
return " wz "
async def seasonLootAsync ( self , platform , gamertag ) :
raise InvalidEndpoint
async def mapListAsync ( self , platform ) :
raise InvalidEndpoint
# WZ2
class __WZ2 ( __GameDataCommons ) :
"""
Warzone 2 class : A class to get players warzone 2 stats , warzone 2 combat history and specific warzone 2 match details
classCategory : game
gameId / gameTitle : mw or wz
gameType : wz2
"""
@property
def _game ( self ) - > str :
return " mw2 "
@property
def _type ( self ) - > str :
return " wz2 "
async def seasonLootAsync ( self , platform , gamertag ) :
raise InvalidEndpoint
async def mapListAsync ( self , platform ) :
raise InvalidEndpoint
# MW
class __MW ( __GameDataCommons ) :
"""
ModernWarfare class : A class to get players modernwarfare stats , modernwarfare combat history , a player ' s modernwarfare season loot, modernwarfare map list and specific modernwarfare match details
classCategory : game
gameId / gameTitle : mw
gameType : mp
"""
@property
def _game ( self ) - > str :
return " mw "
@property
def _type ( self ) - > str :
return " mp "
# CW
class __CW ( __GameDataCommons ) :
"""
ColdWar class : A class to get players coldwar stats , coldwar combat history , a player ' s coldwar season loot, coldwar map list and specific coldwar match details
classCategory : game
gameId / gameTitle : cw
gameType : mp
"""
@property
def _game ( self ) - > str :
return " cw "
@property
def _type ( self ) - > str :
return " mp "
# VG
class __VG ( __GameDataCommons ) :
"""
Vanguard class : A class to get players vanguard stats , vanguard combat history , a player ' s vanguard season loot, vanguard map list and specific vanguard match details
classCategory : game
gameId / gameTitle : vg
gameType : pm
"""
@property
def _game ( self ) - > str :
return " vg "
@property
def _type ( self ) - > str :
return " mp "
# MW2
class __MW2 ( __GameDataCommons ) :
"""
ModernWarfare 2 class : A class to get players modernwarfare 2 stats , modernwarfare 2 combat history , a player ' s modernwarfare 2 season loot, modernwarfare 2 map list and specific modernwarfare 2 match details
classCategory : game
gameId / gameTitle : mw
gameType : mp
"""
@property
def _game ( self ) - > str :
return " mw2 "
@property
def _type ( self ) - > str :
return " mp "
# USER
class __USER ( _Common ) :
def info ( self ) :
if self . loggedIn :
# First, try to load the user info from a local file if it exists.
if os . path . exists ( ' userInfo.json ' ) :
with open ( ' userInfo.json ' , ' r ' ) as file :
rawData = json . load ( file )
else :
# If the file doesn't exist, make the HTTP request.
rawData = requests . get ( f " https://profile.callofduty.com/cod/userInfo/ { self . sso_token ( ) } " ,
headers = API . _Common . requestHeaders )
# Process the raw data (either from the file or HTTP response)
data = { ' userName ' : rawData [ ' userInfo ' ] [ ' userName ' ] , ' identities ' : [ ] }
for i in rawData [ ' identities ' ] :
data [ ' identities ' ] . append ( {
' platform ' : i [ ' provider ' ] ,
' gamertag ' : i [ ' username ' ] ,
' accountID ' : i [ ' accountID ' ]
} )
return data
else :
raise NotLoggedIn
def __priv ( self ) :
d = self . info ( )
return d [ ' identities ' ] [ 0 ] [ ' platform ' ] , quote ( d [ ' identities ' ] [ 0 ] [ ' gamertag ' ] . encode ( " utf-8 " ) )
async def friendFeedAsync ( self ) :
p , g = self . __priv ( )
data = await self . _Common__sendRequest (
f " /userfeed/v1/friendFeed/platform/ { p } /gamer/ { g } /friendFeedEvents/en " )
return data
def friendFeed ( self ) :
return asyncio . run ( self . friendFeedAsync ( ) )
async def eventFeedAsync ( self ) :
data = await self . _Common__sendRequest ( f " /userfeed/v1/friendFeed/rendered/en/ { self . sso_token ( ) } " )
return data
def eventFeed ( self ) :
return asyncio . run ( self . eventFeedAsync ( ) )
async def loggedInIdentitiesAsync ( self ) :
data = await self . _Common__sendRequest ( f " /crm/cod/v2/identities/ { self . sso_token ( ) } " )
return data
def loggedInIdentities ( self ) :
return asyncio . run ( self . loggedInIdentitiesAsync ( ) )
async def codPointsAsync ( self ) :
p , g = self . __priv ( )
data = await self . _Common__sendRequest ( f " /inventory/v1/title/mw/platform/ { p } /gamer/ { g } /currency " )
return data
def codPoints ( self ) :
return asyncio . run ( self . codPointsAsync ( ) )
async def connectedAccountsAsync ( self ) :
p , g = self . __priv ( )
data = await self . _Common__sendRequest ( f " /crm/cod/v2/accounts/platform/ { p } /gamer/ { g } " )
return data
def connectedAccounts ( self ) :
return asyncio . run ( self . connectedAccountsAsync ( ) )
async def settingsAsync ( self ) :
p , g = self . __priv ( )
data = await self . _Common__sendRequest ( f " /preferences/v1/platform/ { p } /gamer/ { g } /list " )
return data
def settings ( self ) :
return asyncio . run ( self . settingsAsync ( ) )
# SHOP
class __SHOP ( _Common ) :
"""
Shop class : A class to get bundle details and battle pass loot
classCategory : other
Methods
== == == =
Sync
- - - -
purchasableItems ( game : games )
returns purchasable items for a specific gameId / gameTitle
bundleInformation ( game : games , bundleId : int )
returns bundle details for the specific gameId / gameTitle and bundleId
battlePassLoot ( game : games , platform : platforms , season : int )
returns battle pass loot for specific game and season on given platform
Async
- - - -
purchasableItemsAsync ( game : games )
returns purchasable items for a specific gameId / gameTitle
bundleInformationAsync ( game : games , bundleId : int )
returns bundle details for the specific gameId / gameTitle and bundleId
battlePassLootAsync ( game : games , platform : platforms , season : int )
returns battle pass loot for specific game and season on given platform
"""
async def purchasableItemsAsync ( self , game : games ) :
data = await self . _Common__sendRequest ( f " /inventory/v1/title/ { game . value } /platform/uno/purchasable/public/en " )
return data
def purchasableItems ( self , game : games ) :
return asyncio . run ( self . purchasableItemsAsync ( game ) )
async def bundleInformationAsync ( self , game : games , bundleId : int ) :
data = await self . _Common__sendRequest ( f " /inventory/v1/title/ { game . value } /bundle/ { bundleId } /en " )
return data
def bundleInformation ( self , game : games , bundleId : int ) :
return asyncio . run ( self . bundleInformationAsync ( game , bundleId ) )
async def battlePassLootAsync ( self , game : games , platform : platforms , season : int ) :
data = await self . _Common__sendRequest (
f " /loot/title/ { game . value } /platform/ { platform . value } /list/loot_season_ { season } /en " )
return data
def battlePassLoot ( self , game : games , platform : platforms , season : int ) :
return asyncio . run ( self . battlePassLootAsync ( game , platform , season ) )
# ALT
class __ALT ( _Common ) :
async def searchAsync ( self , platform , gamertag : str ) :
lookUpType , gamertag , platform = self . _Common__helper ( platform , gamertag )
data = await self . _Common__sendRequest ( f " /crm/cod/v2/platform/ { platform . value } /username/ { gamertag } /search " )
return data
def search ( self , platform , gamertag : str ) :
return asyncio . run ( self . searchAsync ( platform , gamertag ) )
# Exceptions
class NotLoggedIn ( Exception ) :
def __str__ ( self ) :
return " Not logged in! "
class InvalidToken ( Exception ) :
def __init__ ( self , token ) :
self . token = token
def __str__ ( self ) :
return f " Token is invalid, token: { self . token } "
class InvalidPlatform ( Exception ) :
def __init__ ( self , platform : platforms ) :
self . message : str
if platform == platforms . Steam :
self . message = " Steam cannot be used till further updates. "
else :
self . message = " Invalid platform, use platform class! "
super ( ) . __init__ ( self . message )
def __str__ ( self ) :
return self . message
class InvalidEndpoint ( Exception ) :
def __str__ ( self ) :
return " This endpoint is not available for selected title "
class StatusError ( Exception ) :
def __str__ ( self ) :
return " Status Error, Check if your sso token is valid or try again later. "