Module mydata_did.v1_0.utils.util
DIDDoc utility methods.
Copyright 2017-2019 Government of Canada Public Services and Procurement Canada - buyandsell.gc.ca
Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License.
Expand source code
"""
DIDDoc utility methods.
Copyright 2017-2019 Government of Canada
Public Services and Procurement Canada - buyandsell.gc.ca
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
"""
import datetime
import semver
from urllib.parse import urlparse
from multibase import decode
if __name__ == "__main__":
    from mydata_did.v1_0.utils.regex import MYDATA_DID_PATTERN
else:
    from .regex import MYDATA_DID_PATTERN
def resource(ref: str, delimiter: str = None) -> str:
    """
    Extract the resource for an identifier.
    Given a (URI) reference, return up to its delimiter (exclusively), or all of it if
    there is none.
    Args:
        ref: reference
        delimiter: delimiter character
            (default None maps to '#', or ';' introduces identifiers)
    """
    return ref.split(delimiter if delimiter else "#")[0]
def derive_did_type(uri: str) -> str:
    mydata_did_pattern_match = MYDATA_DID_PATTERN.match(uri)
    if mydata_did_pattern_match:
        return mydata_did_pattern_match.group('did_type')
    return None
def canon_did(uri: str) -> str:
    """
    Convert a URI into a DID if need be, left-stripping 'did:mydata:' if present.
    Args:
        uri: input URI or DID
    Raises:
        ValueError: for invalid input.
    """
    if ok_did(uri):
        return uri
    if uri.startswith("did:mydata:"):
        mydata_did_pattern_match = MYDATA_DID_PATTERN.match(uri)
        if mydata_did_pattern_match:
            prefix_end = 13 if mydata_did_pattern_match.group(
                'did_type') else 11
            rv = uri[prefix_end:]
            if ok_did(rv):
                return rv
    raise ValueError(
        "Bad specification {} does not correspond to a MyData DID".format(uri)
    )
def canon_ref(did: str, ref: str, delimiter: str = None, did_type: str = None):
    """
    Given a reference in a DID document, return it in its canonical form of a URI.
    Args:
        did: DID acting as the identifier of the DID document
        ref: reference to canonicalize, either a DID or a fragment pointing to a
            location in the DID doc
        delimiter: delimiter character marking fragment (default '#') or
            introducing identifier (';') against DID resource
    """
    if not ok_did(did):
        raise ValueError(
            "Bad DID {} cannot act as DID document identifier".format(did))
    if ok_did(ref):  # e.g., LjgpST2rjsoxYegQDRm7EL
        return "did:mydata:{}".format(did) if not did_type else "did:mydata:{}:{}".format(did_type, did)
    if ok_did(resource(ref, delimiter)):  # e.g., LjgpST2rjsoxYegQDRm7EL#keys-1
        return "did:mydata:{}".format(ref) if not did_type else "did:mydata:{}:{}".format(did_type, did)
    if ref.startswith(
        "did:mydata:"
    ):  # e.g., did:mydata:LjgpST2rjsoxYegQDRm7EL, did:mydata:LjgpST2rjsoxYegQDRm7EL#3
        mydata_did_pattern_match = MYDATA_DID_PATTERN.match(
            resource(ref, delimiter))
        if mydata_did_pattern_match:
            prefix_end = 13 if mydata_did_pattern_match.group(
                'did_type') else 11
            rv = ref[prefix_end:]
            if ok_did(resource(rv, delimiter)):
                return ref
        raise ValueError(
            "Bad URI {} does not correspond to a MyData DID".format(ref))
    if urlparse(ref).scheme:  # e.g., https://example.com/messages/8377464
        return ref
    return "did:mydata:{}{}{}".format(did, delimiter if delimiter else "#", ref) if not did_type else "did:mydata:{}:{}{}{}".format(did_type, did, delimiter if delimiter else "#", ref)
def ok_did(token: str) -> bool:
    """
    Whether input token looks like a valid decentralized identifier.
    Args:
        token: candidate string
    Returns: whether input token looks like a valid schema identifier
    """
    try:
        return len(decode(token)) == 34 if token else False
    except ValueError:
        return False
def current_datetime_in_iso8601() -> str:
    """
    Return current datetime in ISO8601 format.
    """
    return str(datetime.datetime.utcnow().replace(tzinfo=datetime.timezone.utc).isoformat())
def str_to_bool(s: str) -> bool:
    """
    Convert a string to a boolean.
    Args:
        s: string to convert
    Returns: boolean value
    """
    if not isinstance(s, str):
        return False
    if s.lower() in ['true', 't', '1']:
        return True
    elif s.lower() in ['false', 'f', '0']:
        return False
    else:
        raise ValueError('Cannot convert string to boolean: {}'.format(s))
def bool_to_str(b: bool) -> str:
    """
    Convert a boolean to a string.
    Args:
        b: boolean to convert
    Returns: string value
    """
    return 'true' if b else 'false'
def int_to_semver_str(int_version: int) -> str:
    """
    Convert integer version to semver string.
    """
    return str(semver.VersionInfo(str(int_version)))
def comma_separated_str_to_list(s: str) -> list:
    """
    Convert a comma separated string to a list.
    Args:
        s: string to convert
    Returns: list value
    """
    return s.split(',')
def get_slices(page, page_size=10):
    """
    Get the start and end indices for the given page and page size.
    Args:
        page: page number
        page_size: page size
    Returns: start and end indices
    """
    start = (page - 1) * page_size
    end = start + page_size
    return start, end
if __name__ == "__main__":
    print(canon_did("did:mydata:0:z6MkfiSdYhnLnS6jfwSf2yS2CiwwjZGmFUFL5QbyL2Xu8z2E"))
Functions
def bool_to_str(b: bool) ‑> str- 
Convert a boolean to a string.
Args
b- boolean to convert
 
Returns: string value
Expand source code
def bool_to_str(b: bool) -> str: """ Convert a boolean to a string. Args: b: boolean to convert Returns: string value """ return 'true' if b else 'false' def canon_did(uri: str) ‑> str- 
Convert a URI into a DID if need be, left-stripping 'did:mydata:' if present.
Args
uri- input URI or DID
 
Raises
ValueError- for invalid input.
 
Expand source code
def canon_did(uri: str) -> str: """ Convert a URI into a DID if need be, left-stripping 'did:mydata:' if present. Args: uri: input URI or DID Raises: ValueError: for invalid input. """ if ok_did(uri): return uri if uri.startswith("did:mydata:"): mydata_did_pattern_match = MYDATA_DID_PATTERN.match(uri) if mydata_did_pattern_match: prefix_end = 13 if mydata_did_pattern_match.group( 'did_type') else 11 rv = uri[prefix_end:] if ok_did(rv): return rv raise ValueError( "Bad specification {} does not correspond to a MyData DID".format(uri) ) def canon_ref(did: str, ref: str, delimiter: str = None, did_type: str = None)- 
Given a reference in a DID document, return it in its canonical form of a URI.
Args
did- DID acting as the identifier of the DID document
 ref- reference to canonicalize, either a DID or a fragment pointing to a location in the DID doc
 delimiter- delimiter character marking fragment (default '#') or introducing identifier (';') against DID resource
 
Expand source code
def canon_ref(did: str, ref: str, delimiter: str = None, did_type: str = None): """ Given a reference in a DID document, return it in its canonical form of a URI. Args: did: DID acting as the identifier of the DID document ref: reference to canonicalize, either a DID or a fragment pointing to a location in the DID doc delimiter: delimiter character marking fragment (default '#') or introducing identifier (';') against DID resource """ if not ok_did(did): raise ValueError( "Bad DID {} cannot act as DID document identifier".format(did)) if ok_did(ref): # e.g., LjgpST2rjsoxYegQDRm7EL return "did:mydata:{}".format(did) if not did_type else "did:mydata:{}:{}".format(did_type, did) if ok_did(resource(ref, delimiter)): # e.g., LjgpST2rjsoxYegQDRm7EL#keys-1 return "did:mydata:{}".format(ref) if not did_type else "did:mydata:{}:{}".format(did_type, did) if ref.startswith( "did:mydata:" ): # e.g., did:mydata:LjgpST2rjsoxYegQDRm7EL, did:mydata:LjgpST2rjsoxYegQDRm7EL#3 mydata_did_pattern_match = MYDATA_DID_PATTERN.match( resource(ref, delimiter)) if mydata_did_pattern_match: prefix_end = 13 if mydata_did_pattern_match.group( 'did_type') else 11 rv = ref[prefix_end:] if ok_did(resource(rv, delimiter)): return ref raise ValueError( "Bad URI {} does not correspond to a MyData DID".format(ref)) if urlparse(ref).scheme: # e.g., https://example.com/messages/8377464 return ref return "did:mydata:{}{}{}".format(did, delimiter if delimiter else "#", ref) if not did_type else "did:mydata:{}:{}{}{}".format(did_type, did, delimiter if delimiter else "#", ref) def comma_separated_str_to_list(s: str) ‑> list- 
Convert a comma separated string to a list.
Args
s- string to convert
 
Returns: list value
Expand source code
def comma_separated_str_to_list(s: str) -> list: """ Convert a comma separated string to a list. Args: s: string to convert Returns: list value """ return s.split(',') def current_datetime_in_iso8601() ‑> str- 
Return current datetime in ISO8601 format.
Expand source code
def current_datetime_in_iso8601() -> str: """ Return current datetime in ISO8601 format. """ return str(datetime.datetime.utcnow().replace(tzinfo=datetime.timezone.utc).isoformat()) def derive_did_type(uri: str) ‑> str- 
Expand source code
def derive_did_type(uri: str) -> str: mydata_did_pattern_match = MYDATA_DID_PATTERN.match(uri) if mydata_did_pattern_match: return mydata_did_pattern_match.group('did_type') return None def get_slices(page, page_size=10)- 
Get the start and end indices for the given page and page size.
Args
page- page number
 page_size- page size
 
Returns: start and end indices
Expand source code
def get_slices(page, page_size=10): """ Get the start and end indices for the given page and page size. Args: page: page number page_size: page size Returns: start and end indices """ start = (page - 1) * page_size end = start + page_size return start, end def int_to_semver_str(int_version: int) ‑> str- 
Convert integer version to semver string.
Expand source code
def int_to_semver_str(int_version: int) -> str: """ Convert integer version to semver string. """ return str(semver.VersionInfo(str(int_version))) def ok_did(token: str) ‑> bool- 
Whether input token looks like a valid decentralized identifier.
Args
token- candidate string
 
Returns: whether input token looks like a valid schema identifier
Expand source code
def ok_did(token: str) -> bool: """ Whether input token looks like a valid decentralized identifier. Args: token: candidate string Returns: whether input token looks like a valid schema identifier """ try: return len(decode(token)) == 34 if token else False except ValueError: return False def resource(ref: str, delimiter: str = None) ‑> str- 
Extract the resource for an identifier.
Given a (URI) reference, return up to its delimiter (exclusively), or all of it if there is none.
Args
ref- reference
 delimiter- delimiter character (default None maps to '#', or ';' introduces identifiers)
 
Expand source code
def resource(ref: str, delimiter: str = None) -> str: """ Extract the resource for an identifier. Given a (URI) reference, return up to its delimiter (exclusively), or all of it if there is none. Args: ref: reference delimiter: delimiter character (default None maps to '#', or ';' introduces identifiers) """ return ref.split(delimiter if delimiter else "#")[0] def str_to_bool(s: str) ‑> bool- 
Convert a string to a boolean.
Args
s- string to convert
 
Returns: boolean value
Expand source code
def str_to_bool(s: str) -> bool: """ Convert a string to a boolean. Args: s: string to convert Returns: boolean value """ if not isinstance(s, str): return False if s.lower() in ['true', 't', '1']: return True elif s.lower() in ['false', 'f', '0']: return False else: raise ValueError('Cannot convert string to boolean: {}'.format(s))