import yaml from utils import set_operator_role_and_location, set_eu_market_status, check_within_scope_act, check_prohibited def check_overall_compliance_ui(cards): # {"project_file": None, "data_files": [], "model_files": []} if cards["project_file"] == None: return "no project compliance card loaded" if len(cards["data_files"]) == 0 or len(cards["model_files"]) == 0: return "missing data or model compliance card" project_cc = cards['project_file'] dispositive_variables = { "ai_project_type": { "ai_system": project_cc['ai_system']['ai_system']['value'], "gpai_model": project_cc['gpai_model']['gpai_model']['value'], "high_risk_ai_system": False, "gpai_model_systemic_risk": False }, "operator_details": { "provider": project_cc['operator_details']['provider']['value'], "eu_located": project_cc['operator_details']['eu_located']['value'], "output_used": project_cc['operator_details']['output_used']['value'] }, "eu_market_status": { "placed_on_market": project_cc['eu_market_status']['placed_on_market']['value'], "put_into_service": project_cc['eu_market_status']['put_into_service']['value'] }, "project_intended_purposes": [], "project_cc_pass": False, "project_msg": [], "data_cc_compliant": [], "model_cc_compliant": [], "data_cc_non-compliant": {}, "model_cc_non-compliant": {}, } if any(item['value'] for item in project_cc['high_risk_ai_system'].values()) == True: dispositive_variables['ai_project_type']["high_risk_ai_system"] = True # check intended purposes for card in cards['data_files']: data_cc = card[1] dispositive_variables = check_intended_purpose(dispositive_variables, project_cc, data_cc) for card in cards['model_files']: model_cc = card[1] dispositive_variables = check_intended_purpose(dispositive_variables, project_cc, model_cc) # for each model_cc and data_cc - run analysis with ref to project_cc dispositive_variables = run_compliance_analysis_on_project(dispositive_variables, project_cc) for card in cards['data_files']: data_cc = card[1] dispositive_variables = run_compliance_analysis_on_data(dispositive_variables, data_cc) for card in cards['model_files']: model_cc = card[1] dispositive_variables = run_compliance_analysis_on_model(dispositive_variables, model_cc) return dispositive_variables def check_overall_compliance(cards): with open(cards['project_file'], 'r') as project_filepath: print(project_filepath) project_cc = yaml.safe_load(project_filepath.read()) dispositive_variables = { "ai_project_type": { "ai_system": project_cc['ai_system']['ai_system']['value'], "gpai_model": project_cc['gpai_model']['gpai_model']['value'], "high_risk_ai_system": False, "gpai_model_systemic_risk": False }, "operator_details": { "provider": project_cc['operator_details']['provider']['value'], "eu_located": project_cc['operator_details']['eu_located']['value'], "output_used": project_cc['operator_details']['output_used']['value'] }, "eu_market_status": { "placed_on_market": project_cc['eu_market_status']['placed_on_market']['value'], "put_into_service": project_cc['eu_market_status']['put_into_service']['value'] }, "project_intended_purposes": [], "project_cc_pass": False, "data_cc_pass": False, "model_cc_pass": False, "project_msg": [] } # check intended purposes for card in cards['data_files']: with open(card, 'r') as data_filepath: data_cc = yaml.safe_load(data_filepath.read()) dispositive_variables = check_intended_purpose(dispositive_variables, project_cc, data_cc) for card in cards['model_files']: with open(card, 'r') as model_filepath: model_cc = yaml.safe_load(model_filepath.read()) dispositive_variables = check_intended_purpose(dispositive_variables, project_cc, model_cc) # for each model_cc and data_cc - run analysis with ref to project_cc dispositive_variables = run_compliance_analysis_on_project(dispositive_variables, project_cc) for card in cards['data_files']: with open(card, 'r') as data_filepath: data_cc = yaml.safe_load(data_filepath.read()) dispositive_variables = run_compliance_analysis_on_data(dispositive_variables, data_cc) for card in cards['model_files']: with open(card, 'r') as model_filepath: model_cc = yaml.safe_load(model_filepath.read()) dispositive_variables = run_compliance_analysis_on_model(dispositive_variables, model_cc) return dispositive_variables def run_compliance_analysis_on_project(dispositive_variables, project_cc_yaml): # Project Type if project_cc_yaml['ai_system']['ai_system']['value']: dispositive_variables['ai_project_type']['ai_system'] = True if project_cc_yaml['gpai_model']['gpai_model']['value']: dispositive_variables['ai_project_type']['gpai_model'] = True if project_cc_yaml['ai_system']['ai_system']['value'] == True and project_cc_yaml['gpai_model']['gpai_model']['value'] == True: dispositive_variables['project_msg'].append("Your project cannot be both an AI system and a GPAI model. Please revise your Project CC accordingly.") return dispositive_variables if dispositive_variables['ai_project_type']['ai_system'] == True: for value in project_cc_yaml['high_risk_ai_system']: if value and sum(map(bool, [ project_cc_yaml['high_risk_ai_system_exceptions']['filter_exception_rights']['value'], project_cc_yaml['high_risk_ai_system_exceptions']['filter_exception_narrow']['value'], project_cc_yaml['high_risk_ai_system_exceptions']['filter_exception_human']['value'], project_cc_yaml['high_risk_ai_system_exceptions']['filter_exception_deviation']['value'], project_cc_yaml['high_risk_ai_system_exceptions']['filter_exception_prep']['value']]) ) >= 1: dispositive_variables['ai_project_type']["high_risk_ai_system"] = False if dispositive_variables['ai_project_type']['gpai_model'] == True: if project_cc_yaml['gpai_model_systemic_risk']['evaluation']['value'] or project_cc_yaml['gpai_model_systemic_risk']['flops']['value']: dispositive_variables['ai_project_type']["gpai_model_systemic_risk"] = True # Operator Type dispositive_variables = set_operator_role_and_location(dispositive_variables, project_cc_yaml) dispositive_variables = set_eu_market_status(dispositive_variables, project_cc_yaml) # Check if project is within scope of the Compliance Cards project. If not, inform user. if project_cc_yaml['operator_details']['provider']['value'] == True: dispositive_variables['project_msg'].append("Project is within the scope of the Compliance Cards system. Let's continue...") else: # dispositive_variables['project_msg'].append("Project is not within the scope of the initial version of the Compliance Cards system.") dispositive_variables['project_cc_pass'] = True return dispositive_variables # Check if the project is within scope of the Act. If it's not, the analysis is over. if check_within_scope_act(dispositive_variables, project_cc_yaml): dispositive_variables['project_msg'].append("Project is within the scope of Act. Let's continue...") else: # dispositive_variables['project_msg'].append("Project is not within the scope of what is regulated by the Act.") dispositive_variables['project_cc_pass'] = True return dispositive_variables # Check for prohibited practices. If any exist, the analysis is over. if check_prohibited(project_cc_yaml) == True: dispositive_variables['project_msg'].append("Project is non-compliant due to a prohibited practice.") return dispositive_variables else: print("Project does not contain prohibited practies. Let's continue...") # If project is high-risk AI system, check that is has met all the requirements for such systems: if dispositive_variables['ai_project_type']["high_risk_ai_system"] == True: for key in project_cc_yaml['risk_management_system']: if project_cc_yaml['risk_management_system'][f'{key}']['value'] == True: dispositive_variables['project_msg'].append("Because of project-level characteristics, this high-risk AI system fails the risk management requirements under Article 9.") for key in project_cc_yaml['technical_documentation']: if project_cc_yaml['technical_documentation'][f'{key}']['value'] == True: dispositive_variables['project_msg'].append("Because of project-level characteristics, this high-risk AI system fails the risk management requirements under Article 11.") for key in project_cc_yaml['record_keeping']: if project_cc_yaml['record_keeping'][f'{key}']['value'] == True: dispositive_variables['project_msg'].append("Because of project-level characteristics, this high-risk AI system fails the risk management requirements under Article 12.") for key in project_cc_yaml['transparency_and_provision_of_information_to_deployers']: if project_cc_yaml['transparency_and_provision_of_information_to_deployers'][f'{key}']['value'] == True: dispositive_variables['project_msg'].append("Because of project-level characteristics, this high-risk AI system fails the transparency requirements under Article 13.") for key in project_cc_yaml['human_oversight']: if project_cc_yaml['human_oversight'][f'{key}']['value'] == True: dispositive_variables['project_msg'].append("Because of project-level characteristics, this high-risk AI system fails the human oversight requirements under Article 14.") for key in project_cc_yaml['accuracy_robustness_cybersecurity']: if project_cc_yaml['accuracy_robustness_cybersecurity'][f'{key}']['value'] == True: dispositive_variables['project_msg'].append("Because of project-level characteristics, this high-risk AI system fails the accuracy, robustness, and cybersecurity requirements under Article 15.") for key in project_cc_yaml['quality_management_system']: if project_cc_yaml['quality_management_system'][f'{key}']['value'] == True: dispositive_variables['project_msg'].append("Because of project-level characteristics, this high-risk AI system fails the accuracy, robustness, and cybersecurity requirements under Article 17.") if dispositive_variables['ai_project_type']["gpai_model"] == True: if dispositive_variables['ai_project_type']["gpai_model_systemic_risk"] == True: for key in project_cc_yaml['gpai_models_with_systemic_risk_obligations']: if project_cc_yaml['gpai_models_with_systemic_risk_obligations'][f'{key}']['value'] == True: dispositive_variables['project_msg'].append("GPAI model with systematic risk fails the transparency requirements under Article 55.") for obligation_cat in project_cc_yaml['gpai_model_obligations']: for obligation in project_cc_yaml['gpai_model_obligations'][f'{obligation_cat}']: if project_cc_yaml['gpai_model_obligations'][f'{obligation_cat}'][f'{obligation}']['value'] == True: dispositive_variables['project_msg'].append("GPAI model fails the transparency requirements under Article 53.") return dispositive_variables def run_compliance_analysis_on_data(dispositive_variables, data_cc_yaml): card_label = data_cc_yaml['card_details']['card_label'] if not card_label in dispositive_variables['data_cc_non-compliant']: dispositive_variables['data_cc_non-compliant'][card_label] = {"msg": []} if dispositive_variables['ai_project_type']["high_risk_ai_system"] == True: for key in data_cc_yaml['high_risk_ai_system_requirements']: if data_cc_yaml['high_risk_ai_system_requirements'][f'{key}']['value'] == False: dispositive_variables['data_cc_non-compliant'][card_label]['msg'].append(f"This high-risk AI system fails the {key} requirements under {data_cc_yaml['high_risk_ai_system_requirements'][f'{key}']['article']}.") dispositive_variables['project_cc_pass'] = False if dispositive_variables['ai_project_type']["gpai_model"] == True: for value in data_cc_yaml['gpai_model_requirements']: if data_cc_yaml['gpai_model_requirements'][f'{value}'] == False: dispositive_variables['data_cc_non-compliant'][card_label]['msg'].append(f"") return dispositive_variables def run_compliance_analysis_on_model(dispositive_variables, model_cc_yaml): card_label = model_cc_yaml['card_details']['card_label'] if not card_label in dispositive_variables['model_cc_non-compliant']: dispositive_variables['data_cc_non-compliant'][card_label] = {"msg": []} # If project involves a high-risk AI system, then make sure all the relevant model requirements are met (relevant attributes are positive) if dispositive_variables['ai_project_type']["high_risk_ai_system"] == True: for value in model_cc_yaml['high_risk_ai_system_requirements']: if model_cc_yaml['high_risk_ai_system_requirements'][f'{value}'] == False: dispositive_variables['data_cc_non-compliant'][card_label]['msg'].append(f"This high-risk AI system fails the {key} requirements under {model_cc_yaml['high_risk_ai_system_requirements'][f'{key}']['article']}.") dispositive_variables['project_cc_pass'] = False # If project involves a GPAI model, then make sure all the relevant model requirements are met (relevant attributes are positive) if dispositive_variables['ai_project_type']["gpai_model"] == True: for key in model_cc_yaml['gpai_model_requirements']: if model_cc_yaml['gpai_model_requirements'][f'{key}']['value'] == False: dispositive_variables['data_cc_non-compliant'][card_label]['msg'].append(f"This high-risk AI system fails the {key} requirements under {model_cc_yaml['gpai_model_requirements'][f'{key}']['article']}.") dispositive_variables['project_cc_pass'] = False # If the GPAI model additionally carries systemic risk, then make sure all the relevant model requirements are met (relevant attributes are positive) if dispositive_variables['ai_project_type']["gpai_model_systemic_risk"] == True: for key in model_cc_yaml['gpai_model_with_systemic_risk_requirements']: if model_cc_yaml['gpai_model_with_systemic_risk_requirements'][f'{key}']['value'] == False: dispositive_variables['data_cc_non-compliant'][card_label]['msg'].append(f"This high-risk AI system fails the {key} requirements under {model_cc_yaml['gpai_model_with_systemic_risk_requirements'][f'{key}']['article']}.") dispositive_variables['project_cc_pass'] = False return dispositive_variables def check_intended_purpose(dispositive_variables, project_cc, other_cc): project_intended_purposes = [] dataset_intended_purposes = [] model_intended_purposes = [] if dispositive_variables['ai_project_type']['high_risk_ai_system'] == False: dispositive_variables['project_msg'].append(f"not high-risk") return dispositive_variables if dispositive_variables['ai_project_type']['high_risk_ai_system'] == True: for key in project_cc['high_risk_ai_system']: if project_cc['high_risk_ai_system'][f'{key}']['value']: project_intended_purposes.append(key) # data intended purposes if other_cc['card_details']['card_type'] == 'data': data_cc = other_cc card_label = data_cc['card_details']['card_label'] if not card_label in dispositive_variables['data_cc_non-compliant']: dispositive_variables['data_cc_non-compliant'][card_label] = {"msg": []} for key in data_cc['intended_purpose']: if data_cc['intended_purpose'][f'{key}']['value']: dataset_intended_purposes.append(key) for purpose in project_intended_purposes: if purpose not in dataset_intended_purposes: # dispositive_variables['project_msg'].append(f"You are not compliant because {purpose} is not a valid purpose for {data_cc['card_details']['card_label']}") if not "intended_purpose" in dispositive_variables['data_cc_non-compliant'][card_label]: dispositive_variables['data_cc_non-compliant'][card_label]["intended_purpose"] = [] dispositive_variables['data_cc_non-compliant'][card_label]['intended_purpose'].append((f"{purpose}")) else: dispositive_variables['data_cc_compliant'].append(data_cc['card_details']['card_label']) # model intended purposes if other_cc['card_details']['card_type'] == 'model': model_cc = other_cc card_label = model_cc['card_details']['card_label'] if not card_label in dispositive_variables['model_cc_non-compliant']: dispositive_variables['model_cc_non-compliant'][card_label] = {"msg": []} for key in model_cc['intended_purpose']: if model_cc['intended_purpose'][f'{key}']['value']: model_intended_purposes.append(key) for purpose in project_intended_purposes: if purpose not in model_intended_purposes: # dispositive_variables['project_msg'].append(f"You are not compliant because {purpose} is not a valid purpose for {model_cc['card_details']['card_label']}") if not "intended_purpose" in dispositive_variables['model_cc_non-compliant'][card_label]: dispositive_variables['model_cc_non-compliant'][card_label]["intended_purpose"] = [] dispositive_variables['model_cc_non-compliant'][card_label]['intended_purpose'].append((f"{purpose}")) else: dispositive_variables['model_cc_compliant'].append(model_cc['card_details']['card_label']) dispositive_variables['project_intended_purposes'] = project_intended_purposes return dispositive_variables