2023-06-10 17:39:32 +00:00
|
|
|
import requests
|
2023-06-11 18:51:03 +00:00
|
|
|
from nullptr.models.system import System
|
|
|
|
from nullptr.models.waypoint import Waypoint
|
|
|
|
from nullptr.models.marketplace import Marketplace
|
2023-06-12 08:51:01 +00:00
|
|
|
from nullptr.models.jumpgate import Jumpgate
|
2023-06-15 19:10:33 +00:00
|
|
|
from nullptr.models.ship import Ship
|
2023-06-11 18:51:03 +00:00
|
|
|
from .util import *
|
2023-06-12 10:23:08 +00:00
|
|
|
from time import sleep
|
2023-06-10 17:39:32 +00:00
|
|
|
class ApiError(Exception):
|
|
|
|
def __init__(self, msg, code):
|
|
|
|
super().__init__(msg)
|
|
|
|
self.code = code
|
2023-06-12 10:23:08 +00:00
|
|
|
|
|
|
|
class ApiLimitError(Exception):
|
|
|
|
pass
|
|
|
|
|
2023-06-10 17:39:32 +00:00
|
|
|
class Api:
|
|
|
|
def __init__(self, store, agent):
|
|
|
|
self.agent = agent
|
|
|
|
self.store = store
|
2023-06-15 19:10:33 +00:00
|
|
|
self.requests_sent = 0
|
2023-06-11 18:51:03 +00:00
|
|
|
self.meta = None
|
2023-06-15 19:10:33 +00:00
|
|
|
self.last_result = None
|
2023-06-10 17:39:32 +00:00
|
|
|
self.root = 'https://api.spacetraders.io/v2/'
|
|
|
|
|
|
|
|
def token(self):
|
|
|
|
if self.agent.token is None:
|
|
|
|
raise ApiError('no token. Please register', 1337)
|
|
|
|
return self.agent.token
|
2023-06-12 10:23:08 +00:00
|
|
|
|
2023-06-10 18:49:50 +00:00
|
|
|
def request(self, method, path, data=None, need_token=True, params={}):
|
2023-06-12 10:23:08 +00:00
|
|
|
try:
|
|
|
|
return self.request_once(method, path, data, need_token, params)
|
2023-06-25 18:21:49 +00:00
|
|
|
except (ApiLimitError, requests.exceptions.Timeout):
|
2023-06-12 10:23:08 +00:00
|
|
|
print('oops, hit the limit. take a break')
|
|
|
|
sleep(10)
|
|
|
|
return self.request_once(method, path, data, need_token, params)
|
|
|
|
|
|
|
|
def request_once(self, method, path, data=None, need_token=True, params={}):
|
2023-06-10 17:39:32 +00:00
|
|
|
headers = {}
|
|
|
|
if need_token:
|
|
|
|
headers['Authorization'] = 'Bearer ' + self.token()
|
2023-06-15 19:10:33 +00:00
|
|
|
self.requests_sent += 1
|
2023-06-10 17:39:32 +00:00
|
|
|
if method == 'get':
|
|
|
|
params['limit'] = 20
|
|
|
|
r = requests.request(method, self.root+path, json=data, headers=headers, params=params)
|
2023-06-12 10:23:08 +00:00
|
|
|
if r.status_code == 429:
|
|
|
|
raise ApiLimitError()
|
2023-06-10 17:39:32 +00:00
|
|
|
result = r.json()
|
|
|
|
self.last_result = result
|
|
|
|
if result is None:
|
|
|
|
raise ApiError('http call failed', r.status_code)
|
2023-06-11 18:51:03 +00:00
|
|
|
if 'meta' in result:
|
|
|
|
self.last_meta = result['meta']
|
|
|
|
if 'data' not in result:
|
2023-06-10 17:39:32 +00:00
|
|
|
error_code = sg(result, 'error.code', -1)
|
|
|
|
self.last_error = error_code
|
|
|
|
error_message = sg(result, 'error.message')
|
|
|
|
raise ApiError(error_message, error_code)
|
|
|
|
else:
|
|
|
|
self.last_error = 0
|
|
|
|
return result['data']
|
|
|
|
|
|
|
|
def register(self, faction):
|
|
|
|
callsign = self.agent.symbol
|
|
|
|
data = {
|
|
|
|
'symbol': callsign,
|
|
|
|
'faction': faction
|
|
|
|
}
|
|
|
|
result = self.request('post', 'register', data, need_token=False)
|
|
|
|
token = mg(result, 'token')
|
|
|
|
self.agent.token = token
|
|
|
|
|
|
|
|
def info(self):
|
|
|
|
data = self.request('get', 'my/agent')
|
|
|
|
self.agent.update(data)
|
|
|
|
return self.agent
|
2023-06-10 18:49:50 +00:00
|
|
|
|
|
|
|
def list_systems(self, page=1):
|
|
|
|
data = self.request('get', 'systems', params={'page': page})
|
2023-06-11 18:51:03 +00:00
|
|
|
#pprint(self.last_meta)
|
2023-06-10 18:49:50 +00:00
|
|
|
return self.store.update_list(System, data)
|
2023-06-11 18:51:03 +00:00
|
|
|
|
|
|
|
def list_waypoints(self, system):
|
|
|
|
data = self.request('get', f'systems/{system}/waypoints/')
|
2023-06-12 20:36:58 +00:00
|
|
|
# pprint(data)
|
2023-06-11 18:51:03 +00:00
|
|
|
return self.store.update_list(Waypoint, data)
|
|
|
|
|
|
|
|
def marketplace(self, waypoint):
|
|
|
|
system = waypoint.system()
|
2023-06-12 08:51:01 +00:00
|
|
|
symbol = str(waypoint)
|
2023-06-11 18:51:03 +00:00
|
|
|
data = self.request('get', f'systems/{system}/waypoints/{waypoint}/market')
|
2023-06-16 11:03:19 +00:00
|
|
|
return self.store.update(Marketplace, data)
|
2023-06-12 08:51:01 +00:00
|
|
|
|
|
|
|
def jumps(self, waypoint):
|
|
|
|
data = self.request('get', f'systems/{waypoint.system()}/waypoints/{waypoint}/jump-gate')
|
|
|
|
symbol = str(waypoint)
|
2023-06-17 12:59:52 +00:00
|
|
|
return self.store.update(Jumpgate, data, symbol)
|
2023-06-12 08:51:01 +00:00
|
|
|
|
2023-06-15 19:10:33 +00:00
|
|
|
def list_ships(self):
|
|
|
|
data = self.request('get', 'my/ships')
|
|
|
|
return self.store.update_list(Ship, data)
|
|
|
|
|
2023-06-16 11:03:19 +00:00
|
|
|
def list_contracts(self):
|
|
|
|
data = self.request('get', 'my/contracts')
|
|
|
|
return self.store.update_list('Contract', data)
|
|
|
|
|
|
|
|
def negotiate(self, ship):
|
|
|
|
data = self.request('post', f'my/ships/{ship}/negotiate/contract')
|
|
|
|
if data is not None and 'contract' in data:
|
|
|
|
contract = self.store.update('Contract', data['contract'])
|
|
|
|
return contract
|
2023-06-17 12:59:52 +00:00
|
|
|
|
|
|
|
def deliver(self, ship, typ, contract):
|
|
|
|
units = ship.get_cargo(typ)
|
|
|
|
if units == 0:
|
|
|
|
print("Resource not in cargo")
|
|
|
|
return {}
|
|
|
|
data = {
|
|
|
|
'shipSymbol': str(ship),
|
|
|
|
'tradeSymbol': typ.upper(),
|
|
|
|
'units': units
|
|
|
|
}
|
2023-06-18 18:47:34 +00:00
|
|
|
data = self.request('post', f'my/contracts/{contract.symbol.lower()}/deliver', data)
|
2023-06-17 12:59:52 +00:00
|
|
|
if 'cargo' in data:
|
|
|
|
ship.update(data)
|
|
|
|
if 'contract' in data:
|
|
|
|
contract.update(data['contract'])
|
|
|
|
return contract
|
|
|
|
|
|
|
|
def fulfill(self, contract):
|
2023-06-18 18:47:34 +00:00
|
|
|
data = self.request('post', f'my/contracts/{contract.symbol.lower()}/fulfill')
|
2023-06-17 12:59:52 +00:00
|
|
|
if 'contract' in data:
|
|
|
|
contract.update(data['contract'])
|
|
|
|
if 'agent' in data:
|
|
|
|
self.agent.update(data['agent'])
|
|
|
|
return contract
|
2023-06-16 11:03:19 +00:00
|
|
|
|
2023-06-15 20:37:33 +00:00
|
|
|
def navigate(self, ship, wp):
|
|
|
|
data = {'waypointSymbol': str(wp)}
|
|
|
|
response = self.request('post', f'my/ships/{ship}/navigate', data)
|
|
|
|
ship.update(response)
|
2023-06-12 08:51:01 +00:00
|
|
|
|
2023-06-15 20:37:33 +00:00
|
|
|
def dock(self, ship):
|
|
|
|
data = self.request('post', f'my/ships/{ship}/dock')
|
|
|
|
ship.update(data)
|
|
|
|
return data
|
|
|
|
|
|
|
|
def orbit(self, ship):
|
|
|
|
data = self.request('post', f'my/ships/{ship}/orbit')
|
|
|
|
ship.update(data)
|
|
|
|
return data
|
|
|
|
|
|
|
|
def refuel(self, ship):
|
|
|
|
data = self.request('post', f'my/ships/{ship}/refuel')
|
2023-06-16 12:41:11 +00:00
|
|
|
if 'fuel' in data:
|
|
|
|
ship.update(data)
|
|
|
|
if 'agent' in data:
|
|
|
|
self.agent.update(data['agent'])
|
2023-06-15 20:37:33 +00:00
|
|
|
return data
|
2023-06-16 12:41:11 +00:00
|
|
|
|
|
|
|
def accept_contract(self, contract):
|
|
|
|
data = self.request('post', f'my/contracts/{contract.symbol.lower()}/accept')
|
|
|
|
if 'contract' in data:
|
|
|
|
contract.update(data['contract'])
|
|
|
|
if 'agent' in data:
|
|
|
|
self.agent.update(data['agent'])
|
|
|
|
return contract
|
2023-06-16 21:05:47 +00:00
|
|
|
|
|
|
|
def sell(self, ship, typ):
|
|
|
|
units = ship.get_cargo(typ)
|
|
|
|
data = {
|
|
|
|
'symbol': typ,
|
|
|
|
'units': units
|
|
|
|
}
|
|
|
|
data = self.request('post', f'my/ships/{ship}/sell', data)
|
|
|
|
if 'cargo' in data:
|
|
|
|
ship.update(data)
|
|
|
|
if 'agent' in data:
|
|
|
|
self.agent.update(data['agent'])
|
|
|
|
return data
|
|
|
|
|
|
|
|
def buy(self, ship, typ, amt):
|
|
|
|
data = {
|
|
|
|
'symbol': typ,
|
|
|
|
'units': amt
|
|
|
|
}
|
|
|
|
data = self.request('post', f'my/ships/{ship}/purchase', data)
|
|
|
|
if 'cargo' in data:
|
|
|
|
ship.update(data)
|
|
|
|
if 'agent' in data:
|
|
|
|
self.agent.update(data['agent'])
|
|
|
|
return data
|
|
|
|
|
|
|
|
def jettison(self, ship, typ):
|
|
|
|
units = ship.get_cargo(typ)
|
|
|
|
if units == 0:
|
|
|
|
print('cargo not found')
|
|
|
|
return
|
|
|
|
data = {
|
|
|
|
'symbol': typ,
|
|
|
|
'units': units
|
|
|
|
}
|
|
|
|
data = self.request('post', f'my/ships/{ship.symbol}/jettison', data)
|
|
|
|
if 'cargo' in data:
|
|
|
|
ship.update(data)
|
|
|
|
if 'agent' in data:
|
|
|
|
self.agent.update(data['agent'])
|
|
|
|
return data
|
2023-06-17 12:59:52 +00:00
|
|
|
|
|
|
|
def purchase(self, typ, wp):
|
|
|
|
data = {
|
|
|
|
'shipType': typ,
|
|
|
|
'waypointSymbol': str(wp)
|
|
|
|
}
|
|
|
|
data = self.request('post', 'my/ships', data)
|
|
|
|
if 'agent' in data:
|
|
|
|
self.agent.update(data['agent'])
|
|
|
|
if 'ship' in data:
|
|
|
|
ship = self.store.update('Ship', data['ship'])
|
|
|
|
return ship
|
|
|
|
|
|
|
|
def jump(self, ship, system):
|
2023-06-20 19:46:05 +00:00
|
|
|
if type(system) == System:
|
|
|
|
system = system.symbol
|
2023-06-17 12:59:52 +00:00
|
|
|
data = {
|
2023-06-20 19:46:05 +00:00
|
|
|
"systemSymbol": system
|
2023-06-17 12:59:52 +00:00
|
|
|
}
|
|
|
|
data = self.request('post', f'my/ships/{ship}/jump', data)
|
|
|
|
if 'nav' in data:
|
|
|
|
ship.update(data)
|
|
|
|
return ship
|
|
|
|
|
|
|
|
def shipyard(self, wp):
|
|
|
|
return self.request('get', f'systems/{wp.system()}/waypoints/{wp}/shipyard')
|
2023-06-17 18:18:14 +00:00
|
|
|
|
|
|
|
def extract(self, ship, survey=None):
|
|
|
|
data = {}
|
|
|
|
if survey is not None:
|
|
|
|
data['survey'] = survey.api_dict()
|
|
|
|
try:
|
|
|
|
data = self.request('post', f'my/ships/{ship}/extract', data=data)
|
|
|
|
except ApiError as e:
|
|
|
|
if e.code in [ 4221, 4224]:
|
|
|
|
survey.exhausted = True
|
|
|
|
else:
|
|
|
|
raise e
|
|
|
|
ship.update(data)
|
2023-06-17 21:07:53 +00:00
|
|
|
return data
|
2023-06-17 18:18:14 +00:00
|
|
|
|
|
|
|
def survey(self, ship):
|
|
|
|
data = self.request('post', f'my/ships/{ship}/survey')
|
|
|
|
ship.update(data)
|
|
|
|
result = self.store.update_list('Survey', mg(data, 'surveys'))
|
|
|
|
return result
|
2023-06-17 21:07:53 +00:00
|
|
|
|