Compare commits

...

5 Commits
0.6 ... 0.9

Author SHA1 Message Date
Romuald Bierbasz
d6e6efc633 SDK-2571: Refactor logging 2019-02-28 10:31:12 +01:00
Paweł Kierski
a114c9721c Add Unknown for all enums 2019-02-22 11:26:17 +01:00
Romuald Juchnowicz-Bierbasz
6c0389834b Increment version 2019-02-21 15:29:39 +01:00
Romuald Juchnowicz-Bierbasz
bc7d1c2914 SDK-2538: Use Optional 2019-02-21 15:17:38 +01:00
Romuald Juchnowicz-Bierbasz
d69e1aaa08 SDK-2538: Add LicenseType enum 2019-02-21 15:11:49 +01:00
6 changed files with 58 additions and 24 deletions

View File

@@ -12,6 +12,7 @@ class Platform(Enum):
Battlenet = "battlenet" Battlenet = "battlenet"
class Feature(Enum): class Feature(Enum):
Unknown = "Unknown"
ImportInstalledGames = "ImportInstalledGames" ImportInstalledGames = "ImportInstalledGames"
ImportOwnedGames = "ImportOwnedGames" ImportOwnedGames = "ImportOwnedGames"
LaunchGame = "LaunchGame" LaunchGame = "LaunchGame"
@@ -23,11 +24,19 @@ class Feature(Enum):
ImportUsers = "ImportUsers" ImportUsers = "ImportUsers"
VerifyGame = "VerifyGame" VerifyGame = "VerifyGame"
class LicenseType(Enum):
Unknown = "Unknown"
SinglePurchase = "SinglePurchase"
FreeToPlay = "FreeToPlay"
OtherUserLicense = "OtherUserLicense"
class LocalGameState(Enum): class LocalGameState(Enum):
Unknown = "Unknown"
Installed = "Installed" Installed = "Installed"
Running = "Running" Running = "Running"
class PresenceState(Enum): class PresenceState(Enum):
Unknown = "Unknown"
Online = "online" Online = "online"
Offline = "offline" Offline = "offline"
Away = "away" Away = "away"

View File

@@ -143,8 +143,8 @@ class Server():
self._send_error(request.id, MethodNotFound()) self._send_error(request.id, MethodNotFound())
except JsonRpcError as error: except JsonRpcError as error:
self._send_error(request.id, error) self._send_error(request.id, error)
except Exception as error: #pylint: disable=broad-except except Exception: #pylint: disable=broad-except
logging.error("Unexpected exception raised in plugin handler: %s", repr(error)) logging.exception("Unexpected exception raised in plugin handler")
asyncio.create_task(handle()) asyncio.create_task(handle())

View File

@@ -1,9 +1,11 @@
import asyncio import asyncio
import json import json
import logging import logging
import logging.handlers
import dataclasses import dataclasses
from enum import Enum from enum import Enum
from collections import OrderedDict from collections import OrderedDict
import sys
from galaxy.api.jsonrpc import Server, NotificationClient from galaxy.api.jsonrpc import Server, NotificationClient
from galaxy.api.consts import Feature from galaxy.api.consts import Feature
@@ -21,6 +23,7 @@ class JSONEncoder(json.JSONEncoder):
class Plugin(): class Plugin():
def __init__(self, platform, reader, writer, handshake_token): def __init__(self, platform, reader, writer, handshake_token):
logging.info("Creating plugin for platform %s", platform.value)
self._platform = platform self._platform = platform
self._feature_methods = OrderedDict() self._feature_methods = OrderedDict()
@@ -167,7 +170,10 @@ class Plugin():
async def pass_control(): async def pass_control():
while self._active: while self._active:
logging.debug("Passing control to plugin") logging.debug("Passing control to plugin")
self.tick() try:
self.tick()
except Exception:
logging.exception("Unexpected exception raised in plugin tick")
await asyncio.sleep(1) await asyncio.sleep(1)
await asyncio.gather(pass_control(), self._server.run()) await asyncio.gather(pass_control(), self._server.run())
@@ -309,22 +315,40 @@ class Plugin():
async def get_game_times(self): async def get_game_times(self):
raise NotImplementedError() raise NotImplementedError()
def create_and_run_plugin(plugin_class, argv): def create_and_run_plugin(plugin_class, argv):
if not issubclass(plugin_class, Plugin): root = logging.getLogger()
raise TypeError("plugin_class must be subclass of Plugin") root.setLevel(logging.DEBUG)
if len(argv) >= 4:
handler = logging.handlers.RotatingFileHandler(argv[3], "a", 10000000, 10)
root.addHandler(handler)
if len(argv) < 3: if len(argv) < 3:
raise ValueError("Not enough parameters, required: token, port") logging.critical("Not enough parameters, required: token, port")
sys.exit(1)
token = argv[1] token = argv[1]
try: try:
port = int(argv[2]) port = int(argv[2])
except ValueError as e: except ValueError:
raise ValueError("Failed to parse port value, {}".format(e)) logging.critical("Failed to parse port value: %s", argv[2])
sys.exit(2)
if not (1 <= port <= 65535): if not (1 <= port <= 65535):
raise ValueError("Port value out of range (1, 65535)") logging.critical("Port value out of range (1, 65535)")
sys.exit(3)
if not issubclass(plugin_class, Plugin):
logging.critical("plugin_class must be subclass of Plugin")
sys.exit(4)
async def coroutine(): async def coroutine():
reader, writer = await asyncio.open_connection("127.0.0.1", port) reader, writer = await asyncio.open_connection("127.0.0.1", port)
plugin = plugin_class(reader, writer, token) plugin = plugin_class(reader, writer, token)
await plugin.run() await plugin.run()
asyncio.run(coroutine())
try:
asyncio.run(coroutine())
except Exception:
logging.exception("Error while running plugin")
sys.exit(5)

View File

@@ -1,7 +1,7 @@
from dataclasses import dataclass from dataclasses import dataclass
from typing import List from typing import List, Optional
from galaxy.api.consts import LocalGameState, PresenceState from galaxy.api.consts import LicenseType, LocalGameState, PresenceState
@dataclass @dataclass
class Authentication(): class Authentication():
@@ -10,8 +10,8 @@ class Authentication():
@dataclass @dataclass
class LicenseInfo(): class LicenseInfo():
license_type: str license_type: LicenseType
owner: str = None owner: Optional[str] = None
@dataclass @dataclass
class Dlc(): class Dlc():
@@ -39,8 +39,8 @@ class LocalGame():
@dataclass @dataclass
class Presence(): class Presence():
presence_state: PresenceState presence_state: PresenceState
game_id: str = None game_id: Optional[str] = None
presence_status: str = None presence_status: Optional[str] = None
@dataclass @dataclass
class UserInfo(): class UserInfo():

View File

@@ -2,7 +2,7 @@ from setuptools import setup, find_packages
setup( setup(
name="galaxy.plugin.api", name="galaxy.plugin.api",
version="0.6", version="0.9",
description="Galaxy python plugin API", description="Galaxy python plugin API",
author='Galaxy team', author='Galaxy team',
author_email='galaxy@gog.com', author_email='galaxy@gog.com',

View File

@@ -2,6 +2,7 @@ import asyncio
import json import json
from galaxy.api.types import Game, Dlc, LicenseInfo from galaxy.api.types import Game, Dlc, LicenseInfo
from galaxy.api.consts import LicenseType
from galaxy.api.errors import UnknownError from galaxy.api.errors import UnknownError
def test_success(plugin, readline, write): def test_success(plugin, readline, write):
@@ -13,15 +14,15 @@ def test_success(plugin, readline, write):
readline.side_effect = [json.dumps(request), ""] readline.side_effect = [json.dumps(request), ""]
plugin.get_owned_games.return_value = [ plugin.get_owned_games.return_value = [
Game("3", "Doom", None, LicenseInfo("SinglePurchase", None)), Game("3", "Doom", None, LicenseInfo(LicenseType.SinglePurchase, None)),
Game( Game(
"5", "5",
"Witcher 3", "Witcher 3",
[ [
Dlc("7", "Hearts of Stone", LicenseInfo("SinglePurchase", None)), Dlc("7", "Hearts of Stone", LicenseInfo(LicenseType.SinglePurchase, None)),
Dlc("8", "Temerian Armor Set", LicenseInfo("FreeToPlay", None)), Dlc("8", "Temerian Armor Set", LicenseInfo(LicenseType.FreeToPlay, None)),
], ],
LicenseInfo("SinglePurchase", None)) LicenseInfo(LicenseType.SinglePurchase, None))
] ]
asyncio.run(plugin.run()) asyncio.run(plugin.run())
plugin.get_owned_games.assert_called_with() plugin.get_owned_games.assert_called_with()
@@ -89,7 +90,7 @@ def test_failure(plugin, readline, write):
} }
def test_add_game(plugin, write): def test_add_game(plugin, write):
game = Game("3", "Doom", None, LicenseInfo("SinglePurchase", None)) game = Game("3", "Doom", None, LicenseInfo(LicenseType.SinglePurchase, None))
async def couritine(): async def couritine():
plugin.add_game(game) plugin.add_game(game)
@@ -127,7 +128,7 @@ def test_remove_game(plugin, write):
} }
def test_update_game(plugin, write): def test_update_game(plugin, write):
game = Game("3", "Doom", None, LicenseInfo("SinglePurchase", None)) game = Game("3", "Doom", None, LicenseInfo(LicenseType.SinglePurchase, None))
async def couritine(): async def couritine():
plugin.update_game(game) plugin.update_game(game)