o
    w7e+                     @   s   d 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	 ddl
mZmZ ddlmZmZmZ ddlmZ dZeeZdd	d
Zdd ZdefddZdedededefddZdd Zdd Zedkrke  dS dS )aT  Query standardized instance metadata provided to machine, returning a JSON
structure.

Some instance-data values may be binary on some platforms, such as userdata and
vendordata. Attempt to decompress and decode UTF-8 any binary values.

Any binary values in the instance metadata will be base64-encoded and prefixed
with "ci-b64:" in the output. userdata and, where applicable, vendordata may
be provided to the machine gzip-compressed (and therefore as binary data).
query will attempt to decompress these to a string before emitting the JSON
output; if this fails, they are treated as binary.
    N)EACCES)atomic_helperlogutil)addLogHandlerCLIread_cfg_paths)convert_jinja_instance_dataget_jinja_variable_aliasrender_jinja_payload)REDACT_SENSITIVE_VALUEqueryc                 C   s   | s	t jttd} | jdddddd | jdd	td
t d d | jdddddd | jddtdd | jddtdd | jdtddd | jddddddd | jdd td!d"d# | S )$a#  Build or extend an arg parser for query utility.

    @param parser: Optional existing ArgumentParser instance representing the
        query subcommand which will be extended to support the args of
        this utility.

    @returns: ArgumentParser with proper argument configuration.
    )progdescriptionz-dz--debug
store_trueFz+Add verbose messages during template render)actiondefaulthelpz-iz--instance-dataz,Path to instance-data.json file. Default is instance_data)typer   z-lz--list-keyszBList query keys available at the provided instance-data <varname>.z-uz--user-datazHPath to user-data file. Default is /var/lib/cloud/instance/user-data.txtz-vz--vendor-datazLPath to vendor-data file. Default is /var/lib/cloud/instance/vendor-data.txtvarname?zA dot-delimited specific variable to query from instance-data. For example: v1.local_hostname. If the value is not JSON serializable, it will be base64-encoded and will contain the prefix "ci-b64:". )r   nargsr   z-az--alldump_allz Dump all available instance-data)r   r   destr   z-fz--formatformatzOptionally specify a custom output format string. Any instance-data variable can be specified between double-curly braces. For example -f "{{ v2.cloud_name }}")r   r   r   )argparseArgumentParserNAME__doc__add_argumentstrr   get_runpathparser r$   5/usr/lib/python3/dist-packages/cloudinit/cmd/query.py
get_parser$   st   	
	
		r&   c                 C   s>   t j| dd}z|dW S  ty   t j|ddd Y S w )zAttempt to return a string of user-data from ud_file_path

    Attempt to decode or decompress if needed.
    If unable to decode the content, raw bytes will be returned.

    @returns: String of uncompressed userdata if possible, otherwise bytes.
    F)decodezutf-8T)quietr'   )r   	load_filer'   UnicodeDecodeErrordecomp_gzip)ud_file_pathbdatar$   r$   r%   load_userdata|   s   r.   returnc              
   C   s  t  }t }| r| }n#|d}|dkr-|d}t j|r#|}ntd|| |}n|}|r4|}nt j|j	d}|rA|}	nt j|j	d}	|d}
zt
|}W n! ttfyv } z|jtkrktd|  td	|  d
}~ww t
|} z
t
t
|
}W n ttfy   d
}Y nw |dkrdt|f | d< dt|	f | d< dt|
f | d< | S t|| d< t|	| d< || d< | S )a  Return a dict of merged instance-data, vendordata and userdata.

    The dict will contain supplemental userdata and vendordata keys sourced
    from default user-data and vendor-data files.

    Non-root users will have redacted INSTANCE_JSON_FILE content and redacted
    vendordata and userdata values.

    :raise: IOError/OSError on absence of instance-data.json file or invalid
        access perms.
    r   r   instance_data_sensitivez4Missing root-readable %s. Using redacted %s instead.zuser-data.txtzvendor-data.txtcombined_cloud_configz$No read permission on '%s'. Try sudozMissing instance-data file: %sNz<%s> file:%suserdata
vendordata)osgetuidr   r!   pathexistsLOGwarningjoininstance_linkr   r)   IOErrorOSErrorerrnor   error	load_jsonr   r.   )r   	user_datavendor_datauidpathsinstance_data_fnredacted_data_fnsensitive_data_fnuser_data_fnvendor_data_fncombined_cloud_config_fninstance_jsoner1   r$   r$   r%   _read_instance_data   sv   







rM   jinja_vars_without_aliasesjinja_vars_with_aliasesr   	list_keysc           
      C   s   d}| }| dD ]M}z|| }W n  ty1 } z|r#dj||d}nd|}t||d}~ww ||v r;|| }n|D ]}	t|	|krK||	 } nq=|rR|d7 }||7 }q	|S )a  Return the value of the dot-delimited varname path in instance-data

    Split a dot-delimited jinja variable name path into components, walk the
    path components into the instance_data and look up a matching jinja
    variable name or cloud-init's underscore-delimited key aliases.

    :raises: ValueError when varname represents an invalid key name or path or
        if list-keys is provided by varname isn't a dict object.
     .z*instance-data '{key_path}' has no '{leaf}')leafkey_pathz Undefined instance-data key '{}'N)splitKeyErrorr   
ValueErrorr	   )
rN   rO   r   rP   walked_key_pathresponsekey_path_partrL   msgkeyr$   r$   r%   (_find_instance_data_leaf_by_varname_path   s2   



r]   c              
   C   sv  t t|jrtjntj t|j|j|j	|j
gs$td t   dS zt|j|j|j}W n ttfy;   Y dS w |j	r]dj	|j	d}t|d||jrOdndd}|r[t| d	S dS t|}|jrt|dd
}zt|||j|jd}W n ttfy } zt| W Y d}~dS d}~ww |jrt|tstd|j dS dt| }t|tst !|}t| d	S )z3Handle calls to 'cloud-init query' as a subcommand.zDExpected one of the options: --all, --format, --list-keys or varname   z## template: jinja
{fmt})fmtzquery commandlineTF)payload
payload_fnr   debugr   )include_key_aliases)rN   rO   r   rP   Nz+--list-keys provided but '%s' is not a dict
)"r   r8   rb   r   DEBUGWARNINGanyrP   r   r   r   r?   r&   
print_helprM   r   rA   rB   r<   r=   r
   printr   r]   rV   rW   
isinstancedictr:   sortedkeysr    r   
json_dumps)nameargsr   r`   rendered_payloadrY   rO   rL   r$   r$   r%   handle_args  sl   





rr   c                  C   s   t  } ttt|   dS )z,Tool to query specific instance-data values.N)r&   sysexitrr   r   
parse_argsr"   r$   r$   r%   mainC  s   rv   __main__)N) r   r   r4   rs   r>   r   	cloudinitr   r   r   cloudinit.cmd.develr   r   !cloudinit.handlers.jinja_templater   r	   r
   cloudinit.sourcesr   r   	getLoggerr8   r&   r.   rk   rM   r    boolr]   rr   rv   __name__r$   r$   r$   r%   <module>   s8   

XP
+=
