mirror of
https://silica.codes/BedrockReverse/PyFab.git
synced 2025-04-06 05:05:43 +12:00
Upload
This commit is contained in:
parent
76acbb87ad
commit
7072ca4634
2 changed files with 205 additions and 0 deletions
169
PlayFab.py
Normal file
169
PlayFab.py
Normal file
|
@ -0,0 +1,169 @@
|
|||
import requests
|
||||
import os
|
||||
import json
|
||||
import binascii
|
||||
import base64
|
||||
import struct
|
||||
import hashlib
|
||||
import datetime
|
||||
|
||||
from cryptography.hazmat.primitives.asymmetric import rsa
|
||||
from cryptography.hazmat.primitives.asymmetric import padding
|
||||
|
||||
# Production PlayFab Environment
|
||||
TITLE_ID = "20CA2"
|
||||
TITLE_SHARED_SECRET = "S8RS53ZEIGMYTYG856U3U19AORWXQXF41J7FT3X9YCWAC7I35X"
|
||||
|
||||
# Internal PlayFab Environment
|
||||
#TITLE_ID = "E9D1"
|
||||
#TITLE_SHARED_SECRET = "RX4U3XF1KAG8W6WOOTUKO4BY1P4Z48UMHWEND7KSJHDIKCIWD6"
|
||||
|
||||
PLAYFAB_HEADERS = {
|
||||
"User-Agent": "libhttpclient/1.0.0.0",
|
||||
"Content-Type": "application/json",
|
||||
"Accept-Language": "en-US"
|
||||
}
|
||||
|
||||
PLAYFAB_SESSION = requests.Session()
|
||||
PLAYFAB_SESSION.headers.update(PLAYFAB_HEADERS)
|
||||
|
||||
PLAYFAB_DOMAIN = "https://" + TITLE_ID.lower() + ".playfabapi.com"
|
||||
|
||||
SETTING_FILE = "settings.json"
|
||||
PLAYFAB_SETTINGS = {}
|
||||
|
||||
def sendPlayFabRequest(endpoint, data, hdrs={}):
|
||||
rsp = PLAYFAB_SESSION.post(PLAYFAB_DOMAIN+endpoint, json=data, headers=hdrs).json()
|
||||
if rsp['code'] != 200:
|
||||
print(rsp)
|
||||
else:
|
||||
return rsp['data']
|
||||
|
||||
def genCustomId():
|
||||
return "MCPF"+binascii.hexlify(os.urandom(16)).decode("UTF-8").upper()
|
||||
|
||||
def genPlayerSecret():
|
||||
return base64.b64encode(os.urandom(32)).decode("UTF-8")
|
||||
|
||||
def getMojangCsp():
|
||||
return base64.b64decode(sendPlayFabRequest("/Client/GetTitlePublicKey", {
|
||||
"TitleId":TITLE_ID,
|
||||
"TitleSharedSecret": TITLE_SHARED_SECRET
|
||||
})['RSAPublicKey'])
|
||||
|
||||
def importCspKey(csp):
|
||||
e = struct.unpack("I", csp[0x10:0x14])[0]
|
||||
n = bytearray(csp[0x14:])
|
||||
n.reverse()
|
||||
n = int(binascii.hexlify(n), 16)
|
||||
return rsa.RSAPublicNumbers(e, n).public_key()
|
||||
|
||||
|
||||
|
||||
def genPlayFabTimestamp():
|
||||
return datetime.datetime.now().isoformat()+"Z"
|
||||
|
||||
def genPlayFabSignature(requestBody, timestamp):
|
||||
sha256 = hashlib.sha256()
|
||||
sha256.update(requestBody.encode("UTF-8") + b"." + timestamp.encode("UTF-8") + b"." + configGet("PLAYER_SECRET").encode("UTF-8"))
|
||||
return base64.b64encode(sha256.digest())
|
||||
|
||||
def configGet(key):
|
||||
global PLAYFAB_SETTINGS
|
||||
if key in PLAYFAB_SETTINGS:
|
||||
return PLAYFAB_SETTINGS[key]
|
||||
return None
|
||||
|
||||
|
||||
def configSet(key, newValue):
|
||||
global PLAYFAB_SETTINGS
|
||||
if os.path.exists(SETTING_FILE):
|
||||
PLAYFAB_SETTINGS = json.loads(open(SETTING_FILE, "r").read())
|
||||
|
||||
PLAYFAB_SETTINGS[key] = newValue
|
||||
open(SETTING_FILE, "w").write(json.dumps(PLAYFAB_SETTINGS))
|
||||
return newValue
|
||||
|
||||
def LoginWithCustomId():
|
||||
global TITLE_ID
|
||||
|
||||
customId = configGet("CUSTOM_ID")
|
||||
playerSecret = configGet("PLAYER_SECRET")
|
||||
createNewAccount = False
|
||||
|
||||
if customId == None:
|
||||
customId = genCustomId()
|
||||
createNewAccount = True
|
||||
|
||||
if playerSecret == None:
|
||||
playerSecret = genPlayerSecret()
|
||||
createNewAccount = True
|
||||
|
||||
configSet("CUSTOM_ID", customId)
|
||||
configSet("PLAYER_SECRET", playerSecret)
|
||||
|
||||
payload = {
|
||||
"CreateAccount" : None,
|
||||
"CustomId": None,
|
||||
"EncryptedRequest" : None,
|
||||
"InfoRequestParameters" : {
|
||||
"GetCharacterInventories" : False,
|
||||
"GetCharacterList" : False,
|
||||
"GetPlayerProfile" : True,
|
||||
"GetPlayerStatistics" : False,
|
||||
"GetTitleData" : False,
|
||||
"GetUserAccountInfo" : True,
|
||||
"GetUserData" : False,
|
||||
"GetUserInventory" : False,
|
||||
"GetUserReadOnlyData" : False,
|
||||
"GetUserVirtualCurrency" : False,
|
||||
"PlayerStatisticNames" : None,
|
||||
"ProfileConstraints" : None,
|
||||
"TitleDataKeys" : None,
|
||||
"UserDataKeys" : None,
|
||||
"UserReadOnlyDataKeys" : None
|
||||
},
|
||||
"PlayerSecret" : None,
|
||||
"TitleId" : TITLE_ID
|
||||
}
|
||||
|
||||
req = None
|
||||
if createNewAccount:
|
||||
toEnc = json.dumps({"CustomId":customId, "PlayerSecret": playerSecret}).encode("UTF-8")
|
||||
pubkey = importCspKey(getMojangCsp())
|
||||
|
||||
payload["CreateAccount"] = True
|
||||
payload["EncryptedRequest"] = base64.b64encode(pubkey.encrypt(toEnc, padding.PKCS1v15())).decode("UTF-8")
|
||||
|
||||
req = sendPlayFabRequest("/Client/LoginWithCustomID", payload)
|
||||
else:
|
||||
payload["CustomId"] = customId
|
||||
ts = genPlayFabTimestamp()
|
||||
sig = genPlayFabSignature(json.dumps(payload), ts)
|
||||
req = sendPlayFabRequest("/Client/LoginWithCustomID", payload, {"X-PlayFab-Signature": sig, "X-PlayFab-Timestamp": ts})
|
||||
entitytoken = req["EntityToken"]["EntityToken"]
|
||||
PLAYFAB_SESSION.headers.update({"X-EntityToken": entitytoken})
|
||||
return req
|
||||
|
||||
def GetEntityToken(playfabId, accType):
|
||||
req = sendPlayFabRequest("/Authentication/GetEntityToken", {
|
||||
"Entity" : {
|
||||
"Id" : playfabId,
|
||||
"Type" : accType
|
||||
}
|
||||
})
|
||||
entitytoken = req["EntityToken"]
|
||||
PLAYFAB_SESSION.headers.update({"X-EntityToken": entitytoken})
|
||||
return req
|
||||
|
||||
def Search(query, sfilter, orderBy, select, top, skip):
|
||||
return sendPlayFabRequest("/Catalog/Search", {
|
||||
"count": True,
|
||||
"query": query,
|
||||
"filter": sfilter,
|
||||
"orderBy": orderBy,
|
||||
"scid": "4fc10100-5f7a-4470-899b-280835760c07",
|
||||
"select": select,
|
||||
"top": top,
|
||||
"skip": skip
|
||||
})
|
36
Scraper.py
Normal file
36
Scraper.py
Normal file
|
@ -0,0 +1,36 @@
|
|||
import PlayFab
|
||||
import json
|
||||
|
||||
# Login and Switch to Master Player Account
|
||||
PlayFab.GetEntityToken(PlayFab.LoginWithCustomId()['PlayFabId'], 'master_player_account')
|
||||
|
||||
MAX_SEARCH = 300
|
||||
|
||||
# Get total number of items in marketplace
|
||||
totalItems = PlayFab.Search("", "", "creationDate ASC", None, 1, 0)["Count"]
|
||||
|
||||
# Initalize some variables
|
||||
searchFilter = ""
|
||||
skip = 0
|
||||
resultsDict = {}
|
||||
|
||||
print("Total items in marketplace: "+str(totalItems))
|
||||
|
||||
# Search loop
|
||||
while len(resultsDict) < totalItems:
|
||||
searchResults = PlayFab.Search("", searchFilter, "creationDate ASC", "contents,images,title,description,keywords", MAX_SEARCH, skip)
|
||||
|
||||
for item in searchResults["Items"]:
|
||||
resultsDict[item["Id"]] = item
|
||||
|
||||
print(str(len(resultsDict))+ "/" + str(totalItems));
|
||||
|
||||
|
||||
skip += MAX_SEARCH
|
||||
|
||||
if skip > 10000:
|
||||
searchFilter = "(CreationDate ge " + searchResults["Items"][-1]["CreationDate"] +")";
|
||||
skip = 0
|
||||
|
||||
open("playfab-catalog.json", "wb").write(json.dumps(resultsDict, indent=4).encode("UTF-8"))
|
||||
print("Bubbles")
|
Loading…
Add table
Reference in a new issue