diff --git a/inventory_provider/routes/default.py b/inventory_provider/routes/default.py index f65d6124835c09f70459f9d2defe01a36e23dbd6..047cc10e3534d092879545c3f3a3ab8cf0e882e8 100644 --- a/inventory_provider/routes/default.py +++ b/inventory_provider/routes/default.py @@ -4,42 +4,43 @@ from flask import Blueprint, jsonify, current_app from inventory_provider.routes import common from inventory_provider.tasks.common import get_current_redis, get_latch -routes = Blueprint("inventory-data-default-routes", __name__) +routes = Blueprint('inventory-data-default-routes', __name__) API_VERSION = '0.1' VERSION_SCHEMA = { - "$schema": "http://json-schema.org/draft-07/schema#", + '$schema': 'http://json-schema.org/draft-07/schema#', - "definitions": { - "latch": { - "type": "object", - "properties": { - "current": {"type": "integer"}, - "next": {"type": "integer"}, - "this": {"type": "integer"}, - "failure": {"type": "boolean"}, - "pending": {"type": "boolean"}, + 'definitions': { + 'latch': { + 'type': 'object', + 'properties': { + 'current': {'type': 'integer'}, + 'next': {'type': 'integer'}, + 'this': {'type': 'integer'}, + 'failure': {'type': 'boolean'}, + 'pending': {'type': 'boolean'}, + 'timestamp': {'type': 'number'} }, - "required": ["current", "next", "this", "pending", "failure"], - "additionalProperties": False + 'required': ['current', 'next', 'this', 'pending', 'failure'], + 'additionalProperties': False } }, - "type": "object", - "properties": { - "api": { - "type": "string", - "pattern": r'\d+\.\d+' + 'type': 'object', + 'properties': { + 'api': { + 'type': 'string', + 'pattern': r'\d+\.\d+' }, - "module": { - "type": "string", - "pattern": r'\d+\.\d+' + 'module': { + 'type': 'string', + 'pattern': r'\d+\.\d+' }, - "latch": {"$ref": "#/definitions/latch"} + 'latch': {'$ref': '#/definitions/latch'} }, - "required": ["api", "module"], - "additionalProperties": False + 'required': ['api', 'module'], + 'additionalProperties': False } @@ -48,7 +49,7 @@ def after_request(resp): return common.after_request(resp) -@routes.route("/version", methods=['GET', 'POST']) +@routes.route('/version', methods=['GET', 'POST']) @common.require_accepts_json def version(): """ diff --git a/inventory_provider/tasks/common.py b/inventory_provider/tasks/common.py index ac85afb7de7094517c95cf2ed5f2929b211f6afc..427d51d8a18e4128f3dd3b174274e65fa17f5f8f 100644 --- a/inventory_provider/tasks/common.py +++ b/inventory_provider/tasks/common.py @@ -1,5 +1,6 @@ import json import logging +import time import jsonschema import redis @@ -11,17 +12,18 @@ DEFAULT_REDIS_SENTINEL_TIMEOUT = 0.1 DEFAULT_SENTINEL_SOCKET_TIMEOUT = 0.1 DB_LATCH_SCHEMA = { - "$schema": "http://json-schema.org/draft-07/schema#", - "type": "object", - "properties": { - "current": {"type": "integer"}, - "next": {"type": "integer"}, - "this": {"type": "integer"}, - "pending": {"type": "boolean"}, - "failure": {"type": "boolean"} + '$schema': 'http://json-schema.org/draft-07/schema#', + 'type': 'object', + 'properties': { + 'current': {'type': 'integer'}, + 'next': {'type': 'integer'}, + 'this': {'type': 'integer'}, + 'pending': {'type': 'boolean'}, + 'failure': {'type': 'boolean'}, + 'timestamp': {'type': 'number'} }, - "required": ["current", "next", "this"], - "additionalProperties": False + 'required': ['current', 'next', 'this'], + 'additionalProperties': False } TASK_LOG_SCHEMA = { @@ -86,17 +88,24 @@ def update_latch_status(config, pending=False, failure=False): logger.debug('updating latch status: pending={}, failure={}'.format( pending, failure)) + now = time.time() for db in config['redis-databases']: r = _get_redis(config, dbid=db) latch = get_latch(r) if not latch: continue + if not pending and not failure: + if not latch['pending'] and not latch['failure']: + logger.error( + 'updating latch for db {db} with pending=failure=True, ' + f'but latch is already {latch}') + latch['timestamp'] = now latch['pending'] = pending latch['failure'] = failure r.set('db:latch', json.dumps(latch)) -def set_latch(config, new_current, new_next): +def set_latch(config, new_current, new_next, timestamp): logger.debug('setting latch: new current={}, new next={}'.format( new_current, new_next)) @@ -107,7 +116,8 @@ def set_latch(config, new_current, new_next): 'next': new_next, 'this': db, 'pending': False, - 'failure': False + 'failure': False, + 'timestamp': timestamp } r = _get_redis(config, dbid=db) @@ -129,7 +139,11 @@ def latch_db(config): next_idx = db_ids.index(latch['next']) next_idx = (next_idx + 1) % len(db_ids) - set_latch(config, new_current=latch['next'], new_next=db_ids[next_idx]) + set_latch( + config, + new_current=latch['next'], + new_next=db_ids[next_idx], + timestamp=time.time()) def _get_redis(config, dbid=None): diff --git a/inventory_provider/tasks/worker.py b/inventory_provider/tasks/worker.py index d0613236b21c8b987a92b94a878751ea14f7a9c4..9102731490f31548ccb06818d5e109196635dce2 100644 --- a/inventory_provider/tasks/worker.py +++ b/inventory_provider/tasks/worker.py @@ -541,7 +541,8 @@ def _erase_next_db(config): set_latch( config, new_current=saved_latch['current'], - new_next=saved_latch['next']) + new_next=saved_latch['next'], + timestamp=saved_latch.get('timestamp', 0)) @app.task(base=InventoryTask, bind=True, name='internal_refresh_phase_2') diff --git a/test/test_celery_worker_global.py b/test/test_celery_worker_global.py index 330f8d815e4040eb06abef7d2d0594075e72e41c..ecd1d25e192ecaa06ee4528684292366a8472e2a 100644 --- a/test/test_celery_worker_global.py +++ b/test/test_celery_worker_global.py @@ -77,7 +77,7 @@ def test_next_redis(data_config, mocked_redis): :param mocked_redis: :return: """ - common.set_latch(data_config, 10, 20) + common.set_latch(data_config, 10, 20, 100) r = common.get_next_redis(data_config) assert r @@ -85,6 +85,7 @@ def test_next_redis(data_config, mocked_redis): latch = common.get_latch(r) assert latch['current'] == 10 assert latch['next'] == 20 + assert latch['timestamp'] == 100 def test_next_redis_with_none(data_config, mocked_redis): diff --git a/test/test_general_routes.py b/test/test_general_routes.py index 254d9db122b4d944bdd66df67f8a41e5ca67712c..8bc82db71c96426f68d502f8fe4dac4e2414cd40 100644 --- a/test/test_general_routes.py +++ b/test/test_general_routes.py @@ -5,59 +5,59 @@ from inventory_provider.routes import common from inventory_provider.routes.default import VERSION_SCHEMA DEFAULT_REQUEST_HEADERS = { - "Content-type": "application/json", - "Accept": ["application/json"] + 'Content-type': 'application/json', + 'Accept': ['application/json'] } def test_version_request(client, mocked_redis): rv = client.post( - "version", + 'version', headers=DEFAULT_REQUEST_HEADERS) assert rv.status_code == 200 jsonschema.validate( - json.loads(rv.data.decode("utf-8")), + json.loads(rv.data.decode('utf-8')), VERSION_SCHEMA) def test_load_json_docs(data_config, mocked_redis): INTERFACE_SCHEMA = { - "$schema": "http://json-schema.org/draft-07/schema#", + '$schema': 'http://json-schema.org/draft-07/schema#', - "definitions": { - "interface": { - "type": "object", - "properties": { - "logical-system": {"type": "string"}, - "name": {"type": "string"}, - "description": {"type": "string"}, - "bundle": { - "type": "array", - "items": {"type": "string"} + 'definitions': { + 'interface': { + 'type': 'object', + 'properties': { + 'logical-system': {'type': 'string'}, + 'name': {'type': 'string'}, + 'description': {'type': 'string'}, + 'bundle': { + 'type': 'array', + 'items': {'type': 'string'} }, - "ipv4": { - "type": "array", - "items": {"type": "string"} + 'ipv4': { + 'type': 'array', + 'items': {'type': 'string'} }, - "ipv6": { - "type": "array", - "items": {"type": "string"} + 'ipv6': { + 'type': 'array', + 'items': {'type': 'string'} } }, - "required": ["name", "description", "ipv4", "ipv6"], - "additionalProperties": False + 'required': ['name', 'description', 'ipv4', 'ipv6'], + 'additionalProperties': False } }, - "type": "object", - "properties": { - "key": {"type": "string"}, - "value": {"$ref": "#/definitions/interface"} + 'type': 'object', + 'properties': { + 'key': {'type': 'string'}, + 'value': {'$ref': '#/definitions/interface'} }, - "required": ["key", "value"], - "additionalProperties": False + 'required': ['key', 'value'], + 'additionalProperties': False } for ifc in common.load_json_docs(