Security News
Fluent Assertions Faces Backlash After Abandoning Open Source Licensing
Fluent Assertions is facing backlash after dropping the Apache license for a commercial model, leaving users blindsided and questioning contributor rights.
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.
The documentation for this project is hosted at Read the Docs.
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.
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.
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:
# Your code here
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:
# Your code here
asyncio.run(main())
With that, the stage is set for some actual code.
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.
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)
# NOTE: Any methods that might incur network traffic are asynchronous.
# If the data type has been cached locally, no network communication
# is required.
# This will only generate a request once per faction, as the faction
# data type is cached forever by default.
print(await char.faction())
# The online status is never cached as it is bound to change at any
# moment.
print(await char.is_online())
asyncio.run(main())
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.
To receive data through the event stream, you must define a trigger. A trigger is made up of three things:
The event type definitions are available in the auraxium.event
namespace.
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").
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).
"""
# Do stuff here
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():
# NOTE: Depending on player activity, this script will likely exceed the
# ~6 requests per minute and IP address limit for the default service ID.
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)
# NOTE: This value is likely different from char.data.battle_rank as
# the REST API tends to lag by a few minutes.
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()
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.
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()
.
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 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'
).
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.
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.
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)
# https://census.daybreakgames.com/s:example/get/ps2:v2/character?c:limit=20&c:join=characters_online_status
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())
# {'character_list': [...], 'returned': 1}
Refer to the ps2-census documentation for details.
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.
FAQs
A python wrapper for the PlanetSide 2 Census API.
We found that auraxium demonstrated a healthy version release cadence and project activity because the last version was released less than a year ago. It has 1 open source maintainer collaborating on the project.
Did you know?
Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.
Security News
Fluent Assertions is facing backlash after dropping the Apache license for a commercial model, leaving users blindsided and questioning contributor rights.
Research
Security News
Socket researchers uncover the risks of a malicious Python package targeting Discord developers.
Security News
The UK is proposing a bold ban on ransomware payments by public entities to disrupt cybercrime, protect critical services, and lead global cybersecurity efforts.