Robert Bedner 3 жил өмнө
parent
commit
1ed92dfbe1

+ 1 - 1
.vscode/settings.json

@@ -9,7 +9,7 @@
     ],
     "python.testing.pytestEnabled": true,
     "python.testing.nosetestsEnabled": false,
-    "python.testing.unittestEnabled": false,
+    "python.testing.unittestEnabled": true,
     "python.linting.pylintEnabled": false,
     "python.linting.flake8Enabled": true,
     "python.linting.enabled": true,

+ 1 - 1
mazda/Workshop_Order_Report.csv

@@ -1,5 +1,5 @@
 "creationDate";"currency";"documentType";"invoiceCategory";"invoiceDate";"amount";"discount";"orderLineNumber";"orderNumber";"portion";"unitPrice";"invoiceNumber";"lineNumber";"orderItemType";"category";"descriptionOperation";"hours";"operationCode";"standardHours";"descriptionOther";"type";"descriptionPart";"isDamageCausal";"manufacturer";"partNumber";"quantity";"serialNumber";"unit";"company";"descriptionPurchase";"invoiceCode";"invoiceDate";"invoiceNumber";"orderCompletionDate";"orderDate";"orderNumber";"licensePlate";"nextMotDueDate";"odometer";"odometerUnit";"vin";"vehicleIntakeDate";"timeRangeBegin";"timeRangeEnd"
-"2021-05-29";"EUR";"invoice";"warranty";"2021-05-29";12,47;0,00;25;4671715;1,00;12,47;600011;25;"part";"";"";;"";;"";"";"ENDBESCHLAG";0;"VO";"32341045";1,00;"";"";"";"";"";"";"";"2021-06-25";"2021-06-23";4671715;"NE-CD 2302";"2023-04-01";78317;"KM";"YV1FW45C1E1166502";2021-06-25 00:00;"2021-06-23";"2021-06-23"
+"2021-06-29";"EUR";"invoice";"warranty";"2021-05-29";12,47;0,00;25;4671715;1,00;12,47;600011;25;"part";"";"";;"";;"";"";"ENDBESCHLAG";0;"VO";"32341045";1,00;"";"";"";"";"";"";"";"2021-06-25";"2021-06-23";4671715;"NE-CD 2302";"2023-04-01";78317;"KM";"YV1FW45C1E1166502";2021-06-25 00:00;"2021-06-23";"2021-06-23"
 "2021-06-29";"EUR";"invoice";"warranty";"2021-06-29";21,99;0,00;30;4671715;0,00;109,97;600011;30;"operation";"97702";"Endbeschlag 1 Seite, Montage";0,20;"97702";0,20;"";"";"";0;"";"";;"";"";"";"";"";"";"";"2021-06-25";"2021-06-23";4671715;"NE-CD 2302";"2023-04-01";78317;"KM";"YV1FW45C1E1166502";2021-06-25 00:00;"2021-06-23";"2021-06-23"
 "2021-06-29";"EUR";"cancelation";"warranty";"2021-06-29";-12,47;0,00;25;6430885;-1,00;12,47;600012;25;"part";"";"";;"";;"";"";"ENDBESCHLAG";0;"VO";"32341045";-1,00;"";"";"";"";"";"";"";"2021-06-25";"2021-06-29";6430885;"NE-CD 2302";"2023-04-01";78317;"KM";"YV1FW45C1E1166502";2021-06-25 00:00;"2021-06-23";"2021-06-23"
 "2021-06-29";"EUR";"cancelation";"warranty";"2021-06-29";-21,99;0,00;30;6430885;0,00;109,97;600012;30;"operation";"97702";"Endbeschlag 1 Seite, Montage";-0,20;"97702";-0,20;"";"";"";0;"";"";;"";"";"";"";"";"";"";"2021-06-25";"2021-06-29";6430885;"NE-CD 2302";"2023-04-01";78317;"KM";"YV1FW45C1E1166502";2021-06-25 00:00;"2021-06-23";"2021-06-23"

+ 1 - 1
mazda/mazda_export.json

@@ -1,5 +1,5 @@
 {
-  "creationDate": "2021-08-16T16:28:23.148Z",
+  "creationDate": "2021-08-30T08:43:42.431Z",
   "invoices": [
     {
       "currency": "EUR",

+ 158 - 163
mazda/mazda_upload.py

@@ -6,8 +6,23 @@ from requests_oauthlib import OAuth2Session
 from datetime import datetime
 
 
-MazdaConfig = namedtuple('MazdaConfig', 'domain webservice module auth_url token_url '
-                         + 'client_id client_secret username password dealer_number')
+MazdaConfig = namedtuple('MazdaConfig', 'domain webservice module auth_url token_url client_id client_secret username password dealer_number')
+
+cfg = MazdaConfig(**{
+    'domain': 'https://mappsacc.mazdaeur.com',
+    'webservice': '/dogma-restapi-dms/api',
+    'module': '/vehicles/workshop/order-report',
+    'auth_url': '/oauth/authorize',
+    'token_url': '/oauth/token',
+    'client_id': 'E7FC943B-B73F-F48E-B71A-419EA4CD4AC7',
+    'client_secret': '^bH=rk@c58zrr^Apc#9fzy$c',
+    'username': 'mmd88888.cdk',
+    'password': 'MazdaCX30',
+    'dealer_number': '88888/MMD'
+})
+
+redirect_uri = 'https://localhost/'
+base_dir = '/home/robert/projekte/python/mazda/'
 
 
 def date_format(d: datetime):
@@ -19,171 +34,151 @@ def date_format(d: datetime):
     return date_str[:-3] + 'Z'
 
 
-class mazda_upload:
-    cfg = MazdaConfig(**{
-        'domain': 'https://mappsacc.mazdaeur.com',
-        'webservice': '/dogma-restapi-dms/api',
-        'module': '/vehicles/workshop/order-report',
-        'auth_url': '/oauth/authorize',
-        'token_url': '/oauth/token',
-        'client_id': 'E7FC943B-B73F-F48E-B71A-419EA4CD4AC7',
-        'client_secret': '^bH=rk@c58zrr^Apc#9fzy$c',
-        'username': 'mmd88888.cdk',
-        'password': 'MazdaCX30',
-        'dealer_number': '88888/MMD'
-    })
-
-    redirect_uri = 'https://localhost/'
-    base_dir = '/home/robert/projekte/python/mazda/'
-
-    def token_save(self, token):
-        json.dump(token, open(self.base_dir + 'token.json', 'w'), indent=2)
-
-    def token_load(self):
-        try:
-            return json.load(open(self.base_dir + 'token.json', 'r'))
-        except FileNotFoundError:
-            return None
-
-    def convert_csv(self, csv_file, json_file, year, month):
-        date_min = datetime(year, month, 1, 0, 0, 0)
-        date_max = datetime(year, month + 1, 1, 0, 0, 0)
-
-        date_cols = ['invoiceDate', 'orderDate', 'orderCompletionDate', 'vehicleIntakeDate', 'nextMotDueDate']
-        df = pd.read_csv(self.base_dir + csv_file, encoding='latin-1', decimal=',', sep=';', parse_dates=date_cols)
-        df = df.fillna(0)[(df['invoiceDate'] >= date_min) & (df['invoiceDate'] <= date_max)]
-        df = df.sort_values(by=['invoiceDate', 'invoiceNumber', 'orderNumber', 'lineNumber'])
-        df['vin'] = np.where(df['vin'] == 0, '0' * 17, df['vin'])
-        # print(df[['currency','documentType','invoiceCategory','invoiceDate','invoiceNumber']].drop_duplicates().info())
-        invoices_filter = ['currency', 'documentType', 'invoiceCategory', 'invoiceDate', 'invoiceNumber']
-        invoices = df[invoices_filter].drop_duplicates().to_dict('records')
-        invoice_items_filter = ['invoiceNumber', 'orderLineNumber', 'orderNumber', 'amount', 'discount', 'portion', 'unitPrice']
-        invoice_items = df[invoice_items_filter].groupby('invoiceNumber')
-
-        for invoice in invoices:
-            invoice['invoiceDate'] = date_format(invoice['invoiceDate'])
-            items = invoice_items.get_group(invoice['invoiceNumber'])
-            items.pop('invoiceNumber')
-            invoice['invoiceItems'] = items.to_dict('records')
-
-        orders = df[['orderNumber', 'orderDate', 'orderCompletionDate', 'vehicleIntakeDate']].drop_duplicates().to_dict('records')
-        orders_vehicle_filter = ['orderNumber', 'licensePlate', 'nextMotDueDate', 'odometer', 'odometerUnit', 'vin']
-        orders_vehicle = df[orders_vehicle_filter].drop_duplicates().groupby('orderNumber')
-        orders_items = df[[
-            'orderNumber', 'lineNumber', 'orderItemType',
-            'category', 'descriptionOperation', 'hours', 'operationCode', 'standardHours',
-            'descriptionOther', 'type',
-            'descriptionPart', 'isDamageCausal', 'manufacturer', 'partNumber', 'quantity', 'serialNumber', 'unit',
-            'company', 'descriptionPurchase', 'invoiceCode', 'invoiceDate', 'invoiceNumber'
-        ]].drop_duplicates().groupby('orderNumber')
-
-        for order in orders:
-            order['vehicle'] = orders_vehicle.get_group(order['orderNumber']).to_dict('records')[0]
-            order['vehicle']['nextMotDueDate'] = date_format(order['vehicle']['nextMotDueDate'])
-
-            order['orderDate'] = date_format(order['orderDate'])
-            order['orderCompletionDate'] = date_format(order['orderCompletionDate'])
-            order['vehicleIntakeDate'] = date_format(order['vehicleIntakeDate'])
-
-            items = orders_items.get_group(order['orderNumber']).to_dict('records')
-            order['items'] = []
-            for item in items:
-                if item['orderItemType'] == 'operation':
-                    order['items'].append({
-                        'lineNumber': item['lineNumber'],
-                        'operation': {
-                            'category': item['category'],
-                            'description': item['descriptionOperation'],
-                            'hours': item['hours'],
-                            'operationCode': item['operationCode'],
-                            'standardHours': item['standardHours']
-                        }
-                    })
-                elif item['orderItemType'] == 'part':
-                    order['items'].append({
-                        'lineNumber': item['lineNumber'],
-                        'part': {
-                            'description': item['descriptionPart'],
-                            'isDamageCausal': item['isDamageCausal'],
-                            'manufacturer': item['manufacturer'],
-                            'partNumber': item['partNumber'],
-                            'quantity': item['quantity'],
-                            'serialNumber': item['serialNumber'],
-                            'unit': item['unit']
-                        }
-                    })
-                elif item['orderItemType'] == 'other':
-                    order['items'].append({
-                        'lineNumber': item['lineNumber'],
-                        'other': {
-                            'description': item['descriptionOther'],
-                            'type': item['type']
-                        }
-                    })
-                else:
-                    order['items'].append({
-                        'lineNumber': item['lineNumber'],
-                        'purchaseInvoice': {
-                            'company': item['company'],
-                            'description': item['descriptionPurchase'],
-                            'invoiceCode': item['invoiceCode'],
-                            'invoiceDate': date_format(item['invoiceDate']),
-                            'invoiceNumber': item['invoiceNumber']
-                        }
-                    })
-
-        res = {
-            'creationDate': date_format(datetime.now()),
-            'invoices': invoices,
-            'orders': orders,
-            'timeRangeBegin': date_format(date_min),
-            'timeRangeEnd': date_format(date_max)
-        }
-
-        json.dump(res, open(self.base_dir + json_file, 'w'), indent=2)
-        return res
-
-    def login(self):
-        token = self.token_load()
-        if token is None or token['expires_at'] < datetime.now().timestamp():
-            self.oauth = OAuth2Session(self.cfg.client_id, redirect_uri=self.redirect_uri)
-            authorization_url, _ = self.oauth.authorization_url(self.cfg.domain + self.cfg.auth_url)
-            return authorization_url
-        else:
-            extra = {
-                'client_id': self.cfg.client_id,
-                'client_secret': self.cfg.client_secret
-            }
-            self.oauth = OAuth2Session(self.cfg.client_id, token=token, auto_refresh_url=self.cfg.domain + self.cfg.token_url,
-                                       auto_refresh_kwargs=extra, token_updater=self.token_save)
-            return None
-
-    def fetch_token(self, redirect_response):
-        token = self.oauth.fetch_token(self.cfg.domain + self.cfg.token_url,
-                                       client_secret=self.cfg.client_secret,
-                                       authorization_response=redirect_response)
-        self.token_save(token)
-
-    def upload(self, data):
-        headers = {
-            'accept': 'application/vnd.mazdaeur.dms.v4+json',
-            'x-mme-organisation': self.cfg.dealer_number,
-            'X-mazda-org': self.cfg.dealer_number,
-            'Content-Type': 'application/json',
-            # 'Authorization': 'Bearer ' + token
-        }
-        r = self.oauth.post(self.cfg.domain + self.cfg.webservice + self.cfg.module, json.dumps(data), headers=headers)
-        print(r.status_code)
-        with open(self.base_dir + 'post_error.log', 'w') as fwh:
-            fwh.write(r.text)
+# After updating the token you will most likely want to save it.
+def token_save(token):
+    json.dump(token, open(base_dir + 'token.json', 'w'), indent=2)
+
+
+def token_load():
+    try:
+        return json.load(open(base_dir + 'token.json', 'r'))
+    except FileNotFoundError:
+        return None
+
+
+def convert_csv(csv_file, json_file, year, month):
+    date_min = datetime(year, month, 1, 0, 0, 0)
+    date_max = datetime(year, month + 1, 1, 0, 0, 0)
+
+    date_cols = ['invoiceDate', 'orderDate', 'orderCompletionDate', 'vehicleIntakeDate', 'nextMotDueDate']
+    df = pd.read_csv(csv_file, encoding='latin-1', decimal=',', sep=';', parse_dates=date_cols)
+    df = df.fillna(0)[(df['invoiceDate'] >= date_min) & (df['invoiceDate'] <= date_max)]
+    df = df.sort_values(by=['invoiceDate', 'invoiceNumber', 'orderNumber', 'lineNumber'])
+    df['vin'] = np.where(df['vin'] == 0, '0' * 17, df['vin'])
+    # print(df[['currency','documentType','invoiceCategory','invoiceDate','invoiceNumber']].drop_duplicates().info())
+    invoices_filter = ['currency', 'documentType', 'invoiceCategory', 'invoiceDate', 'invoiceNumber']
+    invoices = df[invoices_filter].drop_duplicates().to_dict('records')
+    invoice_items_filter = ['invoiceNumber', 'orderLineNumber', 'orderNumber', 'amount', 'discount', 'portion', 'unitPrice']
+    invoice_items = df[invoice_items_filter].groupby('invoiceNumber')
+
+    for invoice in invoices:
+        invoice['invoiceDate'] = date_format(invoice['invoiceDate'])
+        items = invoice_items.get_group(invoice['invoiceNumber'])
+        items.pop('invoiceNumber')
+        invoice['invoiceItems'] = items.to_dict('records')
+
+    orders = df[['orderNumber', 'orderDate', 'orderCompletionDate', 'vehicleIntakeDate']].drop_duplicates().to_dict('records')
+    orders_vehicle_filter = ['orderNumber', 'licensePlate', 'nextMotDueDate', 'odometer', 'odometerUnit', 'vin']
+    orders_vehicle = df[orders_vehicle_filter].drop_duplicates().groupby('orderNumber')
+    orders_items = df[[
+        'orderNumber', 'lineNumber', 'orderItemType',
+        'category', 'descriptionOperation', 'hours', 'operationCode', 'standardHours',
+        'descriptionOther', 'type',
+        'descriptionPart', 'isDamageCausal', 'manufacturer', 'partNumber', 'quantity', 'serialNumber', 'unit',
+        'company', 'descriptionPurchase', 'invoiceCode', 'invoiceDate', 'invoiceNumber'
+    ]].drop_duplicates().groupby('orderNumber')
+
+    for order in orders:
+        order['vehicle'] = orders_vehicle.get_group(order['orderNumber']).to_dict('records')[0]
+        order['vehicle']['nextMotDueDate'] = date_format(order['vehicle']['nextMotDueDate'])
+
+        order['orderDate'] = date_format(order['orderDate'])
+        order['orderCompletionDate'] = date_format(order['orderCompletionDate'])
+        order['vehicleIntakeDate'] = date_format(order['vehicleIntakeDate'])
+
+        items = orders_items.get_group(order['orderNumber']).to_dict('records')
+        order['items'] = []
+        for item in items:
+            if item['orderItemType'] == 'operation':
+                order['items'].append({
+                    'lineNumber': item['lineNumber'],
+                    'operation': {
+                        'category': item['category'],
+                        'description': item['descriptionOperation'],
+                        'hours': item['hours'],
+                        'operationCode': item['operationCode'],
+                        'standardHours': item['standardHours']
+                    }
+                })
+            elif item['orderItemType'] == 'part':
+                order['items'].append({
+                    'lineNumber': item['lineNumber'],
+                    'part': {
+                        'description': item['descriptionPart'],
+                        'isDamageCausal': item['isDamageCausal'],
+                        'manufacturer': item['manufacturer'],
+                        'partNumber': item['partNumber'],
+                        'quantity': item['quantity'],
+                        'serialNumber': item['serialNumber'],
+                        'unit': item['unit']
+                    }
+                })
+            elif item['orderItemType'] == 'other':
+                order['items'].append({
+                    'lineNumber': item['lineNumber'],
+                    'other': {
+                        'description': item['descriptionOther'],
+                        'type': item['type']
+                    }
+                })
+            else:
+                order['items'].append({
+                    'lineNumber': item['lineNumber'],
+                    'purchaseInvoice': {
+                        'company': item['company'],
+                        'description': item['descriptionPurchase'],
+                        'invoiceCode': item['invoiceCode'],
+                        'invoiceDate': date_format(item['invoiceDate']),
+                        'invoiceNumber': item['invoiceNumber']
+                    }
+                })
+
+    res = {
+        'creationDate': date_format(datetime.now()),
+        'invoices': invoices,
+        'orders': orders,
+        'timeRangeBegin': date_format(date_min),
+        'timeRangeEnd': date_format(date_max)
+    }
+
+    json.dump(res, open(json_file, 'w'), indent=2)
+    return res
+
+
+def upload(data):
+    headers = {
+        'accept': 'application/vnd.mazdaeur.dms.v4+json',
+        'x-mme-organisation': cfg.dealer_number,
+        'X-mazda-org': cfg.dealer_number,
+        'Content-Type': 'application/json',
+        # 'Authorization': 'Bearer ' + token
+    }
+    extra = {
+        'client_id': cfg.client_id,
+        'client_secret': cfg.client_secret
+    }
+
+    token = token_load()
+    if token is None or token['expires_at'] < datetime.now().timestamp():
+        oauth = OAuth2Session(cfg.client_id, redirect_uri=redirect_uri)
+        authorization_url, state = oauth.authorization_url(cfg.domain + cfg.auth_url)
+        print('Please go here and authorize: ' + authorization_url)
+        redirect_response = input('Paste the full redirect URL here:')
+        token = oauth.fetch_token(cfg.domain + cfg.token_url, client_secret=cfg.client_secret, authorization_response=redirect_response)
+        token_save(token)
+    else:
+        oauth = OAuth2Session(cfg.client_id, token=token, auto_refresh_url=cfg.domain + cfg.token_url,
+                              auto_refresh_kwargs=extra, token_updater=token_save)
+    r = oauth.post(cfg.domain + cfg.webservice + cfg.module, json.dumps(data), headers=headers)
+    print(r.status_code)
+    with open(base_dir + 'post_error.log', 'w') as fwh:
+        fwh.write(r.text)
 
 
 def main():
-    mu = mazda_upload()
-    data = mu.convert_csv('Workshop_Order_Report.csv', 'mazda_export.json', 2021, 6)
+    data = convert_csv(base_dir + 'Workshop_Order_Report.csv', base_dir + 'mazda_export.json', 2021, 6)
     # data = json.load(open(base_dir + 'mazda_export.json', 'r'))
-    mu.login()
-    mu.upload(data)
+    upload(data)
 
 
 if __name__ == '__main__':

+ 1 - 1
mazda/post_error.log

@@ -1,5 +1,5 @@
 {
-  "creationDate" : "2021-08-10T15:06:45Z",
+  "creationDate" : "2021-08-30T08:43:42Z",
   "timeRangeBegin" : "2021-06-01T00:00:00Z",
   "timeRangeEnd" : "2021-07-01T00:00:00Z",
   "orders" : [ {

+ 5 - 5
mazda/token.json

@@ -1,19 +1,19 @@
 {
-  "access_token": "eyJhbGciOiJSUzI1NiIsInR5cCI6IkpXVCIsImtpZCI6Im11bS1hY2MtdjEifQ.eyJzdWIiOiJtbWQ4ODg4OC5jZGsiLCJhdWQiOlsiUGFydHNHVyIsImRvZ21hLXJlc3RhcGktZG1zIl0sIm9yZ2FuaXNhdGlvbnMiOlsiODg4ODgvTU1EIl0sInVzZXJfbmFtZSI6Im1tZDg4ODg4LmNkayIsInNjb3BlIjoiIiwiaXNzIjoiaHR0cHM6Ly9tYXBwc2FjYy5tYXpkYWV1ci5jb20vb2F1dGgiLCJleHAiOjE2Mjg2MDI4NjksImRlZmF1bHREb21haW4iOiI4ODg4OC9NTUQiLCJ1c2VyR3VpZCI6IjRDNUM1RUMyLTEzMzQtMjVFQS1DMjNCLUE1NTU2RDkzQ0RCMiIsImlhdCI6MTYyODU5OTI2OSwianRpIjoiYzc2ZjE3OTktOTMzMy00ZGM1LWFkZjgtNTMyYmE4MTgyODcxIiwiY2xpZW50X2lkIjoiRTdGQzk0M0ItQjczRi1GNDhFLUI3MUEtNDE5RUE0Q0Q0QUM3In0.bAXS4Fb2hEIO4PtOzYuNqK-42HyIhuVYezoXOKZWVCSUlxZSVn2axvXprzW2P8G1IY303SpwfLRamTdQNwjaw4G3tFJ1gWYjkkZrd4MsuRVu-LjsIPUlteFXzicgRmDCTqmLTiNISj6KrEnEi65hfvC6NpP6RglrlyotAZgAMOdquSwbUzlPaqqLvxfho80_xKIsf-6L6yswSnn2_7heUaZfP9acGR_chUiaN_RX2pL6wL2-03nuwHfKBNgIb3gT6vm70WVvHj2NBgRI7cLrJV3FDrjQc-jSNkNqMsKL3liiso36YV15RqhXmZOsOuwmJ5FGbq-VA5D-QQUW_L7dmA",
+  "access_token": "eyJhbGciOiJSUzI1NiIsInR5cCI6IkpXVCIsImtpZCI6Im11bS1hY2MtdjEifQ.eyJzdWIiOiJtbWQ4ODg4OC5jZGsiLCJhdWQiOlsiUGFydHNHVyIsImRvZ21hLXJlc3RhcGktZG1zIl0sIm9yZ2FuaXNhdGlvbnMiOlsiODg4ODgvTU1EIl0sInVzZXJfbmFtZSI6Im1tZDg4ODg4LmNkayIsInNjb3BlIjoiIiwiaXNzIjoiaHR0cHM6Ly9tYXBwc2FjYy5tYXpkYWV1ci5jb20vb2F1dGgiLCJleHAiOjE2MzAzMDk0NTYsImRlZmF1bHREb21haW4iOiI4ODg4OC9NTUQiLCJ1c2VyR3VpZCI6IjRDNUM1RUMyLTEzMzQtMjVFQS1DMjNCLUE1NTU2RDkzQ0RCMiIsImlhdCI6MTYzMDMwNTg1NiwianRpIjoiOTRiMzc5MzAtMGNlOS00NWM5LTkxM2EtYjY4MWViNjFjMjhhIiwiY2xpZW50X2lkIjoiRTdGQzk0M0ItQjczRi1GNDhFLUI3MUEtNDE5RUE0Q0Q0QUM3In0.h3MqgAk7T95xjhv2bGqXp__wCc5lT-_aBMSV7xPnbQdqY9bh1Qb5Si56lEIXSL0qye2giONxlXb6FMtEDz1XUnpUrWaqfVzuj10WFtRWBcsCh9x3dAZd00Wit7XLzNJklurfZbJYtJjqg4hkgHl7g4F74Wn5ZsgN9u2eFoBcXVtOv4pAVjAa7VTXfM5pXmlODF5QHiCnmpdprk4Esr5IeieZUCH9-WvNj-mA3vk0pFKpkdi8KZT3coHk1148aONYxjdjEZOmd8gtmSpbWplJaoll19SDq-koT2-ZUT-bOEyIkeaskq3Hd_ZMD8fMKYtvcw3-jHfi4J0Md-skDoCfLA",
   "token_type": "bearer",
-  "refresh_token": "0bcadea5-7f56-4c6a-9231-440f7fa10478",
+  "refresh_token": "a34dd1a5-922a-4638-b54e-4736c9af12ed",
   "expires_in": 3599,
   "sub": "mmd88888.cdk",
   "iss": "https://mappsacc.mazdaeur.com/oauth",
-  "iat": 1628599269,
+  "iat": 1630305856,
   "defaultDomain": "88888/MMD",
   "userGuid": "4C5C5EC2-1334-25EA-C23B-A5556D93CDB2",
   "organisations": [
     "88888/MMD"
   ],
-  "jti": "c76f1799-9333-4dc5-adf8-532ba8182871",
+  "jti": "94b37930-0ce9-45c9-913a-b681eb61c28a",
   "scope": [
     ""
   ],
-  "expires_at": 1628602868.0546541
+  "expires_at": 1630309455.36723
 }

+ 15 - 0
sandbox/ohno.py

@@ -0,0 +1,15 @@
+import numpy as np
+
+
+class Board:
+    _board: np.ndarray
+
+    def __init__(self, size):
+        self.size = size
+        self._board = np.zeros((self.size, self.size))
+
+    def set_initial(self, new_board):
+        self._board = np.array(new_board)
+
+    def solve(self):
+        return [self._board.tolist()]

+ 17 - 0
sandbox/test_ohno.py

@@ -0,0 +1,17 @@
+import unittest
+import ohno
+
+
+class test_ohno(unittest.TestCase):
+    def test_simple(self):
+        board = ohno.Board(4)
+        board.set_initial([[-1, -1, -1, -1],
+                           [-1, -1, -1, -1],
+                           [-1, -1, -1, -1],
+                           [-1, -1, -1, -1]])
+        solved = board.solve()
+        self.assertEqual(len(solved), 1)
+
+
+if __name__ == '__main__':
+    unittest.main()