Auraxium
Auraxium is an object-oriented, pure-Python wrapper for the PlanetSide 2 API.
It provides a simple object model that can be used by players and outfits without requiring deep knowledge of the API and its idiosyncrasies.
- Clean, Pythonic API
- Asynchronous endpoints to keep apps responsive during high API load
- Low-level interface for more optimised, custom queries
- Support for the real-time event streaming service (ESS)
- User-configurable caching system
- Fully type annotated
The documentation for this project is hosted at Read the Docs.
Table of Contents
Overview
The Census API used by PlanetSide 2 is powerful, but its design also carries a steep learning curve that makes a lot of basic API interactions rather tedious.
Auraxium streamlines this by hiding the game-agnostic queries behind an object model specific to PlanetSide 2. Whenever data is accessed that is not currently loaded, the library dynamically generates and performs the necessary queries in the background before resuming execution.
All queries that may incur network traffic and latency are asynchronous, which keeps multi-user applications - such as Discord bots - responsive.
Getting Started
All API interactions are performed through the auraxium.Client
object. It is the main endpoint used to interact with the API and contains a few essential references, like the current event loop, the connection pool, or the unique service ID used to identify your app.
Regarding service IDs: You can use the default value of s:example
for testing, but you may run into rate limiting issues if your app generates more than ~10 queries a minute.
You can apply for your custom service ID here; the process is free, and you usually hear back within a few hours.
Some of these references are also required for any queries carried out behind the scenes, so the client object is also handed around behind the scenes; be mindful when updating them as this may cause issues with ongoing background queries.
Boilerplate code
The aforementioned auraxium.Client
object must be closed using the auraxium.Client.close()
method before it is destroyed to avoid issues.
Alternatively, you can use the asynchronous context manager interface to automatically close it when leaving the block:
import auraxium
async with auraxium.Client() as client:
Since Auraxium is an asynchronous library, we also need to wrap our code in a coroutine to be able to use the async
keyword.
This gives us the following snippet:
import asyncio
import auraxium
async def main():
async with auraxium.Client() as client:
asyncio.run(main())
With that, the stage is set for some actual code.
Usage
The game-specific object representations for PlanetSide 2 can be found in the auraxium.ps2
sub module. Common ones include ps2.Character
, ps2.Outfit
, or ps2.Item
.
Note: The original data used to build a given object representation is always available via that object's .data
attribute, which will be a type-hinted, named tuple.
Retrieving Data
The auraxium.Client
class exposes several methods used to access the REST API data, like Client.get()
, used to return a single match, or Client.find()
, used to return a list of matching entries.
It also provides some utility methods, like Client.get_by_id()
and Client.get_by_name()
. They behave much like the more general Client.get()
but are cached to provide better performance for common lookups.
This means that repeatedly accessing an object through .get_by_id()
will only generate network traffic once, after which it is retrieved from cache (refer to the Caching section for more information).
Here is the above boilerplate code again, this time with a simple script that prints various character properties:
import asyncio
import auraxium
from auraxium import ps2
async def main():
async with auraxium.Client() as client:
char = await client.get_by_name(ps2.Character, 'auroram')
print(char.name)
print(char.data.prestige_level)
print(await char.faction())
print(await char.is_online())
asyncio.run(main())
Event Streaming
In addition to the REST interface wrapped by Auraxium's object model, PlanetSide 2 also exposes an event stream that can be used to react to in-game events in next to real time.
This can be used to track outfit member performance, implement your own stat tracker, or monitor server population.
The Auraxium client supports this endpoint through a trigger-action system.
Triggers
To receive data through the event stream, you must define a trigger. A trigger is made up of three things:
- One or more events that tells it to wake up
- Any number of conditions that decide whether to run or not
- An action that will be run if the conditions are met
Events
The event type definitions are available in the auraxium.event
namespace.
Conditions
Trigger conditions can be attached to a trigger to limit what events it will respond to, in addition to the event type.
This is useful if you have a commonly encountered event (like event.DEATH
) and would like your action to only run if the event data matches some other requirement (for example "the killing player must be part of my outfit").
Actions
The trigger's action is a method or function that will be run when the event fires and all conditions evaluate to True.
If the action is a coroutine according to inspect.iscoroutinefunction()
, it will be awaited.
The only argument passed to the function set as the trigger action is the event received:
async def example_action(event: Event) -> None:
"""Example function to showcase the signature used for actions.
Keep in mind that this could also be a regular function (i.e. one
defined without the "async" keyword).
"""
Registering Triggers
The easiest way to register a trigger to the client is via the auraxium.event.EventClient.trigger()
decorator. It takes the event/s to listen for as the arguments and creates a trigger using the decorated function as the trigger action.
Important: Keep in mind that the websocket connection will be continuously looping, waiting for new events to come in.
This means that using auraxium.event.EventClient()
as a context manager may cause issues since the context manager will close the connection when the context manager is exited.
import asyncio
from auraxium import event, ps2
async def main():
client = event.EventClient(service_id='s:example')
@client.trigger(event.BattleRankUp)
async def print_levelup(evt):
char = await client.get_by_id(ps2.Character, evt.character_id)
new_battle_rank = evt.battle_rank
print(f'{await char.name_long()} has reached BR {new_battle_rank}!')
loop = asyncio.new_event_loop()
loop.create_task(main())
loop.run_forever()
Technical Details
The following section contains more detailed implementation details for those who want to know; it is safe to ignore if you are only getting started.
Object Hierarchy
All classes in the Auraxium object model inherit from Ps2Object
. It defines the API table and ID field to use for generic queries and implements methods like .get()
or .find()
.
Cache Objects
Cached objects are based off the Cached
class, which introduces a class-specific cache for matching instances before falling back to the regular implementation.
It also adds methods for updating the class cache settings at runtime.
See the Caching section for details on the caching system.
Named Objects
Named objects are based off the Named
class and always cached. This base class guarantees a .name
] attribute and allows the use of the .get_by_name()
method, which is also cached.
This caching strategy is almost identical to the one used for IDs, except that it uses a string constructed of the lower-case name and locale identifier to store objects (e.g. 'en_sunderer'
).
Caching
Auraxium uses timed least-recently-used (TLRU) caches for its objects.
They have a size constraint (i.e. how many objects may be cached at any given time), as well as a maximum age per item (referred to as TTU, "time-to-use"). The TTU is used to ensure frequently used items are updated occasionally and not too far out of date.
When new items are added to the cache, it first removes any expired items (i.e. time_added - now > ttu
).
It then removes as many least-recently-used items as necessary to accommodate the new elements.
The LRU side of things is implemented via an collections.OrderedDict
; every time an item is retrieved from the cache (and is not expired), it is moved back to the start of the dictionary, the last items of the dictionary are then chopped off as needed.
Network Connections
For as long as it is active, the auraxium.Client
object will always have a aiohttp.ClientSession
running in case the REST API must be accessed.
The websocket connection, which is required for event streaming, is only active when there are triggers registered and active.
If the last trigger is removed, the websocket connection is quietly closed after a delay. If a new trigger is added, it will automatically be recreated in the background.
Object Model Alternatives
For some users or applications, Auraxium's object model may be a bad fit, like for highly nested, complex queries or for users that are already familiar with the Census API.
Here are a few Python alternatives for these cases:
-
The URL generator used by Auraxium to generate the queries for the object model can also be used on its own.
This still requires some understanding of the Census API data model but takes away the syntactic pitfalls involved.
It only generates queries, so you will have to pick your own flavour of HTTP library (like requests or aiohttp) to make the queries.
"""Usage example for the auraxium.census module."""
from auraxium import census
query = census.Query('character', service_id='s:example')
query.add_term('name.first_lower', 'auroram')
query.limit(20)
join = query.create_join('characters_online_status')
url = str(query.url())
print(url)
Refer to the census module documentation for details.
-
For an even simpler syntax, you can check out spascou/ps2-census, which was inspired by an earlier version of Auraxium.
It too sticks closely to the original Census API, but also provides methods for retrieving the queried data.
It also features a query factory system that allows creation of common queries from templates.
"""Usage example for spascou's ps2-census module."""
import ps2_census as ps2
query = ps2.Query(ps2.Collection.CHARACTER, service_id='s:example')
query.filter('name.first_lower', 'auroram')
query.limit(20)
query.join(ps2.Join(ps2.Collection.CHARACTERS_ONLINE_STATUS))
print(query.get())
Refer to the ps2-census documentation for details.
Contributing
If you have found a bug or would like to suggest a new feature or change, feel free to get in touch via the repository issues.
Please check out CONTRIBUTING.md before opening any pull requests for details.