U
    Jhi                     @   s   d dl Z 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mZ d dlmZ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mZmZ d dlmZ e  Z!e"e#e$Z%dZ&G d	d
 d
ej'Z(dS )    N)exists)AnyDictListOptionalTupleUnion)
apiaptcontractevent_logger
exceptionshttpmessagessecret_managersystemutil)base)ApplicationStatusCanDisableFailureCanDisableFailureReason)status_cache_filez<^linux-image-([\d]+[.-][\d]+[.-][\d]+-[\d]+-[A-Za-z0-9_-]+)$c                       s8  e Zd ZdZdZdZdZdZdZdZ	dZ
eeeedf ddd	Zeedd
dZeedddZeee dddZeee dddZeee dddZeee dddZedddZedddZeejedddZdCeeeee f d fddZ eddd Z!e"j#ed!d"d#Z$edd$d%Z%e"j#d&d'd(Z&e"j#d&d)d*Z'e"j#d&d+d,Z(d-d. Z)d/d0 Z*ee+ee,j- f dd1d2Z.d3d4 Z/dDe0ee1f e0ee1f eed5 fd6d7Z2dEe"j#eee  edd8d9d:Z3e"j#dd!d;d<Z4dFeee  d=d>d?Z5dGe"j#ed@dAdBZ6  Z7S )HRepoEntitlementz1/etc/apt/sources.list.d/ubuntu-{name}.{extension}z$/etc/apt/preferences.d/ubuntu-{name}z	{}/ubuntuNFT)returnc                 C   s   d S N selfr   r   </usr/lib/python3/dist-packages/uaclient/entitlements/repo.pyrepo_pin_priority;   s    z!RepoEntitlement.repo_pin_priorityc                 C   s.   d}t  j}|tjkrd}| jj| j|dS )NZsourceslist)name	extension)r   get_release_infoseriesr
   ZSERIES_NOT_USING_DEB822repo_file_tmplformatr!   )r   r"   r$   r   r   r   	repo_file?   s
    

zRepoEntitlement.repo_filec                 C   s
   | j d S )Nz {})repo_url_tmplr   r   r   r   repo_policy_check_tmplG   s    z&RepoEntitlement.repo_policy_check_tmplc                 C   s<   g }| j di }|r8|di }t|dg }|}|S )zdebs to install on enablemententitlement
directivesadditionalPackages)entitlement_cfggetcopy)r   packagesr*   r+   Zadditional_packagesr   r   r   r0   K   s    
zRepoEntitlement.packagesc                 C   s   | j di di dS )Nr*   r+   aptURLr-   r.   r   r   r   r   apt_url\   s     zRepoEntitlement.apt_urlc                 C   s   | j di di dg S )Nr*   r+   ZaptSnapshotURLsr2   r   r   r   r   snapshot_urlsd   s      zRepoEntitlement.snapshot_urlsc                 C   s   | j di di dS )Nr*   r+   suitesr2   r   r   r   r   
apt_suitesl   s     zRepoEntitlement.apt_suitesc                 C   sz   | j d}|sv| jjd }| j d di }|ds`t| j}||| j}|r`|d}|sv|}t	
d| j |S )NZresourceTokenZmachineTokenr*   obligationsZenableByDefaultzWNo resourceToken present in contract for service %s. Using machine token as credentials)r-   r.   machine_token_filemachine_tokenr   ZUAContractClientcfgZget_resource_machine_accessr!   LOGZwarningtitle)r   tokenr9   r7   ZclientZmachine_accessr   r   r   get_resource_tokent   s0    
 
 
z"RepoEntitlement.get_resource_tokenc                 C   s    t jt| jd}t| |S )z%Check if system needs to be rebooted.)Zinstalled_pkgs)r   Zshould_rebootsetr0   eventZneeds_reboot)r   Zreboot_requiredr   r   r   _check_for_reboot   s
    
z!RepoEntitlement._check_for_rebootc                 C   s   d S r   r   r   r   r   r   repo_key_file   s    zRepoEntitlement.repo_key_file)ignore_dependent_servicesr   c                    sV   t  j|d\}}|dkr"||fS | jsN| jrNdttjtjj	| j
| j
dfS ||fS )N)rC   Fentitlement_namer<   )supercan_disableoriginpurger   r   ZNO_PURGE_WITHOUT_ORIGINr   ZREPO_PURGE_FAIL_NO_ORIGINr&   r<   )r   rC   resultreason	__class__r   r   rG      s     
 
zRepoEntitlement.can_disablec                 C   s.   | j d k	ot| j dk}| js"|s&dS dS d S )Nr         )r0   lenaccess_only)r   Zwill_installr   r   r   enable_steps   s    
zRepoEntitlement.enable_steps)progressr   c                 C   sh   | tjj| jd | | | jrZ| jrZt| j	dkrd|
dtjjd| j	d n
| | dS )zEnable specific entitlement.

        @return: True on success, False otherwise.
        @raises: UbuntuProError on failure to install suggested packages
        servicer   info r0   T)rS   r   ZCONFIGURING_APT_ACCESSr&   r<   setup_apt_configZsupports_access_onlyrQ   rP   r0   emitZSKIPPING_INSTALLING_PACKAGESjoininstall_packages)r   rS   r   r   r   _perform_enable   s    


zRepoEntitlement._perform_enablec                 C   s   | j s
dS dS d S )NrN   rO   )rI   r   r   r   r   disable_steps   s    zRepoEntitlement.disable_steps)rS   c                 C   s   | j r| jr|dtj |dd t| j}| ||sBdS g }g }|D ]6}tj|| jd}|rz|	|t
|f qN|	| qN| |||sdS t| dr|   | | | j r| jr|tjj| jd | | | | dS )NrV    F)Zexclude_originremove_packagesr<   T)rI   rH   rZ   r   ZPURGE_EXPERIMENTALr
   Z get_installed_packages_by_originpurge_kernel_checkZget_remote_versions_for_packageappendmaxprompt_for_purgehasattrr`   remove_apt_configrS   ZPURGING_PACKAGESr&   r<   execute_reinstallexecute_removal)r   rS   Zrepo_origin_packagespackages_to_reinstallpackages_to_removepackageZalternativesr   r   r   _perform_disable   sH     
  



z RepoEntitlement._perform_disablec                    s   g  |D ]&}t t|j}|r |d q r| sDt |	dt
jj| jd |	dd  t j}|	dt
jj|d t } fdd|D }|s|	dt
j dS |	d	tjd
t
jifg dS )a*  
        Checks if the purge operation involves a kernel.

        When package called 'linux-image-*' is in the package list, warn the
        user that a kernel is being removed. Then, show the user what the
        current kernel is.

        If the current kernel is to be removed, and there are no other valid
        Ubuntu Kernels installed in the system, return False to abort the
        operation.

        If there is another Ubuntu kernel - besides the one installed - then
        prompt the user for confirmation before proceeding.
           rV   rT   rW   )Zkernel_versionc                    s   g | ]}| kr|qS r   r   ).0versionZlinux_image_versionsr   r   
<listcomp>4  s   z6RepoEntitlement.purge_kernel_check.<locals>.<listcomp>Fmessage_operationmsgT)researchRE_KERNEL_PKGr!   rc   groupZis_interactiver   Z#NonInteractiveKernelPurgeDisallowedrZ   r   ZPURGE_KERNEL_REMOVALr&   r<   r[   r   Zget_kernel_infoZuname_releaseZPURGE_CURRENT_KERNELZget_installed_ubuntu_kernelsZPURGE_NO_ALTERNATIVE_KERNELr   prompt_for_confirmationZPURGE_KERNEL_CONFIRMATION)r   package_listrS   rl   mZcurrent_kernelZinstalled_kernelsZalternative_kernelsr   rq   r   rb   	  sF    


z"RepoEntitlement.purge_kernel_checkc                 C   s   d}|r6| dtj | dtdd |D  d}|rh| dtj | dtdd |D  d}|r| dtjdtjifg dS )	NFrV   c                 S   s   g | ]
}|j qS r   r!   ro   rl   r   r   r   rr   V  s     z4RepoEntitlement.prompt_for_purge.<locals>.<listcomp>Tc                 S   s   g | ]\}}|j qS r   r|   )ro   rl   _r   r   r   rr   `  s     rs   rt   )rZ   r   ZWARN_PACKAGES_REMOVALr   Zcreate_package_list_strZWARN_PACKAGES_REINSTALLry   ZPROCEED_YES_NO)r   rk   rj   rS   promptr   r   r   re   J  s8    	z RepoEntitlement.prompt_for_purgec                    s8   t    fdd|D }|r4t |tjj|d d S )Nc                    s   g | ]}|j  kr|j qS r   r|   r}   Zinstalled_packagesr   r   rr   w  s   
z3RepoEntitlement.execute_removal.<locals>.<listcomp>rX   )r
   get_installed_packages_namesZpurge_packagesr   ZUNINSTALLING_PACKAGES_FAILEDr&   )r   rk   Z	to_remover   r   r   ri   q  s    
zRepoEntitlement.execute_removalc                    s,   t    fdd|D }|r(t | d S )Nc                    s*   g | ]"\}}|j  krd |j |jqS )z{}={})r!   r&   Zver_str)ro   rl   rp   r   r   r   rr     s   
z5RepoEntitlement.execute_reinstall.<locals>.<listcomp>)r
   r   Zreinstall_packages)r   rj   Zto_reinstallr   r   r   rh     s    
z!RepoEntitlement.execute_reinstallc           
      C   s   t jtjj| jdf}| j}|di di }|d}|sTt jtjj| jdfS |d}|sxt jtj	j| jdfS t
jtjd}|D ]8}t| j|||}|rt jtjj| jdf} qq| jr| jD ]*}	t
|	st jtjj| j|	df  S q|S )Nra   r*   r+   r1   r5   )Z	error_msg)rU   rl   )r   DISABLEDr   ZSERVICE_NOT_CONFIGUREDr&   r<   r-   r.   ZNO_APT_URL_FOR_SERVICEZNO_SUITES_FOR_SERVICEr
   Zget_apt_cache_policyZAPT_POLICY_FAILEDru   rv   r)   ZENABLEDZSERVICE_IS_ACTIVEcheck_packages_are_installedr0   Zis_installedZ SERVICE_DISABLED_MISSING_PACKAGEr!   )
r   Zcurrent_statusr-   r+   repo_urlrepo_suitesZpolicyZsuiteZservice_matchrl   r   r   r   application_status  sN     

 

 
z"RepoEntitlement.application_statusc                 C   sF   | j }tdd t| dD r,dS |s4dS t|t|kS )zCheck if apt url delta should be applied.

        :param apt_url: string containing the apt url to be used.

        :return: False if apt url is already found on the source file.
                 True otherwise.
        c                 s   s   | ]}| d V  qdS )#N)
startswith)ro   liner   r   r   	<genexpr>  s   z<RepoEntitlement._check_apt_url_is_applied.<locals>.<genexpr>
FT)r'   allr   Z	load_filestripsplitbool)r   r3   Zapt_filer   r   r   _check_apt_url_is_applied  s    z)RepoEntitlement._check_apt_url_is_applied)orig_accessdeltasallow_enabler   c                    s6  t  |||rdS |di }|di }|d}|d}t }|rZ|rZ|  }	n|  \}	}
|	tjkrtdS | 	|st
d| j| ttjj| jd |di }|di d}|rt| j|| j | t  | t  |r2t
d	| ttjjd
|d | jt |d dS )a1  Process any contract access deltas for this entitlement.

        :param orig_access: Dictionary containing the original
            resourceEntitlement access details.
        :param deltas: Dictionary which contains only the changed access keys
        and values.
        :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.

        :return: True when delta operations are processed; False when noop.
        Tr*   r+   r1   r,   Fz.New aptURL, updating %s apt sources list to %srT   z%New additionalPackages, installing %r, rX   )rz   )rF   process_contract_deltasr.   r   readZ"_check_application_status_on_cacher   r   r   r   r;   rV   r!   r@   r   ZREPO_UPDATING_APT_SOURCESr&   r
   remove_auth_apt_repor'   r4   rg   r	   ProgressWrapperrY   Z REPO_REFRESH_INSTALLING_PACKAGESr[   r\   )r   r   r   r   Zdelta_entitlementZdelta_directivesZdelta_apt_urlZdelta_packagesZstatus_cacher   r~   Zorig_entitlementZold_urlrL   r   r   r     sV    




   z'RepoEntitlement.process_contract_deltas)rS   rz   cleanup_on_failurer   c                 C   s   |s
| j }|sdS |d| jd z| | W n* tjk
r^   |rX| t	   Y nX |
tjj| jd | jrddi}ddd	g}nd}g }ztj|||d
 W n< tjk
r   |rtd| j | t	   Y nX dS )zInstall contract recommended packages for the entitlement.

        :param package_list: Optional package list to use instead of
            self.packages.
        :param cleanup_on_failure: Cleanup apt files if apt install fails.
        Nrs   Zpre_installra   ZDEBIAN_FRONTENDZnoninteractivez--allow-downgradesz$-o Dpkg::Options::="--force-confdef"z$-o Dpkg::Options::="--force-confold")r0   apt_optionsoverride_env_varsz.Apt install failed, removing apt config for {})r0   rZ   Z	messagingr.   Z_update_sources_listr   UbuntuProErrorrg   r	   r   rS   r   ZINSTALLING_SERVICE_PACKAGESr&   r<   apt_noninteractiver
   run_apt_install_commandr;   rV   r!   )r   rS   rz   r   r   r   r   r   r   r\   !  sJ    
z RepoEntitlement.install_packagesc              	   C   sd  d}d}d}| j js| j jrNtd| j jtj}td| j jtj}tjj	}n@| j j
s^| j jrtd| j j
tj}td| j jtj}tjj}tj|||d | j}| j}|d di }|  }|d}	|	stj| jd|d	}
|
stj| jd|d
}|stj| jd| jr^| js:tj| j| jd| jj| jd}t||
| j| j g }ttjsx| d ttj!s| d |r|"dt#j$jd%|d ztj&|d W n( tj'k
r   | (t)*   Y nX t+|| j,|
||| j-| j.| j/ |0t#j1j| jd zt2| W n, tj'k
r^   | j(t)* dd  Y nX dS )zSetup apt config based on the resourceToken and directives.
        Also sets up apt proxy if necessary.

        :raise UbuntuProError: on failure to setup any aspect of this apt
           configuration
        Nr   Zhttps)
http_proxyhttps_proxyZproxy_scoper*   r+   aptKeyrE   r1   r5   rD   r|   zapt-transport-httpszca-certificatesrV   r   rX   F)run_apt_update)3r:   Zglobal_apt_http_proxyZglobal_apt_https_proxyr   Zvalidate_proxyZPROXY_VALIDATION_APT_HTTP_URLZPROXY_VALIDATION_APT_HTTPS_URLr
   ZAptProxyScopeZGLOBALZua_apt_http_proxyZua_apt_https_proxyZUACLIENTZsetup_apt_proxyr'   r-   r.   r>   r   ZRepoNoAptKeyr!   MissingAptURLDirectiveZRepoNoSuitesr   rH   ZRepoPinFailNoOriginr<   repo_pref_file_tmplr&   Zadd_ppa_pinningr   ZAPT_METHOD_HTTPS_FILErc   ZCA_CERTIFICATES_FILErZ   r   ZINSTALLING_PACKAGESr[   r   r   rg   r	   r   Zadd_auth_apt_repor(   rB   r4   check_updates_pocketrS   ZAPT_UPDATING_LISTZupdate_sources_list)r   rS   r   r   Zscoperepo_filenameZresource_cfgr+   r=   r   r   r   repo_pref_fileZprerequisite_pkgsr   r   r   rY   \  s    
  





z RepoEntitlement.setup_apt_config)override_snapshot_urlsc                 C   s   |d k	r|}n| j }|  }| jdi di d}| j|}z|d\}}W n tk
rr   d}|}Y nX tj	
| |g| D ]}|rt||| qd S )Nr*   r+   r1   :Zbearer)r4   r>   r-   r.   r(   r&   r   
ValueErrorr   ZsecretsZ
add_secretr
   Zadd_apt_auth_conf_entry)r   r   r4   Zcredentialsr   ZusernameZpasswordZurlr   r   r   update_apt_auth  s*     
zRepoEntitlement.update_apt_auth)rS   r   c           	      C   s   t  j}| j}| j | j di }|di }|d}|sPtj	| jd| j
|}|tjj| jd t||| j| j t|| | jr| jj| jd}t | |r|tj t  dS )zRemove any repository apt configuration files.

        :param run_apt_update: If after removing the apt update
            command after removing the apt files.
        r*   r+   r1   r   ra   r|   N)r   r#   r$   r'   r8   Zentitlementsr!   r.   r   r   r(   r&   rS   r   ZREMOVING_APT_CONFIGURATIONr<   r
   r   r4   rB   Zremove_apt_list_filesr   r   Zensure_file_absentZAPT_UPDATING_LISTSZrun_apt_update_command)	r   rS   r   r$   r   r*   Zaccess_directivesr   r   r   r   r   rg     s6    

 
   
z!RepoEntitlement.remove_apt_config)F)F)NT)N)T)8__name__
__module____qualname__r%   r   r(   rH   r   r   Zsupports_purger   propertyr   intstrr   r'   r)   r   r0   r   r3   r4   r6   r>   r   rA   abcabstractmethodrB   r   r   rG   rR   r	   r   r]   r^   rm   rb   re   ri   rh   r   r   ZNamedMessager   r   r   r   r   r\   rY   r   rg   __classcell__r   r   rL   r   r   $   s    
	,E'1 

H  
;h 
 r   ))r   r/   Zloggingru   os.pathr   typingr   r   r   r   r   r   Zuaclientr	   r
   r   r   r   r   r   r   r   r   Zuaclient.entitlementsr   Z(uaclient.entitlements.entitlement_statusr   r   r   Zuaclient.files.state_filesr   Zget_event_loggerr@   Z	getLoggerZreplace_top_level_logger_namer   r;   rw   ZUAEntitlementr   r   r   r   r   <module>   s    0