�
u��ec@sddlZddlZddlZddlZddlZddlZddlZddlmZddl m
Z
mZyddlZWne
k
r�nXddlZejr�eZndZdZdZdZdefd ��YZd
efd��YZdefd
��YZdS(i����N(tLock(tsos_get_command_outputt
is_executabletfiletlinktnodetdirtArchivecBs�eZdZed��Zejd�ZdZe Z
e�Zd�Z
d�Zd�Zd�Zd�Zd �Zdd
�Zdd�Zd
�Zd�Zd�Zd�Zd�Zd�Zd�Zd�Zd�ZRS(s!Abstract base class for archives.cCs|jS(s6Returns the archive class's name as a string.
(t__name__(tcls((s//usr/lib/python2.7/site-packages/sos/archive.pytarchive_type*stsostunsetcCsd|j�|fS(Ns[archive:%s] %s(R
(tselftmsg((s//usr/lib/python2.7/site-packages/sos/archive.pyt_format_msg7scCs
||_dS(N(t_debug(R
tdebug((s//usr/lib/python2.7/site-packages/sos/archive.pyt set_debug:scCs|jj|j|��dS(N(tlogterrorR(R
R((s//usr/lib/python2.7/site-packages/sos/archive.pyt log_error=scCs|jj|j|��dS(N(RtwarningR(R
R((s//usr/lib/python2.7/site-packages/sos/archive.pytlog_warn@scCs|jj|j|��dS(N(RtinfoR(R
R((s//usr/lib/python2.7/site-packages/sos/archive.pytlog_infoCscCs*|js
dS|jj|j|��dS(N(RRRR(R
R((s//usr/lib/python2.7/site-packages/sos/archive.pyt log_debugFs cCs
t�dS(N(tNotImplementedError(R
tsrctdest((s//usr/lib/python2.7/site-packages/sos/archive.pytadd_fileNstwcCs
t�dS(N(R(R
tcontentRtmode((s//usr/lib/python2.7/site-packages/sos/archive.pyt
add_stringQscCs
t�dS(N(R(R
R R((s//usr/lib/python2.7/site-packages/sos/archive.pyt
add_binaryTscCs
t�dS(N(R(R
tsourcet link_name((s//usr/lib/python2.7/site-packages/sos/archive.pytadd_linkWscCs
t�dS(N(R(R
tpath((s//usr/lib/python2.7/site-packages/sos/archive.pytadd_dirZscCs
t�dS(N(R(R
R'R!tdevice((s//usr/lib/python2.7/site-packages/sos/archive.pytadd_node]scCs
t�dS(s�Return a temporary directory that clients of the archive may
use to write content to. The content of the path is guaranteed
to be included in the generated archive.N(R(R
((s//usr/lib/python2.7/site-packages/sos/archive.pytget_tmp_dir`scCs
t�dS(s�Return the maximum file name length this archive can support.
This is the lesser of the name length limit of the archive
format and any temporary file system based cache.N(R(R
((s//usr/lib/python2.7/site-packages/sos/archive.pytname_maxfscCsdS(sTReturn a string representing the path to the temporary
archive. For archive classes that implement in-line handling
this will be the archive file itself. Archives that use a
directory based cache prior to packaging should return the
path to the temporary directory where the report content is
locatedN((R
((s//usr/lib/python2.7/site-packages/sos/archive.pytget_archive_pathlscCsdS(s:Clean up any temporary resources used by an Archive class.N((R
((s//usr/lib/python2.7/site-packages/sos/archive.pytcleanupuscCs|j�dS(sFinalize an archive object via method. This may involve creating
An archive that is subsequently compressed or simply closing an
archive that supports in-line handling. If method is automatic then
the following methods are tried in order: xz, bz2 and gzipN(tclose(R
tmethod((s//usr/lib/python2.7/site-packages/sos/archive.pytfinalizeysN(Rt
__module__t__doc__tclassmethodR
tloggingt getLoggerRt_nametFalseRRt
_path_lockRRRRRRtNoneRR"R#R&R(R*R+R,R-R.R1(((s//usr/lib/python2.7/site-packages/sos/archive.pyR's. tFileCacheArchivecBs�eZdZdZdZdZd�Zd�Zd�Zdd�Z ded�Zdd�Z
d d
�Zd�Zd�Zd
�Zd�Zd�Zd�Zd�Zdd�Zd�Zd�Zd�Zd�ZRS(sd Abstract superclass for archive types that use a temporary cache
directory in the file system. tcCs�||_||_||_||_||_||_tjj||�|_ |j
�tj|j d�WdQX|jd|j f�dS(Ni�s*initialised empty FileCacheArchive at '%s'(
R7t_tmp_dirt_policyt_threadstenc_optstsysroottosR'tjoint
_archive_rootR9tmakedirsR(R
tnamettmpdirtpolicytthreadsR@RA((s//usr/lib/python2.7/site-packages/sos/archive.pyt__init__�s
cCs=tjj|�r'|jtj�}ntjj|j|�S(N(RBR'tisabstlstriptsepRCRD(R
RF((s//usr/lib/python2.7/site-packages/sos/archive.pyt dest_path�scCsL|j|j�r|S|dtjkr6|d}ntjj|j|�S(Nii(t
startswithRARBRMR'RC(R
R'((s//usr/lib/python2.7/site-packages/sos/archive.pytjoin_sysroot�s
i�csa|jd|�|j�|}�fd�}|jd�sG|}n7tjj|j|��rh|ntjj|�d}|}g}xG|dkr�|dkr�tjj|�\}} |j| �|}q�W|j ��}
d}xp|D]h}tjj
|
|�}
||
�sq�ntjj
||�}tjj|
�s�|jd|
�tjj|�r/tjj|�r/tj
|�}
tjj|�d}tjj
||
�}|j|d|�}tjj|�}tjj|
�rtjj|
|�}
n|jd|
|
f�tj|
|
�qY|jd |
�tj|
|�|}q�q�W|S(
s�Create leading path components
The standard python `os.makedirs` is insufficient for our
needs: it will only create directories, and ignores the fact
that some path components may be symbolic links.
:param src: The source path in the host file system for which
leading components should be created, or the path
to an sos_* virtual directory inside the archive.
Host paths must be absolute (initial '/'), and
sos_* directory paths must be a path relative to
the root of the archive.
:param mode: An optional mode to be used when creating path
components.
:returns: A rewritten destination path in the case that one
or more symbolic links in intermediate components
of the path have altered the path destination.
sMaking leading paths for %scs|jtjj�d��S(s>Test whether path ``path`` is inside the archive.
R<(RORBR'RC(R'(troot(s//usr/lib/python2.7/site-packages/sos/archive.pyt
in_archive�st/iR<sMaking path %sR!sMaking symlink '%s' -> '%s'sMaking directory %s(RRDRORBR'tisdirRPtsplittappendtreverseRCtexiststislinktreadlinkt_make_leading_pathstnormpathRKtrelpathtsymlinktmkdir(R
RR!RRRtsrc_dirR't
path_compstheadttailtabs_pathtsrc_pathtcompttargett
target_dirt
target_src((RQs//usr/lib/python2.7/site-packages/sos/archive.pyR[�sN !
$
c
Cs�|p|j|�}|tkr*|}ntjj|�d}|sJ|Stjj|�r�tjj|�r�td|��nJtjj|�s�|tkr�|ntjj|�d}|j|�nd�}|r�|Stjj|�r�tj |�}d} |t
kr>tj|j
�r>t| |df��n|tkrvtj|j
�rvt| |df��n|tkr�||j
�r�t| |df��n|tkr�tj|j
�r�t| |df��nd S|S(
s4Check a new destination path in the archive.
Since it is possible for multiple plugins to collect the same
paths, and since plugins can now run concurrently, it is possible
for two threads to race in archive methods: historically the
archive class only needed to test for the actual presence of a
path, since it was impossible for another `Archive` client to
enter the class while another method invocation was being
dispatched.
Deal with this by implementing a locking scheme for operations
that modify the path structure of the archive, and by testing
explicitly for conflicts with any existing content at the
specified destination path.
It is not an error to attempt to create a path that already
exists in the archive so long as the type of the object to be
added matches the type of object already found at the path.
It is an error to attempt to re-create an existing path with
a different path type (for example, creating a symbolic link
at a path already occupied by a regular file).
:param src: the source path to be copied to the archive
:param path_type: the type of object to be copied
:param dest: an optional destination path
:param force: force file creation even if the path exists
:returns: An absolute destination path if the path should be
copied now or `None` otherwise
is'path '%s' exists and is not a directorycSs:ttj|�tj|�tj|�tj|�g�S(N(tanytstattS_ISBLKtS_ISCHRtS_ISFIFOtS_ISSOCK(R!((s//usr/lib/python2.7/site-packages/sos/archive.pyt
is_special.s
s path '%s' exists and is not a %ssregular files
symbolic linksspecial filet directoryN(RNtP_DIRRBR'RURXRTt
ValueErrorR[tlstattP_FILERktS_ISREGtst_modetP_LINKtS_ISLNKtP_NODEtS_ISDIRR:(
R
Rt path_typeRtforcetdest_dirR`Rptsttve_msg((s//usr/lib/python2.7/site-packages/sos/archive.pyt_check_path�s8 %
( cCs�|j��|s|}n|j|t�}|s5dSt|dd�s�ytj||�Wnntk
r�}|jd�s�|jd�r�q�|j d||f�n$t
k
r�}|j d|�nXy�tj|�}|jd�s�|jd�r.tj
||�tj|d|j|jf�ntj||�tj||j|j�Wn*tk
r�}|jd||f�nXd|}nI|jd �t|d
��%}x|D]}|j|�q�WWdQXd}|jd||jf�WdQXdS(
Ntreads/sys/s/proc/scaught '%s' copying '%s'sFile not collected: '%s'tnss&caught '%s' setting attributes of '%s's'%s'iRs open files!added %s to FileCacheArchive '%s'(R9R�RutgetattrR:tshutiltcopytIOErrorRORtOSErrorRBRktcopymodetutimetst_atime_nstst_mtime_nstcopystattchowntst_uidtst_gidt ExceptionRtseektopentwriteRD(R
RRteRkt file_nametftline((s//usr/lib/python2.7/site-packages/sos/archive.pyRJsB
"
Rc Cs�|j��|}|j|tdt�}tj||dd�}t|t�rd|jdd�}n|j |�t
jj|�r�yt
j||�Wq�tk
r�}|jd||f�q�Xn|jd||jf�WdQXdS(NR}tencodingsutf-8tutf8tignores!Unable to add '%s' to archive: %ss-added string at '%s' to FileCacheArchive '%s'(R9R�RutTruetcodecsR�t
isinstancetbytestdecodeR�RBR'RXR�R�R�RRRD(R
R RR!RR�R�((s//usr/lib/python2.7/site-packages/sos/archive.pyR"{s
c Cso|j�`|j|t�}|s&dStj|ddd�}|j|�|jd||jf�WdQXdS(NtwbR�s,added binary content at '%s' to archive '%s'( R9R�RuR�R�R:R�RRD(R
R RR�((s//usr/lib/python2.7/site-packages/sos/archive.pyR#�s
c Cs>|jd||f�|j�c|j|t�}|s=dStjj|�stj||�|jd|||jf�nWdQX|jd|||f�tjj |�}tjj
tjj||��}|j|�}d�}tjj
|�s-tjj|�r�|}tjj |�}tjj|tj|��}tjj||�}|||�r�|jd||f�dS|jd||f�|j||�q-tjj|�r�|jd|�|j|�q-tjj|�r|jd|�|j|�q-|jd ||f�n|jd
�dS(Nsadding symlink at '%s' -> '%s's-added symlink at '%s' to '%s' in archive '%s's.Link follow up: source=%s link_name=%s dest=%scSs�tjj|�}tjj|�sHtjjtjj||��}ntjj|�}||krjtSytj|�Wn)tk
r�}|j dkr�tS�nXt
S(s�Return ``True`` if the symbolic link ``link_name`` is part
of a file system loop, or ``False`` otherwise.
i((RBR'tdirnameRKtrealpathRCR�RkR�terrnoR8(R%R$tlink_dirR�((s//usr/lib/python2.7/site-packages/sos/archive.pytis_loop�s$s#Link '%s' - '%s' loops: skipping...s'Adding link %s -> %s for link follow ups Adding dir %s for link follow ups!Adding file %s for link follow ups)No link follow up: source=%s link_name=%ssleaving add_link()(RR9R�RxRBR'tlexistsR^RDR�R�RCRNRXRYRZR]R&RTR(tisfileR( R
R$R%Rt
source_dirthost_path_nametdest_path_nameR�R~((s//usr/lib/python2.7/site-packages/sos/archive.pyR&�sJ
! cCs$|j�|j|t�WdQXdS(smCreate a directory in the archive.
:param path: the path in the host file system to add
N(R9R�Rr(R
R'((s//usr/lib/python2.7/site-packages/sos/archive.pyR(�s
cCs�|j|t�}|sdStjj|�s�ytj|||�WnLtk
r�}|jtjkr�d}|j d||f�dS|�nXt
j||�ndS(NsOperation not permittedsadd_node: %s - mknod '%s'(R�RzRBR'RXtmknodR�R�tEPERMRR�R�(R
R'R!R)RR�R((s//usr/lib/python2.7/site-packages/sos/archive.pyR*�s
cCs7dtjkr/tjd}tj|j|�SdSdS(NtPC_NAME_MAXi�(RBtpathconf_namestpathconfRD(R
tpc_name_max((s//usr/lib/python2.7/site-packages/sos/archive.pyR,�s
cCs|jS(N(RD(R
((s//usr/lib/python2.7/site-packages/sos/archive.pyR+scCs|jS(N(RD(R
((s//usr/lib/python2.7/site-packages/sos/archive.pyR-scCsCtjtjj|j|�d|�|jd||jf�dS(ssCreate path, including leading components.
Used by sos.sosreport to set up sos_* directories.
R!s2created directory at '%s' in FileCacheArchive '%s'N(RBRER'RCRDR(R
R'R!((s//usr/lib/python2.7/site-packages/sos/archive.pyREs% cCs+|j|�}tj|ddddd�S(NtrR�sutf-8terrorsR�(RNR�R�(R
R'((s//usr/lib/python2.7/site-packages/sos/archive.pyt open_filescCs,tjj|j�r(tj|j�ndS(N(RBR'RTRDR�trmtree(R
((s//usr/lib/python2.7/site-packages/sos/archive.pyR.scCs|jd|j|f�|j�|j�|jd|jtj|j�jf�||_y|j �}Wn7t
k
r�}d}|jd||f�|j�SX|j
dr�y|j|�SWqt
k
r�}d}|jd||f�|SXn|SdS(Ns)finalizing archive '%s' using method '%s'sbuilt archive at '%s' (size=%d)s+An error occurred compressing the archive: s%s %stencrypts)An error occurred encrypting the archive:(RRDt_build_archiveR.t
_archive_nameRBRktst_sizeR0t _compressR�RRFR@t_encrypt(R
R0tresR�texp_msg((s//usr/lib/python2.7/site-packages/sos/archive.pyR1s*
cCs,|jdd�}|d7}d|}d}|jdr[|d|jd7}||7}n|jdr�d|jdjd d
�}i|d6}|d7}d
|}||7}nt|ddd|�}|ddkr�|S|ddkr|jdrd}qd}nd|d}t|��dS(s�Encrypts the compressed archive using GPG.
If encryption fails for any reason, it should be logged by sos but not
cause execution to stop. The assumption is that the unencrypted archive
would still be of use to the user, and/or that the end user has another
means of securing the archive.
Returns the name of the encrypted archive, or raises an exception to
signal that encryption failed and the unencrypted archive name should
be used.
s
sosreport-ssecured-sosreport-s.gpgsgpg --batch -o %s tkeys--trust-model always -e -r %s tpasswords%ss'"R<tsos_gpgs-c --passphrase-fd 0 s!/bin/bash -c "echo $sos_gpg | %s"ttimeoutitenvtstatusisSpecified key not in keyringsCould not read passphrasesgpg exited with code %sN(treplaceR:R@RR�(R
tarchivetarc_nametenc_cmdR�tpasswdR�R((s//usr/lib/python2.7/site-packages/sos/archive.pyR�5s,
N(RR2R3R=RDR�RJRNRPR[R:R8R�RR"R#R&R(R*R,R+R-RER�R.R1R�(((s//usr/lib/python2.7/site-packages/sos/archive.pyR;�s.
\K1 H tTarFileArchivecBseeZdZd ZeZd�Zd d�Zd�Z d�Z
d�Zd�Zd�Z
d�ZRS(
s: archive class using python TarFile to create tar archivescCsPtt|�j||||||�d|_tjj||j��|_dS(Nttar( tsuperR�RJt_suffixRBR'RCRFR�(R
RFRGRHRIR@RA((s//usr/lib/python2.7/site-packages/sos/archive.pyRJgs
cCsn|j|_d|j|jd<d|j|jd<|rF||_n|j|_|j|_|j |_
dS(Ns%.9ftatimetctime(tst_mtimetmtimetst_atimetpax_headerstst_ctimeR!RwR�tuidR�tgid(R
ttar_infotfstatR!((s//usr/lib/python2.7/site-packages/sos/archive.pytset_tarinfo_from_statmscCs�|jttjj|j�d�}|s8|j}nytj|�}Wntk
r_|SX|j r�|j
|�}|r�||jd<q�n|j||�|S(Ni����sRHT.security.selinux(
RFtlenRBR'RUR7RDRkR�t_with_selinux_contexttget_selinux_contextR�R�(R
ttarinfot orig_pathR�tcontext((s//usr/lib/python2.7/site-packages/sos/archive.pytcopy_permissions_filterzs&
cCs6ytj|�\}}|SWntk
r1dSXdS(N(tselinuxt
getfileconR�R:(R
R'trctc((s//usr/lib/python2.7/site-packages/sos/archive.pyR��s
cCsd|j|jfS(Ns%s.%s(R7R�(R
((s//usr/lib/python2.7/site-packages/sos/archive.pyRF�scCstt|�j�S(N(R�R�R,(R
((s//usr/lib/python2.7/site-packages/sos/archive.pyR,�scCsXtj|jdd�}|j|jdtjj|j�dd|j �|j
�dS(NR!Rtarcnameitfilter(ttarfileR�R�taddRDRBR'RUR7R�R/(R
R�((s//usr/lib/python2.7/site-packages/sos/archive.pyR��s(
c
Cseg}xDdddgD]3}t|�r8|j|�q|jd|�qW|j|krk|jg}nd}t|�}x�|D]�}d|jdd�}|jj||j�}y�d ||j �f}t
|d
tdd�}|d
r!|j|d�td||d
f��n|j
|7_
|j �SWq�tk
rV} | }q�Xq�W|�dS(Ntxztbzip2tgzips#"%s" compression method unavailablesNo compression utilities found.t.tipR<s%s %ststderrR�iR�toutputs%s exited with %s(RRVRR0R�R�R>tget_cmd_for_compress_methodR?RFRR�RR�(
R
tmethodsR0R�t
last_errortcmdtsuffixtexec_cmdR�R�((s//usr/lib/python2.7/site-packages/sos/archive.pyR��s0
N(RR2R3R:R0R8R�RJR�R�R�RFR,R�R�(((s//usr/lib/python2.7/site-packages/sos/archive.pyR�as
(RBR�R�R5R�R�Rkt threadingRt
sos.utilitiesRRR�tImportErrortsixtPY3tinttlongRuRxRzRrtobjectRR;R�(((s//usr/lib/python2.7/site-packages/sos/archive.pyt<module>s.
[�� |