o
    (fw                     @   s  d dl Z d dlZd dlZd dlmZ d dlmZmZmZm	Z	m
Z
 d dlm  mZ d dlmZmZmZmZmZmZmZmZmZ d dlmZ d dlmZ d dlmZ d dlm Z  d d	l!m"Z"m#Z# d d
l$m%Z% d dl&m'Z' dZ(dZ)dZ*dZ+dZ,dZ-dZ.dZ/dZ0dZ1dZ2dddddZ3e4 Z5e6e7e8Z9edddgZ:G dd de%j;Z<de=fdd Z>	!dHd"ed#ee?ef d$ee?ef d%e@d&e@d'dfd(d)ZA	*	!dId"ed+ee?ef d,ee?ef d%e@d&e@d'e
ee@f fd-d.ZBd/ejCd'ejDfd0d1ZEd"efd2d3ZFd"ed'ee fd4d5ZGd"ed6e?d'ee?ef fd7d8ZHd9ee?e?f d:ee?e?f d'eIfd;d<ZJ	dJd=ee?ef d>e?d?e?de	e? d'eeIee?ef f f
d@dAZK		dKd+ee?ef dBe	e? de	e? d'dfdCdDZLd"edEee?ef d'ee: fdFdGZMdS )L    N)
namedtuple)AnyDictListOptionalTuple)	cloudsevent_logger
exceptionshttpmessagessecret_managersystemutilversion)_enabled_services)_is_attached)UAConfig)ATTACH_FAIL_DATE_FORMAT)attachment_data_filemachine_id_file)serviceclient)get_user_or_root_log_file_pathz/v1/context/machines/tokenz3/v1/contracts/{contract}/context/machines/{machine}z/v1/resourcesz3/v1/resources/{resource}/context/machines/{machine}z/v1/clouds/{cloud_type}/tokenz3/v1/contracts/{contract}/machine-activity/{machine}z/v1/contractz/v1/magic-attach            )series_overridesseriescloudvariantEnableByDefaultServicenamer    c                       sj  e Zd ZdZ	d(dee ddf fddZeje	j
g dd	d(d	d
Zdeeef fddZdedeeef fddZeje	j
g dddejfddZ	d(dededee deeef fddZdd Zdedeeef fddZdeeef fddZdefdd Z	d(ded!edee deeef fd"d#Z	d(ded!edee defd$d%Zd&d' Z  ZS ))UAContractClientcontract_urlNcfgreturnc                    s   t  j|d t | _d S )Nr%   )super__init__mtfget_machine_token_filemachine_token_file)selfr%   	__class__ 3/usr/lib/python3/dist-packages/uaclient/contract.pyr)   E   s   zUAContractClient.__init__)r   r   r   )retry_sleepsc                 C   s   |st | j}|  }|dd|i |  }| |d< ||d}t|}| j	t
||d}|jdkr:t |jdkrCt| |jdkrRtjt
|j|jd	|j}	tj|	d
d |	dg D ]}
tj|
dd qe|	S )a}  Requests machine attach to the provided machine_id.

        @param contract_token: Token string providing authentication to
            ContractBearer service endpoint.
        @param machine_id: Optional unique system machine id. When absent,
            contents of /etc/machine-id will be used.

        @return: Dict of the JSON response containing the machine-token.
        Authorization	Bearer {}lastAttachment	machineIdactivityInfo)dataheaders  i     urlcodebodymachineToken resourceTokenstoken)r   get_machine_idr%   r:   updateformat_get_activity_info	isoformat_support_old_machine_inforequest_urlAPI_V1_ADD_CONTRACT_MACHINEr?   r
   AttachInvalidTokenError_raise_attach_forbidden_messageContractAPIErrorr@   	json_dictr   secrets
add_secretget)r-   contract_tokenattachment_dt
machine_idr:   activity_infor9   backcompat_dataresponseresponse_jsonrD   r0   r0   r1   add_contract_machineL   s8   




z%UAContractClient.add_contract_machinec                 C   sT   |   }| jt|d |d |d |d dd}|jdkr'tjt|j|jd|jS )	z=Requests list of entitlements available to this machine type.architecturer   kernelvirtr\   r   r]   r^   )query_paramsr<   r=   )rH   rK   API_V1_AVAILABLE_RESOURCESr?   r
   rO   r@   rP   )r-   rW   rY   r0   r0   r1   available_resourcesw   s    
	z$UAContractClient.available_resourcesrT   c                 C   sN   |   }|dd|i | jt|d}|jdkr$tjt|j|jd|j	S )Nr3   r4   r:   r<   r=   )
r:   rF   rG   rK   API_V1_GET_CONTRACT_USING_TOKENr?   r
   rO   r@   rP   )r-   rT   r:   rY   r0   r0   r1   get_contract_using_token   s   
z)UAContractClient.get_contract_using_tokeninstancec                C   s~   | j tj|jd|jd}|jdkr0|jdd}|r&t	| t
j|dt
jt|j|jd|j}tj|dd |S )	zRequests contract token for auto-attach images for Pro clouds.

        @param instance: AutoAttachCloudInstance for the cloud.

        @return: Dict of the JSON response containing the contract-token.
        )
cloud_type)r9   r<   messagerB   )	error_msgr=   contractToken)rK   ,API_V1_GET_CONTRACT_TOKEN_FOR_CLOUD_INSTANCErG   rg   identity_docr?   rP   rS   LOGdebugr
   InvalidProImagerO   r@   r   rQ   rR   )r-   rf   rY   msgrZ   r0   r0   r1   %get_contract_token_for_cloud_instance   s*   



z6UAContractClient.get_contract_token_for_cloud_instancemachine_tokenresourcerV   c           	      C   s   |st | j}|  }|dd|i tj||d}| j||d}|jdkr3t	j
t|j|jd|jdrA|jd |jd< |j}|dg D ]}tj|d	d
 qJ|S )a  Requests machine access context for a given resource

        @param machine_token: The authentication token needed to talk to
            this contract service endpoint.
        @param resource: Entitlement name.
        @param machine_id: Optional unique system machine id. When absent,
            contents of /etc/machine-id will be used.

        @return: Dict of the JSON response containing entitlement accessInfo.
        r3   r4   )rs   machinerc   r<   r=   expiresrC   rD   rB   )r   rE   r%   r:   rF   rG   "API_V1_GET_RESOURCE_MACHINE_ACCESSrK   r?   r
   rO   r@   rS   rP   r   rQ   rR   )	r-   rr   rs   rV   r:   r>   rY   rZ   rD   r0   r0   r1   get_resource_machine_access   s(   
z,UAContractClient.get_resource_machine_accessc                 C   s   | j j}| j jd}t| j}|  }tj	||d}| 
 }|dd	|i | j|||d}|jdkrAtj||j|jd|jrU| j j}|j|d< | j | d	S d	S )
zReport current activity token and enabled services.

        This will report to the contracts backend all the current
        enabled services in the system.
        rA   contractrt   r3   r4   )r:   r9   r<   r=   r8   N)r,   contract_idrr   rS   r   rE   r%   rH   API_V1_UPDATE_ACTIVITY_TOKENrG   r:   rF   rK   r?   r
   rO   r@   rP   write)r-   rz   rr   rV   request_datar>   r:   rY   r0   r0   r1   update_activity_token   s*   


z&UAContractClient.update_activity_tokenmagic_tokenc                 C   s   |   }|dd|i | jt|d}|jdkrt |jdkr't |jdkr6tj	t|j|j
d|j}g d}|D ]}tj||d	 q?|S )
zRequest magic attach token info.

        When the magic token is registered, it will contain new fields
        that will allow us to know that the attach process can proceed
        r3   r4   rc   r;     r<   r=   rD   userCoderj   rB   )r:   rF   rG   rK   "API_V1_GET_MAGIC_ATTACH_TOKEN_INFOr?   r
   MagicAttachTokenErrorMagicAttachUnavailablerO   r@   rP   r   rQ   rR   rS   )r-   r   r:   rY   rZ   secret_fieldsfieldr0   r0   r1   get_magic_attach_token_info  s(   


z,UAContractClient.get_magic_attach_token_infoc                 C   sx   |   }| jt|dd}|jdkrt |jdkr$tjt|j|jd|j}g d}|D ]}t	j
||d q-|S )z)Create a magic attach token for the user.POSTr:   methodr   r<   r=   r   rB   )r:   rK   API_V1_NEW_MAGIC_ATTACHr?   r
   r   rO   r@   rP   r   rQ   rR   rS   )r-   r:   rY   rZ   r   r   r0   r0   r1   new_magic_attach_token$  s&   

z'UAContractClient.new_magic_attach_tokenc                 C   s   |   }|dd|i | jt|dd}|jdkrt |jdkr(t |jdkr1t	 |jdkr@tj
t|j|jd	d
S )z)Revoke a magic attach token for the user.r3   r4   DELETEr   i  r;   r   r<   r=   N)r:   rF   rG   rK   API_V1_REVOKE_MAGIC_ATTACHr?   r
    MagicAttachTokenAlreadyActivatedr   r   rO   r@   )r-   r   r:   rY   r0   r0   r1   revoke_magic_attach_token;  s(   



z*UAContractClient.revoke_magic_attach_tokenrz   c              	   C   s   |st | j}|  }|dd|i tj||d}|  }| j|d||d |d |d |d d	d
}|j	dkrFt
j||j	|jd|jdrT|jd |jd< |jS )a|  Get the updated machine token from the contract server.

        @param machine_token: The machine token needed to talk to
            this contract service endpoint.
        @param contract_id: Unique contract id provided by contract service
        @param machine_id: Optional unique system machine id. When absent,
            contents of /etc/machine-id will be used.
        r3   r4   rx   GETr\   r   r]   r^   r_   )r   r:   r`   r<   r=   ru   )r   rE   r%   r:   rF   rG   API_V1_GET_CONTRACT_MACHINErH   rK   r?   r
   rO   r@   rS   rP   )r-   rr   rz   rV   r:   r>   rW   rY   r0   r0   r1   get_contract_machineR  s4   

z%UAContractClient.get_contract_machinec           	      C   s   |st | j}|  }|dd|i ||  d}t|}tj||d}| j	||d|d}|j
dkr@tj||j
|jd|jd	rN|jd	 |jd	< |jS )
a  Request machine token refresh from contract server.

        @param machine_token: The machine token needed to talk to
            this contract service endpoint.
        @param contract_id: Unique contract id provided by contract service.
        @param machine_id: Optional unique system machine id. When absent,
            contents of /etc/machine-id will be used.

        @return: Dict of the JSON response containing refreshed machine-token
        r3   r4   r6   rx   r   )r:   r   r9   r<   r=   ru   )r   rE   r%   r:   rF   rG   rH   rJ   API_V1_UPDATE_CONTRACT_MACHINErK   r?   r
   rO   r@   rS   rP   )	r-   rr   rz   rV   r:   r9   rX   r>   rY   r0   r0   r1   update_contract_machine}  s*   

z(UAContractClient.update_contract_machinec                 C   s   t  jt  jt  jt  t  t  t	
 d}t| jjrOt| jj}t }| jjp4t | j| jjdd |D dd |D |rJ|j ndd}ni }i ||S )z9Return a dict of activity info data for contract requests)distributionr]   r   r\   desktopr^   clientVersionc                 S   s   g | ]}|j qS r0   )r"   .0servicer0   r0   r1   
<listcomp>  s    z7UAContractClient._get_activity_info.<locals>.<listcomp>c                 S   s   i | ]
}|j r|j|jqS r0   )variant_enabledr"   variant_namer   r0   r0   r1   
<dictcomp>  s    z7UAContractClient._get_activity_info.<locals>.<dictcomp>N)
activityIDactivityToken	resourcesresourceVariantsr5   )r   get_release_infor   get_kernel_infouname_releaser   get_dpkg_arch
is_desktopget_virt_typer   get_versionr   r%   is_attachedr   enabled_servicesr   readr,   activity_idrE   activity_tokenattached_atrI   )r-   machine_infor   attachment_datarW   r0   r0   r1   rH     s8   

z#UAContractClient._get_activity_infoN)__name__
__module____qualname__cfg_url_base_attrr   r   r)   r   retrysockettimeoutr[   r   strr   rb   re   r   AutoAttachCloudInstancerq   rw   r~   r   r   r   r   r   rH   __classcell__r0   r0   r.   r1   r#   B   sh    *$

&(

/
(r#   request_bodyc              	   C   sJ   |  di }|  d|| d| d| d| ddt jdd	S )
a?  
    Transforms a request_body that has the new activity_info into a body that
    includes both old and new forms of machineInfo/activityInfo

    This is necessary because there may be old ua-airgapped contract
    servers deployed that we need to support.
    This function is used for attach and refresh calls.
    r8   r7   r\   r   r]   r   Linux)r   r]   r   typerelease)r7   r8   r\   os)rS   r   r   r   )r   rW   r0   r0   r1   rJ     s   	rJ   Tr%   past_entitlementsnew_entitlementsallow_enabler   r&   c                 C   sj  ddl m} d}g }g }|| D ]|}	z||	 }
W n	 ty!   Y qw g }zt| ||	i |
||d\}}W nM tjy[ } zt| d}|	|	 t
d|	|
 W Y d}~qd}~w ty } zt| |	| |	|	 td|	|
 W Y d}~qd}~ww |r|rt|	 qt| t|dkrtjd	d
 t||D d|rtjdd
 |D ddS )a  Iterate over all entitlements in new_entitlement and apply any delta
    found according to past_entitlements.

    :param cfg: UAConfig instance
    :param past_entitlements: dict containing the last valid information
        regarding service entitlements.
    :param new_entitlements: dict containing the current information regarding
        service entitlements.
    :param allow_enable: Boolean set True if allowed to perform the enable
        operation. When False, a message will be logged to inform the user
        about the recommended enabled service.
    :param series_overrides: Boolean set True if series overrides should be
        applied to the new_access dict.
    r   )entitlements_enable_orderF)r%   orig_access
new_accessr   r   Tz+Failed to process contract delta for %s: %rNz5Unexpected error processing contract delta for %s: %rc                 S   s*   g | ]\}}|t jjt|t d fqS ))ri   log_path)r   UNEXPECTED_ERRORrG   r   r   )r   r"   	exceptionr0   r0   r1   r   /  s    z.process_entitlements_delta.<locals>.<listcomp>)failed_servicesc                 S   s   g | ]}|t jfqS r0   )r   !E_ATTACH_FAILURE_DEFAULT_SERVICES)r   r"   r0   r0   r1   r   <  s    )uaclient.entitlementsr   KeyErrorprocess_entitlement_deltarS   r
   UbuntuProErrorrm   r   appenderror	Exceptioneventservice_processedservices_failedlenAttachFailureUnknownErrorzipAttachFailureDefaultServices)r%   r   r   r   r   r   delta_errorunexpected_errorsr   r"   new_entitlementdeltasservice_enableder0   r0   r1   process_entitlements_delta  sr   







r   Fr   r   c              
   C   s   ddl m} |rt| t||}d}|rh|di d}|s*|di d}|s3tj||d|di di d	d
}	z	|| ||	d}
W n tjy_ } zt	
d| |d}~ww |
j|||d}||fS )a,  Process a entitlement access dictionary deltas if they exist.

    :param cfg: UAConfig instance
    :param orig_access: Dict with original entitlement access details before
        contract refresh deltas
    :param new_access: Dict with updated entitlement access details after
        contract refresh
    :param allow_enable: Boolean set True if allowed to perform the enable
        operation. When False, a message will be logged to inform the user
        about the recommended enabled service.
    :param series_overrides: Boolean set True if series overrides should be
        applied to the new_access dict.

    :raise UbuntuProError: on failure to process deltas.
    :return: A tuple containing a dict of processed deltas and a
             boolean indicating if the service was fully processed
    r   entitlement_factoryFentitlementr   )orignewentitlementsobligationsuse_selectorrB   r%   r"   r    z3Skipping entitlement deltas for "%s". No such classNr   )r   r   apply_contract_overridesr   get_dict_deltasrS   r
    InvalidContractDeltasServiceTypeEntitlementNotFoundErrorrm   rn   process_contract_deltas)r%   r   r   r   r   r   r   retr"   r    r   excr0   r0   r1   r   C  sD   

r   rY   c                 C   s   | j d}|rJ|d }|d }|dkr(|d t}tj|||d dd|dkr@|d t}tj|||d dd	|d
krJtj|dt )Ninfo
contractIdreasonzno-longer-effectivetimez%m-%d-%Y)rz   datecontract_expiry_dateznot-effective-yet)rz   r   contract_effective_dateznever-effective)rz   )	rP   rS   strftimer   r
   AttachForbiddenExpiredAttachForbiddenNotYetAttachForbiddenNeverAttachExpiredToken)rY   r   rz   r   r   r0   r0   r1   rN     s*   rN   c           	      C   s   t | }| }|j}|d }|d d d }t| d}|j||d}|| tj	  |
di 
dt| }t| t| || dd	 d
S )zRequest contract refresh from ua-contracts service.

    :raise UbuntuProError: on failure to update contract or error processing
        contract deltas
    :raise ConnectivityError: On failure during a connection
    rA   machineTokenInfocontractInfoidr'   )rr   rz   r7   Fr   N)r*   r+   r   rr   r#   r   r|   r   rE   cache_clearrS   r   r   )	r%   r,   orig_entitlements
orig_tokenrr   rz   contract_clientresprV   r0   r0   r1   refresh  s*   






r	  c                 C   s   t | }| }|dg S )zDQuery available resources from the contract server for this machine.r   )r#   rb   rS   )r%   clientr   r0   r0   r1   get_available_resources  s   r  rD   c                 C   s   t | }||S )z/Query contract information for a specific token)r#   re   )r%   rD   r
  r0   r0   r1   get_contract_information  s   
r  override_selectorselector_valuesc                 C   s<   d}|   D ]\}}||f|  vr dS |t| 7 }q|S )Nr   )itemsOVERRIDE_SELECTOR_WEIGHTS)r  r  override_weightselectorvaluer0   r0   r1   _get_override_weight  s   r  r   series_namerg   c           
      C   sz   i }||d}|r||d< |  di  |i }|r||td < t| dg }|D ]}t| d|}	|	r:|||	< q*|S )N)r   r   r    r   r   	overridesr  )popr  copydeepcopyrS   r  )
r   r  rg   r    r  r  r   general_overridesoverrideweightr0   r0   r1   _select_overrides  s"   


r  r   c                 C   s   ddl m} tt| td| v gstd| |du r!t j	n|}| \}}| 
di }t||||}t| D ]%\}	}
|
 D ]\}}| d 
|}t|trY|| qC|| d |< qCq;dS )a  Apply series-specific overrides to an entitlement dict.

    This function mutates orig_access dict by applying any series-overrides to
    the top-level keys under 'entitlement'. The series-overrides are sparse
    and intended to supplement existing top-level dict values. So, sub-keys
    under the top-level directives, obligations and affordance sub-key values
    will be preserved if unspecified in series-overrides.

    To more clearly indicate that orig_access in memory has already had
    the overrides applied, the 'series' key is also removed from the
    orig_access dict.

    :param orig_access: Dict with original entitlement access details
    r   )get_cloud_typer   z?Expected entitlement access dict. Missing "entitlement" key: {}N)uaclient.clouds.identityr  all
isinstancedictRuntimeErrorrG   r   r   r   rS   r  sortedr  rF   )r   r   r    r  r  rg   _orig_entitlementr  _weightoverrides_to_applykeyr  currentr0   r0   r1   r     s*   

r   r   c              	   C   s   ddl m} g }| D ]H\}}|di dd}z	|| ||d}W n
 tjy-   Y qw |di di }|d}	|||	rT| \}
}|
rT|t	||d	 q|S )
Nr   r   r   r   rB   r   r   resourceToken)r"   r    )
r   r   r  rS   r
   r   _should_enable_by_default
can_enabler   r!   )r%   r   r   enable_by_default_servicesent_name	ent_valuer    entr   r+  r-  r%  r0   r0   r1   get_enabled_by_default_services'  s,   
r2  )T)FTr   )NN)Nr  loggingr   collectionsr   typingr   r   r   r   r   uaclient.files.machine_tokenfilesrr   r*   uaclientr   r	   r
   r   r   r   r   r   r   -uaclient.api.u.pro.status.enabled_services.v1r   (uaclient.api.u.pro.status.is_attached.v1r   uaclient.configr   uaclient.defaultsr   uaclient.files.state_filesr   r   uaclient.httpr   uaclient.logr   rL   r   r   ra   rv   rk   r{   rd   r   r   r   r  get_event_loggerr   	getLoggerreplace_top_level_logger_namer   rm   r!   UAServiceClientr#   r"  rJ   r   boolr   r   HTTPResponseNamedMessagerN   r	  r  r  intr  r  r   r2  r0   r0   r0   r1   <module>   s    ,    


^



?
!






1
