diff --git a/docs/_static/i18n.csv b/docs/_static/i18n.csv index 45a581b6..0d82e86e 100644 --- a/docs/_static/i18n.csv +++ b/docs/_static/i18n.csv @@ -140,10 +140,49 @@ parties/contactPoint/telephone,Telephone,True, parties/contactPoint/faxNumber,Fax number,True, parties/contactPoint/url,URL,False, parties/roles,Party roles,False, +parties/beneficialOwners,Beneficial owners,False, +parties/beneficialOwners,Person,False, +parties/beneficialOwners/id,ID,True, +parties/beneficialOwners/name,Name,True, +parties/beneficialOwners/identifier,Identifier,False, +parties/beneficialOwners/identifier,Identifier,False, +parties/beneficialOwners/identifier/scheme,Scheme,True, +parties/beneficialOwners/identifier/id,ID,True, +parties/beneficialOwners/identifier/legalName,Legal Name,True, +parties/beneficialOwners/identifier/uri,URI,False, +parties/beneficialOwners/nationalities,Nationalities,False, +parties/beneficialOwners/address,Address,False, +parties/beneficialOwners/address,Address,False, +parties/beneficialOwners/address/streetAddress,Street address,True, +parties/beneficialOwners/address/locality,Locality,True, +parties/beneficialOwners/address/region,Region,True, +parties/beneficialOwners/address/postalCode,Postal code,True, +parties/beneficialOwners/address/countryName,Country name,True, +parties/beneficialOwners/email,Email,False, +parties/beneficialOwners/faxNumber,Fax number,True, +parties/beneficialOwners/telephone,Telephone,True, +parties/beneficialOwners/jobTitle,Job title,True, parties/people,People,False, parties/people,Person,False, -parties/people/id,Identifier,True, +parties/people/id,ID,True, parties/people/name,Name,True, +parties/people/identifier,Identifier,False, +parties/people/identifier,Identifier,False, +parties/people/identifier/scheme,Scheme,True, +parties/people/identifier/id,ID,True, +parties/people/identifier/legalName,Legal Name,True, +parties/people/identifier/uri,URI,False, +parties/people/nationalities,Nationalities,False, +parties/people/address,Address,False, +parties/people/address,Address,False, +parties/people/address/streetAddress,Street address,True, +parties/people/address/locality,Locality,True, +parties/people/address/region,Region,True, +parties/people/address/postalCode,Postal code,True, +parties/people/address/countryName,Country name,True, +parties/people/email,Email,False, +parties/people/faxNumber,Fax number,True, +parties/people/telephone,Telephone,True, parties/people/jobTitle,Job title,True, publicAuthority,Public authority,False, publicAuthority,Organization reference,False, diff --git a/docs/examples/blank.json b/docs/examples/blank.json index 488b17a2..959870f3 100644 --- a/docs/examples/blank.json +++ b/docs/examples/blank.json @@ -165,6 +165,29 @@ "name": "", "jobTitle": "" } + ], + "beneficialOwners": [ + { + "id": "string", + "name": "string", + "identifier": { + "scheme": "string", + "id": "string" + }, + "nationalities": [ + "string" + ], + "address": { + "streetAddress": "string", + "locality": "string", + "region": "string", + "postalCode": "string", + "countryName": "string" + }, + "email": "string", + "faxNumber": "string", + "telephone": "string" + } ] } ], diff --git a/docs/examples/example.json b/docs/examples/example.json index 0e501ca7..328e16cd 100644 --- a/docs/examples/example.json +++ b/docs/examples/example.json @@ -312,6 +312,29 @@ "roles": [ "supplier", "tenderer" + ], + "beneficialOwners": [ + { + "id": "1", + "name": "Juan Perez", + "identifier": { + "scheme": "PRY-IDCARD", + "id": "12345" + }, + "nationalities": [ + "PY" + ], + "address": { + "streetAddress": "Avenida Eusebio Ayala 1347", + "locality": "Asunción", + "region": "Gran Asunción", + "postalCode": "1001", + "countryName": "Paraguay" + }, + "email": "jperez@example.com", + "faxNumber": "+595210000001", + "telephone": "+595210000000" + } ] }, { diff --git a/docs/reference/codelists.md b/docs/reference/codelists.md index f1246301..acc262b0 100644 --- a/docs/reference/codelists.md +++ b/docs/reference/codelists.md @@ -67,6 +67,13 @@ Projects with a `status` of 'completed' may be displayed in a list of archived p :file: ../../build/current_lang/codelists/milestoneType.csv ``` +### country + +```{csv-table-no-translate} +:header-rows: 1 +:file: ../../build/current_lang/codelists/country.csv +``` + ## Open codelists ### DocumentType diff --git a/docs/reference/schema.md b/docs/reference/schema.md index 23e72b19..9f92ce4b 100644 --- a/docs/reference/schema.md +++ b/docs/reference/schema.md @@ -456,7 +456,7 @@ Each `Organization` has the following fields: ```{jsonschema} ../../build/current_lang/project-schema.json :pointer: /definitions/Organization -:collapse: name,id,identifier,additionalIdentifiers,address,contactPoint,roles,people +:collapse: name,id,identifier,additionalIdentifiers,address,contactPoint,roles,beneficialOwners,people :addtargets: ``` @@ -574,6 +574,7 @@ When working with data, users ought to be aware that addresses might not always This sub-schema is referenced by the following properties: * [`Location/address`](project-schema.json,/definitions/Location,address) * [`Organization/address`](project-schema.json,/definitions/Organization,address) +* [`Person/address`](project-schema.json,/definitions/Person,address) Each `Address` has the following fields: @@ -734,6 +735,7 @@ For example, if identifying a company in Colombia, look up its identifier in the This sub-schema is referenced by the following properties: * [`Organization/identifier`](project-schema.json,/definitions/Organization,identifier) * [`Organization/additionalIdentifiers`](project-schema.json,/definitions/Organization,additionalIdentifiers) +* [`Person/identifier`](project-schema.json,/definitions/Person,identifier) Each `Identifier` has the following fields: @@ -891,6 +893,7 @@ Use this object when you need to disclose the details of people associated with, ``` This sub-schema is referenced by the following properties: +* [`Organization/beneficialOwners`](project-schema.json,/definitions/Organization,beneficialOwners) * [`Organization/people`](project-schema.json,/definitions/Organization,people) Each `Person` has the following fields: @@ -901,7 +904,7 @@ Each `Person` has the following fields: ```{jsonschema} ../../build/current_lang/project-schema.json :pointer: /definitions/Person -:collapse: id,name,jobTitle +:collapse: id,name,identifier,nationalities,address,email,faxNumber,telephone,jobTitle :addtargets: ``` diff --git a/manage.py b/manage.py index 781437cb..10bd0cf6 100755 --- a/manage.py +++ b/manage.py @@ -334,21 +334,32 @@ def update_sub_schema_reference(schema): ]) # Paths that don't appear in the example data at all - paths_to_skip = ['forecasts/0/observations/0/value', 'metrics/0/observations/0/value'] + paths_to_skip = ['forecasts/0/observations/0/value', + 'metrics/0/observations/0/value', + 'parties/0/beneficialOwners/0', + 'parties/0/people/0/address', + 'parties/0/people/0/identifier'] # Add examples definition["references"] = get_definition_references(schema, defn) for ref in definition["references"]: - if ref[0] not in schema['definitions'] and '/'.join(ref) not in paths_to_skip: - if ref[-1] == '0': - ref.pop(-1) + if ref[0] not in schema['definitions']: + skip = False - definition["content"].extend([ - "```{jsoninclude} ../../docs/examples/example.json\n", - f":jsonpointer: /projects/0/{'/'.join(ref)}\n", - f":title: {'/'.join(ref)}\n", - "```\n\n" - ]) + for path_to_skip in paths_to_skip: + if path_to_skip == '/'.join(ref)[:len(path_to_skip)]: + skip = True + + if not skip: + if ref[-1] == '0': + ref.pop(-1) + + definition["content"].extend([ + "```{jsoninclude} ../../docs/examples/example.json\n", + f":jsonpointer: /projects/0/{'/'.join(ref)}\n", + f":title: {'/'.join(ref)}\n", + "```\n\n" + ]) definition["content"].extend([ "````\n\n", @@ -453,7 +464,8 @@ def copy_element(name, replacements=None, root='definitions'): ocds_base_url = 'https://standard.open-contracting.org/1.1/en/' - builder = ProfileBuilder('1__1__5', {'budget': 'master', 'transaction_milestones': 'master'}) + builder = ProfileBuilder('1__1__5', + {'budget': 'master', 'transaction_milestones': 'master', 'beneficialOwners': 'master'}) ppp_schema = get(f'{ppp_base_url}release-schema.json').json() ppp_schema = builder.patched_release_schema(schema=ppp_schema) @@ -474,6 +486,7 @@ def copy_element(name, replacements=None, root='definitions'): 'relatedProjectScheme.csv', 'relatedProject.csv', 'classificationScheme.csv', + 'country.csv', } ocds_codelists = { 'currency.csv', @@ -499,7 +512,6 @@ def copy_element(name, replacements=None, root='definitions'): 'LinkedRelease', # Similar to linked release in OCDS 'Modification', 'RelatedProject', # Similar to relatedProcess in OCDS - 'Person', 'SimpleIdentifier', } ocds_definitions = { @@ -519,6 +531,7 @@ def copy_element(name, replacements=None, root='definitions'): 'Transaction', 'Milestone', 'MilestoneReference', + 'Person', } compare(schema['definitions'], infra_definitions, ocds_definitions, 'schema/project-level/project-schema.json#/definitions', 'definitions') @@ -750,6 +763,14 @@ def copy_element(name, replacements=None, root='definitions'): # Original from standard: "The title of the milestone being referenced, this must match the title of a milestone described elsewhere in a release about this contracting process." # noqa: E501 schema['definitions']['MilestoneReference']['properties']['title']['description'] = "The title of the milestone being referenced, this must match the title of a milestone in this project or contracting process's `.milestones`." # noqa: E501 + copy_element('Person') + schema['definitions']['Person']['properties']['jobTitle'] = { + "title": "Job title", + "description": "The job title of the person (for example, Financial Manager).", + "type": "string", + "minLength": 1 + } + remove_null_and_pattern_properties(schema) remove_integer_identifier_types(schema) remove_deprecated_properties(schema) diff --git a/schema/project-level/codelists/country.csv b/schema/project-level/codelists/country.csv new file mode 100644 index 00000000..67670fcd --- /dev/null +++ b/schema/project-level/codelists/country.csv @@ -0,0 +1,251 @@ +Code,Title +AD,Andorra +AE,United Arab Emirates +AF,Afghanistan +AG,Antigua and Barbuda +AI,Anguilla +AL,Albania +AM,Armenia +AO,Angola +AQ,Antarctica +AR,Argentina +AS,American Samoa +AT,Austria +AU,Australia +AW,Aruba +AX,Åland Islands +AZ,Azerbaijan +BA,Bosnia and Herzegovina +BB,Barbados +BD,Bangladesh +BE,Belgium +BF,Burkina Faso +BG,Bulgaria +BH,Bahrain +BI,Burundi +BJ,Benin +BL,Saint Barthélemy +BM,Bermuda +BN,Brunei Darussalam +BO,Bolivia (Plurinational State of) +BQ,"Bonaire, Sint Eustatius and Saba" +BR,Brazil +BS,Bahamas +BT,Bhutan +BV,Bouvet Island +BW,Botswana +BY,Belarus +BZ,Belize +CA,Canada +CC,Cocos (Keeling) Islands +CD,Congo (the Democratic Republic of the) +CF,Central African Republic +CG,Congo +CH,Switzerland +CI,Côte d'Ivoire +CK,Cook Islands +CL,Chile +CM,Cameroon +CN,China +CO,Colombia +CR,Costa Rica +CU,Cuba +CV,Cabo Verde +CW,Curaçao +CX,Christmas Island +CY,Cyprus +CZ,Czechia +DE,Germany +DJ,Djibouti +DK,Denmark +DM,Dominica +DO,Dominican Republic +DZ,Algeria +EC,Ecuador +EE,Estonia +EG,Egypt +EH,Western Sahara +ER,Eritrea +ES,Spain +ET,Ethiopia +FI,Finland +FJ,Fiji +FK,Falkland Islands [Malvinas] +FM,Micronesia (Federated States of) +FO,Faroe Islands +FR,France +GA,Gabon +GB,United Kingdom of Great Britain and Northern Ireland +GD,Grenada +GE,Georgia +GF,French Guiana +GG,Guernsey +GH,Ghana +GI,Gibraltar +GL,Greenland +GM,Gambia +GN,Guinea +GP,Guadeloupe +GQ,Equatorial Guinea +GR,Greece +GS,South Georgia and the South Sandwich Islands +GT,Guatemala +GU,Guam +GW,Guinea-Bissau +GY,Guyana +HK,Hong Kong +HM,Heard Island and McDonald Islands +HN,Honduras +HR,Croatia +HT,Haiti +HU,Hungary +ID,Indonesia +IE,Ireland +IL,Israel +IM,Isle of Man +IN,India +IO,British Indian Ocean Territory +IQ,Iraq +IR,Iran (Islamic Republic of) +IS,Iceland +IT,Italy +JE,Jersey +JM,Jamaica +JO,Jordan +JP,Japan +KE,Kenya +KG,Kyrgyzstan +KH,Cambodia +KI,Kiribati +KM,Comoros +KN,Saint Kitts and Nevis +KP,Korea (the Democratic People's Republic of) +KR,Korea (the Republic of) +KW,Kuwait +KY,Cayman Islands +KZ,Kazakhstan +LA,Lao People's Democratic Republic +LB,Lebanon +LC,Saint Lucia +LI,Liechtenstein +LK,Sri Lanka +LR,Liberia +LS,Lesotho +LT,Lithuania +LU,Luxembourg +LV,Latvia +LY,Libya +MA,Morocco +MC,Monaco +MD,Moldova (the Republic of) +ME,Montenegro +MF,Saint Martin (French part) +MG,Madagascar +MH,Marshall Islands +MK,North Macedonia +ML,Mali +MM,Myanmar +MN,Mongolia +MO,Macao +MP,Northern Mariana Islands +MQ,Martinique +MR,Mauritania +MS,Montserrat +MT,Malta +MU,Mauritius +MV,Maldives +MW,Malawi +MX,Mexico +MY,Malaysia +MZ,Mozambique +NA,Namibia +NC,New Caledonia +NE,Niger +NF,Norfolk Island +NG,Nigeria +NI,Nicaragua +NL,Netherlands +NO,Norway +NP,Nepal +NR,Nauru +NU,Niue +NZ,New Zealand +OM,Oman +PA,Panama +PE,Peru +PF,French Polynesia +PG,Papua New Guinea +PH,Philippines +PK,Pakistan +PL,Poland +PM,Saint Pierre and Miquelon +PN,Pitcairn +PR,Puerto Rico +PS,"Palestine, State of" +PT,Portugal +PW,Palau +PY,Paraguay +QA,Qatar +RE,Réunion +RO,Romania +RS,Serbia +RU,Russian Federation +RW,Rwanda +SA,Saudi Arabia +SB,Solomon Islands +SC,Seychelles +SD,Sudan +SE,Sweden +SG,Singapore +SH,"Saint Helena, Ascension and Tristan da Cunha" +SI,Slovenia +SJ,Svalbard and Jan Mayen +SK,Slovakia +SL,Sierra Leone +SM,San Marino +SN,Senegal +SO,Somalia +SR,Suriname +SS,South Sudan +ST,Sao Tome and Principe +SV,El Salvador +SX,Sint Maarten (Dutch part) +SY,Syrian Arab Republic +SZ,Eswatini +TC,Turks and Caicos Islands +TD,Chad +TF,French Southern Territories +TG,Togo +TH,Thailand +TJ,Tajikistan +TK,Tokelau +TL,Timor-Leste +TM,Turkmenistan +TN,Tunisia +TO,Tonga +TR,Turkey +TT,Trinidad and Tobago +TV,Tuvalu +TW,Taiwan (Province of China) +TZ,"Tanzania, the United Republic of" +UA,Ukraine +UG,Uganda +UM,United States Minor Outlying Islands +US,United States of America +UY,Uruguay +UZ,Uzbekistan +VA,Holy See +VC,Saint Vincent and the Grenadines +VE,Venezuela (Bolivarian Republic of) +VG,Virgin Islands (British) +VI,Virgin Islands (U.S.) +VN,Viet Nam +VU,Vanuatu +WF,Wallis and Futuna +WS,Samoa +XK,Kosovo +YE,Yemen +YT,Mayotte +ZA,South Africa +ZM,Zambia +ZW,Zimbabwe diff --git a/schema/project-level/project-schema.json b/schema/project-level/project-schema.json index 31606722..88c485db 100644 --- a/schema/project-level/project-schema.json +++ b/schema/project-level/project-schema.json @@ -1244,6 +1244,16 @@ "uniqueItems": true, "minItems": 1 }, + "beneficialOwners": { + "title": "Beneficial owners", + "description": "The beneficial owners of the organization.", + "type": "array", + "items": { + "$ref": "#/definitions/Person" + }, + "uniqueItems": true, + "minItems": 1 + }, "people": { "title": "People", "description": "People associated with, representing, or working on behalf of this organization in respect of this project.", @@ -1773,17 +1783,323 @@ "title": "Person", "description": "A natural person.", "type": "object", + "required": [ + "id" + ], "properties": { "id": { - "title": "Identifier", - "description": "A local identifier for this person.", + "title": "ID", + "description": "A local identifier for the person.", "type": "string", "minLength": 1 }, "name": { "title": "Name", "description": "The full name of the person.", - "type": "string", + "type": [ + "string" + ], + "minLength": 1 + }, + "identifier": { + "title": "Identifier", + "description": "An official identifier for the person. The scheme should follow the pattern {JURISDICTION}-{TYPE}, where JURISDICTION is a three-letter, uppercase ISO 3166-1 alpha-3 country code, and TYPE is one of \"PASSPORT\", \"TAXID\" or \"IDCARD\".", + "$ref": "#/definitions/Identifier" + }, + "nationalities": { + "title": "Nationalities", + "description": "The nationalities of the person, from the closed country codelist.", + "type": [ + "array" + ], + "items": { + "type": "string", + "enum": [ + "AD", + "AE", + "AF", + "AG", + "AI", + "AL", + "AM", + "AO", + "AQ", + "AR", + "AS", + "AT", + "AU", + "AW", + "AX", + "AZ", + "BA", + "BB", + "BD", + "BE", + "BF", + "BG", + "BH", + "BI", + "BJ", + "BL", + "BM", + "BN", + "BO", + "BQ", + "BR", + "BS", + "BT", + "BV", + "BW", + "BY", + "BZ", + "CA", + "CC", + "CD", + "CF", + "CG", + "CH", + "CI", + "CK", + "CL", + "CM", + "CN", + "CO", + "CR", + "CU", + "CV", + "CW", + "CX", + "CY", + "CZ", + "DE", + "DJ", + "DK", + "DM", + "DO", + "DZ", + "EC", + "EE", + "EG", + "EH", + "ER", + "ES", + "ET", + "FI", + "FJ", + "FK", + "FM", + "FO", + "FR", + "GA", + "GB", + "GD", + "GE", + "GF", + "GG", + "GH", + "GI", + "GL", + "GM", + "GN", + "GP", + "GQ", + "GR", + "GS", + "GT", + "GU", + "GW", + "GY", + "HK", + "HM", + "HN", + "HR", + "HT", + "HU", + "ID", + "IE", + "IL", + "IM", + "IN", + "IO", + "IQ", + "IR", + "IS", + "IT", + "JE", + "JM", + "JO", + "JP", + "KE", + "KG", + "KH", + "KI", + "KM", + "KN", + "KP", + "KR", + "KW", + "KY", + "KZ", + "LA", + "LB", + "LC", + "LI", + "LK", + "LR", + "LS", + "LT", + "LU", + "LV", + "LY", + "MA", + "MC", + "MD", + "ME", + "MF", + "MG", + "MH", + "MK", + "ML", + "MM", + "MN", + "MO", + "MP", + "MQ", + "MR", + "MS", + "MT", + "MU", + "MV", + "MW", + "MX", + "MY", + "MZ", + "NA", + "NC", + "NE", + "NF", + "NG", + "NI", + "NL", + "NO", + "NP", + "NR", + "NU", + "NZ", + "OM", + "PA", + "PE", + "PF", + "PG", + "PH", + "PK", + "PL", + "PM", + "PN", + "PR", + "PS", + "PT", + "PW", + "PY", + "QA", + "RE", + "RO", + "RS", + "RU", + "RW", + "SA", + "SB", + "SC", + "SD", + "SE", + "SG", + "SH", + "SI", + "SJ", + "SK", + "SL", + "SM", + "SN", + "SO", + "SR", + "SS", + "ST", + "SV", + "SX", + "SY", + "SZ", + "TC", + "TD", + "TF", + "TG", + "TH", + "TJ", + "TK", + "TL", + "TM", + "TN", + "TO", + "TR", + "TT", + "TV", + "TW", + "TZ", + "UA", + "UG", + "UM", + "US", + "UY", + "UZ", + "VA", + "VC", + "VE", + "VG", + "VI", + "VN", + "VU", + "WF", + "WS", + "XK", + "YE", + "YT", + "ZA", + "ZM", + "ZW" + ] + }, + "codelist": "country.csv", + "openCodelist": false, + "uniqueItems": true, + "minItems": 1 + }, + "address": { + "title": "Address", + "description": "The address of the person.", + "$ref": "#/definitions/Address" + }, + "email": { + "title": "Email", + "description": "The e-mail address of the person.", + "type": [ + "string" + ], + "format": "email", + "minLength": 1 + }, + "faxNumber": { + "title": "Fax number", + "description": "The fax number of the person. This must include the international dialing code.", + "type": [ + "string" + ], + "minLength": 1 + }, + "telephone": { + "title": "Telephone", + "description": "The telephone number of the person. This must include the international dialing code.", + "type": [ + "string" + ], "minLength": 1 }, "jobTitle": { @@ -1793,9 +2109,6 @@ "minLength": 1 } }, - "required": [ - "id" - ], "minProperties": 1 }, "Transaction": {