diff --git a/.github/workflows/commitlint.yml b/.github/workflows/commitlint.yml new file mode 100644 index 000000000..a8ff39eee --- /dev/null +++ b/.github/workflows/commitlint.yml @@ -0,0 +1,36 @@ +name: Commitlint + +on: [push] + +jobs: + commitlint: + runs-on: ubuntu-latest + steps: + - name: Checkout + uses: actions/checkout@v4.1.1 + with: + fetch-depth: 0 + + - name: Setup Node + uses: actions/setup-node@v4.0.1 + with: + node-version-file: '.nvmrc' + cache: 'npm' + + - name: Install Dependencies + run: npm ci + + - name: Print versions + run: | + git --version + node --version + npm --version + npx commitlint --version + + # Run the commitlint action, considering its own dependencies and yours as well 🚀 + # `github.workspace` is the path to your repository. + - uses: wagoid/commitlint-github-action@v5 + env: + NODE_PATH: ${{ github.workspace }}/node_modules + with: + commitDepth: 1 diff --git a/.github/workflows/verify.yml b/.github/workflows/verify.yml new file mode 100644 index 000000000..90ec08d77 --- /dev/null +++ b/.github/workflows/verify.yml @@ -0,0 +1,53 @@ +name: Code quality checks + +on: + pull_request: + +jobs: + py-scripts-lint: + name: Check for formatting of python scripts + runs-on: ubuntu-latest + steps: + - name: Checkout + uses: actions/checkout@v4.1.1 + + - uses: actions/setup-python@v5 + with: + python-version: '3.10' + + - name: Setting up python libraries + run: ./scripts/setup-python.sh + + # Reference: https://black.readthedocs.io/en/stable/integrations/github_actions.html + - name: Check formatting for Python files + uses: psf/black@stable + with: + options: '--check --verbose' + + formatting-lint: + name: Check for formatting & lint errors + runs-on: ubuntu-latest + + steps: + - name: Checkout + uses: actions/checkout@v4.1.1 + + - name: Setup Node + uses: actions/setup-node@v3.7.0 + with: + node-version-file: .nvmrc + cache: 'npm' + + - name: Install Dependencies + run: npm ci + + - name: Run Lint Checks + run: | + npm run lint + + - run: git diff --exit-code + + - name: Error message + if: ${{ failure() }} + run: | + echo 'ESLint check is failing. Please run `npm run lint` on your working copy and commit the changes.' diff --git a/.husky/pre-commit b/.husky/pre-commit index d4a43dd13..14ccce4a2 100755 --- a/.husky/pre-commit +++ b/.husky/pre-commit @@ -1,4 +1,4 @@ #!/usr/bin/env sh . "$(dirname -- "$0")/_/husky.sh" -npm run pre-commit +npm run pre-commit \ No newline at end of file diff --git a/migration/migrate.py b/migration/migrate.py index 79c958304..6106730e2 100644 --- a/migration/migrate.py +++ b/migration/migrate.py @@ -2,116 +2,126 @@ import json import os -DEST_WEBAPP_FILE_PATH = 'rudder-webapp/src/components/destinations/schemas/' -DEST_CONFIG_BE_FILE_PATH = 'rudder-config-backend/src/scripts/destinationConfigs/' -DEST_SCHEMA_FILE_PATH = 'rudder-config-backend/src/scripts/schemaList/destinations/' +DEST_WEBAPP_FILE_PATH = "rudder-webapp/src/components/destinations/schemas/" +DEST_CONFIG_BE_FILE_PATH = "rudder-config-backend/src/scripts/destinationConfigs/" +DEST_SCHEMA_FILE_PATH = "rudder-config-backend/src/scripts/schemaList/destinations/" -SOURCE_WEBAPP_FILE_PATH = 'rudder-webapp/src/components/sources/source/warehouseSource/warehouseSourceList/' -SOURCE_CONFIG_BE_FILE_PATH = 'rudder-config-backend/src/scripts/sourceConfigs/' +SOURCE_WEBAPP_FILE_PATH = ( + "rudder-webapp/src/components/sources/source/warehouseSource/warehouseSourceList/" +) +SOURCE_CONFIG_BE_FILE_PATH = "rudder-config-backend/src/scripts/sourceConfigs/" # SOURCE_SCHEMA_FILE_PATH = 'rudder-config-backend/src/scripts/schemaList/destinations/' -CONFIGURATIONS_DIR_PATH = '../src/configurations' -DEST_CONFIG_PATH = f'{CONFIGURATIONS_DIR_PATH}/destinations' -SRC_CONFIG_PATH = f'{CONFIGURATIONS_DIR_PATH}/sources' +CONFIGURATIONS_DIR_PATH = "../src/configurations" +DEST_CONFIG_PATH = f"{CONFIGURATIONS_DIR_PATH}/destinations" +SRC_CONFIG_PATH = f"{CONFIGURATIONS_DIR_PATH}/sources" + def update_destination(): - dest_list = [d[:-5].lower() for d in os.listdir(f'../../{DEST_CONFIG_BE_FILE_PATH}')] + dest_list = [ + d[:-5].lower() for d in os.listdir(f"../../{DEST_CONFIG_BE_FILE_PATH}") + ] for dest in dest_list: final_data = [] # read db_config - with open(f'../../{DEST_CONFIG_BE_FILE_PATH}{dest.upper()}.json', 'r') as f: + with open(f"../../{DEST_CONFIG_BE_FILE_PATH}{dest.upper()}.json", "r") as f: db_config = json.loads(f.read()) - print (db_config) + print(db_config) final_data.append(db_config) - print ("========================") + print("========================") # read ui_config - with open(f'../../{DEST_WEBAPP_FILE_PATH}{dest.upper()}.json', 'r') as f: + with open(f"../../{DEST_WEBAPP_FILE_PATH}{dest.upper()}.json", "r") as f: ui_config = {"uiConfig": json.loads(f.read())} - print (ui_config) + print(ui_config) final_data.append(ui_config) - print ("========================") + print("========================") # read schema - if f'{dest.upper()}.json' in os.listdir(f'../../{DEST_SCHEMA_FILE_PATH}'): - with open(f'../../{DEST_SCHEMA_FILE_PATH}{dest.upper()}.json', 'r') as f: + if f"{dest.upper()}.json" in os.listdir(f"../../{DEST_SCHEMA_FILE_PATH}"): + with open(f"../../{DEST_SCHEMA_FILE_PATH}{dest.upper()}.json", "r") as f: schema = {"schema": json.loads(f.read())} else: schema = {"schema": None} - print (schema) + print(schema) final_data.append(schema) - print ("========================") + print("========================") # metadata - with open('dest_meta_template.json', 'r') as f: + with open("dest_meta_template.json", "r") as f: metadata = json.loads(f.read()) - print (metadata) + print(metadata) final_data.append(metadata) - print ("========================") + print("========================") ######################## ## write new files ######################## if dest not in os.listdir(DEST_CONFIG_PATH): - os.system(f'mkdir {DEST_CONFIG_PATH}/{dest}') - print (f'created directory for {dest}') + os.system(f"mkdir {DEST_CONFIG_PATH}/{dest}") + print(f"created directory for {dest}") - file_names = ['db-config', 'ui-config', 'schema', 'metadata'] + file_names = ["db-config", "ui-config", "schema", "metadata"] for index, f_name in enumerate(file_names): - print (f'writing {f_name} for {dest}...') - with open(f'{DEST_CONFIG_PATH}/{dest}/{f_name}.json', 'w') as f: + print(f"writing {f_name} for {dest}...") + with open(f"{DEST_CONFIG_PATH}/{dest}/{f_name}.json", "w") as f: f.write(json.dumps(final_data[index], indent=2)) - print (f'complete for {dest}...') - print ('------------------------------------------------') + print(f"complete for {dest}...") + print("------------------------------------------------") + def update_source(): - source_list = [d[:-5].lower() for d in os.listdir(f'../../{SOURCE_CONFIG_BE_FILE_PATH}')] + source_list = [ + d[:-5].lower() for d in os.listdir(f"../../{SOURCE_CONFIG_BE_FILE_PATH}") + ] for source in source_list: final_data = [] # read db_config - with open(f'../../{SOURCE_CONFIG_BE_FILE_PATH}{source.upper()}.json', 'r') as f: + with open(f"../../{SOURCE_CONFIG_BE_FILE_PATH}{source.upper()}.json", "r") as f: db_config = json.loads(f.read()) - print (db_config) + print(db_config) final_data.append(db_config) - print ("========================") + print("========================") # read ui_config - if f'{source.upper()}.json' in os.listdir(f'../../{SOURCE_WEBAPP_FILE_PATH}'): - with open(f'../../{SOURCE_WEBAPP_FILE_PATH}{source.upper()}.json', 'r') as f: + if f"{source.upper()}.json" in os.listdir(f"../../{SOURCE_WEBAPP_FILE_PATH}"): + with open( + f"../../{SOURCE_WEBAPP_FILE_PATH}{source.upper()}.json", "r" + ) as f: ui_config = {"uiConfig": json.loads(f.read())} else: ui_config = {"uiConfig": None} - print (ui_config) + print(ui_config) final_data.append(ui_config) - print ("========================") + print("========================") # read schema (we don't need schema for sources) # if f'{source.upper()}.json' in os.listdir(f'../../{SOURCE_SCHEMA_FILE_PATH}'): # with open(f'../../{SOURCE_SCHEMA_FILE_PATH}{source.upper()}.json', 'r') as f: # schema = {"schema": json.loads(f.read())} # else: schema = {"schema": None} - print (schema) + print(schema) final_data.append(schema) - print ("========================") + print("========================") # metadata - with open('source_meta_template.json', 'r') as f: + with open("source_meta_template.json", "r") as f: metadata = json.loads(f.read()) - print (metadata) + print(metadata) final_data.append(metadata) - print ("========================") + print("========================") ######################## ## write new files ######################## if source not in os.listdir(SRC_CONFIG_PATH): - os.system(f'mkdir {SRC_CONFIG_PATH}/{source}') - print (f'created directory for {source}') + os.system(f"mkdir {SRC_CONFIG_PATH}/{source}") + print(f"created directory for {source}") - file_names = ['db_config', 'ui_config', 'schema', 'metadata'] + file_names = ["db_config", "ui_config", "schema", "metadata"] for index, f_name in enumerate(file_names): - print (f'writing {f_name} for {source}...') - with open(f'{SRC_CONFIG_PATH}/{source}/{f_name}.json', 'w') as f: + print(f"writing {f_name} for {source}...") + with open(f"{SRC_CONFIG_PATH}/{source}/{f_name}.json", "w") as f: f.write(json.dumps(final_data[index], indent=2)) - print (f'complete for {source}...') - print ('------------------------------------------------') + print(f"complete for {source}...") + print("------------------------------------------------") if __name__ == "__main__": diff --git a/package.json b/package.json index 534f118ac..167e25fd0 100755 --- a/package.json +++ b/package.json @@ -21,8 +21,10 @@ "test:silent": "npm run test -- --silent", "release": "npx standard-version", "check:lint": "eslint \"src/**/*.*\" -f json -o reports/eslint.json || exit 0", + "format:py": "python3 -m black .", "format": "prettier . --write", "lint:fix": "eslint \"src/**/*.*\" --fix", + "lint": "npm run format && npm run lint:fix", "prepare": "husky install", "pre-commit": "npm run test && npx lint-staged", "commit-msg": "commitlint --edit", @@ -79,7 +81,8 @@ "glob": "^9.3.2" }, "lint-staged": { - "*.{json,js,ts,md}": "prettier --write" + "*.{json,js,ts,md}": "prettier --write", + "*.{py}": "python3 -m black" }, "config": { "commitizen": { diff --git a/scripts/configGenerator.py b/scripts/configGenerator.py index 1a1fdaf4d..f7e38ba0c 100644 --- a/scripts/configGenerator.py +++ b/scripts/configGenerator.py @@ -4,23 +4,24 @@ import os import sys -ConfigData = TypedDict('ConfigData', {'db_config': str, 'ui_config': str}) +ConfigData = TypedDict("ConfigData", {"db_config": str, "ui_config": str}) + def generateConfigs(data) -> ConfigData: # Read the content of template-db-config.json - with open('scripts/template-db-config.json', 'r') as file: + with open("scripts/template-db-config.json", "r") as file: template_db_config = json.load(file) # Read the content of template-ui-config.json - with open('scripts/template-ui-config.json', 'r') as file: + with open("scripts/template-ui-config.json", "r") as file: template_ui_config = json.load(file) # Create db-config object with the same content db_config = template_db_config - db_config['displayName'] = data['displayName'] - db_config['name'] = data['displayName'] - formFields = data['formFields'] + db_config["displayName"] = data["displayName"] + db_config["name"] = data["displayName"] + formFields = data["formFields"] # Create db-config object with the same content ui_config = template_ui_config @@ -33,14 +34,13 @@ def appendFieldsInGroups(settings, field): if groups: first_group = groups[0] if "fields" in first_group: - del field['required'] + del field["required"] first_group["fields"].append(field) - def updateUiConfig(field): if "uiConfig" in ui_config and "baseTemplate" in ui_config["uiConfig"]: base_template = ui_config["uiConfig"]["baseTemplate"] - if base_template and field['required'] == True: + if base_template and field["required"] == True: connection_settings = base_template[0] appendFieldsInGroups(connection_settings, field) elif base_template: @@ -48,28 +48,27 @@ def updateUiConfig(field): appendFieldsInGroups(configuration_settings, field) return ui_config - # Iterate over JSON objects in the array for obj in formFields: # update field in ui-config ui_config = updateUiConfig(obj) # update db-config for key, value in obj.items(): - if key == 'configKey': - db_config['config']['destConfig']['defaultConfig'].append( - value) - if key == 'secret' and value == True: - db_config['config']['secretKeys'].append( - db_config['config']['destConfig']['defaultConfig'][-1]) + if key == "configKey": + db_config["config"]["destConfig"]["defaultConfig"].append(value) + if key == "secret" and value == True: + db_config["config"]["secretKeys"].append( + db_config["config"]["destConfig"]["defaultConfig"][-1] + ) db_config = json.dumps(db_config) ui_config = json.dumps(ui_config) - return {'db_config':db_config, 'ui_config': ui_config} + return {"db_config": db_config, "ui_config": ui_config} -if __name__ == '__main__': - file_path = sys.argv[1] if len(sys.argv) > 1 else 'test/configData/inputData.json' - with open(file_path, 'r') as file: +if __name__ == "__main__": + file_path = sys.argv[1] if len(sys.argv) > 1 else "test/configData/inputData.json" + with open(file_path, "r") as file: # Load the JSON data data = json.load(file) @@ -79,16 +78,15 @@ def updateUiConfig(field): if not os.path.exists(directory): os.makedirs(directory) - with open(file_path, 'w') as file: + with open(file_path, "w") as file: # Write the new content - file.write(configData['db_config']) - + file.write(configData["db_config"]) file_path = f'src/configurations/destinations/{data["displayName"]}/ui-config.json' directory = os.path.dirname(file_path) if not os.path.exists(directory): os.makedirs(directory) - with open(file_path, 'w') as file: + with open(file_path, "w") as file: # Write the new content - file.write(configData['ui_config']) + file.write(configData["ui_config"]) diff --git a/scripts/constants.py b/scripts/constants.py index 54b717202..5d9c7b926 100644 --- a/scripts/constants.py +++ b/scripts/constants.py @@ -1,11 +1,11 @@ import os -__DEFAULT_CONFIG_DIR = 'src/configurations' +__DEFAULT_CONFIG_DIR = "src/configurations" CONFIG_DIR = __DEFAULT_CONFIG_DIR -if 'CONFIG_DIR' in os.environ: - CONFIG_DIR = os.environ['CONFIG_DIR'] +if "CONFIG_DIR" in os.environ: + CONFIG_DIR = os.environ["CONFIG_DIR"] -TEST_INTEGRATION_NAME_PREFIX = 'test_' +TEST_INTEGRATION_NAME_PREFIX = "test_" -TEST_INTEGRATION_NAME_SUFFIX = '_ignore' +TEST_INTEGRATION_NAME_SUFFIX = "_ignore" diff --git a/scripts/deployToDB.py b/scripts/deployToDB.py index 6c513fb92..97b4b3d00 100644 --- a/scripts/deployToDB.py +++ b/scripts/deployToDB.py @@ -18,12 +18,12 @@ ######################### # ENV VARIABLES -CONTROL_PLANE_URL=sys.argv[1] +CONTROL_PLANE_URL = sys.argv[1] print(CONTROL_PLANE_URL) -USERNAME=os.environ['API_USER'] #sys.argv[2] +USERNAME = os.environ["API_USER"] # sys.argv[2] print(USERNAME) -PASSWORD=os.environ['API_PASSWORD'] #sys.argv[3] -#print(PASSWORD) +PASSWORD = os.environ["API_PASSWORD"] # sys.argv[3] +# print(PASSWORD) ######################### # CONSTANTS HEADER = {"Content-Type": "application/json"} @@ -34,131 +34,152 @@ ######################### # UTIL METHODS def parse_response(resp): - if resp.status_code >= 200 and resp.status_code <= 300: - return resp.status_code, resp.json() - else: - return resp.status_code, str(resp.content) + if resp.status_code >= 200 and resp.status_code <= 300: + return resp.status_code, resp.json() + else: + return resp.status_code, str(resp.content) + def get_persisted_store(base_url, selector): - request_url = f'{base_url}/{selector}-definitions' + request_url = f"{base_url}/{selector}-definitions" response = requests.get(request_url) return json.loads(response.text) + def get_config_definition(base_url, selector, name): - request_url = f'{base_url}/{selector}-definitions/{name}' + request_url = f"{base_url}/{selector}-definitions/{name}" response = requests.get(request_url) return response + def get_file_content(name, selector): - file_selectors = ['db-config.json', 'ui-config.json', 'schema.json'] + file_selectors = ["db-config.json", "ui-config.json", "schema.json"] - directory = f'./{CONFIG_DIR}/{selector}s/{name}' + directory = f"./{CONFIG_DIR}/{selector}s/{name}" available_files = os.listdir(directory) file_content = {} for file_selector in file_selectors: if file_selector in available_files: - with open (f'{directory}/{file_selector}', 'r') as f: + with open(f"{directory}/{file_selector}", "r") as f: file_content.update(json.loads(f.read())) return file_content + def update_config_definition(selector, name, fileData): - url = f'{CONTROL_PLANE_URL}/{selector}-definitions/{name}' + url = f"{CONTROL_PLANE_URL}/{selector}-definitions/{name}" resp = requests.post(url=url, headers=HEADER, data=json.dumps(fileData), auth=AUTH) return parse_response(resp) + def create_config_definition(selector, fileData): - url = f'{CONTROL_PLANE_URL}/{selector}-definitions/' + url = f"{CONTROL_PLANE_URL}/{selector}-definitions/" resp = requests.post(url=url, headers=HEADER, data=json.dumps(fileData), auth=AUTH) return parse_response(resp) + def update_config(data_diff, selector): results = [] for diff in data_diff: - name = diff['name'] + name = diff["name"] fileData = get_file_content(name, selector) nameInConfig = fileData["name"] - if diff['action'] == 'create': - url = f'{CONTROL_PLANE_URL}/{selector}-definitions' + if diff["action"] == "create": + url = f"{CONTROL_PLANE_URL}/{selector}-definitions" else: - url = f'{CONTROL_PLANE_URL}/{selector}-definitions/{nameInConfig}' + url = f"{CONTROL_PLANE_URL}/{selector}-definitions/{nameInConfig}" - resp = requests.post(url=url, headers=HEADER, data=json.dumps(fileData), auth=AUTH) + resp = requests.post( + url=url, headers=HEADER, data=json.dumps(fileData), auth=AUTH + ) status, response = parse_response(resp) - diff['update'] = {"status": status, "response": response} + diff["update"] = {"status": status, "response": response} # results.append(diff) results.append(name) - return json.dumps(results, indent=2) + def update_diff_db(selector): final_report = [] ## data sets - current_items = os.listdir(f'./{CONFIG_DIR}/{selector}s') + current_items = os.listdir(f"./{CONFIG_DIR}/{selector}s") for item in current_items: # check if item is a directory - if not os.path.isdir(f'./{CONFIG_DIR}/{selector}s/{item}'): + if not os.path.isdir(f"./{CONFIG_DIR}/{selector}s/{item}"): continue updated_data = get_file_content(item, selector) - persisted_data = get_config_definition(CONTROL_PLANE_URL, selector, updated_data["name"]) + persisted_data = get_config_definition( + CONTROL_PLANE_URL, selector, updated_data["name"] + ) if persisted_data.status_code == 200: - diff = jsondiff.diff(json.loads(persisted_data.text), updated_data, marshal=True) + diff = jsondiff.diff( + json.loads(persisted_data.text), updated_data, marshal=True + ) # ignore the $delete - values present in DB but missing in files. Anyways this doesn't get reflected in DB as keys are missing in files itself. # Best practice is to make sure all keys are maintained in the config files irrespective of them being null. - del diff['$delete'] - - if len(diff.keys()) > 0: # changes exist - #print(diff) - status, response = update_config_definition(selector, updated_data["name"], updated_data) - final_report.append({"name": updated_data["name"], "action":"update", "status": status}) + del diff["$delete"] + + if len(diff.keys()) > 0: # changes exist + # print(diff) + status, response = update_config_definition( + selector, updated_data["name"], updated_data + ) + final_report.append( + {"name": updated_data["name"], "action": "update", "status": status} + ) else: - final_report.append({"name": updated_data["name"], "action":"na", "status": ""}) + final_report.append( + {"name": updated_data["name"], "action": "na", "status": ""} + ) else: status, response = create_config_definition(selector, updated_data) - final_report.append({"name": updated_data["name"], "action":"create", "status": status}) + final_report.append( + {"name": updated_data["name"], "action": "create", "status": status} + ) return final_report + def get_stale_data(selector, report): stale_config_report = [] persisted_data_set = get_persisted_store(CONTROL_PLANE_URL, selector) - persisted_items = [item['name'] for item in persisted_data_set] - file_items = [item['name'] for item in report] + persisted_items = [item["name"] for item in persisted_data_set] + file_items = [item["name"] for item in report] for item in persisted_items: if item not in file_items: - stale_config_report.append({item}) + stale_config_report.append({item}) return stale_config_report -if __name__ == '__main__': + +if __name__ == "__main__": print("Running Destination Definitions Updates") - dest_final_report = update_diff_db('destination') + dest_final_report = update_diff_db("destination") print("Destination Definition Update Report") print(dest_final_report) print("Destination Stale Config Report") - print(get_stale_data('destination', dest_final_report)) + print(get_stale_data("destination", dest_final_report)) print("Running Source Definitions Updates") - src_final_report = update_diff_db('source') + src_final_report = update_diff_db("source") print("Source Definition Update Report") print(src_final_report) print("Source Stale Config Report") - print(get_stale_data('source', src_final_report)) + print(get_stale_data("source", src_final_report)) print("Running Wht Lib Projects Definitions Updates") - wht_final_report = update_diff_db('wht-lib-project') + wht_final_report = update_diff_db("wht-lib-project") print("Wht lib project Definition Update Report") print(wht_final_report) print("Wht lib project Stale Config Report") - print(get_stale_data('wht-lib-project', wht_final_report)) - \ No newline at end of file + print(get_stale_data("wht-lib-project", wht_final_report)) diff --git a/scripts/schemaGenerator.py b/scripts/schemaGenerator.py index 36d9d9ee3..60f97a0cf 100644 --- a/scripts/schemaGenerator.py +++ b/scripts/schemaGenerator.py @@ -1,4 +1,4 @@ -''' +""" Usage: schemaGenerator.py [-h] [-name name | -all] [-update] selector 1. selector - “source” or “destination” 2. all - runs the validator for all the selector. @@ -7,7 +7,8 @@ Example: 1. python3 scripts/schemaGenerator.py -name="adobe_analytics" destination 2. python3 scripts/schemaGenerator.py -all source -''' +""" + import os import warnings from enum import Enum @@ -15,7 +16,8 @@ from utils import get_json_from_file, get_json_diff, apply_json_diff, get_formatted_json from constants import CONFIG_DIR -EXCLUDED_DEST = ['postgres', 'bq', 'azure_synapse', 'clickhouse', 'deltalake', 'kafka'] +EXCLUDED_DEST = ["postgres", "bq", "azure_synapse", "clickhouse", "deltalake", "kafka"] + class FieldTypeEnum(Enum): STRING = "string" @@ -29,13 +31,14 @@ def is_old_format(uiConfig): return False return True + def get_options_list_for_enum(field): """Creates the list of options given in field and return the list Args: field (object): Individual field in ui-config. Returns: list: list of options - """ + """ options_list = [] for i in range(0, len(field["options"])): if isinstance(field["options"][i], int) or isinstance(field["options"][i], str): @@ -43,14 +46,19 @@ def get_options_list_for_enum(field): else: options_list.append(field["options"][i]["value"]) # allow empty field in enum if field in not required. - if "default" not in field and "defaultOption" not in field and field.get("required", False) == False: - options_list.append("") + if ( + "default" not in field + and "defaultOption" not in field + and field.get("required", False) == False + ): + options_list.append("") return options_list + def generalize_regex_pattern(field): """Generates the pattern for schema based on the type of field. - For type : singleSelect and dynamicSelectForm, the pattern is generated by iterating over options. - - For other types, + - For other types, - If the field contains regex, then regex is the pattern; it gets prefixed with a default prefix if regex does not have it. - Else, the default prefix is appended with ^(.{0,100}). @@ -59,88 +67,116 @@ def generalize_regex_pattern(field): Returns: string: generated pattern for the field. - """ + """ defaultSubPattern = "(^\\{\\{.*\\|\\|(.*)\\}\\}$)" defaultEnvPattern = "(^env[.].+)" pattern = "" if "regex" in field: pattern = field["regex"] - if defaultSubPattern not in pattern and (('value' not in field or field['value'] != 'purpose') and ('configKey' not in field or field['configKey'] != 'purpose')): + if defaultSubPattern not in pattern and ( + ("value" not in field or field["value"] != "purpose") + and ("configKey" not in field or field["configKey"] != "purpose") + ): pattern = "|".join([defaultSubPattern, pattern]) - if defaultEnvPattern not in pattern and (('value' not in field or field['value'] != 'purpose') and ('configKey' not in field or field['configKey'] != 'purpose')): + if defaultEnvPattern not in pattern and ( + ("value" not in field or field["value"] != "purpose") + and ("configKey" not in field or field["configKey"] != "purpose") + ): indexToPlace = pattern.find(defaultSubPattern) + len(defaultSubPattern) - pattern = pattern[:indexToPlace] + '|' + defaultEnvPattern + pattern[indexToPlace:] + pattern = ( + pattern[:indexToPlace] + + "|" + + defaultEnvPattern + + pattern[indexToPlace:] + ) # TODO: we should not use a case here for the individual properties. Just pass the desired pattern as regex property # in ketch purpose fields and delete next case - elif ('value' in field and field['value'] == 'purpose') or ('configKey' in field and field['configKey'] == 'purpose'): - pattern = '^(.{0,100})$' + elif ("value" in field and field["value"] == "purpose") or ( + "configKey" in field and field["configKey"] == "purpose" + ): + pattern = "^(.{0,100})$" else: - pattern = "|".join([defaultSubPattern, defaultEnvPattern, '^(.{0,100})$']) + pattern = "|".join([defaultSubPattern, defaultEnvPattern, "^(.{0,100})$"]) return pattern def is_dest_field_dependent_on_source(field, dbConfig, schema_field_name): - """Checks if the given field is source-specific by using dbConfig. - In dbConfig all the sources are listed in 'supportedSourceTypes', + """Checks if the given field is source-specific by using dbConfig. + In dbConfig all the sources are listed in 'supportedSourceTypes', and their fields are listed inside 'destConfig' with the key as the source. Args: field (object): Individual field in ui-config. dbConfig (object): Configurations of db-config.json. - schema_field_name (string): Specifies which key has the field's name in schema. + schema_field_name (string): Specifies which key has the field's name in schema. For old schema types, it is 'value' else 'configKey'. Returns: boolean: True if the field is source dependent else, False. - """ + """ if not dbConfig: return False for sourceType in dbConfig["supportedSourceTypes"]: - if sourceType in dbConfig["destConfig"] and field[schema_field_name] in dbConfig["destConfig"][sourceType]: + if ( + sourceType in dbConfig["destConfig"] + and field[schema_field_name] in dbConfig["destConfig"][sourceType] + ): return True return False + def is_field_present_in_default_config(field, dbConfig, schema_field_name): """Checks if the given field is present in defaultConfig list present in dbConfig. Args: field (object): Individual field in ui-config. dbConfig (object): Configurations of db-config.json. - schema_field_name (string): Specifies which key has the field's name in schema. + schema_field_name (string): Specifies which key has the field's name in schema. For old schema types, it is 'value' else 'configKey'. Returns: boolean: True if field is in defaultConfig else False. - """ + """ if not dbConfig: return False - if "destConfig" in dbConfig and "defaultConfig" in dbConfig["destConfig"] and field[schema_field_name] in dbConfig["destConfig"]["defaultConfig"]: + if ( + "destConfig" in dbConfig + and "defaultConfig" in dbConfig["destConfig"] + and field[schema_field_name] in dbConfig["destConfig"]["defaultConfig"] + ): return True return False + def generate_schema_for_default_checkbox(field, dbConfig, schema_field_name): """Creates a schema object of defaultCheckbox. Args: field (object): Individual field in ui-config. dbConfig (object): Configurations of db-config.json. - schema_field_name (string): Specifies which key has the field's name in schema. + schema_field_name (string): Specifies which key has the field's name in schema. For old schema types, it is 'value' else 'configKey'. Returns: object - """ - isSourceDependent = is_dest_field_dependent_on_source(field, dbConfig, schema_field_name) + """ + isSourceDependent = is_dest_field_dependent_on_source( + field, dbConfig, schema_field_name + ) defaultCheckboxSchemaObj = {} if isSourceDependent: defaultCheckboxSchemaObj["type"] = FieldTypeEnum.OBJECT.value defaultCheckboxSchemaObj["properties"] = {} # iterates over supported sources and sets the field for that source if field is present inside that source for sourceType in dbConfig["supportedSourceTypes"]: - if sourceType in dbConfig["destConfig"] and field[schema_field_name] in dbConfig["destConfig"][sourceType]: + if ( + sourceType in dbConfig["destConfig"] + and field[schema_field_name] in dbConfig["destConfig"][sourceType] + ): defaultCheckboxSchemaObj["properties"][sourceType] = { - "type": FieldTypeEnum.BOOLEAN.value} + "type": FieldTypeEnum.BOOLEAN.value + } else: defaultCheckboxSchemaObj["type"] = FieldTypeEnum.BOOLEAN.value if "default" in field: @@ -154,22 +190,28 @@ def generate_schema_for_checkbox(field, dbConfig, schema_field_name): Args: field (object): Individual field in ui-config. dbConfig (object): Configurations of db-config.json. - schema_field_name (string): Specifies which key has the field's name in schema. + schema_field_name (string): Specifies which key has the field's name in schema. For old schema types, it is 'value' else 'configKey'. Returns: object """ - isSourceDependent = is_dest_field_dependent_on_source(field, dbConfig, schema_field_name) + isSourceDependent = is_dest_field_dependent_on_source( + field, dbConfig, schema_field_name + ) checkboxSchemaObj = {} if isSourceDependent: checkboxSchemaObj["type"] = FieldTypeEnum.OBJECT.value checkboxSchemaObj["properties"] = {} # iterates over supported sources and sets the field for that source if field is present inside that source for sourceType in dbConfig["supportedSourceTypes"]: - if sourceType in dbConfig["destConfig"] and field[schema_field_name] in dbConfig["destConfig"][sourceType]: + if ( + sourceType in dbConfig["destConfig"] + and field[schema_field_name] in dbConfig["destConfig"][sourceType] + ): checkboxSchemaObj["properties"][sourceType] = { - "type": FieldTypeEnum.BOOLEAN.value} + "type": FieldTypeEnum.BOOLEAN.value + } else: checkboxSchemaObj["type"] = FieldTypeEnum.BOOLEAN.value if "default" in field: @@ -183,27 +225,35 @@ def generate_schema_for_textinput(field, dbConfig, schema_field_name): Args: field (object): Individual field in ui-config. dbConfig (object): Configurations of db-config.json. - schema_field_name (string): Specifies which key has the field's name in schema. + schema_field_name (string): Specifies which key has the field's name in schema. For old schema types, it is 'value' else 'configKey'. Returns: object """ textInputSchemaObj = {} - isSourceDependent = is_dest_field_dependent_on_source(field, dbConfig, schema_field_name) + isSourceDependent = is_dest_field_dependent_on_source( + field, dbConfig, schema_field_name + ) if isSourceDependent: textInputSchemaObj["type"] = FieldTypeEnum.OBJECT.value textInputSchemaObj["properties"] = {} # iterates over supported sources and sets the field for that source if field is present inside that source for sourceType in dbConfig["supportedSourceTypes"]: - if sourceType in dbConfig["destConfig"] and field[schema_field_name] in dbConfig["destConfig"][sourceType]: + if ( + sourceType in dbConfig["destConfig"] + and field[schema_field_name] in dbConfig["destConfig"][sourceType] + ): textInputSchemaObj["properties"][sourceType] = { - "type": FieldTypeEnum.STRING.value} - if 'regex' in field: - textInputSchemaObj["properties"][sourceType]["pattern"] = generalize_regex_pattern(field) + "type": FieldTypeEnum.STRING.value + } + if "regex" in field: + textInputSchemaObj["properties"][sourceType]["pattern"] = ( + generalize_regex_pattern(field) + ) else: textInputSchemaObj = {"type": FieldTypeEnum.STRING.value} - if 'regex' in field: + if "regex" in field: textInputSchemaObj["pattern"] = generalize_regex_pattern(field) return textInputSchemaObj @@ -214,14 +264,14 @@ def generate_schema_for_textarea_input(field, dbConfig, schema_field_name): Args: field (object): Individual field in ui-config. dbConfig (object): Configurations of db-config.json. - schema_field_name (string): Specifies which key has the field's name in schema. + schema_field_name (string): Specifies which key has the field's name in schema. For old schema types, it is 'value' else 'configKey'. Returns: object """ textareaInputObj = {"type": FieldTypeEnum.STRING.value} - if 'regex' in field: + if "regex" in field: textareaInputObj["pattern"] = generalize_regex_pattern(field) return textareaInputObj @@ -232,25 +282,25 @@ def generate_schema_for_single_select(field, dbConfig, schema_field_name): Args: field (object): Individual field in ui-config. dbConfig (object): Configurations of db-config.json. - schema_field_name (string): Specifies which key has the field's name in schema. + schema_field_name (string): Specifies which key has the field's name in schema. For old schema types, it is 'value' else 'configKey'. Returns: object """ singleSelectObj = {} - if "mode" in field and field["mode"] == 'multiple': - singleSelectObj = {"type": FieldTypeEnum.ARRAY.value} + if "mode" in field and field["mode"] == "multiple": + singleSelectObj = {"type": FieldTypeEnum.ARRAY.value} singleSelectObj["items"] = { "type": FieldTypeEnum.STRING.value, - "enum": get_options_list_for_enum(field) + "enum": get_options_list_for_enum(field), } if "default" or "defaultOption" in field: if isinstance(field["defaultOption"]["value"], list): singleSelectObj["default"] = field["defaultOption"]["value"] elif field["defaultOption"]["value"]: singleSelectObj["default"] = [field["defaultOption"]["value"]] - elif 'default' in field: + elif "default" in field: singleSelectObj["default"] = field["default"] else: singleSelectObj = {"type": FieldTypeEnum.STRING.value} @@ -258,16 +308,21 @@ def generate_schema_for_single_select(field, dbConfig, schema_field_name): if "default" or "defaultOption" in field: if "defaultOption" in field: singleSelectObj["default"] = field["defaultOption"]["value"] - elif 'default' in field: + elif "default" in field: singleSelectObj["default"] = field["default"] - isSourceDependent = is_dest_field_dependent_on_source(field, dbConfig, schema_field_name) + isSourceDependent = is_dest_field_dependent_on_source( + field, dbConfig, schema_field_name + ) if isSourceDependent: newSingleSelectObj = {"type": FieldTypeEnum.OBJECT.value} newSingleSelectObj["properties"] = {} # iterates over supported sources and sets the field for that source if field is present inside that source for sourceType in dbConfig["supportedSourceTypes"]: - if sourceType in dbConfig["destConfig"] and field[schema_field_name] in dbConfig["destConfig"][sourceType]: + if ( + sourceType in dbConfig["destConfig"] + and field[schema_field_name] in dbConfig["destConfig"][sourceType] + ): newSingleSelectObj["properties"][sourceType] = singleSelectObj singleSelectObj = newSingleSelectObj return singleSelectObj @@ -279,7 +334,7 @@ def generate_schema_for_dynamic_custom_form(field, dbConfig, schema_field_name): Args: field (object): Individual field in ui-config. dbConfig (object): Configurations of db-config.json. - schema_field_name (string): Specifies which key has the field's name in schema. + schema_field_name (string): Specifies which key has the field's name in schema. For old schema types, it is 'value' else 'configKey'. Returns: @@ -297,46 +352,71 @@ def generate_schema_for_dynamic_custom_form(field, dbConfig, schema_field_name): if "rowFields" in field: customFieldsKey = "rowFields" - allOfSchemaObj = generate_schema_for_dynamic_custom_form_allOf(field[customFieldsKey], dbConfig, schema_field_name) + allOfSchemaObj = generate_schema_for_dynamic_custom_form_allOf( + field[customFieldsKey], dbConfig, schema_field_name + ) for customField in field[customFieldsKey]: - customFieldSchemaObj = uiTypetoSchemaFn.get(customField["type"])(customField, dbConfig, schema_field_name) - isCustomFieldDependentOnSource = is_dest_field_dependent_on_source(customField, dbConfig, schema_field_name) + customFieldSchemaObj = uiTypetoSchemaFn.get(customField["type"])( + customField, dbConfig, schema_field_name + ) + isCustomFieldDependentOnSource = is_dest_field_dependent_on_source( + customField, dbConfig, schema_field_name + ) if "preRequisites" in customField: continue - if 'pattern' not in customFieldSchemaObj and not isCustomFieldDependentOnSource and customFieldSchemaObj["type"] == FieldTypeEnum.STRING.value and customField["type"] != "singleSelect" and customField["type"] != "dynamicSelectForm": + if ( + "pattern" not in customFieldSchemaObj + and not isCustomFieldDependentOnSource + and customFieldSchemaObj["type"] == FieldTypeEnum.STRING.value + and customField["type"] != "singleSelect" + and customField["type"] != "dynamicSelectForm" + ): customFieldSchemaObj["pattern"] = generalize_regex_pattern(customField) # If the custom field is source dependent, we remove the source keys as it's not required inside custom fields, rather they need to be moved to top. if isCustomFieldDependentOnSource: for sourceType in dbConfig["supportedSourceTypes"]: - if sourceType in dbConfig["destConfig"] and field[schema_field_name] in dbConfig["destConfig"][sourceType]: - customFieldSchemaObj = customFieldSchemaObj["properties"][sourceType] + if ( + sourceType in dbConfig["destConfig"] + and field[schema_field_name] in dbConfig["destConfig"][sourceType] + ): + customFieldSchemaObj = customFieldSchemaObj["properties"][ + sourceType + ] break - dynamicCustomFormItemObj["properties"][customField[schema_field_name]] = customFieldSchemaObj + dynamicCustomFormItemObj["properties"][ + customField[schema_field_name] + ] = customFieldSchemaObj if allOfSchemaObj: - dynamicCustomFormItemObj['allOf'] = allOfSchemaObj + dynamicCustomFormItemObj["allOf"] = allOfSchemaObj dynamicCustomFormObj["items"] = dynamicCustomFormItemObj - isSourceDependent = is_dest_field_dependent_on_source(field, dbConfig, schema_field_name) + isSourceDependent = is_dest_field_dependent_on_source( + field, dbConfig, schema_field_name + ) # If the field is source dependent, new schema object is created by setting the fields inside the source. if isSourceDependent: newDynamicCustomFormObj = {"type": FieldTypeEnum.OBJECT.value} newDynamicCustomFormObj["properties"] = {} for sourceType in dbConfig["supportedSourceTypes"]: - if sourceType in dbConfig["destConfig"] and field[schema_field_name] in dbConfig["destConfig"][sourceType]: + if ( + sourceType in dbConfig["destConfig"] + and field[schema_field_name] in dbConfig["destConfig"][sourceType] + ): newDynamicCustomFormObj["properties"][sourceType] = dynamicCustomFormObj dynamicCustomFormObj = newDynamicCustomFormObj return dynamicCustomFormObj - -def generate_schema_for_dynamic_custom_form_allOf(customFields, dbConfig, schema_field_name): +def generate_schema_for_dynamic_custom_form_allOf( + customFields, dbConfig, schema_field_name +): """Creates the allOf structure of schema, empty if not required. - Finds the list of unique preRequisites. - For each unique preRequisites, the properties are found by matching the current preRequisites. @@ -359,7 +439,9 @@ def generate_schema_for_dynamic_custom_form_allOf(customFields, dbConfig, schema continue isPresent = False for preRequisites in preRequisitesList: - if compare_pre_requisite_fields(preRequisites, field["preRequisites"]["fields"], True): + if compare_pre_requisite_fields( + preRequisites, field["preRequisites"]["fields"], True + ): isPresent = True break if not isPresent: @@ -373,8 +455,12 @@ def generate_schema_for_dynamic_custom_form_allOf(customFields, dbConfig, schema for field in customFields: if "preRequisites" not in field: continue - if compare_pre_requisite_fields(field["preRequisites"]["fields"], preRequisites, True): - thenObj["properties"][field[schema_field_name]] = uiTypetoSchemaFn.get(field["type"])(field, dbConfig, schema_field_name) + if compare_pre_requisite_fields( + field["preRequisites"]["fields"], preRequisites, True + ): + thenObj["properties"][field[schema_field_name]] = uiTypetoSchemaFn.get( + field["type"] + )(field, dbConfig, schema_field_name) if "required" in field and field["required"] == True: thenObj["required"].append(field[schema_field_name]) allOfItemObj["then"] = thenObj @@ -391,18 +477,19 @@ def generate_schema_for_dynamic_form(field, dbConfig, schema_field_name): Args: field (object): Individual field in ui-config. dbConfig (object): Configurations of db-config.json. - schema_field_name (string): Specifies which key has the field's name in schema. + schema_field_name (string): Specifies which key has the field's name in schema. For old schema types, it is 'value' else 'configKey'. Returns: object """ + def generate_key(forFieldWithTo): obj = { "type": FieldTypeEnum.STRING.value, } - if(field["type"] == 'dynamicSelectForm'): - if (forFieldWithTo != (field.get("reverse", False)==False)): + if field["type"] == "dynamicSelectForm": + if forFieldWithTo != (field.get("reverse", False) == False): obj["pattern"] = generalize_regex_pattern(field) else: if "defaultOption" in field: @@ -413,25 +500,35 @@ def generate_key(forFieldWithTo): return obj dynamicFormSchemaObject = {} - dynamicFormSchemaObject['type'] = FieldTypeEnum.ARRAY.value + dynamicFormSchemaObject["type"] = FieldTypeEnum.ARRAY.value dynamicFormItemObject = {} dynamicFormItemObject["type"] = FieldTypeEnum.OBJECT.value - dynamicFormItemObject['properties'] = {} + dynamicFormItemObject["properties"] = {} dynamicFormItemObjectProps = [ - (field['keyLeft'], generate_key), (field['keyRight'], generate_key)] + (field["keyLeft"], generate_key), + (field["keyRight"], generate_key), + ] for dynamicFromItemObjectProp in dynamicFormItemObjectProps: - dynamicFormItemObject['properties'][dynamicFromItemObjectProp[0] - ] = dynamicFromItemObjectProp[1](dynamicFromItemObjectProp[0] == "to") - dynamicFormSchemaObject['items'] = dynamicFormItemObject - - isSourceDependent = is_dest_field_dependent_on_source(field, dbConfig, schema_field_name) + dynamicFormItemObject["properties"][dynamicFromItemObjectProp[0]] = ( + dynamicFromItemObjectProp[1](dynamicFromItemObjectProp[0] == "to") + ) + dynamicFormSchemaObject["items"] = dynamicFormItemObject + + isSourceDependent = is_dest_field_dependent_on_source( + field, dbConfig, schema_field_name + ) # If the field is source dependent, new schema object is created by setting the fields inside the source. if isSourceDependent: newDynamicFormFormObj = {"type": FieldTypeEnum.OBJECT.value} newDynamicFormFormObj["properties"] = {} for sourceType in dbConfig["supportedSourceTypes"]: - if sourceType in dbConfig["destConfig"] and field[schema_field_name] in dbConfig["destConfig"][sourceType]: - newDynamicFormFormObj["properties"][sourceType] = dynamicFormSchemaObject + if ( + sourceType in dbConfig["destConfig"] + and field[schema_field_name] in dbConfig["destConfig"][sourceType] + ): + newDynamicFormFormObj["properties"][ + sourceType + ] = dynamicFormSchemaObject dynamicFormSchemaObject = newDynamicFormFormObj return dynamicFormSchemaObject @@ -442,7 +539,7 @@ def generate_schema_for_dynamic_select_form(field, dbConfig, schema_field_name): Args: field (object): Individual field in ui-config. dbConfig (object): Configurations of db-config.json. - schema_field_name (string): Specifies which key has the field's name in schema. + schema_field_name (string): Specifies which key has the field's name in schema. For old schema types, it is 'value' else 'configKey'. Returns: @@ -450,13 +547,14 @@ def generate_schema_for_dynamic_select_form(field, dbConfig, schema_field_name): """ return generate_schema_for_dynamic_form(field, dbConfig, schema_field_name) + def generate_schema_for_mapping(field, dbConfig, schema_field_name): """Creates a schema object of mapping. Args: field (object): Individual field in ui-config. dbConfig (object): Configurations of db-config.json. - schema_field_name (string): Specifies which key has the field's name in schema. + schema_field_name (string): Specifies which key has the field's name in schema. For old schema types, it is 'value' else 'configKey'. Returns: @@ -471,7 +569,7 @@ def generate_schema_for_tag_input(field, dbConfig, schema_field_name): Args: field (object): Individual field in ui-config. dbConfig (object): Configurations of db-config.json. - schema_field_name (string): Specifies which key has the field's name in schema. + schema_field_name (string): Specifies which key has the field's name in schema. For old schema types, it is 'value' else 'configKey'. Returns: @@ -480,23 +578,28 @@ def generate_schema_for_tag_input(field, dbConfig, schema_field_name): tagObject = {} tagObject["type"] = FieldTypeEnum.ARRAY.value tagItem = {} - tagItem['type'] = FieldTypeEnum.OBJECT.value + tagItem["type"] = FieldTypeEnum.OBJECT.value tagItemProps = { - str(field['tagKey']): { + str(field["tagKey"]): { "type": FieldTypeEnum.STRING.value, - "pattern": generalize_regex_pattern(field) + "pattern": generalize_regex_pattern(field), } } - tagItem['properties'] = tagItemProps + tagItem["properties"] = tagItemProps tagObject["items"] = tagItem - isSourceDependent = is_dest_field_dependent_on_source(field, dbConfig, schema_field_name) + isSourceDependent = is_dest_field_dependent_on_source( + field, dbConfig, schema_field_name + ) if isSourceDependent: tagObjectCopy = tagObject tagObject = {} tagObject = {"type": FieldTypeEnum.OBJECT.value} tagObject["properties"] = {} for sourceType in dbConfig["supportedSourceTypes"]: - if sourceType in dbConfig["destConfig"] and field[schema_field_name] in dbConfig["destConfig"][sourceType]: + if ( + sourceType in dbConfig["destConfig"] + and field[schema_field_name] in dbConfig["destConfig"][sourceType] + ): tagObject["properties"][sourceType] = tagObjectCopy return tagObject @@ -507,20 +610,20 @@ def generate_schema_for_time_range_picker(field, dbConfig, schema_field_name): Args: field (object): Individual field in ui-config. dbConfig (object): Configurations of db-config.json. - schema_field_name (string): Specifies which key has the field's name in schema. + schema_field_name (string): Specifies which key has the field's name in schema. For old schema types, it is 'value' else 'configKey'. Returns: object """ timeRangeObj = {} - timeRangeObj['type'] = FieldTypeEnum.OBJECT.value + timeRangeObj["type"] = FieldTypeEnum.OBJECT.value timeRangeProps = { - field['startTime']['value']: {'type': FieldTypeEnum.STRING.value}, - field['endTime']['value']: {'type': FieldTypeEnum.STRING.value} + field["startTime"]["value"]: {"type": FieldTypeEnum.STRING.value}, + field["endTime"]["value"]: {"type": FieldTypeEnum.STRING.value}, } - timeRangeObj['properties'] = timeRangeProps - timeRangeObj['required'] = list(timeRangeProps.keys()) + timeRangeObj["properties"] = timeRangeProps + timeRangeObj["required"] = list(timeRangeProps.keys()) return timeRangeObj @@ -530,18 +633,16 @@ def generate_schema_for_time_picker(field, dbConfig, schema_field_name): Args: field (object): Individual field in ui-config. dbConfig (object): Configurations of db-config.json. - schema_field_name (string): Specifies which key has the field's name in schema. + schema_field_name (string): Specifies which key has the field's name in schema. For old schema types, it is 'value' else 'configKey'. Returns: object """ - return { - "type": FieldTypeEnum.STRING.value - } + return {"type": FieldTypeEnum.STRING.value} -def compare_pre_requisite_fields(fieldA, fieldB, isV2 = False): +def compare_pre_requisite_fields(fieldA, fieldB, isV2=False): """Compares two preRequisiteFields fieldA and fieldB for each property and checks if their value matches. Args: @@ -552,12 +653,12 @@ def compare_pre_requisite_fields(fieldA, fieldB, isV2 = False): Returns: boolean: If all the properties have the same 'name' and 'selectedValue', then it returns True else False. """ - valueKey = 'selectedValue' - nameKey = 'name' + valueKey = "selectedValue" + nameKey = "name" if isV2: - valueKey = 'value' - nameKey = 'configKey' + valueKey = "value" + nameKey = "configKey" if type(fieldA) != type(fieldB): return False @@ -565,13 +666,17 @@ def compare_pre_requisite_fields(fieldA, fieldB, isV2 = False): if len(fieldA) != len(fieldB): return False for i in range(0, len(fieldA)): - if fieldA[i][nameKey] != fieldB[i][nameKey] or fieldA[i][valueKey] != fieldB[i][valueKey]: + if ( + fieldA[i][nameKey] != fieldB[i][nameKey] + or fieldA[i][valueKey] != fieldB[i][valueKey] + ): return False else: if fieldA[nameKey] != fieldB[nameKey] or fieldA[valueKey] != fieldB[valueKey]: - return False + return False return True + def get_unique_pre_requisite_fields(uiConfig): """Returns the list of unique preRequisiteFields present in a uiConfig. @@ -580,16 +685,18 @@ def get_unique_pre_requisite_fields(uiConfig): Returns: list: containing unique preRequisiteFields. - """ + """ preRequisiteFieldsList = [] for group in uiConfig: - fields = group.get('fields', []) + fields = group.get("fields", []) for field in fields: if "preRequisiteField" not in field: continue isPresent = False for preRequisiteField in preRequisiteFieldsList: - if compare_pre_requisite_fields(preRequisiteField, field["preRequisiteField"]): + if compare_pre_requisite_fields( + preRequisiteField, field["preRequisiteField"] + ): isPresent = True break if not isPresent: @@ -597,7 +704,7 @@ def get_unique_pre_requisite_fields(uiConfig): return preRequisiteFieldsList -def generate_if_object(preRequisiteField, isV2 = False): +def generate_if_object(preRequisiteField, isV2=False): """Creates an if object for the given preRequisiteField. The preRequisiteField becomes an if condition in the schema. Args: @@ -608,18 +715,16 @@ def generate_if_object(preRequisiteField, isV2 = False): object: if block for given preRequisiteField. """ ifObj = {"properties": {}, "required": []} - valueKey = 'selectedValue' - nameKey = 'name' + valueKey = "selectedValue" + nameKey = "name" if isV2: - valueKey = 'value' - nameKey = 'configKey' + valueKey = "value" + nameKey = "configKey" if type(preRequisiteField) == list: for field in preRequisiteField: - ifObj["properties"][field[nameKey]] = { - "const": field[valueKey] - } + ifObj["properties"][field[nameKey]] = {"const": field[valueKey]} ifObj["required"].append(field[nameKey]) else: ifObj["properties"][preRequisiteField[nameKey]] = { @@ -638,12 +743,12 @@ def generate_schema_for_allOf(uiConfig, dbConfig, schema_field_name): Args: uiConfig (object): file content of ui-config.json. dbConfig (object): Configurations of db-config.json. - schema_field_name (string): Specifies which key has the field's name in schema. + schema_field_name (string): Specifies which key has the field's name in schema. For old schema types, it is 'value' else 'configKey'. Returns: object: allOf object of schema - """ + """ allOfItemList = [] preRequisiteFieldsList = get_unique_pre_requisite_fields(uiConfig) for preRequisiteField in preRequisiteFieldsList: @@ -651,12 +756,18 @@ def generate_schema_for_allOf(uiConfig, dbConfig, schema_field_name): thenObj = {"properties": {}, "required": []} allOfItemObj = {"if": ifObj} for group in uiConfig: - fields = group.get('fields', []) + fields = group.get("fields", []) for field in fields: if "preRequisiteField" not in field: continue - if compare_pre_requisite_fields(field["preRequisiteField"], preRequisiteField): - thenObj["properties"][field[schema_field_name]] = uiTypetoSchemaFn.get(field["type"])(field, dbConfig, schema_field_name) + if compare_pre_requisite_fields( + field["preRequisiteField"], preRequisiteField + ): + thenObj["properties"][field[schema_field_name]] = ( + uiTypetoSchemaFn.get(field["type"])( + field, dbConfig, schema_field_name + ) + ) if "required" in field and field["required"] == True: thenObj["required"].append(field[schema_field_name]) allOfItemObj["then"] = thenObj @@ -665,14 +776,15 @@ def generate_schema_for_allOf(uiConfig, dbConfig, schema_field_name): allOfItemList = generate_schema_for_anyOf(allOfItemList, schema_field_name) return allOfItemList + def get_common_and_opposite_fields(propertiesA, propertiesB, schema_field_name): """Takes properties of two if Objects and returns a list of common and opposite properties. Common properties have the same value in both A and B. Args: propertiesA (object): if block - propertiesB (object): - schema_field_name (string): Specifies which key has the field's name in schema. + propertiesB (object): + schema_field_name (string): Specifies which key has the field's name in schema. For old schema types, it is 'value' else 'configKey'. Returns: @@ -684,10 +796,10 @@ def get_common_and_opposite_fields(propertiesA, propertiesB, schema_field_name): propertiesA = { 'storage' : {'const' : 'S3'}, 'useStorage' : {'const': True}} propertiesB = { 'storage' : {'const' : 'S3'}, 'useStorage' : {'const': False}} - Returns + Returns commonProperties = [{'key': 'storage', 'value':'S3'}] oppositeProperties = [{'key': 'useStorage', 'value': 'True'}] - """ + """ keysListA = list(propertiesA.keys()) keysListB = list(propertiesB.keys()) commonProperties = [] @@ -696,32 +808,40 @@ def get_common_and_opposite_fields(propertiesA, propertiesB, schema_field_name): if key not in keysListB: return None, None if propertiesA[key]["const"] == propertiesB[key]["const"]: - commonProperties.append({"key": key, schema_field_name: propertiesA[key]["const"]}) - elif type(propertiesA[key]["const"]) == bool and propertiesA[key]["const"] != propertiesB[key]["const"]: - oppositeProperties.append({"key": key, schema_field_name: propertiesA[key]["const"]}) + commonProperties.append( + {"key": key, schema_field_name: propertiesA[key]["const"]} + ) + elif ( + type(propertiesA[key]["const"]) == bool + and propertiesA[key]["const"] != propertiesB[key]["const"] + ): + oppositeProperties.append( + {"key": key, schema_field_name: propertiesA[key]["const"]} + ) else: return None, None return commonProperties, oppositeProperties + def check_if_conditions_match(ifPropsA, ifObjectB, schema_field_name): """Compares the ifPropsA and ifObjectB if they have the same properties and values. Args: ifPropsA (list): consists of key and "schema_field_name" pairs. ifObjectB (object): if block consisting of multiple properties with the value. - schema_field_name (string): Specifies which key has the field's name in schema. + schema_field_name (string): Specifies which key has the field's name in schema. For old schema types, it is 'value' else 'configKey'. Returns: boolean: True if all the properties in IfPropsA are in ifObjectB with same value, else False. - + Example: - schema_field_name = 'value' + schema_field_name = 'value' ifPropsA = [{'key': 'storage', 'value': 'S3'}, {'key': 'deploy', 'value': 'web'}] ifObjectA = {'storage': {'const': 'S3'}, 'deploy': {'const': 'web'}} - + Returns: True - """ + """ for ifProp in ifPropsA: if ifProp["key"] not in ifObjectB: return False @@ -729,18 +849,19 @@ def check_if_conditions_match(ifPropsA, ifObjectB, schema_field_name): return False return True + def find_index_to_place_anyOf(ifProp, allOfItemList, schema_field_name): """Returns the index of the item in allOfItemList consisting of matching if conditions as that of ifProp. Args: ifProp (object): consists of key and "schema_field_name" pairs. allOfItemList (list): consists of a list of objects with "if-then" properties. - schema_field_name (string): Specifies which key has the field's name in schema. + schema_field_name (string): Specifies which key has the field's name in schema. For old schema types, it is 'value' else 'configKey'. Returns: int: index of if-block matching with ifProp else -1. - + Example: schema_field_name = 'value' ifProp = [{'key': 'storage', 'value': 'S3'}, {'key': 'deploy', 'value': 'web'}] @@ -750,85 +871,109 @@ def find_index_to_place_anyOf(ifProp, allOfItemList, schema_field_name): ] Returns: 1 - """ + """ if not ifProp: return -1 length = len(allOfItemList) for index in range(length): - if "if" in allOfItemList[index] and check_if_conditions_match(ifProp, allOfItemList[index]["if"]["properties"], schema_field_name): + if "if" in allOfItemList[index] and check_if_conditions_match( + ifProp, allOfItemList[index]["if"]["properties"], schema_field_name + ): return index return -1 + def generate_schema_for_anyOf(allOfItemList, schema_field_name): """Takes in two parameters allOfItemList and schema_field_name, and returns an updated allOf Items list. - - It checks for all the pairs of allOf Items ("if-then" blocks), if their "if" conditions have an "if-else" based structure rather than "if-if". + - It checks for all the pairs of allOf Items ("if-then" blocks), if their "if" conditions have an "if-else" based structure rather than "if-if". - Items following the "if-else" structure are deleted and replaced by an anyOf structure. Args: allOfItemList (list): consists of a list of objects with "if-then" properties. - schema_field_name (string): Specifies which key has the field's name in schema. + schema_field_name (string): Specifies which key has the field's name in schema. For old schema types, it is 'value' else 'configKey'. Returns: list: updated allOf Items list - """ + """ length = len(allOfItemList) delIndices = [] for i in range(0, length): - for j in range(i+1, length): + for j in range(i + 1, length): ifPropertiesA = allOfItemList[i]["if"]["properties"] thenPropertiesA = allOfItemList[i]["then"] ifPropertiesB = allOfItemList[j]["if"]["properties"] thenPropertiesB = allOfItemList[j]["then"] - commonIfProp, oppositeIfProp = get_common_and_opposite_fields(ifPropertiesA, ifPropertiesB, schema_field_name) + commonIfProp, oppositeIfProp = get_common_and_opposite_fields( + ifPropertiesA, ifPropertiesB, schema_field_name + ) if oppositeIfProp: anyOfObj = [{}, {}] for k in range(0, len(oppositeIfProp)): if ifPropertiesA[oppositeIfProp[k]["key"]]["const"] == True: anyOfObj[1] = thenPropertiesA - anyOfObj[1]["properties"][oppositeIfProp[k]["key"]] = {"const": True} + anyOfObj[1]["properties"][oppositeIfProp[k]["key"]] = { + "const": True + } anyOfObj[1]["required"].append(oppositeIfProp[k]["key"]) anyOfObj[0] = thenPropertiesB - anyOfObj[0]["properties"][oppositeIfProp[k]["key"]] = {"const": False} + anyOfObj[0]["properties"][oppositeIfProp[k]["key"]] = { + "const": False + } else: anyOfObj[1] = thenPropertiesB - anyOfObj[1]["properties"][oppositeIfProp[k]["key"]] = {"const": True} + anyOfObj[1]["properties"][oppositeIfProp[k]["key"]] = { + "const": True + } anyOfObj[1]["required"].append(oppositeIfProp[k]["key"]) anyOfObj[0] = thenPropertiesA - anyOfObj[0]["properties"][oppositeIfProp[k]["key"]] = {"const": False} - # AnyOf object is placed at index of "if-then" block having same if properties as of common properties else at end. - indexToPlace = find_index_to_place_anyOf(commonIfProp, allOfItemList, schema_field_name) + anyOfObj[0]["properties"][oppositeIfProp[k]["key"]] = { + "const": False + } + # AnyOf object is placed at index of "if-then" block having same if properties as of common properties else at end. + indexToPlace = find_index_to_place_anyOf( + commonIfProp, allOfItemList, schema_field_name + ) if indexToPlace != -1: allOfItemList[indexToPlace]["then"]["anyOf"] = anyOfObj delIndices.append(i) delIndices.append(j) - allOfItemList = [allOfItemList[index] for index in range(len(allOfItemList)) if index not in delIndices] + allOfItemList = [ + allOfItemList[index] + for index in range(len(allOfItemList)) + if index not in delIndices + ] return allOfItemList + def generate_connection_mode(dbConfig): """Creates the connection mode object present in new schema types. - + Args: dbConfig (object): Configurations of db-config.json. Returns: object - """ + """ connectionObj = {"type": FieldTypeEnum.OBJECT.value} connectionObj["properties"] = {} for sourceType in dbConfig["supportedSourceTypes"]: - if sourceType in dbConfig["supportedConnectionModes"]: - connectionItemObj = {"type": FieldTypeEnum.STRING.value} - connectionModesEnum=[] - length = len(dbConfig["supportedConnectionModes"][sourceType]) - for i in range(0, length): - connectionModesEnum.append(dbConfig["supportedConnectionModes"][sourceType][i]) - connectionItemObj["enum"] = connectionModesEnum - connectionObj["properties"][sourceType] = connectionItemObj + if sourceType in dbConfig["supportedConnectionModes"]: + connectionItemObj = {"type": FieldTypeEnum.STRING.value} + connectionModesEnum = [] + length = len(dbConfig["supportedConnectionModes"][sourceType]) + for i in range(0, length): + connectionModesEnum.append( + dbConfig["supportedConnectionModes"][sourceType][i] + ) + connectionItemObj["enum"] = connectionModesEnum + connectionObj["properties"][sourceType] = connectionItemObj return connectionObj -def generate_schema_properties(uiConfig, dbConfig, schemaObject, properties, name, selector): +def generate_schema_properties( + uiConfig, dbConfig, schemaObject, properties, name, selector +): """Generates corresponding schema properties by iterating over each of the ui-config fields. Args: @@ -838,75 +983,98 @@ def generate_schema_properties(uiConfig, dbConfig, schemaObject, properties, nam properties (object): properties of schema name (string): name of the source or destination. selector (string): either 'source' or 'destination' - """ + """ if is_old_format(uiConfig): for group in uiConfig: - fields = group.get('fields', []) + fields = group.get("fields", []) for field in fields: if "preRequisiteField" in field: continue - generateFunction = uiTypetoSchemaFn.get(field['type'], None) + generateFunction = uiTypetoSchemaFn.get(field["type"], None) if generateFunction: - properties[field['value']] = generateFunction( - field, dbConfig, 'value') - if field.get('required', False) == True and is_field_present_in_default_config(field, dbConfig, "value"): - schemaObject['required'].append(field['value']) + properties[field["value"]] = generateFunction( + field, dbConfig, "value" + ) + if field.get( + "required", False + ) == True and is_field_present_in_default_config( + field, dbConfig, "value" + ): + schemaObject["required"].append(field["value"]) else: - if selector == 'destination': - baseTemplate = uiConfig.get('baseTemplate', []) - sdkTemplate = uiConfig.get('sdkTemplate', {}) - consentSettingsTemplate = uiConfig.get('consentSettingsTemplate', {}) + if selector == "destination": + baseTemplate = uiConfig.get("baseTemplate", []) + sdkTemplate = uiConfig.get("sdkTemplate", {}) + consentSettingsTemplate = uiConfig.get("consentSettingsTemplate", {}) for template in baseTemplate: - for section in template.get('sections', []): - for group in section.get('groups', []): - for field in group.get('fields', []): - generateFunction = uiTypetoSchemaFn.get( - field['type'], None) + for section in template.get("sections", []): + for group in section.get("groups", []): + for field in group.get("fields", []): + generateFunction = uiTypetoSchemaFn.get(field["type"], None) if generateFunction: - properties[field['configKey']] = generateFunction( - field, dbConfig, 'configKey') - if template.get('title', "") == "Initial setup" and is_field_present_in_default_config(field, dbConfig, "configKey") and 'preRequisites' not in field: - schemaObject['required'].append( - field['configKey']) - - for field in sdkTemplate.get('fields', []): - generateFunction = uiTypetoSchemaFn.get(field['type'], None) + properties[field["configKey"]] = generateFunction( + field, dbConfig, "configKey" + ) + if ( + template.get("title", "") == "Initial setup" + and is_field_present_in_default_config( + field, dbConfig, "configKey" + ) + and "preRequisites" not in field + ): + schemaObject["required"].append(field["configKey"]) + + for field in sdkTemplate.get("fields", []): + generateFunction = uiTypetoSchemaFn.get(field["type"], None) if generateFunction: - properties[field['configKey']] = generateFunction( - field, dbConfig, 'configKey') - if field.get('required', False) == True and is_field_present_in_default_config(field, dbConfig, "configKey"): - schemaObject['required'].append(field['configKey']) - - for field in consentSettingsTemplate.get('fields', []): - generateFunction = uiTypetoSchemaFn.get(field['type'], None) + properties[field["configKey"]] = generateFunction( + field, dbConfig, "configKey" + ) + if field.get( + "required", False + ) == True and is_field_present_in_default_config( + field, dbConfig, "configKey" + ): + schemaObject["required"].append(field["configKey"]) + + for field in consentSettingsTemplate.get("fields", []): + generateFunction = uiTypetoSchemaFn.get(field["type"], None) if generateFunction: - properties[field['configKey']] = generateFunction( - field, dbConfig, 'configKey') - if field.get('required', False) == True and is_field_present_in_default_config(field, dbConfig, "configKey"): - schemaObject['required'].append(field['configKey']) + properties[field["configKey"]] = generateFunction( + field, dbConfig, "configKey" + ) + if field.get( + "required", False + ) == True and is_field_present_in_default_config( + field, dbConfig, "configKey" + ): + schemaObject["required"].append(field["configKey"]) # default properties in new ui-config based schemas. - schemaObject['properties']['useNativeSDK'] = generate_schema_for_checkbox({"type":"checkbox", - "value":"useNativeSDK"}, dbConfig, "value") - schemaObject['properties']['connectionMode'] = generate_connection_mode(dbConfig) + schemaObject["properties"]["useNativeSDK"] = generate_schema_for_checkbox( + {"type": "checkbox", "value": "useNativeSDK"}, dbConfig, "value" + ) + schemaObject["properties"]["connectionMode"] = generate_connection_mode( + dbConfig + ) else: # for sources def generate_config_props(config): for group in config: - fields = group.get('fields', []) + fields = group.get("fields", []) for field in fields: - generateFunction = uiTypetoSchemaFn.get( - field["type"], None) + generateFunction = uiTypetoSchemaFn.get(field["type"], None) if generateFunction: - properties[field['value']] = generateFunction( - field, dbConfig, 'value') + properties[field["value"]] = generateFunction( + field, dbConfig, "value" + ) - auth = uiConfig.get('auth', None) - config = uiConfig.get('config', []) + auth = uiConfig.get("auth", None) + config = uiConfig.get("config", []) if auth: - type = auth.get('type') + type = auth.get("type") if type == "form": - auth_config = auth.get('config', []) + auth_config = auth.get("config", []) generate_config_props(auth_config) generate_config_props(config) @@ -923,31 +1091,33 @@ def generate_schema(uiConfig, dbConfig, name, selector): Returns: object: schema - """ + """ newSchema = {} schemaObject = {} - schemaObject['$schema'] = 'http://json-schema.org/draft-07/schema#' - schemaObject['required'] = [] - schemaObject['type'] = "object" - schemaObject['properties'] = {} + schemaObject["$schema"] = "http://json-schema.org/draft-07/schema#" + schemaObject["required"] = [] + schemaObject["type"] = "object" + schemaObject["properties"] = {} allOfSchemaObj = {} if is_old_format(uiConfig): allOfSchemaObj = generate_schema_for_allOf(uiConfig, dbConfig, "value") if allOfSchemaObj: # AnyOf occurring separately, not inside allOf. if len(allOfSchemaObj) == 1: - if isinstance(allOfSchemaObj[0], list): - schemaObject['anyOf'] = allOfSchemaObj[0] - else: - schemaObject['anyOf'] = allOfSchemaObj + if isinstance(allOfSchemaObj[0], list): + schemaObject["anyOf"] = allOfSchemaObj[0] + else: + schemaObject["anyOf"] = allOfSchemaObj else: - schemaObject['allOf'] = allOfSchemaObj - generate_schema_properties(uiConfig, dbConfig, schemaObject, - schemaObject['properties'], name, selector) - newSchema['configSchema'] = schemaObject + schemaObject["allOf"] = allOfSchemaObj + generate_schema_properties( + uiConfig, dbConfig, schemaObject, schemaObject["properties"], name, selector + ) + newSchema["configSchema"] = schemaObject return newSchema + def generate_warnings_for_each_type(uiConfig, dbConfig, schema, curUiType): """Generates warning for each schema difference created by the current ui-type. @@ -956,7 +1126,7 @@ def generate_warnings_for_each_type(uiConfig, dbConfig, schema, curUiType): dbConfig (object): Configurations of db-config.json. schema (object): Existing schema in schema.json curUiType (string): Ui-Type for which warnings are generated. - """ + """ if is_old_format(uiConfig): for uiConfigItem in uiConfig: for field in uiConfigItem["fields"]: @@ -965,77 +1135,138 @@ def generate_warnings_for_each_type(uiConfig, dbConfig, schema, curUiType): if field["type"] == curUiType: if field["value"] not in schema["properties"]: warnings.warn( - f'{field["value"]} field is not in schema \n', UserWarning) + f'{field["value"]} field is not in schema \n', UserWarning + ) else: curSchemaField = schema["properties"][field["value"]] - newSchemaField = uiTypetoSchemaFn.get( - curUiType)(field, dbConfig, "value") + newSchemaField = uiTypetoSchemaFn.get(curUiType)( + field, dbConfig, "value" + ) schemaDiff = get_json_diff(curSchemaField, newSchemaField) if schemaDiff: - warnings.warn("For type:{} field:{} Difference is : \n\n {} \n".format( - curUiType, field["value"], get_formatted_json(schemaDiff)), UserWarning) + warnings.warn( + "For type:{} field:{} Difference is : \n\n {} \n".format( + curUiType, + field["value"], + get_formatted_json(schemaDiff), + ), + UserWarning, + ) else: - baseTemplate = uiConfig.get('baseTemplate', []) - sdkTemplate = uiConfig.get('sdkTemplate', {}) - consentSettingsTemplate = uiConfig.get('consentSettingsTemplate', {}) + baseTemplate = uiConfig.get("baseTemplate", []) + sdkTemplate = uiConfig.get("sdkTemplate", {}) + consentSettingsTemplate = uiConfig.get("consentSettingsTemplate", {}) for template in baseTemplate: - for section in template.get('sections', []): - for group in section.get('groups', []): + for section in template.get("sections", []): + for group in section.get("groups", []): if "preRequisites" in group: continue - for field in group.get('fields', []): + for field in group.get("fields", []): if "preRequisites" in field: continue - generateFunction = uiTypetoSchemaFn.get( - field['type'], None) + generateFunction = uiTypetoSchemaFn.get(field["type"], None) if generateFunction and field["type"] == curUiType: if field["configKey"] not in schema["properties"]: warnings.warn( - f'{field["configKey"]} field is not in schema \n', UserWarning) + f'{field["configKey"]} field is not in schema \n', + UserWarning, + ) else: - curSchemaField = schema["properties"][field["configKey"]] - newSchemaField = uiTypetoSchemaFn.get( - curUiType)(field, dbConfig, "configKey") - schemaDiff = get_json_diff(curSchemaField, newSchemaField) + curSchemaField = schema["properties"][ + field["configKey"] + ] + newSchemaField = uiTypetoSchemaFn.get(curUiType)( + field, dbConfig, "configKey" + ) + schemaDiff = get_json_diff( + curSchemaField, newSchemaField + ) if schemaDiff: - warnings.warn("For type:{} field:{} Difference is : \n\n {} \n".format( - curUiType, field["configKey"], get_formatted_json(schemaDiff)), UserWarning) - - for field in sdkTemplate.get('fields', []): + warnings.warn( + "For type:{} field:{} Difference is : \n\n {} \n".format( + curUiType, + field["configKey"], + get_formatted_json(schemaDiff), + ), + UserWarning, + ) + + for field in sdkTemplate.get("fields", []): if "preRequisites" in field: continue - generateFunction = uiTypetoSchemaFn.get(field['type'], None) + generateFunction = uiTypetoSchemaFn.get(field["type"], None) if generateFunction: if generateFunction and field["type"] == curUiType: if field["configKey"] not in schema["properties"]: warnings.warn( - f'{field["configKey"]} field is not in schema \n', UserWarning) + f'{field["configKey"]} field is not in schema \n', + UserWarning, + ) else: curSchemaField = schema["properties"][field["configKey"]] - newSchemaField = uiTypetoSchemaFn.get( - curUiType)(field, dbConfig, "configKey") + newSchemaField = uiTypetoSchemaFn.get(curUiType)( + field, dbConfig, "configKey" + ) schemaDiff = get_json_diff(curSchemaField, newSchemaField) if schemaDiff: - warnings.warn("For type:{} field:{} Difference is : \n\n {} \n".format( - curUiType, field["configKey"], get_formatted_json(schemaDiff)), UserWarning) - - for field in consentSettingsTemplate.get('fields', []): + warnings.warn( + "For type:{} field:{} Difference is : \n\n {} \n".format( + curUiType, + field["configKey"], + get_formatted_json(schemaDiff), + ), + UserWarning, + ) + + for field in sdkTemplate.get("fields", []): + if "preRequisites" in field: + continue + generateFunction = uiTypetoSchemaFn.get(field["type"], None) + if generateFunction: + if generateFunction and field["type"] == curUiType: + if field["configKey"] not in schema["properties"]: + warnings.warn( + f'{field["configKey"]} field is not in schema \n', + UserWarning, + ) + else: + curSchemaField = schema["properties"][field["configKey"]] + newSchemaField = uiTypetoSchemaFn.get(curUiType)( + field, dbConfig, "configKey" + ) + schemaDiff = get_json_diff(newSchemaField, curSchemaField) + if schemaDiff: + warnings.warn( + "For type:{} field:{} Difference is : \n\n {} \n".format( + curUiType, field["configKey"], schemaDiff + ), + UserWarning, + ) + + for field in consentSettingsTemplate.get("fields", []): if "preRequisites" in field: continue - generateFunction = uiTypetoSchemaFn.get(field['type'], None) + generateFunction = uiTypetoSchemaFn.get(field["type"], None) if generateFunction: if generateFunction and field["type"] == curUiType: if field["configKey"] not in schema["properties"]: warnings.warn( - f'{field["configKey"]} field is not in schema \n', UserWarning) + f'{field["configKey"]} field is not in schema \n', + UserWarning, + ) else: curSchemaField = schema["properties"][field["configKey"]] - newSchemaField = uiTypetoSchemaFn.get( - curUiType)(field, dbConfig, "configKey") - schemaDiff = diff(newSchemaField, curSchemaField) + newSchemaField = uiTypetoSchemaFn.get(curUiType)( + field, dbConfig, "configKey" + ) + schemaDiff = get_json_diff(newSchemaField, curSchemaField) if schemaDiff: - warnings.warn("For type:{} field:{} Difference is : \n\n {} \n".format( - curUiType, field["configKey"], schemaDiff), UserWarning) + warnings.warn( + "For type:{} field:{} Difference is : \n\n {} \n".format( + curUiType, field["configKey"], schemaDiff + ), + UserWarning, + ) uiTypetoSchemaFn = { @@ -1045,29 +1276,33 @@ def generate_warnings_for_each_type(uiConfig, dbConfig, schema, curUiType): "textareaInput": generate_schema_for_textarea_input, "singleSelect": generate_schema_for_single_select, "dynamicCustomForm": generate_schema_for_dynamic_custom_form, - 'dynamicForm': generate_schema_for_dynamic_form, - 'mapping': generate_schema_for_mapping, - 'dynamicSelectForm': generate_schema_for_dynamic_select_form, - 'tagInput': generate_schema_for_tag_input, - 'timeRangePicker': generate_schema_for_time_range_picker, - 'timePicker': generate_schema_for_time_picker + "dynamicForm": generate_schema_for_dynamic_form, + "mapping": generate_schema_for_mapping, + "dynamicSelectForm": generate_schema_for_dynamic_select_form, + "tagInput": generate_schema_for_tag_input, + "timeRangePicker": generate_schema_for_time_range_picker, + "timePicker": generate_schema_for_time_picker, } + def save_schema_to_file(selector, name, schema): # Get the parent directory (one level up) script_directory = os.path.dirname(os.path.abspath(__file__)) directory = os.path.dirname(script_directory) # Define the relative path - relative_path = f'{CONFIG_DIR}/{selector}s/{name}/schema.json' + relative_path = f"{CONFIG_DIR}/{selector}s/{name}/schema.json" file_path = os.path.join(directory, relative_path) # Write the new content - with open(file_path, 'w') as file: + with open(file_path, "w") as file: file.write(get_formatted_json(schema)) -def validate_config_consistency(name, selector, uiConfig, dbConfig, schema, shouldUpdateSchema): - """Generates a schema and compares it with an existing one. + +def validate_config_consistency( + name, selector, uiConfig, dbConfig, schema, shouldUpdateSchema +): + """Generates a schema and compares it with an existing one. If schemaDiff is present, it calls for individual warnings by iterating over each ui-type. Args: @@ -1077,13 +1312,13 @@ def validate_config_consistency(name, selector, uiConfig, dbConfig, schema, shou dbConfig (object): Configurations of db-config.json. schema (object): Existing schema in schema.json. shouldUpdateSchema (boolean): if it should update the existing schema with generated one - """ + """ if schema == None and uiConfig == None: return if uiConfig == None: - print('-'*50) + print("-" * 50) warnings.warn(f"Ui-Config is null for {name} in {selector} \n", UserWarning) - print('-'*50) + print("-" * 50) return generatedSchema = generate_schema(uiConfig, dbConfig, name, selector) @@ -1095,23 +1330,30 @@ def validate_config_consistency(name, selector, uiConfig, dbConfig, schema, shou save_schema_to_file(selector, name, finalSchema) if schemaDiff: - print('-'*50) - print(f'Schema diff for {name} in {selector}s') + print("-" * 50) + print(f"Schema diff for {name} in {selector}s") # call for individual warnings for uiType in uiTypetoSchemaFn.keys(): generate_warnings_for_each_type(uiConfig, dbConfig, schema, uiType) # schema diff for "additionalProperties" if "additionalProperties" not in schema: - print("\n Recommendation: Please set additionalProperties to False in schema.json. \n") + print( + "\n Recommendation: Please set additionalProperties to False in schema.json. \n" + ) # schema diff for "required" if "required" not in schema: - warnings.warn('required field is not in schema \n', UserWarning) + warnings.warn("required field is not in schema \n", UserWarning) else: curRequiredField = schema["required"] newRequiredField = generatedSchema["configSchema"]["required"] requiredFieldDiff = get_json_diff(curRequiredField, newRequiredField) if requiredFieldDiff: - warnings.warn("For required field Difference is : \n\n {} \n".format(get_formatted_json(requiredFieldDiff)), UserWarning) + warnings.warn( + "For required field Difference is : \n\n {} \n".format( + get_formatted_json(requiredFieldDiff) + ), + UserWarning, + ) if "allOf" in generatedSchema["configSchema"]: curAllOfSchema = {} if "allOf" in schema: @@ -1119,7 +1361,12 @@ def validate_config_consistency(name, selector, uiConfig, dbConfig, schema, shou newAllOfSchema = generatedSchema["configSchema"]["allOf"] allOfSchemaDiff = get_json_diff(curAllOfSchema, newAllOfSchema) if allOfSchemaDiff: - warnings.warn("For allOf field Difference is : \n\n {} \n".format(get_formatted_json(allOfSchemaDiff)), UserWarning) + warnings.warn( + "For allOf field Difference is : \n\n {} \n".format( + get_formatted_json(allOfSchemaDiff) + ), + UserWarning, + ) if "anyOf" in generatedSchema["configSchema"]: curAnyOfSchema = {} if "anyOf" in schema: @@ -1127,30 +1374,36 @@ def validate_config_consistency(name, selector, uiConfig, dbConfig, schema, shou newAnyOfSchema = generatedSchema["configSchema"]["anyOf"] anyOfSchemaDiff = get_json_diff(curAnyOfSchema, newAnyOfSchema) if anyOfSchemaDiff: - warnings.warn("For anyOf field Difference is : \n\n {} \n".format(get_formatted_json(anyOfSchemaDiff)), UserWarning) - print('-'*50) + warnings.warn( + "For anyOf field Difference is : \n\n {} \n".format( + get_formatted_json(anyOfSchemaDiff) + ), + UserWarning, + ) + print("-" * 50) else: if shouldUpdateSchema: save_schema_to_file(selector, name, generatedSchema) - print('-'*50) - print(f'Generated schema for {name} in {selector}s') + print("-" * 50) + print(f"Generated schema for {name} in {selector}s") print(get_formatted_json(generatedSchema)) - print('-'*50) + print("-" * 50) + def get_schema_diff(name, selector, shouldUpdateSchema=False): - """ Validates the schema for the given name and selector. + """Validates the schema for the given name and selector. Args: name (string): name of the source or destination. selector (string): either 'source' or 'destination'. shouldUpdateSchema (boolean): if it should update the existing schema with generated one - """ + """ - file_selectors = ['db-config.json', 'ui-config.json', 'schema.json'] - directory = f'./{CONFIG_DIR}/{selector}s/{name}' + file_selectors = ["db-config.json", "ui-config.json", "schema.json"] + directory = f"./{CONFIG_DIR}/{selector}s/{name}" if not os.path.isdir(directory): - print(f'No {selector}s directory found for {name}') + print(f"No {selector}s directory found for {name}") return if name not in EXCLUDED_DEST: @@ -1158,35 +1411,55 @@ def get_schema_diff(name, selector, shouldUpdateSchema=False): file_content = {} for file_selector in file_selectors: if file_selector in available_files: - file_content.update(get_json_from_file(f'{directory}/{file_selector}')) + file_content.update(get_json_from_file(f"{directory}/{file_selector}")) uiConfig = file_content.get("uiConfig") schema = file_content.get("configSchema") dbConfig = file_content.get("config") - validate_config_consistency(name, selector, uiConfig, dbConfig, schema, shouldUpdateSchema) + validate_config_consistency( + name, selector, uiConfig, dbConfig, schema, shouldUpdateSchema + ) + -if __name__ == '__main__': - parser = argparse.ArgumentParser(description='Generates schema.json from ui-config.json and db-config.json and validates against actual scheme.json') +if __name__ == "__main__": + parser = argparse.ArgumentParser( + description="Generates schema.json from ui-config.json and db-config.json and validates against actual scheme.json" + ) group = parser.add_mutually_exclusive_group() - parser.add_argument('selector', metavar='selector', type=str, help='Enter whether -name is a source or destination') - parser.add_argument('-update', action='store_true', help='Will update existing schema with any changes') - group.add_argument('-name', metavar='name', type=str, help='Enter the folder name under selector') - group.add_argument('-all', action='store_true', help='Will run validation for all entities under selector') - + parser.add_argument( + "selector", + metavar="selector", + type=str, + help="Enter whether -name is a source or destination", + ) + parser.add_argument( + "-update", + action="store_true", + help="Will update existing schema with any changes", + ) + group.add_argument( + "-name", metavar="name", type=str, help="Enter the folder name under selector" + ) + group.add_argument( + "-all", + action="store_true", + help="Will run validation for all entities under selector", + ) + args = parser.parse_args() selector = args.selector shouldUpdateSchema = args.update if args.all: - dir_path = f'./{CONFIG_DIR}/{selector}s' + dir_path = f"./{CONFIG_DIR}/{selector}s" if not os.path.isdir(dir_path): - print(f'No {selector}s folder found') + print(f"No {selector}s folder found") exit(1) - + current_items = os.listdir(dir_path) for name in current_items: get_schema_diff(name, selector, shouldUpdateSchema) - + else: name = args.name get_schema_diff(name, selector, shouldUpdateSchema) diff --git a/scripts/setup-python.sh b/scripts/setup-python.sh index b71b35f43..e5518394d 100755 --- a/scripts/setup-python.sh +++ b/scripts/setup-python.sh @@ -5,3 +5,4 @@ pip3 --version pip3 install requests pip3 install jsonschema pip3 install jsondiff +pip3 install black diff --git a/scripts/utils.py b/scripts/utils.py index 49cc83027..b61e8cdc3 100644 --- a/scripts/utils.py +++ b/scripts/utils.py @@ -1,6 +1,7 @@ from jsondiff import JsonDiffer import json + def get_json_diff(oldJson, newJson): """Returns the difference between two JSONs. @@ -10,10 +11,11 @@ def get_json_diff(oldJson, newJson): Returns: object: difference between oldJson and newJson. - """ + """ differ = JsonDiffer(marshal=True) return differ.diff(oldJson, newJson) + def apply_json_diff(oldJson, diff): """Applies the difference on oldJson and returns the newJson. @@ -23,10 +25,11 @@ def apply_json_diff(oldJson, diff): Returns: object: new json. - """ + """ differ = JsonDiffer(marshal=True) return differ.patch(oldJson, diff) + def get_formatted_json(jsonObj): """Formats the json object. @@ -35,9 +38,10 @@ def get_formatted_json(jsonObj): Returns: string: formatted json. - """ + """ return json.dumps(jsonObj, indent=2, ensure_ascii=False) + def get_json_from_file(filePath): """Reads the content of the file and returns the json object. @@ -46,6 +50,6 @@ def get_json_from_file(filePath): Returns: object: json object. - """ - with open(filePath, 'r') as file: - return json.loads(file.read().encode('utf-8', 'ignore')) + """ + with open(filePath, "r") as file: + return json.loads(file.read().encode("utf-8", "ignore")) diff --git a/test/test_configGenerator.py b/test/test_configGenerator.py index 1964e7f77..19cf42839 100644 --- a/test/test_configGenerator.py +++ b/test/test_configGenerator.py @@ -10,20 +10,22 @@ from scripts.configGenerator import generateConfigs -with open('test/configData/inputData.json', 'r') as file: +with open("test/configData/inputData.json", "r") as file: input_data = json.load(file) -with open('test/configData/db-config.json', 'r') as file: +with open("test/configData/db-config.json", "r") as file: db_config = json.load(file) -with open('test/configData/ui-config.json', 'r') as file: +with open("test/configData/ui-config.json", "r") as file: ui_config = json.load(file) + class TestConfigGenerator(unittest.TestCase): def test_config_generator(self): result = generateConfigs(input_data) - self.assertEqual(result['db_config'], json.dumps(db_config)) - self.assertEqual(result['ui_config'], json.dumps(ui_config)) + self.assertEqual(result["db_config"], json.dumps(db_config)) + self.assertEqual(result["ui_config"], json.dumps(ui_config)) + -if __name__ == '__main__': +if __name__ == "__main__": unittest.main()