Compare commits
54 Commits
Author | SHA1 | Date |
---|---|---|
Matteo Cypriani | 476f311559 | |
Matteo Cypriani | 1dee9e218f | |
Matteo Cypriani | 186e598a7d | |
Matteo Cypriani | 721632baf8 | |
Matteo Cypriani | 9469a59718 | |
Matteo Cypriani | 462676e803 | |
Matteo Cypriani | 8ce1846288 | |
Matteo Cypriani | 07f53e4109 | |
Matteo Cypriani | 35b8fed792 | |
Matteo Cypriani | 63870b25ed | |
Matteo Cypriani | 93c37d756d | |
Matteo Cypriani | 7bff021cd3 | |
Matteo Cypriani | 81bd453526 | |
Matteo Cypriani | 43da9301fa | |
Matteo Cypriani | 0995d580c9 | |
Matteo Cypriani | 007d04dd16 | |
Matteo Cypriani | 28e8089316 | |
Matteo Cypriani | 6dbb4152f5 | |
Matteo Cypriani | c3a865920e | |
Matteo Cypriani | 4a2c1224f9 | |
Matteo Cypriani | 7ba62cdfca | |
Matteo Cypriani | de8fe1900c | |
Matteo Cypriani | b2d16599a4 | |
Matteo Cypriani | b2dcbd8368 | |
Matteo Cypriani | c86dae222a | |
Matteo Cypriani | f99dcc1e2c | |
Matteo Cypriani | cbd96480c5 | |
Matteo Cypriani | c4db1808da | |
Matteo Cypriani | e17dbfa1c8 | |
Matteo Cypriani | b83d37a5e9 | |
Matteo Cypriani | 2e0e8ba994 | |
Matteo Cypriani | d40548f287 | |
Matteo Cypriani | 193a0e2784 | |
Matteo Cypriani | 307ffbea86 | |
Matteo Cypriani | 2136a91df4 | |
Matteo Cypriani | 1de548e62f | |
Matteo Cypriani | e008689b9b | |
Matteo Cypriani | 97f8f0d752 | |
Matteo Cypriani | f436127427 | |
Matteo Cypriani | 4c02959204 | |
Matteo Cypriani | 7a6317a5e6 | |
Matteo Cypriani | 7fa513de68 | |
Matteo Cypriani | cd317355f1 | |
Matteo Cypriani | ecfe87cd99 | |
Matteo Cypriani | b8fdd846a4 | |
Jingbei Li | 580776117f | |
Jingbei Li | 0d00e99525 | |
Jingbei Li | a6a5cf3a03 | |
Jingbei Li | 45905e8f35 | |
Jingbei Li | bdc1df9b6f | |
Jingbei Li | 774f371070 | |
Jingbei Li | aad446095c | |
Matteo Cypriani | 67469febfb | |
Matteo Cypriani | 8d60858375 |
|
@ -0,0 +1,5 @@
|
||||||
|
*.pyc
|
||||||
|
*.pyv
|
||||||
|
*.sw*
|
||||||
|
*.tar*
|
||||||
|
*.egg
|
62
CHANGELOG
62
CHANGELOG
|
@ -1,21 +1,41 @@
|
||||||
gcp 0.1.3 (20/06/11):
|
gcp 0.2.1 (2019-03-10, Matteo Cypriani):
|
||||||
- fixed exit status
|
- fix deprecation warnings with dbus.glib
|
||||||
- updated manpage with exit status
|
see https://bugs.debian.org/cgi-bin/bugreport.cgi?bug=923677
|
||||||
- gcp DIR1 DIR2 syntax fixed
|
- minor documentation (README) improvements
|
||||||
- tests
|
|
||||||
gcp 0.1.2 (16/06/11):
|
gcp 0.2.0 (2018-10-14, Matteo Cypriani):
|
||||||
- removed bad fd close
|
- actually switch to Python3
|
||||||
- crash fix when source file can't be openned
|
- cp compatibibility:
|
||||||
- symbolic link skipping (--dereferrence and --no-dereferrence options)
|
+ don't preserve any attributes by default (Jingbei Li)
|
||||||
- unaccessible source dir crash fix
|
+ added -p switch (same as --preserve=mode,ownership,timestamps)
|
||||||
- os.stat precision fix
|
+ added -R switch (same as --recursive)
|
||||||
- manpage
|
- new --fix-filenames option
|
||||||
- install script
|
- reworked packaging
|
||||||
gcp 0.1.1 (30/09/10):
|
|
||||||
- double entry check in journal
|
gcp 0.1.4.dev1 (unreleased, Jingbei Li):
|
||||||
- unicode source_path send via dbus (second instance of gcp) fixed
|
- main Python3 migration work
|
||||||
- errors are now shown after copy
|
|
||||||
- fixed bad closure when a file already exists
|
gcp 0.1.3 (2011-06-20, Goffi):
|
||||||
- added "gcp SOURCE_FILE DEST_FILE" syntax management
|
- fixed exit status
|
||||||
gcp 0.1 (28/09/10):
|
- updated manpage with exit status
|
||||||
**INITIAL PUBLIC RELEASE**
|
- gcp DIR1 DIR2 syntax fixed
|
||||||
|
- tests
|
||||||
|
|
||||||
|
gcp 0.1.2 (2011-06-16, Goffi):
|
||||||
|
- removed bad fd close
|
||||||
|
- crash fix when source file can't be openned
|
||||||
|
- symbolic link skipping (--dereferrence and --no-dereferrence options)
|
||||||
|
- unaccessible source dir crash fix
|
||||||
|
- os.stat precision fix
|
||||||
|
- manpage
|
||||||
|
- install script
|
||||||
|
|
||||||
|
gcp 0.1.1 (2010-09-30, Goffi):
|
||||||
|
- double entry check in journal
|
||||||
|
- unicode source_path send via dbus (second instance of gcp) fixed
|
||||||
|
- errors are now shown after copy
|
||||||
|
- fixed bad closure when a file already exists
|
||||||
|
- added "gcp SOURCE_FILE DEST_FILE" syntax management
|
||||||
|
|
||||||
|
gcp 0.1 (2010-09-28, Goffi):
|
||||||
|
- **INITIAL PUBLIC RELEASE**
|
||||||
|
|
|
@ -1,5 +0,0 @@
|
||||||
include MANIFEST.in distribute_setup.py gcp.1
|
|
||||||
global-include *.py
|
|
||||||
global-include *.po *.mo
|
|
||||||
global-include CHANGELOG COPYING* INSTALL README*
|
|
||||||
global-exclude *.un~ *.swp
|
|
102
README
102
README
|
@ -1,102 +0,0 @@
|
||||||
gcp v0.1.3
|
|
||||||
(c) Jérôme Poisson aka Goffi 2010, 2011
|
|
||||||
|
|
||||||
gcp (Goffi's cp) is a files copier.
|
|
||||||
|
|
||||||
|
|
||||||
** LICENSE **
|
|
||||||
|
|
||||||
gcp is free software: you can redistribute it and/or modify
|
|
||||||
it under the terms of the GNU General Public License as published by
|
|
||||||
the Free Software Foundation, either version 3 of the License, or
|
|
||||||
(at your option) any later version.
|
|
||||||
|
|
||||||
gcp is distributed in the hope that it will be useful,
|
|
||||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
GNU General Public License for more details.
|
|
||||||
|
|
||||||
You should have received a copy of the GNU General Public License
|
|
||||||
along with gcp. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
** WTF ? **
|
|
||||||
gcp is a file copier, loosely inspired from cp, but with high level functionalities like:
|
|
||||||
- progression indicator
|
|
||||||
- gcp continue copying even when there is an issue: he just skip the file with problem, and go on
|
|
||||||
- journalization: gcp write what he is doing, this allow to know which files were effectively copied
|
|
||||||
- fixing names to be compatible with the target filesystem (e.g. removing incompatible chars like "?" or "*" on vfat)
|
|
||||||
- if you launch a copy when an other is already running, the files are added to the first queue, this avoid your hard drive to move its read/write head all the time
|
|
||||||
- files saving: you can keep track of files you have copied, and re-copy them later (useful when, for example, you always copy some free music to all your friends).
|
|
||||||
- gcp will be approximately option-compatible with cp (approximately because the behaviour is not exactly the same, see below)
|
|
||||||
|
|
||||||
/!\ WARNING /!\
|
|
||||||
gcp is at an early stage of development, and really experimental: use at your own risks !
|
|
||||||
|
|
||||||
** How to use it ? **
|
|
||||||
Pretty much like cp (see gcp --help).
|
|
||||||
Please note that the behaviour is not exactly the same as cp, even if gcp want to be option-compatible. Mainly, the destination filenames can be changed (by default, can be deactivated).
|
|
||||||
gcp doesn't implement yet all the options from cp, but it's planed.
|
|
||||||
|
|
||||||
** journalizaion **
|
|
||||||
The journal is planed to be used by gcp itself, buts remains human-readable. It is located in ~/.gcp/journal
|
|
||||||
|
|
||||||
3 states are used:
|
|
||||||
- OK means the file is copied and all operation were successful
|
|
||||||
- PARTIAL means the file is copied, but something went wrong (e.g. changing the permissions of the file)
|
|
||||||
- FAILED: the file is *not* copied
|
|
||||||
|
|
||||||
after the state, a list of things which went wront are show, separated by ", "
|
|
||||||
|
|
||||||
** What's next ? **
|
|
||||||
|
|
||||||
Several improvment are already planed
|
|
||||||
- copy queue management (moving copy order)
|
|
||||||
- advanced console interface
|
|
||||||
- notification (xmpp and maybe mail) when a long copy is finished
|
|
||||||
- retry for files which were not correctly copied
|
|
||||||
- badly encoded unicode filenames fix
|
|
||||||
- file copy integrity check
|
|
||||||
|
|
||||||
... and other are with a "maybe"
|
|
||||||
- graphic interface
|
|
||||||
- desktop (Kde, Gnome, XFCE, ...) integration
|
|
||||||
- distant copy (ftp)
|
|
||||||
- basic server mode, for copying files on network without the need of nfs or other heavy stuff
|
|
||||||
|
|
||||||
** Credits **
|
|
||||||
|
|
||||||
A big big thank to the authors/contributors of...
|
|
||||||
|
|
||||||
progressbar:
|
|
||||||
gcp use ProgressBar (http://pypi.python.org/pypi/progressbar/2.2), a class coded by Nilton Volpato which allow the textual representation of progression.
|
|
||||||
|
|
||||||
GLib:
|
|
||||||
This heavily used library is used here for the main loop, event catching, and for DBus. Get it at http://library.gnome.org/devel/glib/
|
|
||||||
|
|
||||||
DBus:
|
|
||||||
This excellent IPC is in the heart of gcp. Get more information at www.freedesktop.org/wiki/Software/dbus
|
|
||||||
|
|
||||||
python and its amazing standard library:
|
|
||||||
gcp was coded quickly for my own need thanks to this excellent and efficient language and its really huge standard library. Python can be download at www.python.org
|
|
||||||
|
|
||||||
If I forgot any credit, please contact me (mail below) to fix it.
|
|
||||||
|
|
||||||
Big thanks to contributors and package mainteners
|
|
||||||
|
|
||||||
** Contributions **
|
|
||||||
2011: Thomas Preud'homme <robotux@celest.fr>: manpage, stat resolution fix
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
** Contact **
|
|
||||||
|
|
||||||
You can contact me at goffi@goffi.org .
|
|
||||||
You'll find the latest version on my ftp: ftp://ftp.goffi.org/gcp, or check the wiki ( http://wiki.goffi.org/wiki/Gcp )
|
|
||||||
Please report any bug on http://bugs.goffi.org
|
|
||||||
You can also have a look to my other main projects (and maybe to the smaller ones too ;) ):
|
|
||||||
- lm (list movie): a tool to list movies using IMdB data, loosely inspired from ls
|
|
||||||
- SàT: my main project, a jabber/XMPP client, which is a brick to many others things I have in mind
|
|
||||||
|
|
||||||
Don't hesitate to give feedback :)
|
|
|
@ -0,0 +1,167 @@
|
||||||
|
gcp
|
||||||
|
===
|
||||||
|
|
||||||
|
gcp is a user-friendly file copier written in Python. Its name used to stand
|
||||||
|
for "Goffi's CoPier", but was changed into a recursive acronym: Gcp CoPier.
|
||||||
|
|
||||||
|
|
||||||
|
License
|
||||||
|
=======
|
||||||
|
|
||||||
|
gcp is free software: you can redistribute it and/or modify it under the terms
|
||||||
|
of the GNU General Public License as published by the Free Software Foundation,
|
||||||
|
either version 3 of the License, or (at your option) any later version.
|
||||||
|
|
||||||
|
gcp is distributed in the hope that it will be useful, but WITHOUT ANY
|
||||||
|
WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A
|
||||||
|
PARTICULAR PURPOSE. See the GNU General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a [copy of the GNU General Public License](LICENSE)
|
||||||
|
along with gcp. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
|
||||||
|
About
|
||||||
|
=====
|
||||||
|
|
||||||
|
gcp is a file copier, loosely inspired from cp, but with high level
|
||||||
|
functionalities such as:
|
||||||
|
|
||||||
|
- **Progress bar.**
|
||||||
|
- gcp **keeps copying** even when there is an issue: it just skips the file,
|
||||||
|
logs an error and goes on.
|
||||||
|
- **Logging**: gcp writes what it's doing to a log file; this allows you to
|
||||||
|
know which files were effectively copied.
|
||||||
|
- **Fixing file names** to be compatible with the target filesystem (e.g.
|
||||||
|
removing incompatible chars like `?` or `*` on FAT).
|
||||||
|
- **Queue**: if you launch a copy when another copy is already running, the
|
||||||
|
files are added to the first queue; this optimizes hard drive head movement
|
||||||
|
and filesystem fragmentation.
|
||||||
|
- **Files saving**: you can keep track of the files you have copied, and copy
|
||||||
|
them again later (useful when, for example, you copy some free music to your
|
||||||
|
friends on a regular basis).
|
||||||
|
- gcp will be **approximately option-compatible with (GNU) cp** (approximately
|
||||||
|
because the behaviour is not exactly the same, see below).
|
||||||
|
|
||||||
|
**WARNING**: gcp is at a relatively early stage of development, use at your own
|
||||||
|
risks!
|
||||||
|
|
||||||
|
|
||||||
|
Installing
|
||||||
|
==========
|
||||||
|
|
||||||
|
The Python way
|
||||||
|
--------------
|
||||||
|
|
||||||
|
First, install the following packages on your system (these Debian packages
|
||||||
|
names, they may be different on other distros/systems):
|
||||||
|
- libdbus-1-dev
|
||||||
|
- libdbus-glib-1-dev
|
||||||
|
- libgirepository1.0-dev
|
||||||
|
- libcairo2-dev
|
||||||
|
- python3-cairo-dev
|
||||||
|
|
||||||
|
Then install gcp with pip:
|
||||||
|
pip3 install gcp
|
||||||
|
|
||||||
|
On Debian-based systems
|
||||||
|
-----------------------
|
||||||
|
|
||||||
|
apt install gcp
|
||||||
|
|
||||||
|
|
||||||
|
How to use it?
|
||||||
|
==============
|
||||||
|
|
||||||
|
Pretty much like cp (see `gcp --help` and `man gcp`).
|
||||||
|
|
||||||
|
Please note that the behaviour is not exactly the same as cp's, even if gcp
|
||||||
|
aims to be option-compatible. Mainly, the destination filenames can be modified
|
||||||
|
(cf. the `--fix-filenames` option).
|
||||||
|
|
||||||
|
gcp doesn't implement all the options GNU cp has yet, but it's a long-term
|
||||||
|
goal.
|
||||||
|
|
||||||
|
|
||||||
|
Logging
|
||||||
|
=======
|
||||||
|
|
||||||
|
The log file is aimed to be used by gcp itself, buts remains human-readable. It
|
||||||
|
is located in `~/.gcp/journal`.
|
||||||
|
|
||||||
|
3 states are used:
|
||||||
|
- **OK** means the file was copied and all operation were successful.
|
||||||
|
- **PARTIAL** means the file was copied, but something went wrong (file
|
||||||
|
permissions could not be preserved, file name had to be changed, etc.).
|
||||||
|
- **FAILED**: the file was *not* copied.
|
||||||
|
|
||||||
|
After the state, a list of things that went wrong is shown, separated by ", ".
|
||||||
|
|
||||||
|
|
||||||
|
Contribution ideas
|
||||||
|
==================
|
||||||
|
|
||||||
|
Here are some ideas for future developments:
|
||||||
|
- handle XDG
|
||||||
|
- copy queue management (moving copy order)
|
||||||
|
- advanced console interface
|
||||||
|
- notification (XMPP and maybe email) when a long copy is finished
|
||||||
|
- retry for files that were not correctly copied
|
||||||
|
- badly encoded unicode filenames fix
|
||||||
|
- file copy integrity check
|
||||||
|
|
||||||
|
And in an even more distant future:
|
||||||
|
- graphic interface
|
||||||
|
- desktop (Kde, Gnome, XFCE...) integration
|
||||||
|
- distant copy (FTP)
|
||||||
|
- basic server mode, for copying files on network without the need of NFS or
|
||||||
|
other heavy stuff
|
||||||
|
|
||||||
|
|
||||||
|
Credits
|
||||||
|
=======
|
||||||
|
|
||||||
|
A big big thanks to the authors/contributors of...
|
||||||
|
|
||||||
|
* **progressbar**:
|
||||||
|
gcp uses [ProgressBar](https://pypi.python.org/pypi/progressbar), a class
|
||||||
|
coded by Nilton Volpato that allows the textual representation of
|
||||||
|
progression.
|
||||||
|
|
||||||
|
* **GLib**:
|
||||||
|
This heavily used library is used here for the main loop, event catching, and
|
||||||
|
for DBus. Get it at <https://developer.gnome.org/glib/>.
|
||||||
|
|
||||||
|
* **DBus**:
|
||||||
|
This excellent IPC is ut the heart of gcp. Get more information at
|
||||||
|
<https://www.freedesktop.org/wiki/Software/dbus/>.
|
||||||
|
|
||||||
|
* **Python** and its amazing standard library:
|
||||||
|
gcp was coded quickly for my own needs thanks to this excellent and efficient
|
||||||
|
language and its really huge standard library. Python can be download at
|
||||||
|
<https://www.python.org/>.
|
||||||
|
|
||||||
|
If I forgot any credit, please contact me (email below) to fix that.
|
||||||
|
|
||||||
|
Big thanks to contributors and package maintainers.
|
||||||
|
|
||||||
|
|
||||||
|
Contributors
|
||||||
|
============
|
||||||
|
|
||||||
|
* Original author: Jérôme Poisson a.k.a. Goffi <goffi@goffi.org> 2010-2011.
|
||||||
|
* Thomas Preud'homme <robotux@celest.fr> 2011: manpage, stat resolution fix.
|
||||||
|
* Jingbei Li a.k.a. petronny 2016: conversion to Python3.
|
||||||
|
* Matteo Cypriani <mcy@lm7.fr> 2018: `--fix-filenames` option, Python3 fixes.
|
||||||
|
|
||||||
|
|
||||||
|
Contact
|
||||||
|
=======
|
||||||
|
|
||||||
|
Feedback, bug reports, patches, etc. are welcome, either by email or on the
|
||||||
|
repository's issue tracker <https://code.lm7.fr/mcy/gcp/issues>.
|
||||||
|
|
||||||
|
You can also have a look at Goffi's other main project, [Salut à
|
||||||
|
Toi](https://www.salut-a-toi.org/) (SàT), a Jabber/XMPP-based multi-frontend,
|
||||||
|
multipurpose communication tool.
|
||||||
|
|
||||||
|
Don't hesitate to give feedback :)
|
|
@ -1,485 +0,0 @@
|
||||||
#!python
|
|
||||||
"""Bootstrap distribute installation
|
|
||||||
|
|
||||||
If you want to use setuptools in your package's setup.py, just include this
|
|
||||||
file in the same directory with it, and add this to the top of your setup.py::
|
|
||||||
|
|
||||||
from distribute_setup import use_setuptools
|
|
||||||
use_setuptools()
|
|
||||||
|
|
||||||
If you want to require a specific version of setuptools, set a download
|
|
||||||
mirror, or use an alternate download directory, you can do so by supplying
|
|
||||||
the appropriate options to ``use_setuptools()``.
|
|
||||||
|
|
||||||
This file can also be run as a script to install or upgrade setuptools.
|
|
||||||
"""
|
|
||||||
import os
|
|
||||||
import sys
|
|
||||||
import time
|
|
||||||
import fnmatch
|
|
||||||
import tempfile
|
|
||||||
import tarfile
|
|
||||||
from distutils import log
|
|
||||||
|
|
||||||
try:
|
|
||||||
from site import USER_SITE
|
|
||||||
except ImportError:
|
|
||||||
USER_SITE = None
|
|
||||||
|
|
||||||
try:
|
|
||||||
import subprocess
|
|
||||||
|
|
||||||
def _python_cmd(*args):
|
|
||||||
args = (sys.executable,) + args
|
|
||||||
return subprocess.call(args) == 0
|
|
||||||
|
|
||||||
except ImportError:
|
|
||||||
# will be used for python 2.3
|
|
||||||
def _python_cmd(*args):
|
|
||||||
args = (sys.executable,) + args
|
|
||||||
# quoting arguments if windows
|
|
||||||
if sys.platform == 'win32':
|
|
||||||
def quote(arg):
|
|
||||||
if ' ' in arg:
|
|
||||||
return '"%s"' % arg
|
|
||||||
return arg
|
|
||||||
args = [quote(arg) for arg in args]
|
|
||||||
return os.spawnl(os.P_WAIT, sys.executable, *args) == 0
|
|
||||||
|
|
||||||
DEFAULT_VERSION = "0.6.14"
|
|
||||||
DEFAULT_URL = "http://pypi.python.org/packages/source/d/distribute/"
|
|
||||||
SETUPTOOLS_FAKED_VERSION = "0.6c11"
|
|
||||||
|
|
||||||
SETUPTOOLS_PKG_INFO = """\
|
|
||||||
Metadata-Version: 1.0
|
|
||||||
Name: setuptools
|
|
||||||
Version: %s
|
|
||||||
Summary: xxxx
|
|
||||||
Home-page: xxx
|
|
||||||
Author: xxx
|
|
||||||
Author-email: xxx
|
|
||||||
License: xxx
|
|
||||||
Description: xxx
|
|
||||||
""" % SETUPTOOLS_FAKED_VERSION
|
|
||||||
|
|
||||||
|
|
||||||
def _install(tarball):
|
|
||||||
# extracting the tarball
|
|
||||||
tmpdir = tempfile.mkdtemp()
|
|
||||||
log.warn('Extracting in %s', tmpdir)
|
|
||||||
old_wd = os.getcwd()
|
|
||||||
try:
|
|
||||||
os.chdir(tmpdir)
|
|
||||||
tar = tarfile.open(tarball)
|
|
||||||
_extractall(tar)
|
|
||||||
tar.close()
|
|
||||||
|
|
||||||
# going in the directory
|
|
||||||
subdir = os.path.join(tmpdir, os.listdir(tmpdir)[0])
|
|
||||||
os.chdir(subdir)
|
|
||||||
log.warn('Now working in %s', subdir)
|
|
||||||
|
|
||||||
# installing
|
|
||||||
log.warn('Installing Distribute')
|
|
||||||
if not _python_cmd('setup.py', 'install'):
|
|
||||||
log.warn('Something went wrong during the installation.')
|
|
||||||
log.warn('See the error message above.')
|
|
||||||
finally:
|
|
||||||
os.chdir(old_wd)
|
|
||||||
|
|
||||||
|
|
||||||
def _build_egg(egg, tarball, to_dir):
|
|
||||||
# extracting the tarball
|
|
||||||
tmpdir = tempfile.mkdtemp()
|
|
||||||
log.warn('Extracting in %s', tmpdir)
|
|
||||||
old_wd = os.getcwd()
|
|
||||||
try:
|
|
||||||
os.chdir(tmpdir)
|
|
||||||
tar = tarfile.open(tarball)
|
|
||||||
_extractall(tar)
|
|
||||||
tar.close()
|
|
||||||
|
|
||||||
# going in the directory
|
|
||||||
subdir = os.path.join(tmpdir, os.listdir(tmpdir)[0])
|
|
||||||
os.chdir(subdir)
|
|
||||||
log.warn('Now working in %s', subdir)
|
|
||||||
|
|
||||||
# building an egg
|
|
||||||
log.warn('Building a Distribute egg in %s', to_dir)
|
|
||||||
_python_cmd('setup.py', '-q', 'bdist_egg', '--dist-dir', to_dir)
|
|
||||||
|
|
||||||
finally:
|
|
||||||
os.chdir(old_wd)
|
|
||||||
# returning the result
|
|
||||||
log.warn(egg)
|
|
||||||
if not os.path.exists(egg):
|
|
||||||
raise IOError('Could not build the egg.')
|
|
||||||
|
|
||||||
|
|
||||||
def _do_download(version, download_base, to_dir, download_delay):
|
|
||||||
egg = os.path.join(to_dir, 'distribute-%s-py%d.%d.egg'
|
|
||||||
% (version, sys.version_info[0], sys.version_info[1]))
|
|
||||||
if not os.path.exists(egg):
|
|
||||||
tarball = download_setuptools(version, download_base,
|
|
||||||
to_dir, download_delay)
|
|
||||||
_build_egg(egg, tarball, to_dir)
|
|
||||||
sys.path.insert(0, egg)
|
|
||||||
import setuptools
|
|
||||||
setuptools.bootstrap_install_from = egg
|
|
||||||
|
|
||||||
|
|
||||||
def use_setuptools(version=DEFAULT_VERSION, download_base=DEFAULT_URL,
|
|
||||||
to_dir=os.curdir, download_delay=15, no_fake=True):
|
|
||||||
# making sure we use the absolute path
|
|
||||||
to_dir = os.path.abspath(to_dir)
|
|
||||||
was_imported = 'pkg_resources' in sys.modules or \
|
|
||||||
'setuptools' in sys.modules
|
|
||||||
try:
|
|
||||||
try:
|
|
||||||
import pkg_resources
|
|
||||||
if not hasattr(pkg_resources, '_distribute'):
|
|
||||||
if not no_fake:
|
|
||||||
_fake_setuptools()
|
|
||||||
raise ImportError
|
|
||||||
except ImportError:
|
|
||||||
return _do_download(version, download_base, to_dir, download_delay)
|
|
||||||
try:
|
|
||||||
pkg_resources.require("distribute>="+version)
|
|
||||||
return
|
|
||||||
except pkg_resources.VersionConflict:
|
|
||||||
e = sys.exc_info()[1]
|
|
||||||
if was_imported:
|
|
||||||
sys.stderr.write(
|
|
||||||
"The required version of distribute (>=%s) is not available,\n"
|
|
||||||
"and can't be installed while this script is running. Please\n"
|
|
||||||
"install a more recent version first, using\n"
|
|
||||||
"'easy_install -U distribute'."
|
|
||||||
"\n\n(Currently using %r)\n" % (version, e.args[0]))
|
|
||||||
sys.exit(2)
|
|
||||||
else:
|
|
||||||
del pkg_resources, sys.modules['pkg_resources'] # reload ok
|
|
||||||
return _do_download(version, download_base, to_dir,
|
|
||||||
download_delay)
|
|
||||||
except pkg_resources.DistributionNotFound:
|
|
||||||
return _do_download(version, download_base, to_dir,
|
|
||||||
download_delay)
|
|
||||||
finally:
|
|
||||||
if not no_fake:
|
|
||||||
_create_fake_setuptools_pkg_info(to_dir)
|
|
||||||
|
|
||||||
def download_setuptools(version=DEFAULT_VERSION, download_base=DEFAULT_URL,
|
|
||||||
to_dir=os.curdir, delay=15):
|
|
||||||
"""Download distribute from a specified location and return its filename
|
|
||||||
|
|
||||||
`version` should be a valid distribute version number that is available
|
|
||||||
as an egg for download under the `download_base` URL (which should end
|
|
||||||
with a '/'). `to_dir` is the directory where the egg will be downloaded.
|
|
||||||
`delay` is the number of seconds to pause before an actual download
|
|
||||||
attempt.
|
|
||||||
"""
|
|
||||||
# making sure we use the absolute path
|
|
||||||
to_dir = os.path.abspath(to_dir)
|
|
||||||
try:
|
|
||||||
from urllib.request import urlopen
|
|
||||||
except ImportError:
|
|
||||||
from urllib2 import urlopen
|
|
||||||
tgz_name = "distribute-%s.tar.gz" % version
|
|
||||||
url = download_base + tgz_name
|
|
||||||
saveto = os.path.join(to_dir, tgz_name)
|
|
||||||
src = dst = None
|
|
||||||
if not os.path.exists(saveto): # Avoid repeated downloads
|
|
||||||
try:
|
|
||||||
log.warn("Downloading %s", url)
|
|
||||||
src = urlopen(url)
|
|
||||||
# Read/write all in one block, so we don't create a corrupt file
|
|
||||||
# if the download is interrupted.
|
|
||||||
data = src.read()
|
|
||||||
dst = open(saveto, "wb")
|
|
||||||
dst.write(data)
|
|
||||||
finally:
|
|
||||||
if src:
|
|
||||||
src.close()
|
|
||||||
if dst:
|
|
||||||
dst.close()
|
|
||||||
return os.path.realpath(saveto)
|
|
||||||
|
|
||||||
def _no_sandbox(function):
|
|
||||||
def __no_sandbox(*args, **kw):
|
|
||||||
try:
|
|
||||||
from setuptools.sandbox import DirectorySandbox
|
|
||||||
if not hasattr(DirectorySandbox, '_old'):
|
|
||||||
def violation(*args):
|
|
||||||
pass
|
|
||||||
DirectorySandbox._old = DirectorySandbox._violation
|
|
||||||
DirectorySandbox._violation = violation
|
|
||||||
patched = True
|
|
||||||
else:
|
|
||||||
patched = False
|
|
||||||
except ImportError:
|
|
||||||
patched = False
|
|
||||||
|
|
||||||
try:
|
|
||||||
return function(*args, **kw)
|
|
||||||
finally:
|
|
||||||
if patched:
|
|
||||||
DirectorySandbox._violation = DirectorySandbox._old
|
|
||||||
del DirectorySandbox._old
|
|
||||||
|
|
||||||
return __no_sandbox
|
|
||||||
|
|
||||||
def _patch_file(path, content):
|
|
||||||
"""Will backup the file then patch it"""
|
|
||||||
existing_content = open(path).read()
|
|
||||||
if existing_content == content:
|
|
||||||
# already patched
|
|
||||||
log.warn('Already patched.')
|
|
||||||
return False
|
|
||||||
log.warn('Patching...')
|
|
||||||
_rename_path(path)
|
|
||||||
f = open(path, 'w')
|
|
||||||
try:
|
|
||||||
f.write(content)
|
|
||||||
finally:
|
|
||||||
f.close()
|
|
||||||
return True
|
|
||||||
|
|
||||||
_patch_file = _no_sandbox(_patch_file)
|
|
||||||
|
|
||||||
def _same_content(path, content):
|
|
||||||
return open(path).read() == content
|
|
||||||
|
|
||||||
def _rename_path(path):
|
|
||||||
new_name = path + '.OLD.%s' % time.time()
|
|
||||||
log.warn('Renaming %s into %s', path, new_name)
|
|
||||||
os.rename(path, new_name)
|
|
||||||
return new_name
|
|
||||||
|
|
||||||
def _remove_flat_installation(placeholder):
|
|
||||||
if not os.path.isdir(placeholder):
|
|
||||||
log.warn('Unkown installation at %s', placeholder)
|
|
||||||
return False
|
|
||||||
found = False
|
|
||||||
for file in os.listdir(placeholder):
|
|
||||||
if fnmatch.fnmatch(file, 'setuptools*.egg-info'):
|
|
||||||
found = True
|
|
||||||
break
|
|
||||||
if not found:
|
|
||||||
log.warn('Could not locate setuptools*.egg-info')
|
|
||||||
return
|
|
||||||
|
|
||||||
log.warn('Removing elements out of the way...')
|
|
||||||
pkg_info = os.path.join(placeholder, file)
|
|
||||||
if os.path.isdir(pkg_info):
|
|
||||||
patched = _patch_egg_dir(pkg_info)
|
|
||||||
else:
|
|
||||||
patched = _patch_file(pkg_info, SETUPTOOLS_PKG_INFO)
|
|
||||||
|
|
||||||
if not patched:
|
|
||||||
log.warn('%s already patched.', pkg_info)
|
|
||||||
return False
|
|
||||||
# now let's move the files out of the way
|
|
||||||
for element in ('setuptools', 'pkg_resources.py', 'site.py'):
|
|
||||||
element = os.path.join(placeholder, element)
|
|
||||||
if os.path.exists(element):
|
|
||||||
_rename_path(element)
|
|
||||||
else:
|
|
||||||
log.warn('Could not find the %s element of the '
|
|
||||||
'Setuptools distribution', element)
|
|
||||||
return True
|
|
||||||
|
|
||||||
_remove_flat_installation = _no_sandbox(_remove_flat_installation)
|
|
||||||
|
|
||||||
def _after_install(dist):
|
|
||||||
log.warn('After install bootstrap.')
|
|
||||||
placeholder = dist.get_command_obj('install').install_purelib
|
|
||||||
_create_fake_setuptools_pkg_info(placeholder)
|
|
||||||
|
|
||||||
def _create_fake_setuptools_pkg_info(placeholder):
|
|
||||||
if not placeholder or not os.path.exists(placeholder):
|
|
||||||
log.warn('Could not find the install location')
|
|
||||||
return
|
|
||||||
pyver = '%s.%s' % (sys.version_info[0], sys.version_info[1])
|
|
||||||
setuptools_file = 'setuptools-%s-py%s.egg-info' % \
|
|
||||||
(SETUPTOOLS_FAKED_VERSION, pyver)
|
|
||||||
pkg_info = os.path.join(placeholder, setuptools_file)
|
|
||||||
if os.path.exists(pkg_info):
|
|
||||||
log.warn('%s already exists', pkg_info)
|
|
||||||
return
|
|
||||||
|
|
||||||
log.warn('Creating %s', pkg_info)
|
|
||||||
f = open(pkg_info, 'w')
|
|
||||||
try:
|
|
||||||
f.write(SETUPTOOLS_PKG_INFO)
|
|
||||||
finally:
|
|
||||||
f.close()
|
|
||||||
|
|
||||||
pth_file = os.path.join(placeholder, 'setuptools.pth')
|
|
||||||
log.warn('Creating %s', pth_file)
|
|
||||||
f = open(pth_file, 'w')
|
|
||||||
try:
|
|
||||||
f.write(os.path.join(os.curdir, setuptools_file))
|
|
||||||
finally:
|
|
||||||
f.close()
|
|
||||||
|
|
||||||
_create_fake_setuptools_pkg_info = _no_sandbox(_create_fake_setuptools_pkg_info)
|
|
||||||
|
|
||||||
def _patch_egg_dir(path):
|
|
||||||
# let's check if it's already patched
|
|
||||||
pkg_info = os.path.join(path, 'EGG-INFO', 'PKG-INFO')
|
|
||||||
if os.path.exists(pkg_info):
|
|
||||||
if _same_content(pkg_info, SETUPTOOLS_PKG_INFO):
|
|
||||||
log.warn('%s already patched.', pkg_info)
|
|
||||||
return False
|
|
||||||
_rename_path(path)
|
|
||||||
os.mkdir(path)
|
|
||||||
os.mkdir(os.path.join(path, 'EGG-INFO'))
|
|
||||||
pkg_info = os.path.join(path, 'EGG-INFO', 'PKG-INFO')
|
|
||||||
f = open(pkg_info, 'w')
|
|
||||||
try:
|
|
||||||
f.write(SETUPTOOLS_PKG_INFO)
|
|
||||||
finally:
|
|
||||||
f.close()
|
|
||||||
return True
|
|
||||||
|
|
||||||
_patch_egg_dir = _no_sandbox(_patch_egg_dir)
|
|
||||||
|
|
||||||
def _before_install():
|
|
||||||
log.warn('Before install bootstrap.')
|
|
||||||
_fake_setuptools()
|
|
||||||
|
|
||||||
|
|
||||||
def _under_prefix(location):
|
|
||||||
if 'install' not in sys.argv:
|
|
||||||
return True
|
|
||||||
args = sys.argv[sys.argv.index('install')+1:]
|
|
||||||
for index, arg in enumerate(args):
|
|
||||||
for option in ('--root', '--prefix'):
|
|
||||||
if arg.startswith('%s=' % option):
|
|
||||||
top_dir = arg.split('root=')[-1]
|
|
||||||
return location.startswith(top_dir)
|
|
||||||
elif arg == option:
|
|
||||||
if len(args) > index:
|
|
||||||
top_dir = args[index+1]
|
|
||||||
return location.startswith(top_dir)
|
|
||||||
if arg == '--user' and USER_SITE is not None:
|
|
||||||
return location.startswith(USER_SITE)
|
|
||||||
return True
|
|
||||||
|
|
||||||
|
|
||||||
def _fake_setuptools():
|
|
||||||
log.warn('Scanning installed packages')
|
|
||||||
try:
|
|
||||||
import pkg_resources
|
|
||||||
except ImportError:
|
|
||||||
# we're cool
|
|
||||||
log.warn('Setuptools or Distribute does not seem to be installed.')
|
|
||||||
return
|
|
||||||
ws = pkg_resources.working_set
|
|
||||||
try:
|
|
||||||
setuptools_dist = ws.find(pkg_resources.Requirement.parse('setuptools',
|
|
||||||
replacement=False))
|
|
||||||
except TypeError:
|
|
||||||
# old distribute API
|
|
||||||
setuptools_dist = ws.find(pkg_resources.Requirement.parse('setuptools'))
|
|
||||||
|
|
||||||
if setuptools_dist is None:
|
|
||||||
log.warn('No setuptools distribution found')
|
|
||||||
return
|
|
||||||
# detecting if it was already faked
|
|
||||||
setuptools_location = setuptools_dist.location
|
|
||||||
log.warn('Setuptools installation detected at %s', setuptools_location)
|
|
||||||
|
|
||||||
# if --root or --preix was provided, and if
|
|
||||||
# setuptools is not located in them, we don't patch it
|
|
||||||
if not _under_prefix(setuptools_location):
|
|
||||||
log.warn('Not patching, --root or --prefix is installing Distribute'
|
|
||||||
' in another location')
|
|
||||||
return
|
|
||||||
|
|
||||||
# let's see if its an egg
|
|
||||||
if not setuptools_location.endswith('.egg'):
|
|
||||||
log.warn('Non-egg installation')
|
|
||||||
res = _remove_flat_installation(setuptools_location)
|
|
||||||
if not res:
|
|
||||||
return
|
|
||||||
else:
|
|
||||||
log.warn('Egg installation')
|
|
||||||
pkg_info = os.path.join(setuptools_location, 'EGG-INFO', 'PKG-INFO')
|
|
||||||
if (os.path.exists(pkg_info) and
|
|
||||||
_same_content(pkg_info, SETUPTOOLS_PKG_INFO)):
|
|
||||||
log.warn('Already patched.')
|
|
||||||
return
|
|
||||||
log.warn('Patching...')
|
|
||||||
# let's create a fake egg replacing setuptools one
|
|
||||||
res = _patch_egg_dir(setuptools_location)
|
|
||||||
if not res:
|
|
||||||
return
|
|
||||||
log.warn('Patched done.')
|
|
||||||
_relaunch()
|
|
||||||
|
|
||||||
|
|
||||||
def _relaunch():
|
|
||||||
log.warn('Relaunching...')
|
|
||||||
# we have to relaunch the process
|
|
||||||
# pip marker to avoid a relaunch bug
|
|
||||||
if sys.argv[:3] == ['-c', 'install', '--single-version-externally-managed']:
|
|
||||||
sys.argv[0] = 'setup.py'
|
|
||||||
args = [sys.executable] + sys.argv
|
|
||||||
sys.exit(subprocess.call(args))
|
|
||||||
|
|
||||||
|
|
||||||
def _extractall(self, path=".", members=None):
|
|
||||||
"""Extract all members from the archive to the current working
|
|
||||||
directory and set owner, modification time and permissions on
|
|
||||||
directories afterwards. `path' specifies a different directory
|
|
||||||
to extract to. `members' is optional and must be a subset of the
|
|
||||||
list returned by getmembers().
|
|
||||||
"""
|
|
||||||
import copy
|
|
||||||
import operator
|
|
||||||
from tarfile import ExtractError
|
|
||||||
directories = []
|
|
||||||
|
|
||||||
if members is None:
|
|
||||||
members = self
|
|
||||||
|
|
||||||
for tarinfo in members:
|
|
||||||
if tarinfo.isdir():
|
|
||||||
# Extract directories with a safe mode.
|
|
||||||
directories.append(tarinfo)
|
|
||||||
tarinfo = copy.copy(tarinfo)
|
|
||||||
tarinfo.mode = 448 # decimal for oct 0700
|
|
||||||
self.extract(tarinfo, path)
|
|
||||||
|
|
||||||
# Reverse sort directories.
|
|
||||||
if sys.version_info < (2, 4):
|
|
||||||
def sorter(dir1, dir2):
|
|
||||||
return cmp(dir1.name, dir2.name)
|
|
||||||
directories.sort(sorter)
|
|
||||||
directories.reverse()
|
|
||||||
else:
|
|
||||||
directories.sort(key=operator.attrgetter('name'), reverse=True)
|
|
||||||
|
|
||||||
# Set correct owner, mtime and filemode on directories.
|
|
||||||
for tarinfo in directories:
|
|
||||||
dirpath = os.path.join(path, tarinfo.name)
|
|
||||||
try:
|
|
||||||
self.chown(tarinfo, dirpath)
|
|
||||||
self.utime(tarinfo, dirpath)
|
|
||||||
self.chmod(tarinfo, dirpath)
|
|
||||||
except ExtractError:
|
|
||||||
e = sys.exc_info()[1]
|
|
||||||
if self.errorlevel > 1:
|
|
||||||
raise
|
|
||||||
else:
|
|
||||||
self._dbg(1, "tarfile: %s" % e)
|
|
||||||
|
|
||||||
|
|
||||||
def main(argv, version=DEFAULT_VERSION):
|
|
||||||
"""Install or upgrade setuptools and EasyInstall"""
|
|
||||||
tarball = download_setuptools()
|
|
||||||
_install(tarball)
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
|
||||||
main(sys.argv[1:])
|
|
300
fr.po
300
fr.po
|
@ -1,38 +1,40 @@
|
||||||
# Goffi's CoPier.
|
# gcp -- French translation file
|
||||||
# Copyright (C) 2010 Jérôme Poisson
|
#
|
||||||
|
# Copyright:
|
||||||
|
# 2010 Jérôme Poisson (Goffi) <goffi@goffi.org>
|
||||||
|
# 2018 Matteo Cypriani <mcy@lm7.fr>
|
||||||
|
#
|
||||||
# This file is distributed under the same license as the gcp package.
|
# This file is distributed under the same license as the gcp package.
|
||||||
# Jérôme Poisson (Goffi) <goffi@goffi.org>, 2010.
|
#
|
||||||
# Goffi <goffi@goffi.org>, 2010.
|
|
||||||
#, fuzzy
|
|
||||||
msgid ""
|
msgid ""
|
||||||
msgstr ""
|
msgstr ""
|
||||||
"Project-Id-Version: 0.1\n"
|
"Project-Id-Version: 0.2.1\n"
|
||||||
"Report-Msgid-Bugs-To: \n"
|
"Report-Msgid-Bugs-To: \n"
|
||||||
"POT-Creation-Date: 2010-09-30 18:05+0800\n"
|
"POT-Creation-Date: 2010-09-30 18:05+0800\n"
|
||||||
"PO-Revision-Date: 2010-09-30 18:09+0800\n"
|
"PO-Revision-Date: 2018-10-14 20:56+0200\n"
|
||||||
"Last-Translator: Goffi <goffi@goffi.org>\n"
|
"Last-Translator: Matteo Cypriani <mcy@lm7.fr>\n"
|
||||||
"Language-Team: French <goffi@goffi.org>\n"
|
"Language: fr\n"
|
||||||
"MIME-Version: 1.0\n"
|
"MIME-Version: 1.0\n"
|
||||||
"Content-Type: text/plain; charset=UTF-8\n"
|
"Content-Type: text/plain; charset=UTF-8\n"
|
||||||
"Content-Transfer-Encoding: 8bit\n"
|
"Content-Transfer-Encoding: 8bit\n"
|
||||||
|
"Language-Team: \n"
|
||||||
|
"X-Generator: Poedit 2.2\n"
|
||||||
|
|
||||||
#: gcp:43
|
#: gcp:46
|
||||||
msgid "Error during import"
|
msgid "Error during import"
|
||||||
msgstr "Erreur pendant l'import de bibliothèques"
|
msgstr "Erreur pendant l'import de bibliothèques"
|
||||||
|
|
||||||
#: gcp:44
|
#: gcp:47
|
||||||
msgid "Please check dependecies:"
|
msgid "Please check dependecies:"
|
||||||
msgstr "Merci de vérifier les dépendances"
|
msgstr "Merci de vérifier les dépendances :"
|
||||||
|
|
||||||
#: gcp:50
|
#: gcp:54
|
||||||
msgid ""
|
msgid "ProgressBar not available, please download it at https://pypi.org/"
|
||||||
"ProgressBar not available, please download it at http://pypi.python.org/pypi/"
|
|
||||||
"progressbar"
|
|
||||||
msgstr ""
|
msgstr ""
|
||||||
"«ProgressBar» n'est pas disponible, merci de le télécharger à http://pypi."
|
"ProgressBar n'est pas disponible, merci de le télécharger depuishttps://pypi."
|
||||||
"python.org/pypi/progressbar"
|
"org/"
|
||||||
|
|
||||||
#: gcp:51
|
#: gcp:55
|
||||||
msgid ""
|
msgid ""
|
||||||
"Progress bar deactivated\n"
|
"Progress bar deactivated\n"
|
||||||
"--\n"
|
"--\n"
|
||||||
|
@ -40,235 +42,247 @@ msgstr ""
|
||||||
"Barre de progression désactivée\n"
|
"Barre de progression désactivée\n"
|
||||||
"--\n"
|
"--\n"
|
||||||
|
|
||||||
#: gcp:62
|
#: gcp:90
|
||||||
msgid ""
|
|
||||||
"This program comes with ABSOLUTELY NO WARRANTY;\n"
|
|
||||||
"This is free software, and you are welcome to redistribute it\n"
|
|
||||||
"under certain conditions.\n"
|
|
||||||
"---\n"
|
|
||||||
"\n"
|
|
||||||
"This software is an advanced file copier\n"
|
|
||||||
"Get the latest version at http://www.goffi.org\n"
|
|
||||||
msgstr ""
|
|
||||||
"This program comes with ABSOLUTELY NO WARRANTY;\n"
|
|
||||||
"This is free software, and you are welcome to redistribute it\n"
|
|
||||||
"under certain conditions.\n"
|
|
||||||
"---\n"
|
|
||||||
"\n"
|
|
||||||
"Ce logiciel est un copieur de fichiers avancé\n"
|
|
||||||
"Vous pouvez télécharger la dernière version à http://www.goffi.org\n"
|
|
||||||
|
|
||||||
#: gcp:86
|
|
||||||
msgid "Init DbusObject..."
|
msgid "Init DbusObject..."
|
||||||
msgstr "Initialisation de «DbusObject»"
|
msgstr "Initialisation de DbusObject..."
|
||||||
|
|
||||||
#: gcp:106
|
#: gcp:111
|
||||||
msgid "INTERNAL ERROR: invalid arguments"
|
msgid "INTERNAL ERROR: invalid arguments"
|
||||||
msgstr "ERREUR INTERNE: arguments invalides"
|
msgstr "ERREUR INTERNE : arguments invalides"
|
||||||
|
|
||||||
#: gcp:110
|
#: gcp:116
|
||||||
msgid "INTERNAL ERROR: invalid source_path"
|
msgid "INTERNAL ERROR: invalid source_dir"
|
||||||
msgstr "ERREUR INTERNE:.chemin source invalide"
|
msgstr "ERREUR INTERNE : chemin source invalide"
|
||||||
|
|
||||||
#: gcp:165
|
|
||||||
msgid "/!\\ THE FOLLOWING FILES WERE *NOT* SUCCESSFULY COPIED:"
|
|
||||||
msgstr "/!\\ LES FICHIERS SUIVANTS *N'ONT PAS* ÉTÉ COPIÉS:"
|
|
||||||
|
|
||||||
#: gcp:171
|
#: gcp:171
|
||||||
msgid "The following files were copied, but some errors happened:"
|
msgid "/!\\ THE FOLLOWING FILES WERE *NOT* SUCCESSFULY COPIED:"
|
||||||
msgstr ""
|
msgstr "/!\\ LES FICHIERS SUIVANTS N'ONT *PAS* ÉTÉ COPIÉS :"
|
||||||
"Les fichiers suivant ont été copiés, mais quelques erreurs sont survenues:"
|
|
||||||
|
|
||||||
#: gcp:177
|
#: gcp:177
|
||||||
|
msgid "The following files were copied, but some errors happened:"
|
||||||
|
msgstr "Les fichiers suivant ont été copiés, mais des erreurs sont survenues :"
|
||||||
|
|
||||||
|
#: gcp:183
|
||||||
#, python-format
|
#, python-format
|
||||||
msgid "Please check journal: %s"
|
msgid "Please check journal: %s"
|
||||||
msgstr "Merci de vérifier le journal: %s"
|
msgstr "Merci de vérifier le journal : %s"
|
||||||
|
|
||||||
#: gcp:199
|
#: gcp:205
|
||||||
msgid "gcp launched"
|
msgid "gcp launched"
|
||||||
msgstr "gcp lancé"
|
msgstr "gcp lancé"
|
||||||
|
|
||||||
#: gcp:207
|
#: gcp:213
|
||||||
msgid "Init DBus..."
|
msgid "Init DBus..."
|
||||||
msgstr "Initialisation de Dbus..."
|
msgstr "Initialisation de DBus..."
|
||||||
|
|
||||||
#: gcp:237
|
#: gcp:244
|
||||||
msgid "Can't read mounts table"
|
msgid "Can't read mounts table"
|
||||||
msgstr "Impossible de lire la table des montages"
|
msgstr "Impossible de lire la table des montages"
|
||||||
|
|
||||||
#: gcp:244
|
|
||||||
#, python-format
|
|
||||||
msgid "Adding to copy list: %(path)s ==> %(dest_path)s (%(fs_type)s)"
|
|
||||||
msgstr "Ajout à la liste des copies: %(path)s ==> %(dest_path)s (%(fs_type)s)"
|
|
||||||
|
|
||||||
#: gcp:251
|
#: gcp:251
|
||||||
#, python-format
|
#, python-format
|
||||||
msgid "Can't copy %(path)s: %(exception)s"
|
msgid "Adding to copy list: %(path)s ==> %(dest_path)s (%(fs_type)s)"
|
||||||
msgstr "Impossible de copier %(path)s: %(exception)s"
|
msgstr "Ajout à la liste des copies : %(path)s ==> %(dest_path)s (%(fs_type)s)"
|
||||||
|
|
||||||
#: gcp:274
|
#: gcp:258
|
||||||
|
#, python-format
|
||||||
|
msgid "Can't copy %(path)s: %(exception)s"
|
||||||
|
msgstr "Impossible de copier %(path)s : %(exception)s"
|
||||||
|
|
||||||
|
#: gcp:286
|
||||||
#, python-format
|
#, python-format
|
||||||
msgid "Can't append %(path)s to copy list: %(exception)s"
|
msgid "Can't append %(path)s to copy list: %(exception)s"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
"Impossible d'ajouter %(path)s à la liste des fichiers à copier: %(exception)s"
|
"Impossible d'ajouter %(path)s à la liste des fichiers à copier : "
|
||||||
|
"%(exception)s"
|
||||||
|
|
||||||
#: gcp:283
|
#: gcp:289
|
||||||
|
#, python-format
|
||||||
|
msgid "Can't access %(dirpath)s: %(exception)s"
|
||||||
|
msgstr "Impossible d'accéder à %(dirpath)s : %(exception)s"
|
||||||
|
|
||||||
|
#: gcp:301
|
||||||
#, python-format
|
#, python-format
|
||||||
msgid "Invalid dest_path: %s"
|
msgid "Invalid dest_path: %s"
|
||||||
msgstr "Chemin de destination invalide: %s"
|
msgstr "Chemin de destination invalide : %s"
|
||||||
|
|
||||||
#: gcp:288
|
#: gcp:306
|
||||||
#, python-format
|
#, python-format
|
||||||
msgid "The path given in arg doesn't exist or is not accessible: %s"
|
msgid "The path given in arg doesn't exist or is not accessible: %s"
|
||||||
msgstr "Le chemin donné en argument n'existe pas ou n'est pas accessible: %s"
|
msgstr "Le chemin donné en argument n'existe pas ou n'est pas accessible : %s"
|
||||||
|
|
||||||
#: gcp:293
|
|
||||||
#, python-format
|
|
||||||
msgid "omitting directory \"%s\""
|
|
||||||
msgstr "Répertoire \"%s\" ignoré"
|
|
||||||
|
|
||||||
#: gcp:310
|
#: gcp:310
|
||||||
#, python-format
|
#, python-format
|
||||||
msgid "File [%s] already exists, skipping it !"
|
msgid "omitting directory \"%s\""
|
||||||
|
msgstr "répertoire \"%s\" ignoré"
|
||||||
|
|
||||||
|
#: gcp:348
|
||||||
|
#, python-format
|
||||||
|
msgid "File [%s] already exists, skipping it!"
|
||||||
msgstr "Le fichier [%s] existe déjà, je le saute !"
|
msgstr "Le fichier [%s] existe déjà, je le saute !"
|
||||||
|
|
||||||
#: gcp:328
|
#: gcp:368
|
||||||
#, python-format
|
#, python-format
|
||||||
msgid "COPYING %(source)s ==> %(dest)s"
|
msgid "COPYING %(source)s ==> %(dest)s"
|
||||||
msgstr "COPIE %(source)s ==> %(dest)s"
|
msgstr "COPIE %(source)s ==> %(dest)s"
|
||||||
|
|
||||||
#: gcp:429
|
#: gcp:462
|
||||||
#, python-format
|
#, python-format
|
||||||
msgid "%.2f PiB"
|
msgid "%.2f PiB"
|
||||||
msgstr "%.2f Pio"
|
msgstr "%.2f Pio"
|
||||||
|
|
||||||
#: gcp:431
|
#: gcp:464
|
||||||
#, python-format
|
#, python-format
|
||||||
msgid "%.2f TiB"
|
msgid "%.2f TiB"
|
||||||
msgstr "%.2f Tio"
|
msgstr "%.2f Tio"
|
||||||
|
|
||||||
#: gcp:433
|
#: gcp:466
|
||||||
#, python-format
|
#, python-format
|
||||||
msgid "%.2f GiB"
|
msgid "%.2f GiB"
|
||||||
msgstr "%.2f Gio"
|
msgstr "%.2f Gio"
|
||||||
|
|
||||||
#: gcp:435
|
#: gcp:468
|
||||||
#, python-format
|
#, python-format
|
||||||
msgid "%.2f MiB"
|
msgid "%.2f MiB"
|
||||||
msgstr "%.2f Mio"
|
msgstr "%.2f Mio"
|
||||||
|
|
||||||
#: gcp:437
|
#: gcp:470
|
||||||
#, python-format
|
#, python-format
|
||||||
msgid "%.2f KiB"
|
msgid "%.2f KiB"
|
||||||
msgstr "%.2f Kio"
|
msgstr "%.2f Kio"
|
||||||
|
|
||||||
#: gcp:439
|
#: gcp:471
|
||||||
#, python-format
|
#, python-format
|
||||||
msgid "%i B"
|
msgid "%i B"
|
||||||
msgstr "%i o"
|
msgstr "%i o"
|
||||||
|
|
||||||
#: gcp:447 gcp:452
|
#: gcp:479 gcp:485
|
||||||
#, python-format
|
#, python-format
|
||||||
msgid "Copying %s"
|
msgid "Copying %s"
|
||||||
msgstr "Copie de %s"
|
msgstr "Copie de %s"
|
||||||
|
|
||||||
#: gcp:479 gcp:512
|
#: gcp:515 gcp:548
|
||||||
msgid ""
|
msgid ""
|
||||||
"No saved sources with this name, check existing names with --sources-list"
|
"No saved sources with this name, check existing names with --sources-list"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
"Aucun sauvegarde de fichiers sources avec ce nom, veuillez vérifier les "
|
"Aucune sauvegarde de fichiers sources avec ce nom, veuillez vérifier les "
|
||||||
"lists existantes avec --sources-list"
|
"listes existantes avec --sources-list"
|
||||||
|
|
||||||
#: gcp:489
|
#: gcp:525
|
||||||
msgid "Saved sources:"
|
msgid "Saved sources:"
|
||||||
msgstr "Liste de sources sauvées:"
|
msgstr "Liste de sources sauvées :"
|
||||||
|
|
||||||
#: gcp:503
|
#: gcp:539
|
||||||
msgid ""
|
msgid ""
|
||||||
"There is already a saved sources with this name, skipping --sources-save"
|
"There is already a saved sources with this name, skipping --sources-save"
|
||||||
msgstr "Il y a déjà une liste de sources avec ce nom, --sources-save ignoré"
|
msgstr "Il y a déjà une liste de sources avec ce nom, --sources-save ignoré"
|
||||||
|
|
||||||
#: gcp:539
|
#: gcp:578
|
||||||
msgid "copy directories recursively"
|
|
||||||
msgstr "copie les répertoire récursivement"
|
|
||||||
|
|
||||||
#: gcp:542
|
|
||||||
msgid "force overwriting of existing files"
|
msgid "force overwriting of existing files"
|
||||||
msgstr "force le remplacement des fichiers déjà existants"
|
msgstr "forcer le remplacement des fichiers déjà existants"
|
||||||
|
|
||||||
#: gcp:545
|
#: gcp:582
|
||||||
msgid "preserve the specified attributes"
|
msgid "always follow symbolic links in sources"
|
||||||
msgstr "garde les attributs spécifiés"
|
msgstr "toujours suivre les liens symboliques"
|
||||||
|
|
||||||
#: gcp:551
|
#: gcp:586
|
||||||
msgid "don't fix filesystem name incompatibily"
|
msgid "never follow symbolic links in sources"
|
||||||
|
msgstr "ne pas suivre les liens symboliques"
|
||||||
|
|
||||||
|
#: gcp:590
|
||||||
|
#, python-format
|
||||||
|
msgid "same as --preserve=%s"
|
||||||
|
msgstr "raccourci pour --preserve=%s"
|
||||||
|
|
||||||
|
#: gcp:594
|
||||||
|
#, python-format
|
||||||
|
msgid ""
|
||||||
|
"preserve specified attributes; accepted values: 'all', or "
|
||||||
|
"one or more amongst %s"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
"Ne corrige pas les incompatibilités des noms pour le système de fichiers"
|
"préserver les attributs spécifiés; valeurs acceptées : 'all' ou un ou "
|
||||||
|
"plusieurs éléments parmi %s"
|
||||||
|
|
||||||
#: gcp:554
|
#: gcp:599
|
||||||
msgid "deactivate progress bar"
|
msgid "copy directories recursively"
|
||||||
msgstr "désactive la barre de progression"
|
msgstr "copier les répertoire récursivement"
|
||||||
|
|
||||||
#: gcp:557
|
#: gcp:603
|
||||||
msgid "Show what is currently done"
|
msgid "display what is being done"
|
||||||
msgstr "Affiche les opérations effectuées"
|
msgstr "afficher les opérations effectuées"
|
||||||
|
|
||||||
#: gcp:562
|
#: gcp:614
|
||||||
msgid "Save source arguments"
|
msgid ""
|
||||||
msgstr "Sauvegarde la liste des fichiers sources"
|
"fix file names incompatible with the destination file "
|
||||||
|
"system (default: auto)"
|
||||||
#: gcp:565
|
|
||||||
msgid "Save source arguments and replace memory if it already exists"
|
|
||||||
msgstr ""
|
msgstr ""
|
||||||
"Sauvegarde la liste des fichiers sources et la remplace si elle existe déjà"
|
"corriger les noms de fichiers incompatibles avec le système de fichiers "
|
||||||
|
"cible (défaut : auto)"
|
||||||
|
|
||||||
#: gcp:568
|
#: gcp:619
|
||||||
msgid "Load source arguments"
|
msgid ""
|
||||||
msgstr "Réutilise les fichiers sources à copier"
|
"[DEPRECATED] same as --fix-filename=no (overrides --fix-"
|
||||||
|
"filenames)"
|
||||||
#: gcp:571
|
|
||||||
msgid "delete saved sources"
|
|
||||||
msgstr "Supprime la liste des fichiers sources"
|
|
||||||
|
|
||||||
#: gcp:574
|
|
||||||
msgid "List names of saved sources"
|
|
||||||
msgstr "Liste les noms des listes de fichiers sources"
|
|
||||||
|
|
||||||
#: gcp:577
|
|
||||||
msgid "List names of saved sources and files in it"
|
|
||||||
msgstr ""
|
msgstr ""
|
||||||
"Liste les noms des listes de fichiers sources, en incluant les fichiers "
|
"[OBSOLÈTE] identique à --fix-filenames=no (--fix-filenames sera ignoré)"
|
||||||
|
|
||||||
|
#: gcp:624
|
||||||
|
msgid "disable progress bar"
|
||||||
|
msgstr "désactiver la barre de progression"
|
||||||
|
|
||||||
|
#: gcp:631
|
||||||
|
msgid "save sources arguments"
|
||||||
|
msgstr "sauvegarder la liste des fichiers source"
|
||||||
|
|
||||||
|
#: gcp:635
|
||||||
|
msgid "save sources arguments and replace memory if it already exists"
|
||||||
|
msgstr ""
|
||||||
|
"sauvegarder la liste des fichiers source et la remplacer si elle existe déjà"
|
||||||
|
|
||||||
|
#: gcp:639
|
||||||
|
msgid "load sources arguments"
|
||||||
|
msgstr "réutiliser les fichiers source à copier"
|
||||||
|
|
||||||
|
#: gcp:643
|
||||||
|
msgid "delete saved sources list"
|
||||||
|
msgstr "supprimer la liste des fichiers source"
|
||||||
|
|
||||||
|
#: gcp:647
|
||||||
|
msgid "list names of saved sources"
|
||||||
|
msgstr "afficher les noms des listes de fichiers source"
|
||||||
|
|
||||||
|
#: gcp:651
|
||||||
|
msgid "list names of saved sources and files in it"
|
||||||
|
msgstr ""
|
||||||
|
"afficher les noms des listes de fichiers sources, en incluant les fichiers "
|
||||||
"qu'elles contiennent"
|
"qu'elles contiennent"
|
||||||
|
|
||||||
#: gcp:585
|
#: gcp:663
|
||||||
msgid "Progress bar is not available, deactivating"
|
msgid "Progress bar is not available, deactivating"
|
||||||
msgstr "La barre de progression n'est pas disponible, désactivation"
|
msgstr "La barre de progression n'est pas disponible, désactivation"
|
||||||
|
|
||||||
#: gcp:595
|
#: gcp:687
|
||||||
msgid ""
|
#, python-format
|
||||||
"Invalide --preserve value\n"
|
msgid "Invalid --preserve value '%s'"
|
||||||
"valid values are:"
|
msgstr "Valeur de --preserve invalide « %s »"
|
||||||
msgstr ""
|
|
||||||
"La valeur de «--preserve» est invalide\n"
|
|
||||||
"Les valeurs valides sont:"
|
|
||||||
|
|
||||||
#: gcp:616
|
#: gcp:714
|
||||||
#, python-format
|
#, python-format
|
||||||
msgid "There is already one instance of %s running, pluging to it"
|
msgid "There is already one instance of %s running, pluging to it"
|
||||||
msgstr "Il y a déjà une instance de %s lancée, je m'y connecte"
|
msgstr "Il y a déjà une instance de %s lancée, je m'y connecte"
|
||||||
|
|
||||||
#: gcp:622
|
#: gcp:720
|
||||||
msgid "Wrong number of arguments"
|
msgid "Wrong number of arguments"
|
||||||
msgstr "Nombre d'arguments invalide"
|
msgstr "Nombre d'arguments invalide"
|
||||||
|
|
||||||
#: gcp:624
|
#: gcp:722
|
||||||
#, python-format
|
#, python-format
|
||||||
msgid "adding args to gcp: %s"
|
msgid "adding args to gcp: %s"
|
||||||
msgstr "ajout des arguments à gcp: %s"
|
msgstr "ajout des arguments à gcp : %s"
|
||||||
|
|
||||||
#: gcp:633
|
#: gcp:731
|
||||||
msgid "User interruption: good bye"
|
msgid "User interruption: good bye"
|
||||||
msgstr "Interruption par l'utilisateur: au revoir"
|
msgstr "Interruption par l'utilisateur : au revoir"
|
||||||
|
|
||||||
#~ msgid "don't fixe name encoding errors"
|
#~ msgid "don't fixe name encoding errors"
|
||||||
#~ msgstr "Ne corrige pas les erreurs dans l'encodage des noms"
|
#~ msgstr "Ne corrige pas les erreurs dans l'encodage des noms"
|
||||||
|
|
452
gcp
452
gcp
|
@ -1,10 +1,11 @@
|
||||||
#!/usr/bin/python
|
#!/usr/bin/env python3
|
||||||
# -*- coding: utf-8 -*-
|
|
||||||
|
|
||||||
"""
|
"""
|
||||||
gcp: Goffi's CoPier
|
gcp: Gcp CoPier
|
||||||
Copyright (C) 2010, 2011 Jérôme Poisson <goffi@goffi.org>
|
Copyright (c) 2010, 2011 Jérôme Poisson <goffi@goffi.org>
|
||||||
(c) 2011 Thomas Preud'homme <robotux@celest.fr>
|
(c) 2011 Thomas Preud'homme <robotux@celest.fr>
|
||||||
|
(c) 2016 Jingbei Li <i@jingbei.li>
|
||||||
|
(c) 2018, 2019 Matteo Cypriani <mcy@lm7.fr>
|
||||||
|
|
||||||
This program is free software: you can redistribute it and/or modify
|
This program is free software: you can redistribute it and/or modify
|
||||||
it under the terms of the GNU General Public License as published by
|
it under the terms of the GNU General Public License as published by
|
||||||
|
@ -20,66 +21,65 @@ You should have received a copy of the GNU General Public License
|
||||||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
### logging ###
|
|
||||||
import logging
|
import logging
|
||||||
from logging import debug, info, error, warning
|
from logging import debug, info, error, warning
|
||||||
logging.basicConfig(level=logging.INFO,
|
|
||||||
format='%(message)s')
|
|
||||||
###
|
|
||||||
|
|
||||||
import gettext
|
import gettext
|
||||||
gettext.install('gcp', "i18n", unicode=True)
|
|
||||||
|
|
||||||
import sys
|
import sys
|
||||||
import os,os.path
|
import os
|
||||||
from optparse import OptionParser, OptionGroup #To be replaced by argparse ASAP
|
import os.path
|
||||||
import cPickle as pickle
|
from argparse import ArgumentParser, RawDescriptionHelpFormatter
|
||||||
|
import pickle
|
||||||
|
|
||||||
|
logging.basicConfig(level=logging.INFO, format='%(message)s')
|
||||||
|
gettext.install('gcp', "i18n")
|
||||||
|
|
||||||
try:
|
try:
|
||||||
import gobject
|
from gi.repository import GLib
|
||||||
#DBus
|
from dbus.mainloop.glib import DBusGMainLoop
|
||||||
import dbus, dbus.glib
|
DBusGMainLoop(set_as_default=True)
|
||||||
import dbus.service
|
import dbus.service
|
||||||
import dbus.mainloop.glib
|
import dbus
|
||||||
except ImportError,e:
|
except ImportError as e:
|
||||||
error(_("Error during import"))
|
error(_("Error during import"))
|
||||||
error(_("Please check dependecies:"),e)
|
error(_("Please check dependecies:"), e)
|
||||||
exit(1)
|
exit(1)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
from progressbar import ProgressBar, Percentage, Bar, ETA, FileTransferSpeed
|
from progressbar import ProgressBar, Percentage, Bar, ETA, FileTransferSpeed
|
||||||
pbar_available=True
|
pbar_available=True
|
||||||
except ImportError, e:
|
except ImportError as e:
|
||||||
info (_('ProgressBar not available, please download it at http://pypi.python.org/pypi/progressbar'))
|
info (_("ProgressBar not available, please download it at https://pypi.org/"))
|
||||||
info (_('Progress bar deactivated\n--\n'))
|
info (_('Progress bar deactivated\n--\n'))
|
||||||
pbar_available=False
|
pbar_available=False
|
||||||
|
|
||||||
NAME = "gcp (Goffi's copier)"
|
NAME = "gcp (Gcp CoPier)"
|
||||||
NAME_SHORT = "gcp"
|
NAME_SHORT = "gcp"
|
||||||
VERSION = '0.1.3'
|
VERSION = '0.2.1'
|
||||||
|
|
||||||
ABOUT = NAME+u" v"+VERSION+u""" (c) Jérôme Poisson (aka Goffi) 2010, 2011
|
|
||||||
|
|
||||||
|
ABOUT = NAME_SHORT + " " + VERSION + """
|
||||||
---
|
---
|
||||||
"""+NAME+u""" Copyright (C) 2010 Jérôme Poisson
|
""" + NAME + """
|
||||||
""" + _(u"""This program comes with ABSOLUTELY NO WARRANTY;
|
Copyright: 2010-2011 Jérôme Poisson <goffi@goffi.org>
|
||||||
This is free software, and you are welcome to redistribute it
|
2011 Thomas Preud'homme <robotux@celest.fr>
|
||||||
under certain conditions.
|
2016 Jingbei Li <i@jingbei.li>
|
||||||
---
|
2018 Matteo Cypriani <mcy@lm7.fr>
|
||||||
|
This program comes with ABSOLUTELY NO WARRANTY; it is free software,
|
||||||
This software is an advanced file copier
|
and you are welcome to redistribute it under certain conditions.
|
||||||
Get the latest version at http://wiki.goffi.org/wiki/Gcp
|
"""
|
||||||
""")
|
|
||||||
|
|
||||||
const_DBUS_INTERFACE = "org.goffi.gcp"
|
const_DBUS_INTERFACE = "org.goffi.gcp"
|
||||||
const_DBUS_PATH = "/org/goffi/gcp"
|
const_DBUS_PATH = "/org/goffi/gcp"
|
||||||
const_BUFF_SIZE = 4096
|
const_BUFF_SIZE = 4096
|
||||||
const_PRESERVE = set(['mode','ownership','timestamps'])
|
const_PRESERVE = set(['mode','ownership','timestamps'])
|
||||||
|
const_PRESERVE_p = 'mode,ownership,timestamps'
|
||||||
const_FS_FIX = set(['auto','force','no'])
|
const_FS_FIX = set(['auto','force','no'])
|
||||||
const_FILES_DIR = "~/.gcp"
|
const_FILES_DIR = "~/.gcp"
|
||||||
const_JOURNAL_PATH = const_FILES_DIR + "/journal"
|
const_JOURNAL_PATH = const_FILES_DIR + "/journal"
|
||||||
const_SAVED_LIST = const_FILES_DIR + "/saved_list"
|
const_SAVED_LIST = const_FILES_DIR + "/saved_list"
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
class DbusObject(dbus.service.Object):
|
class DbusObject(dbus.service.Object):
|
||||||
|
|
||||||
def __init__(self, gcp, bus, path):
|
def __init__(self, gcp, bus, path):
|
||||||
|
@ -104,15 +104,19 @@ class DbusObject(dbus.service.Object):
|
||||||
@return: success (boolean) and error message if any (string)"""
|
@return: success (boolean) and error message if any (string)"""
|
||||||
try:
|
try:
|
||||||
args = pickle.loads(str(args))
|
args = pickle.loads(str(args))
|
||||||
except TypeError, pickle.UnpicklingError:
|
except TypeError as e:
|
||||||
|
pickle.UnpicklingError = e
|
||||||
return (False, _("INTERNAL ERROR: invalid arguments"))
|
return (False, _("INTERNAL ERROR: invalid arguments"))
|
||||||
try:
|
try:
|
||||||
source_dir = pickle.loads(str(source_dir))
|
source_dir = pickle.loads(str(source_dir))
|
||||||
except TypeError, pickle.UnpicklingError:
|
except TypeError as e:
|
||||||
|
pickle.UnpicklingError = e
|
||||||
return (False, _("INTERNAL ERROR: invalid source_dir"))
|
return (False, _("INTERNAL ERROR: invalid source_dir"))
|
||||||
return self._gcp.parseArguments(args, source_dir)
|
return self._gcp.parseArguments(args, source_dir)
|
||||||
|
|
||||||
|
|
||||||
class Journal():
|
class Journal():
|
||||||
|
|
||||||
def __init__(self, path=const_JOURNAL_PATH):
|
def __init__(self, path=const_JOURNAL_PATH):
|
||||||
self.journal_path = os.path.expanduser(path)
|
self.journal_path = os.path.expanduser(path)
|
||||||
self.journal_fd = open(self.journal_path,'w') #TODO: check and maybe save previous journals
|
self.journal_fd = open(self.journal_path,'w') #TODO: check and maybe save previous journals
|
||||||
|
@ -179,8 +183,6 @@ class Journal():
|
||||||
info(_("Please check journal: %s") % self.journal_path)
|
info(_("Please check journal: %s") % self.journal_path)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
class GCP():
|
class GCP():
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
|
@ -195,7 +197,7 @@ class GCP():
|
||||||
dbus_interface=const_DBUS_INTERFACE)
|
dbus_interface=const_DBUS_INTERFACE)
|
||||||
self._main_instance = False
|
self._main_instance = False
|
||||||
|
|
||||||
except dbus.exceptions.DBusException,e:
|
except dbus.exceptions.DBusException as e:
|
||||||
if e._dbus_error_name=='org.freedesktop.DBus.Error.ServiceUnknown':
|
if e._dbus_error_name=='org.freedesktop.DBus.Error.ServiceUnknown':
|
||||||
self.launchDbusMainInstance()
|
self.launchDbusMainInstance()
|
||||||
debug (_("gcp launched"))
|
debug (_("gcp launched"))
|
||||||
|
@ -227,14 +229,15 @@ class GCP():
|
||||||
|
|
||||||
def __getMountPoints(self):
|
def __getMountPoints(self):
|
||||||
"""Parse /proc/mounts to get currently mounted devices"""
|
"""Parse /proc/mounts to get currently mounted devices"""
|
||||||
#TODO: reparse when a new device is added/a device is removed
|
# TODO: reparse when a new device is added/a device is removed
|
||||||
#(check freedesktop mounting signals)
|
# (check freedesktop mounting signals)
|
||||||
ret = {}
|
ret = {}
|
||||||
try:
|
try:
|
||||||
with open("/proc/mounts",'rb') as mounts:
|
with open("/proc/mounts",'r') as mounts:
|
||||||
for line in mounts.readlines():
|
for line in mounts.readlines():
|
||||||
fs_spec, fs_file, fs_vfstype, fs_mntops, fs_freq, fs_passno = line.split(' ')
|
fs_spec, fs_file, fs_vfstype, \
|
||||||
ret[fs_file] = fs_vfstype
|
fs_mntops, fs_freq, fs_passno = line.split(' ')
|
||||||
|
ret[fs_file] = fs_vfstype
|
||||||
except:
|
except:
|
||||||
error (_("Can't read mounts table"))
|
error (_("Can't read mounts table"))
|
||||||
return ret
|
return ret
|
||||||
|
@ -243,22 +246,22 @@ class GCP():
|
||||||
"""Add a file to the copy list
|
"""Add a file to the copy list
|
||||||
@param path: absolute path of file
|
@param path: absolute path of file
|
||||||
@param options: options as return by optparse"""
|
@param options: options as return by optparse"""
|
||||||
debug (_("Adding to copy list: %(path)s ==> %(dest_path)s (%(fs_type)s)") % {"path":path.decode('utf-8','replace'),
|
debug(_("Adding to copy list: %(path)s ==> %(dest_path)s (%(fs_type)s)")
|
||||||
"dest_path":dest_path.decode('utf-8','replace'),
|
% {"path":path, "dest_path":dest_path,
|
||||||
"fs_type":self.getFsType(dest_path)} )
|
"fs_type":self.getFsType(dest_path)})
|
||||||
try:
|
try:
|
||||||
self.bytes_total+=os.path.getsize(path)
|
self.bytes_total+=os.path.getsize(path)
|
||||||
self.copy_list.insert(0,(path, dest_path, options))
|
self.copy_list.insert(0,(path, dest_path, options))
|
||||||
except OSError,e:
|
except OSError as e:
|
||||||
error(_("Can't copy %(path)s: %(exception)s") % {'path':path.decode('utf-8','replace'), 'exception':e.strerror})
|
error(_("Can't copy %(path)s: %(exception)s")
|
||||||
|
% {'path':path, 'exception':e.strerror})
|
||||||
|
|
||||||
def __appendDirToList(self, dirpath, dest_path, options):
|
def __appendDirToList(self, dirpath, dest_path, options):
|
||||||
"""Add recursively directory to the copy list
|
"""Add recursively directory to the copy list
|
||||||
@param path: absolute path of dir
|
@param path: absolute path of dir
|
||||||
@param options: options as return by optparse"""
|
@param options: options as return by optparse"""
|
||||||
#We first check that the dest path exists, and create it if needed
|
#We first check that the dest path exists, and create it if needed
|
||||||
dest_path = self.__filename_fix(dest_path, options, no_journal=True)
|
dest_path = self.__fix_filenames(dest_path, options, no_journal=True)
|
||||||
if not os.path.exists(dest_path):
|
if not os.path.exists(dest_path):
|
||||||
debug ("Creating directory %s" % dest_path)
|
debug ("Creating directory %s" % dest_path)
|
||||||
os.makedirs(dest_path) #TODO: check permissions
|
os.makedirs(dest_path) #TODO: check permissions
|
||||||
|
@ -268,21 +271,19 @@ class GCP():
|
||||||
for filename in os.listdir(dirpath):
|
for filename in os.listdir(dirpath):
|
||||||
filepath = os.path.join(dirpath,filename)
|
filepath = os.path.join(dirpath,filename)
|
||||||
if os.path.islink(filepath) and not options.dereference:
|
if os.path.islink(filepath) and not options.dereference:
|
||||||
debug ("Skippink symbolic dir: %s" % filepath.decode('utf-8','replace'))
|
debug ("Skippink symbolic dir: %s" % filepath)
|
||||||
continue
|
continue
|
||||||
if os.path.isdir(filepath):
|
if os.path.isdir(filepath):
|
||||||
full_dest_path = os.path.join(dest_path,filename)
|
full_dest_path = os.path.join(dest_path,filename)
|
||||||
self.__appendDirToList(filepath, full_dest_path, options)
|
self.__appendDirToList(filepath, full_dest_path, options)
|
||||||
else:
|
else:
|
||||||
self.__appendToList(filepath, dest_path, options)
|
self.__appendToList(filepath, dest_path, options)
|
||||||
except OSError,e:
|
except OSError as e:
|
||||||
try:
|
try:
|
||||||
error(_("Can't append %(path)s to copy list: %(exception)s") % {'path':filepath.decode('utf-8','replace'),
|
error(_("Can't append %(path)s to copy list: %(exception)s") % {'path':filepath, 'exception':e.strerror})
|
||||||
'exception':e.strerror})
|
|
||||||
except NameError:
|
except NameError:
|
||||||
#We can't list the dir
|
#We can't list the dir
|
||||||
error(_("Can't access %(dirpath)s: %(exception)s") % {'dirpath':dirpath.decode('utf-8','replace'),
|
error(_("Can't access %(dirpath)s: %(exception)s") % {'dirpath':dirpath, 'exception':e.strerror})
|
||||||
'exception':e.strerror})
|
|
||||||
|
|
||||||
def __checkArgs(self, options, source_dir, args):
|
def __checkArgs(self, options, source_dir, args):
|
||||||
"""Check thats args are files, and add them to copy list
|
"""Check thats args are files, and add them to copy list
|
||||||
|
@ -293,17 +294,17 @@ class GCP():
|
||||||
len_args = len(args)
|
len_args = len(args)
|
||||||
try:
|
try:
|
||||||
dest_path = os.path.normpath(os.path.join(source_dir, args.pop()))
|
dest_path = os.path.normpath(os.path.join(source_dir, args.pop()))
|
||||||
except OSError,e:
|
except OSError as e:
|
||||||
error (_("Invalid dest_path: %s"),e)
|
error (_("Invalid dest_path: %s"),e)
|
||||||
|
|
||||||
for path in args:
|
for path in args:
|
||||||
abspath = os.path.normpath(os.path.join(os.path.expanduser(source_dir), path))
|
abspath = os.path.normpath(os.path.join(os.path.expanduser(source_dir), path))
|
||||||
if not os.path.exists(abspath):
|
if not os.path.exists(abspath):
|
||||||
warning(_("The path given in arg doesn't exist or is not accessible: %s") % abspath.decode('utf-8','replace'))
|
warning(_("The path given in arg doesn't exist or is not accessible: %s") % abspath)
|
||||||
else:
|
else:
|
||||||
if os.path.isdir(abspath):
|
if os.path.isdir(abspath):
|
||||||
if not options.recursive:
|
if not options.recursive:
|
||||||
warning (_('omitting directory "%s"') % abspath.decode('utf-8','replace'))
|
warning (_('omitting directory "%s"') % abspath)
|
||||||
else:
|
else:
|
||||||
_basename=os.path.basename(os.path.normpath(path))
|
_basename=os.path.basename(os.path.normpath(path))
|
||||||
full_dest_path = dest_path if options.directdir else os.path.normpath(os.path.join(dest_path, _basename))
|
full_dest_path = dest_path if options.directdir else os.path.normpath(os.path.join(dest_path, _basename))
|
||||||
|
@ -312,49 +313,58 @@ class GCP():
|
||||||
self.__appendToList(abspath, dest_path, options)
|
self.__appendToList(abspath, dest_path, options)
|
||||||
|
|
||||||
def __copyNextFile(self):
|
def __copyNextFile(self):
|
||||||
"""Take the last file in the list, and launch the copy using glib io_watch event
|
"""Takes the last file in the list and launches the copy using glib
|
||||||
@return: True a file was added, False else"""
|
io_watch event.
|
||||||
if self.copy_list:
|
@return: True a file was added, False otherwise."""
|
||||||
source_file, dest_path, options = self.copy_list.pop()
|
if not self.copy_list:
|
||||||
self.journal.startFile(source_file)
|
# Nothing left to copy, we quit
|
||||||
try:
|
|
||||||
source_fd = open(source_file, 'rb')
|
|
||||||
except:
|
|
||||||
self.journal.copyFailed()
|
|
||||||
self.journal.error("can't open source")
|
|
||||||
self.journal.closeFile()
|
|
||||||
return True
|
|
||||||
filename = os.path.basename(source_file)
|
|
||||||
assert(filename)
|
|
||||||
dest_file = self.__filename_fix(options.dest_file,options) if options.dest_file else self.__filename_fix(os.path.join(dest_path,filename),options)
|
|
||||||
if os.path.exists(dest_file) and not options.force:
|
|
||||||
warning (_("File [%s] already exists, skipping it !") % dest_file.decode('utf-8','replace'))
|
|
||||||
self.journal.copyFailed()
|
|
||||||
self.journal.error("already exists")
|
|
||||||
self.journal.closeFile()
|
|
||||||
source_fd.close()
|
|
||||||
return True
|
|
||||||
try:
|
|
||||||
dest_fd = open(dest_file, 'wb')
|
|
||||||
except:
|
|
||||||
self.journal.copyFailed()
|
|
||||||
self.journal.error("can't open dest")
|
|
||||||
self.journal.closeFile()
|
|
||||||
source_fd.close()
|
|
||||||
return True
|
|
||||||
|
|
||||||
gobject.io_add_watch(source_fd,gobject.IO_IN,self._copyFile,
|
|
||||||
(dest_fd, options), priority=gobject.PRIORITY_DEFAULT)
|
|
||||||
if not self.progress:
|
|
||||||
info(_("COPYING %(source)s ==> %(dest)s") % {"source":source_file.decode('utf-8','replace'),
|
|
||||||
"dest":dest_file.decode('utf-8','replace')})
|
|
||||||
return True
|
|
||||||
else:
|
|
||||||
#Nothing left to copy, we quit
|
|
||||||
if self.progress:
|
if self.progress:
|
||||||
self.__pbar_finish()
|
self.__pbar_finish()
|
||||||
self.journal.showErrors()
|
self.journal.showErrors()
|
||||||
self.loop.quit()
|
self.loop.quit()
|
||||||
|
return False
|
||||||
|
|
||||||
|
source_file, dest_path, options = self.copy_list.pop()
|
||||||
|
self.journal.startFile(source_file)
|
||||||
|
try:
|
||||||
|
source_fd = open(source_file, 'rb')
|
||||||
|
except:
|
||||||
|
self.journal.copyFailed()
|
||||||
|
self.journal.error("can't open source")
|
||||||
|
self.journal.closeFile()
|
||||||
|
return True
|
||||||
|
|
||||||
|
filename = os.path.basename(source_file)
|
||||||
|
assert(filename)
|
||||||
|
if options.dest_file:
|
||||||
|
dest_file = self.__fix_filenames(options.dest_file, options)
|
||||||
|
else:
|
||||||
|
dest_file = self.__fix_filenames(os.path.join(dest_path, filename),
|
||||||
|
options)
|
||||||
|
if os.path.exists(dest_file) and not options.force:
|
||||||
|
warning (_("File [%s] already exists, skipping it!") % dest_file)
|
||||||
|
self.journal.copyFailed()
|
||||||
|
self.journal.error("already exists")
|
||||||
|
self.journal.closeFile()
|
||||||
|
source_fd.close()
|
||||||
|
return True
|
||||||
|
|
||||||
|
try:
|
||||||
|
dest_fd = open(dest_file, 'wb')
|
||||||
|
except:
|
||||||
|
self.journal.copyFailed()
|
||||||
|
self.journal.error("can't open dest")
|
||||||
|
self.journal.closeFile()
|
||||||
|
source_fd.close()
|
||||||
|
return True
|
||||||
|
|
||||||
|
GLib.io_add_watch(source_fd, GLib.IO_IN,self._copyFile,
|
||||||
|
(dest_fd, options),
|
||||||
|
priority=GLib.PRIORITY_DEFAULT)
|
||||||
|
if not self.progress:
|
||||||
|
info(_("COPYING %(source)s ==> %(dest)s")
|
||||||
|
% {"source":source_file, "dest":dest_file})
|
||||||
|
return True
|
||||||
|
|
||||||
def __copyFailed(self, reason, source_fd, dest_fd):
|
def __copyFailed(self, reason, source_fd, dest_fd):
|
||||||
"""Write the failure in the journal and close files descriptors"""
|
"""Write the failure in the journal and close files descriptors"""
|
||||||
|
@ -364,8 +374,6 @@ class GCP():
|
||||||
source_fd.close()
|
source_fd.close()
|
||||||
dest_fd.close()
|
dest_fd.close()
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
def _copyFile(self, source_fd, condition, data):
|
def _copyFile(self, source_fd, condition, data):
|
||||||
"""Actually copy the file, callback used with io_add_watch
|
"""Actually copy the file, callback used with io_add_watch
|
||||||
@param source_fd: file descriptor of the file to copy
|
@param source_fd: file descriptor of the file to copy
|
||||||
|
@ -404,7 +412,7 @@ class GCP():
|
||||||
except KeyboardInterrupt:
|
except KeyboardInterrupt:
|
||||||
self._userInterruption()
|
self._userInterruption()
|
||||||
|
|
||||||
def __filename_fix(self, filename, options, no_journal=False):
|
def __fix_filenames(self, filename, options, no_journal=False):
|
||||||
"""Fix filenames incompatibilities/mistake according to options
|
"""Fix filenames incompatibilities/mistake according to options
|
||||||
@param filename: full path to the file
|
@param filename: full path to the file
|
||||||
@param options: options as parsed on command line
|
@param options: options as parsed on command line
|
||||||
|
@ -412,7 +420,7 @@ class GCP():
|
||||||
@return: fixed filename"""
|
@return: fixed filename"""
|
||||||
fixed_filename = filename
|
fixed_filename = filename
|
||||||
|
|
||||||
if options.fs_fix == 'force' or (options.fs_fix == 'auto' and self.getFsType(filename) == 'vfat'):
|
if options.fix_filenames == 'force' or (options.fix_filenames == 'auto' and self.getFsType(filename) == 'vfat'):
|
||||||
fixed_filename = filename.replace('\\','_')\
|
fixed_filename = filename.replace('\\','_')\
|
||||||
.replace(':',';')\
|
.replace(':',';')\
|
||||||
.replace('*','+')\
|
.replace('*','+')\
|
||||||
|
@ -440,24 +448,22 @@ class GCP():
|
||||||
os.chown(dest_file, st_file.st_uid, st_file.st_gid)
|
os.chown(dest_file, st_file.st_uid, st_file.st_gid)
|
||||||
elif preserve == 'timestamps':
|
elif preserve == 'timestamps':
|
||||||
os.utime(dest_file, (st_file.st_atime, st_file.st_mtime))
|
os.utime(dest_file, (st_file.st_atime, st_file.st_mtime))
|
||||||
except OSError,e:
|
except OSError as e:
|
||||||
self.journal.error("preserve-"+preserve)
|
self.journal.error("preserve-"+preserve)
|
||||||
|
|
||||||
def __get_string_size(self, size):
|
def __get_string_size(self, size):
|
||||||
"""Return a nice string representation of a size"""
|
"""Return a nice string representation of a size"""
|
||||||
|
if size >= 2**50:
|
||||||
if size>=2**50:
|
return _("%.2f PiB") % (float(size) / 2**50)
|
||||||
return _("%.2f PiB") % (float(size)/2**50)
|
if size >= 2**40:
|
||||||
elif size>=2**40:
|
return _("%.2f TiB") % (float(size) / 2**40)
|
||||||
return _("%.2f TiB") % (float(size)/2**40)
|
if size >= 2**30:
|
||||||
elif size>=2**30:
|
return _("%.2f GiB") % (float(size) / 2**30)
|
||||||
return _("%.2f GiB") % (float(size)/2**30)
|
if size >= 2**20:
|
||||||
elif size>=2**20:
|
return _("%.2f MiB") % (float(size) / 2**20)
|
||||||
return _("%.2f MiB") % (float(size)/2**20)
|
if size >= 2**10:
|
||||||
elif size>=2**10:
|
return _("%.2f KiB") % (float(size) / 2**10)
|
||||||
return _("%.2f KiB") % (float(size)/2**10)
|
return _("%i B") % size
|
||||||
else:
|
|
||||||
return _("%i B") % size
|
|
||||||
|
|
||||||
def _pbar_update(self):
|
def _pbar_update(self):
|
||||||
"""Update progress bar position, create the bar if it doesn't exist"""
|
"""Update progress bar position, create the bar if it doesn't exist"""
|
||||||
|
@ -465,12 +471,16 @@ class GCP():
|
||||||
try:
|
try:
|
||||||
if self.pbar.maxval != self.bytes_total:
|
if self.pbar.maxval != self.bytes_total:
|
||||||
self.pbar.maxval = self.bytes_total
|
self.pbar.maxval = self.bytes_total
|
||||||
self.pbar.widgets[0] = _("Copying %s") % self.__get_string_size(self.bytes_total)
|
pbar_msg = _("Copying %s") % self.__get_string_size(self.bytes_total)
|
||||||
|
self.pbar.widgets[0] = pbar_msg
|
||||||
except AttributeError:
|
except AttributeError:
|
||||||
if not self.bytes_total:
|
if not self.bytes_total:
|
||||||
#No progress bar if the files have a null size
|
# No progress bar if the files have a null size
|
||||||
return
|
return
|
||||||
self.pbar = ProgressBar(self.bytes_total,[_("Copying %s") % self.__get_string_size(self.bytes_total)," ",Percentage()," ",Bar()," ",FileTransferSpeed()," ",ETA()])
|
pbar_msg = _("Copying %s") % self.__get_string_size(self.bytes_total)
|
||||||
|
self.pbar = ProgressBar(self.bytes_total,
|
||||||
|
[pbar_msg, " ", Percentage(), " ", Bar(),
|
||||||
|
" ", FileTransferSpeed(), " ", ETA()])
|
||||||
self.pbar.start()
|
self.pbar.start()
|
||||||
self.pbar.update(self.bytes_copied)
|
self.pbar.update(self.bytes_copied)
|
||||||
|
|
||||||
|
@ -496,7 +506,7 @@ class GCP():
|
||||||
saved_files={}
|
saved_files={}
|
||||||
|
|
||||||
if options.sources_del:
|
if options.sources_del:
|
||||||
if not saved_files.has_key(options.sources_del):
|
if options.sources_del not in saved_files:
|
||||||
error(_("No saved sources with this name, check existing names with --sources-list"))
|
error(_("No saved sources with this name, check existing names with --sources-list"))
|
||||||
else:
|
else:
|
||||||
del saved_files[options.sources_del]
|
del saved_files[options.sources_del]
|
||||||
|
@ -505,10 +515,9 @@ class GCP():
|
||||||
if not args:
|
if not args:
|
||||||
exit(0)
|
exit(0)
|
||||||
|
|
||||||
|
|
||||||
if options.sources_list or options.sources_full_list:
|
if options.sources_list or options.sources_full_list:
|
||||||
info(_('Saved sources:'))
|
info(_('Saved sources:'))
|
||||||
sources = saved_files.keys()
|
sources = list(saved_files.keys())
|
||||||
sources.sort()
|
sources.sort()
|
||||||
for source in sources:
|
for source in sources:
|
||||||
info("\t[%s]" % source)
|
info("\t[%s]" % source)
|
||||||
|
@ -520,16 +529,16 @@ class GCP():
|
||||||
exit(0)
|
exit(0)
|
||||||
|
|
||||||
if options.sources_save or options.sources_replace:
|
if options.sources_save or options.sources_replace:
|
||||||
if saved_files.has_key(options.sources_save) and not options.sources_replace:
|
if options.sources_save in saved_files and not options.sources_replace:
|
||||||
error(_("There is already a saved sources with this name, skipping --sources-save"))
|
error(_("There is already a saved sources with this name, skipping --sources-save"))
|
||||||
else:
|
else:
|
||||||
if len(args)>1:
|
if len(args)>1:
|
||||||
saved_files[options.sources_save] = map(os.path.abspath,args[:-1])
|
saved_files[options.sources_save] = list(map(os.path.abspath,args[:-1]))
|
||||||
with open(os.path.expanduser(const_SAVED_LIST),'w') as saved_fd:
|
with open(os.path.expanduser(const_SAVED_LIST),'w') as saved_fd:
|
||||||
pickle.dump(saved_files,saved_fd)
|
pickle.dump(saved_files,saved_fd)
|
||||||
|
|
||||||
if options.sources_load:
|
if options.sources_load:
|
||||||
if not saved_files.has_key(options.sources_load):
|
if options.sources_load not in saved_files:
|
||||||
error(_("No saved sources with this name, check existing names with --sources-list"))
|
error(_("No saved sources with this name, check existing names with --sources-list"))
|
||||||
else:
|
else:
|
||||||
saved_args = saved_files[options.sources_load]
|
saved_args = saved_files[options.sources_load]
|
||||||
|
@ -544,74 +553,106 @@ class GCP():
|
||||||
@return: a tuple (boolean, message) where the boolean is the success of the arguments
|
@return: a tuple (boolean, message) where the boolean is the success of the arguments
|
||||||
validation, and message is the error message to print when necessary"""
|
validation, and message is the error message to print when necessary"""
|
||||||
_usage="""
|
_usage="""
|
||||||
%prog [options] FILE DEST
|
%(prog)s [options] FILE DEST
|
||||||
%prog [options] FILE1 [FILE2 ...] DEST-DIR
|
%(prog)s [options] FILE1 [FILE2 ...] DEST-DIR
|
||||||
|
|
||||||
%prog --help for options list
|
|
||||||
"""
|
"""
|
||||||
for idx in range(len(full_args)):
|
for idx in range(len(full_args)):
|
||||||
if isinstance(full_args[idx], unicode):
|
full_args[idx] = full_args[idx].encode('utf-8')
|
||||||
#We don't want unicode as some filenames can be invalid unicode
|
|
||||||
full_args[idx] = full_args[idx].encode('utf-8')
|
|
||||||
|
|
||||||
parser = OptionParser(usage=_usage,version=ABOUT)
|
parser = ArgumentParser(usage=_usage,
|
||||||
|
formatter_class=RawDescriptionHelpFormatter)
|
||||||
|
|
||||||
parser.add_option("-r", "--recursive", action="store_true", default=False,
|
parser.add_argument("-V", "--version",
|
||||||
help=_("copy directories recursively"))
|
action="version", version=ABOUT
|
||||||
|
)
|
||||||
|
|
||||||
parser.add_option("-f", "--force", action="store_true", default=False,
|
group_cplike = parser.add_argument_group("cp-like options")
|
||||||
help=_("force overwriting of existing files"))
|
group_cplike.add_argument("-f", "--force",
|
||||||
|
action="store_true", default=False,
|
||||||
|
help=_("force overwriting of existing files")
|
||||||
|
)
|
||||||
|
group_cplike.add_argument("-L", "--dereference",
|
||||||
|
action="store_true", default=False,
|
||||||
|
help=_("always follow symbolic links in sources")
|
||||||
|
)
|
||||||
|
group_cplike.add_argument("-P", "--no-dereference",
|
||||||
|
action="store_false", dest='dereference',
|
||||||
|
help=_("never follow symbolic links in sources")
|
||||||
|
)
|
||||||
|
group_cplike.add_argument("-p",
|
||||||
|
action="store_true", default=False,
|
||||||
|
help=_("same as --preserve=%s" % const_PRESERVE_p)
|
||||||
|
)
|
||||||
|
group_cplike.add_argument("--preserve",
|
||||||
|
action="store", default='',
|
||||||
|
help=_("preserve specified attributes; accepted values: \
|
||||||
|
'all', or one or more amongst %s") % str(const_PRESERVE)
|
||||||
|
)
|
||||||
|
group_cplike.add_argument("-r", "-R", "--recursive",
|
||||||
|
action="store_true", default=False,
|
||||||
|
help=_("copy directories recursively")
|
||||||
|
)
|
||||||
|
group_cplike.add_argument("-v", "--verbose",
|
||||||
|
action="store_true", default=False,
|
||||||
|
help=_("display what is being done")
|
||||||
|
)
|
||||||
|
parser.add_argument_group(group_cplike)
|
||||||
|
|
||||||
parser.add_option("--preserve", action="store", default='mode,ownership,timestamps',
|
group_gcpspecific = parser.add_argument_group("gcp-specific options")
|
||||||
help=_("preserve the specified attributes"))
|
#parser.add_argument("--no-unicode-fix",
|
||||||
|
# action="store_false", dest='unicode_fix', default=True,
|
||||||
|
# help=_("don't fix name encoding errors") #TODO
|
||||||
|
#)
|
||||||
|
group_gcpspecific.add_argument("--fix-filenames",
|
||||||
|
choices = const_FS_FIX, dest='fix_filenames', default='auto',
|
||||||
|
help=_("fix file names incompatible with the destination \
|
||||||
|
file system (default: auto)")
|
||||||
|
)
|
||||||
|
group_gcpspecific.add_argument("--no-fs-fix",
|
||||||
|
action="store_true", dest='no_fs_fix', default=False,
|
||||||
|
help=_("[DEPRECATED] same as --fix-filename=no (overrides \
|
||||||
|
--fix-filenames)")
|
||||||
|
)
|
||||||
|
group_gcpspecific.add_argument("--no-progress",
|
||||||
|
action="store_false", dest="progress", default=True,
|
||||||
|
help=_("disable progress bar")
|
||||||
|
)
|
||||||
|
parser.add_argument_group(group_gcpspecific)
|
||||||
|
|
||||||
parser.add_option("-L", "--dereference", action="store_true", default=False,
|
group_saving = parser.add_argument_group("sources saving")
|
||||||
help=_("always follow symbolic links in sources"))
|
group_saving.add_argument("--sources-save",
|
||||||
|
action="store",
|
||||||
|
help=_("save sources arguments")
|
||||||
|
)
|
||||||
|
group_saving.add_argument("--sources-replace",
|
||||||
|
action="store",
|
||||||
|
help=_("save sources arguments and replace memory if it already exists")
|
||||||
|
)
|
||||||
|
group_saving.add_argument("--sources-load",
|
||||||
|
action="store",
|
||||||
|
help=_("load sources arguments")
|
||||||
|
)
|
||||||
|
group_saving.add_argument("--sources-del",
|
||||||
|
action="store",
|
||||||
|
help=_("delete saved sources list")
|
||||||
|
)
|
||||||
|
group_saving.add_argument("--sources-list",
|
||||||
|
action="store_true", default=False,
|
||||||
|
help=_("list names of saved sources")
|
||||||
|
)
|
||||||
|
group_saving.add_argument("--sources-full-list",
|
||||||
|
action="store_true", default=False,
|
||||||
|
help=_("list names of saved sources and files in it")
|
||||||
|
)
|
||||||
|
parser.add_argument_group(group_saving)
|
||||||
|
|
||||||
parser.add_option("-P", "--no-dereference", action="store_false", dest='dereference',
|
(options, args) = parser.parse_known_args()
|
||||||
help=_("never follow symbolic links in sources"))
|
|
||||||
|
|
||||||
#parser.add_option("--no-unicode-fix", action="store_false", dest='unicode_fix', default=True,
|
# True only in the special case: we are copying a dir and it doesn't
|
||||||
# help=_("don't fix name encoding errors")) #TODO
|
# exists:
|
||||||
|
options.directdir = False
|
||||||
|
|
||||||
parser.add_option("--fs-fix", action="store", dest='fs_fix', default='auto',
|
# options check
|
||||||
help=_("fix filesystem name incompatibily; can be 'auto' (default), 'force' or 'no'"))
|
|
||||||
|
|
||||||
parser.add_option("--no-fs-fix", action="store_true", dest='no_fs_fix', default=False,
|
|
||||||
help=_("same as --fs-fix=no (overrides --fs-fix)"))
|
|
||||||
|
|
||||||
parser.add_option("--no-progress", action="store_false", dest="progress", default=True,
|
|
||||||
help=_("deactivate progress bar"))
|
|
||||||
|
|
||||||
parser.add_option("-v", "--verbose", action="store_true", default=False,
|
|
||||||
help=_("Show what is currently done"))
|
|
||||||
|
|
||||||
group_saving = OptionGroup(parser, "sources saving")
|
|
||||||
|
|
||||||
group_saving.add_option("--sources-save", action="store",
|
|
||||||
help=_("Save source arguments"))
|
|
||||||
|
|
||||||
group_saving.add_option("--sources-replace", action="store",
|
|
||||||
help=_("Save source arguments and replace memory if it already exists"))
|
|
||||||
|
|
||||||
group_saving.add_option("--sources-load", action="store",
|
|
||||||
help=_("Load source arguments"))
|
|
||||||
|
|
||||||
group_saving.add_option("--sources-del", action="store",
|
|
||||||
help=_("delete saved sources"))
|
|
||||||
|
|
||||||
group_saving.add_option("--sources-list", action="store_true", default=False,
|
|
||||||
help=_("List names of saved sources"))
|
|
||||||
|
|
||||||
group_saving.add_option("--sources-full-list", action="store_true", default=False,
|
|
||||||
help=_("List names of saved sources and files in it"))
|
|
||||||
|
|
||||||
parser.add_option_group(group_saving)
|
|
||||||
|
|
||||||
|
|
||||||
(options, args) = parser.parse_args(full_args)
|
|
||||||
options.directdir = False #True only in the special case: we are copying a dir and it doesn't exists
|
|
||||||
#options check
|
|
||||||
if options.progress and not pbar_available:
|
if options.progress and not pbar_available:
|
||||||
warning (_("Progress bar is not available, deactivating"))
|
warning (_("Progress bar is not available, deactivating"))
|
||||||
options.progress = self.progress = False
|
options.progress = self.progress = False
|
||||||
|
@ -622,22 +663,28 @@ class GCP():
|
||||||
logging.getLogger().setLevel(logging.DEBUG)
|
logging.getLogger().setLevel(logging.DEBUG)
|
||||||
|
|
||||||
if options.no_fs_fix:
|
if options.no_fs_fix:
|
||||||
options.fs_fix = 'no'
|
options.fix_filenames = 'no'
|
||||||
else:
|
|
||||||
if not options.fs_fix in const_FS_FIX:
|
|
||||||
error (_("Invalid --fs-fix value\nvalid values are:"))
|
|
||||||
for value in const_FS_FIX:
|
|
||||||
error('- %s' % value)
|
|
||||||
exit(1)
|
|
||||||
|
|
||||||
preserve = set(options.preserve.split(','))
|
preserve = set()
|
||||||
if not preserve.issubset(const_PRESERVE):
|
|
||||||
error (_("Invalid --preserve value\nvalid values are:"))
|
if options.p:
|
||||||
for value in const_PRESERVE:
|
preserve.update(const_PRESERVE_p.split(','))
|
||||||
error('- %s' % value)
|
|
||||||
exit(1)
|
if options.preserve:
|
||||||
else:
|
preserve.update(options.preserve.split(','))
|
||||||
options.preserve = preserve
|
preserve_all = False
|
||||||
|
for value in preserve:
|
||||||
|
if value == 'all':
|
||||||
|
preserve_all = True
|
||||||
|
continue
|
||||||
|
if value not in const_PRESERVE:
|
||||||
|
error (_("Invalid --preserve value '%s'") % value)
|
||||||
|
exit(1)
|
||||||
|
if preserve_all:
|
||||||
|
preserve.remove('all')
|
||||||
|
preserve.update(const_PRESERVE)
|
||||||
|
|
||||||
|
options.preserve = preserve
|
||||||
|
|
||||||
self.__sourcesSaving(options, args)
|
self.__sourcesSaving(options, args)
|
||||||
|
|
||||||
|
@ -666,11 +713,11 @@ class GCP():
|
||||||
if len(args) < 2:
|
if len(args) < 2:
|
||||||
_error_msg = _("Wrong number of arguments")
|
_error_msg = _("Wrong number of arguments")
|
||||||
return (False, _error_msg)
|
return (False, _error_msg)
|
||||||
debug(_("adding args to gcp: %s") % str(args).decode('utf-8','replace'))
|
debug(_("adding args to gcp: %s") % args)
|
||||||
self.__checkArgs(options, source_dir, args)
|
self.__checkArgs(options, source_dir, args)
|
||||||
if not self.__launched:
|
if not self.__launched:
|
||||||
self.journal = Journal()
|
self.journal = Journal()
|
||||||
gobject.idle_add(self.__copyNextFile)
|
GLib.idle_add(self.__copyNextFile)
|
||||||
self.__launched = True
|
self.__launched = True
|
||||||
return (True,'')
|
return (True,'')
|
||||||
|
|
||||||
|
@ -680,7 +727,7 @@ class GCP():
|
||||||
|
|
||||||
def go(self):
|
def go(self):
|
||||||
"""Launch main loop"""
|
"""Launch main loop"""
|
||||||
self.loop = gobject.MainLoop()
|
self.loop = GLib.MainLoop()
|
||||||
try:
|
try:
|
||||||
self.loop.run()
|
self.loop.run()
|
||||||
except KeyboardInterrupt:
|
except KeyboardInterrupt:
|
||||||
|
@ -699,4 +746,3 @@ if __name__ == "__main__":
|
||||||
exit(1)
|
exit(1)
|
||||||
if gcp.journal.partial:
|
if gcp.journal.partial:
|
||||||
exit(2)
|
exit(2)
|
||||||
|
|
||||||
|
|
160
gcp.1
160
gcp.1
|
@ -2,7 +2,7 @@
|
||||||
.\" First parameter, NAME, should be all caps
|
.\" First parameter, NAME, should be all caps
|
||||||
.\" Second parameter, SECTION, should be 1-8, maybe w/ subsection
|
.\" Second parameter, SECTION, should be 1-8, maybe w/ subsection
|
||||||
.\" other parameters are allowed: see man(7), man(1)
|
.\" other parameters are allowed: see man(7), man(1)
|
||||||
.TH GCP 1 "June 04, 2011"
|
.TH GCP 1 "October 14, 2018"
|
||||||
.\" Please adjust this date whenever revising the manpage.
|
.\" Please adjust this date whenever revising the manpage.
|
||||||
.\"
|
.\"
|
||||||
.\" Some roff macros, for reference:
|
.\" Some roff macros, for reference:
|
||||||
|
@ -15,6 +15,9 @@
|
||||||
.\" .br insert line break
|
.\" .br insert line break
|
||||||
.\" .sp <n> insert n+1 empty lines
|
.\" .sp <n> insert n+1 empty lines
|
||||||
.\" for manpage-specific macros, see man(7)
|
.\" for manpage-specific macros, see man(7)
|
||||||
|
.\" TeX users may be more comfortable with the \fB<whatever>\fP and
|
||||||
|
.\" \fI<whatever>\fP escape sequences to invode bold face and italics,
|
||||||
|
.\" respectively.
|
||||||
.SH NAME
|
.SH NAME
|
||||||
gcp \- Advanced command-line file copier
|
gcp \- Advanced command-line file copier
|
||||||
.SH SYNOPSIS
|
.SH SYNOPSIS
|
||||||
|
@ -28,74 +31,109 @@ gcp \- Advanced command-line file copier
|
||||||
.RI [ FILE2 ... ]
|
.RI [ FILE2 ... ]
|
||||||
.I DEST-DIR
|
.I DEST-DIR
|
||||||
.SH DESCRIPTION
|
.SH DESCRIPTION
|
||||||
This manual page documents briefly the
|
\fBgcp\fP is a file copier, loosely inspired by cp, but with high level
|
||||||
.B gcp
|
functionalities like:
|
||||||
command.
|
.IP \(bu 2
|
||||||
.PP
|
transfer progression indication
|
||||||
.\" TeX users may be more comfortable with the \fB<whatever>\fP and
|
.IP \(bu
|
||||||
.\" \fI<whatever>\fP escape sequences to invode bold face and italics,
|
continuous copying when there is an issue: it skips the problematic file and
|
||||||
.\" respectively.
|
goes on
|
||||||
\fBgcp\fP is a file copier, loosely inspired by cp, but with high level functionalities like:
|
.IP \(bu
|
||||||
\- transfer progression indication
|
copy status logging: which files were effectively copied
|
||||||
\- continuous copying when there is an issue: it skips the problematic file and goes on
|
.IP \(bu
|
||||||
\- copy status logging: which files were effectively copied
|
name mangling to handle target filesystem limitations (e.g. removing
|
||||||
\- name mangling to handle target filesystem limitations (e.g. removing incompatible chars like "?" or "*" on vfat)
|
incompatible chars like "?" or "*" on FAT filesystems)
|
||||||
\- forced copy serialization: new files to copy are added to a global queue to avoid hard drive head seeks
|
.IP \(bu
|
||||||
\- transfer list management: gcp can save a list of files to copy and reuse it later
|
forced copy serialization: new files to copy are added to a global queue to
|
||||||
\- approximate option compatibility with cp (approximate because the behaviour is not exactly the same, see below)
|
avoid hard drive head seeks
|
||||||
|
.IP \(bu
|
||||||
|
transfer list management: gcp can save a list of files to copy and reuse it
|
||||||
|
later
|
||||||
|
.IP \(bu
|
||||||
|
approximate option compatibility with cp (approximate because the behaviour is
|
||||||
|
not exactly the same, see below)
|
||||||
.SH OPTIONS
|
.SH OPTIONS
|
||||||
These programs follow the usual GNU command line syntax, with long
|
These programs follow the usual GNU command line syntax, with long
|
||||||
options starting with two dashes (`-').
|
options starting with two dashes (`-').
|
||||||
By default, calling gcp is equivalent to calling gcp \-\-preserve=mode,ownership,timestamps.
|
|
||||||
.PP
|
.PP
|
||||||
A summary of options is included below.
|
A summary of options is included below.
|
||||||
.SS "General options"
|
.SS General options
|
||||||
.TP
|
|
||||||
.B \-\-version
|
|
||||||
Show version of program and exit.
|
|
||||||
.TP
|
.TP
|
||||||
.B \-h, \-\-help
|
.B \-h, \-\-help
|
||||||
Show summary of options.
|
Show summary of options.
|
||||||
.TP
|
.TP
|
||||||
.B \-r, \-\-recursive
|
.B \-V, \-\-version
|
||||||
Copy directories recursively.
|
Show program version and copyright information and exit.
|
||||||
.TP
|
.SS cp-like options
|
||||||
.B \-L, \-\-dereference
|
|
||||||
always follow symbolic links in sources
|
|
||||||
.TP
|
|
||||||
.B \-P, \-\-no\-dereference
|
|
||||||
never follow symbolic links in sources
|
|
||||||
.TP
|
.TP
|
||||||
.B \-f, \-\-force
|
.B \-f, \-\-force
|
||||||
Overwrite existing files.
|
Overwrite existing files.
|
||||||
.TP
|
.TP
|
||||||
.B \-\-preserve=PRESERVE
|
.B \-L, \-\-dereference
|
||||||
Keep specified attributes. Attributes can be mode, ownership and timestamps.
|
Always follow symbolic links in sources.
|
||||||
When several attributes are passed, they need to be separated by commas. Note
|
|
||||||
that timestamps preservation has some limits, see section LIMITS.
|
|
||||||
.TP
|
.TP
|
||||||
.B \-\-no\-fs\-fix
|
.B \-P, \-\-no\-dereference
|
||||||
Don't fix file system naming incompatibilities.
|
Never follow symbolic links in sources.
|
||||||
.TP
|
.TP
|
||||||
.B \-\-no\-progress
|
.B \-p
|
||||||
Disable progress bar.
|
Same as \fB\-\-preserve=mode,ownership,timestamps\fP
|
||||||
|
.TP
|
||||||
|
.B \-\-preserve=<\fIattributes\fP>
|
||||||
|
Preserve specified attributes. Attributes can be \fImode\fP, \fIownership\fP
|
||||||
|
and \fItimestamps\fP.
|
||||||
|
When several attributes are passed, they need to be separated by commas.
|
||||||
|
Please note that timestamps preservation has some limits, see section
|
||||||
|
\fILIMITATIONS\fP.
|
||||||
|
.TP
|
||||||
|
.B \-r, \-R, \-\-recursive
|
||||||
|
Copy directories recursively.
|
||||||
.TP
|
.TP
|
||||||
.B \-v, \-\-verbose
|
.B \-v, \-\-verbose
|
||||||
Display what is being done.
|
Display what is being done.
|
||||||
.SS "Sources saving"
|
.SS gcp-specific options
|
||||||
.TP
|
.TP
|
||||||
.B \-\-sources\-save=SOURCES_SAVE
|
.B \-\-fix\-filenames=<\fIforce\fP|\fIauto\fP|\fIno\fP>
|
||||||
Save the list of source files in a list named SOURCES_SAVE.
|
gcp has the ability to modify the destination file name if the target file
|
||||||
|
system would not accept the original file name.
|
||||||
|
Offending characters will be replaced with similar-looking ones.
|
||||||
|
.IP
|
||||||
|
This option accept the following values:
|
||||||
|
.RS
|
||||||
.TP
|
.TP
|
||||||
.B \-\-sources\-replace=SOURCES_REPLACE
|
\fIauto\fP (default)
|
||||||
Save the list of source files in a list named SOURCES_REPLACE and
|
gcp will attempt to be smart, i.e. detect incompatibilities and fix them as-needed.
|
||||||
replace it if it already exists.
|
|
||||||
.TP
|
.TP
|
||||||
.B \-\-sources\-load=SOURCES_LOAD
|
\fIforce\fP
|
||||||
Reuse the list of source file named SOURCES_LOAD.
|
Always fix file names that could cause problems on any known filesystem or OS.
|
||||||
|
This is useful e.g. with NTFS, see \fINOTE ON NTFS\fP below.
|
||||||
.TP
|
.TP
|
||||||
.B \-\-sources\-del=SOURCES_DEL
|
\fIno\fP
|
||||||
Delete the list of source files named SOURCES_DEL.
|
Renaming is disabled entirely.
|
||||||
|
.RE
|
||||||
|
.IP
|
||||||
|
Currently, gcp is only aware of FAT incompatibilities:
|
||||||
|
\'\\\', \':\', \'*\', \'?\', \'"\', \'<\', \'>\' and \'|\'.
|
||||||
|
.TP
|
||||||
|
.B \-\-no\-fs\-fix (DEPRECATED)
|
||||||
|
Same as \fB\-\-fix\-filenames=no\fP.
|
||||||
|
This option will be removed in a future release.
|
||||||
|
.TP
|
||||||
|
.B \-\-no\-progress
|
||||||
|
Disable progress bar.
|
||||||
|
.SS Sources saving
|
||||||
|
.TP
|
||||||
|
.B \-\-sources\-save=\fISOURCES\fP
|
||||||
|
Save the list of source files in a list named \fISOURCES\fP.
|
||||||
|
.TP
|
||||||
|
.B \-\-sources\-replace=\fISOURCES\fP
|
||||||
|
Save the list of source files in a list named \fISOURCES\fP;
|
||||||
|
the file is overwritten it already exists.
|
||||||
|
.TP
|
||||||
|
.B \-\-sources\-load=\fISOURCES\fP
|
||||||
|
Use the list of source files named \fISOURCES\fP.
|
||||||
|
.TP
|
||||||
|
.B \-\-sources\-del=\fISOURCES\fP
|
||||||
|
Delete the list of source files named \fISOURCES\fP.
|
||||||
.TP
|
.TP
|
||||||
.B \-\-sources\-list
|
.B \-\-sources\-list
|
||||||
List the names of source file lists.
|
List the names of source file lists.
|
||||||
|
@ -105,21 +143,33 @@ List the names of source file lists, including their content.
|
||||||
.SH EXIT STATUS
|
.SH EXIT STATUS
|
||||||
The exit status can be:
|
The exit status can be:
|
||||||
.IP \[bu] 2
|
.IP \[bu] 2
|
||||||
\fB0\fP if files have been copied correctly or if another instance of gcp is already running and will do the copy.
|
\fB0\fP if files have been copied correctly or if another instance of gcp is
|
||||||
|
already running and will do the copy.
|
||||||
.IP \[bu]
|
.IP \[bu]
|
||||||
\fB1\fP if at least one file has not been copied, or if something went wrong.
|
\fB1\fP if at least one file has not been copied, or if something went wrong.
|
||||||
.IP \[bu]
|
.IP \[bu]
|
||||||
\fB2\fP if all files have been copied but with some issues
|
\fB2\fP if all files have been copied but with some issues
|
||||||
.SH LIMITS
|
.SH LIMITATIONS
|
||||||
Timestamps preservation with \-\-preserve option is limited by the os python
|
Timestamps preservation with \fB\-\-preserve\fP option is limited by the
|
||||||
module on POSIX systems. Currently, python only returns timestamps in float
|
\fIos\fP Python module on POSIX systems. Currently, Python only returns
|
||||||
format, which is a smaller precision than what POSIX provides. Progress on this
|
timestamps in float format, which is a smaller precision than what POSIX
|
||||||
issue can be seen at http://bugs.python.org/issue11457.
|
provides. Progress on this issue can be seen at
|
||||||
|
http://bugs.python.org/issue11457.
|
||||||
|
.PP
|
||||||
|
The \fB\-\-preserve\fP option cannot currently be used without an attribute
|
||||||
|
list (\fBgcp \-\-preserve foo bar\fP will behave as \fBgcp \-\-preserve=foo
|
||||||
|
bar\fP). Use the \fB\-p\fP switch instead.
|
||||||
|
.SH NOTE ON NTFS
|
||||||
|
NTFS will not enforce the same file name limitations than FAT, but files that
|
||||||
|
would not be accepted on a FAT filesystem will still cause problems on Windows.
|
||||||
|
Hence, it is recommended to use \-\-fix-filenames=force when copying to NTFS
|
||||||
|
(when Windows compatibility is desired, anyway).
|
||||||
.SH SEE ALSO
|
.SH SEE ALSO
|
||||||
.BR cp (1).
|
.BR cp (1).
|
||||||
.br
|
.br
|
||||||
.SH AUTHOR
|
.SH AUTHOR
|
||||||
gcp was written by Jérôme Poisson <goffi@goffi.org>.
|
gcp was written by Jérôme Poisson <goffi@goffi.org>.
|
||||||
|
It is currently maintained by Matteo Cypriani <mcy@lm7.fr>.
|
||||||
.PP
|
.PP
|
||||||
This manual page was written by Thomas Preud'homme <robotux@celest.fr>,
|
This manual page was initially written by Thomas Preud'homme
|
||||||
for the Debian project (and may be used by others).
|
<robotux@celest.fr> for the Debian project (and may be used by others).
|
||||||
|
|
164
gcp.po
164
gcp.po
|
@ -1,164 +0,0 @@
|
||||||
# SOME DESCRIPTIVE TITLE.
|
|
||||||
# Copyright (C) YEAR THE PACKAGE'S COPYRIGHT HOLDER
|
|
||||||
# This file is distributed under the same license as the PACKAGE package.
|
|
||||||
# FIRST AUTHOR <EMAIL@ADDRESS>, YEAR.
|
|
||||||
#
|
|
||||||
#, fuzzy
|
|
||||||
msgid ""
|
|
||||||
msgstr ""
|
|
||||||
"Project-Id-Version: PACKAGE VERSION\n"
|
|
||||||
"Report-Msgid-Bugs-To: \n"
|
|
||||||
"POT-Creation-Date: 2010-09-27 13:03+0800\n"
|
|
||||||
"PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
|
|
||||||
"Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
|
|
||||||
"Language-Team: LANGUAGE <LL@li.org>\n"
|
|
||||||
"MIME-Version: 1.0\n"
|
|
||||||
"Content-Type: text/plain; charset=CHARSET\n"
|
|
||||||
"Content-Transfer-Encoding: 8bit\n"
|
|
||||||
|
|
||||||
#: gcp:43
|
|
||||||
msgid "Error during import"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
#: gcp:44
|
|
||||||
msgid "Please check dependecies:"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
#: gcp:50
|
|
||||||
msgid ""
|
|
||||||
"ProgressBar not available, please download it at http://pypi.python.org/pypi/"
|
|
||||||
"progressbar"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
#: gcp:51
|
|
||||||
msgid ""
|
|
||||||
"Progress bar deactivated\n"
|
|
||||||
"--\n"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
#: gcp:62
|
|
||||||
msgid ""
|
|
||||||
"This program comes with ABSOLUTELY NO WARRANTY;\n"
|
|
||||||
"This is free software, and you are welcome to redistribute it\n"
|
|
||||||
"under certain conditions.\n"
|
|
||||||
"---\n"
|
|
||||||
"\n"
|
|
||||||
"This software is an advanced file copier\n"
|
|
||||||
"Get the latest version at http://www.goffi.org\n"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
#: gcp:82
|
|
||||||
msgid "Init DbusObject..."
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
#: gcp:102
|
|
||||||
msgid "INTERNAL ERROR: invalid arguments"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
#: gcp:119
|
|
||||||
msgid "gcp launched"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
#: gcp:126
|
|
||||||
msgid "Init DBus..."
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
#: gcp:155
|
|
||||||
msgid "Can't read mounts table"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
#: gcp:162
|
|
||||||
#, python-format
|
|
||||||
msgid "Adding to copy list: %(path)s ==> %(dest_path)s (%(fs_type)s)"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
#: gcp:170 gcp:192
|
|
||||||
#, python-format
|
|
||||||
msgid "Can't copy %(path)s: %(exception)s"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
#: gcp:179
|
|
||||||
#, python-format
|
|
||||||
msgid "Creating directory %s"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
#: gcp:200
|
|
||||||
#, python-format
|
|
||||||
msgid "Invalid dest_path: %s"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
#: gcp:205
|
|
||||||
#, python-format
|
|
||||||
msgid "The path given in arg doesn't exist or is not accessible: %s"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
#: gcp:210
|
|
||||||
#, python-format
|
|
||||||
msgid "omitting directory \"%s\""
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
#: gcp:226
|
|
||||||
#, python-format
|
|
||||||
msgid "File [%s] already exists, skipping it !"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
#: gcp:233
|
|
||||||
#, python-format
|
|
||||||
msgid "COPYING %(source)s ==> %(dest)s"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
#: gcp:284
|
|
||||||
msgid "Progress: "
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
#: gcp:315
|
|
||||||
msgid "copy directories recursively"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
#: gcp:318
|
|
||||||
msgid "force overwriting of existing files"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
#: gcp:321
|
|
||||||
msgid "preserve the specified attributes"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
#: gcp:324
|
|
||||||
msgid "don't fixe name encoding errors"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
#: gcp:327
|
|
||||||
msgid "deactivate progress bar"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
#: gcp:330
|
|
||||||
msgid "Show what is currently done"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
#: gcp:335
|
|
||||||
msgid "Progress bar is not available, deactivating"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
#: gcp:345
|
|
||||||
msgid ""
|
|
||||||
"Invalide --preserve value\n"
|
|
||||||
"valid values are:"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
#: gcp:354
|
|
||||||
#, python-format
|
|
||||||
msgid "There is already one instance of %s running, pluging to it"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
#: gcp:360
|
|
||||||
msgid "Wrong number of arguments"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
#: gcp:362
|
|
||||||
#, python-format
|
|
||||||
msgid "adding args to gcp: %s"
|
|
||||||
msgstr ""
|
|
||||||
|
|
||||||
#: gcp:373
|
|
||||||
msgid "User interruption: good bye"
|
|
||||||
msgstr ""
|
|
|
@ -0,0 +1,265 @@
|
||||||
|
# gcp -- translation template
|
||||||
|
#
|
||||||
|
# This file is distributed under the same license as the gcp package.
|
||||||
|
#
|
||||||
|
#, fuzzy
|
||||||
|
msgid ""
|
||||||
|
msgstr ""
|
||||||
|
"Project-Id-Version: 0.2.1\n"
|
||||||
|
"Report-Msgid-Bugs-To: \n"
|
||||||
|
"POT-Creation-Date: 2018-10-14 20:51+0200\n"
|
||||||
|
"PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
|
||||||
|
"Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
|
||||||
|
"Language-Team: LANGUAGE <LL@li.org>\n"
|
||||||
|
"Language: en\n"
|
||||||
|
"MIME-Version: 1.0\n"
|
||||||
|
"Content-Type: text/plain; charset=UTF-8\n"
|
||||||
|
"Content-Transfer-Encoding: 8bit\n"
|
||||||
|
|
||||||
|
#: gcp:46
|
||||||
|
msgid "Error during import"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:47
|
||||||
|
msgid "Please check dependecies:"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:54
|
||||||
|
msgid "ProgressBar not available, please download it at https://pypi.org/"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:55
|
||||||
|
msgid ""
|
||||||
|
"Progress bar deactivated\n"
|
||||||
|
"--\n"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:90
|
||||||
|
msgid "Init DbusObject..."
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:111
|
||||||
|
msgid "INTERNAL ERROR: invalid arguments"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:116
|
||||||
|
msgid "INTERNAL ERROR: invalid source_dir"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:171
|
||||||
|
msgid "/!\\ THE FOLLOWING FILES WERE *NOT* SUCCESSFULY COPIED:"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:177
|
||||||
|
msgid "The following files were copied, but some errors happened:"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:183
|
||||||
|
#, python-format
|
||||||
|
msgid "Please check journal: %s"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:205
|
||||||
|
msgid "gcp launched"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:213
|
||||||
|
msgid "Init DBus..."
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:244
|
||||||
|
msgid "Can't read mounts table"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:251
|
||||||
|
#, python-format
|
||||||
|
msgid "Adding to copy list: %(path)s ==> %(dest_path)s (%(fs_type)s)"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:258
|
||||||
|
#, python-format
|
||||||
|
msgid "Can't copy %(path)s: %(exception)s"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:286
|
||||||
|
#, python-format
|
||||||
|
msgid "Can't append %(path)s to copy list: %(exception)s"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:289
|
||||||
|
#, python-format
|
||||||
|
msgid "Can't access %(dirpath)s: %(exception)s"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:301
|
||||||
|
#, python-format
|
||||||
|
msgid "Invalid dest_path: %s"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:306
|
||||||
|
#, python-format
|
||||||
|
msgid "The path given in arg doesn't exist or is not accessible: %s"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:310
|
||||||
|
#, python-format
|
||||||
|
msgid "omitting directory \"%s\""
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:348
|
||||||
|
#, python-format
|
||||||
|
msgid "File [%s] already exists, skipping it!"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:368
|
||||||
|
#, python-format
|
||||||
|
msgid "COPYING %(source)s ==> %(dest)s"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:462
|
||||||
|
#, python-format
|
||||||
|
msgid "%.2f PiB"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:464
|
||||||
|
#, python-format
|
||||||
|
msgid "%.2f TiB"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:466
|
||||||
|
#, python-format
|
||||||
|
msgid "%.2f GiB"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:468
|
||||||
|
#, python-format
|
||||||
|
msgid "%.2f MiB"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:470
|
||||||
|
#, python-format
|
||||||
|
msgid "%.2f KiB"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:471
|
||||||
|
#, python-format
|
||||||
|
msgid "%i B"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:479 gcp:485
|
||||||
|
#, python-format
|
||||||
|
msgid "Copying %s"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:515 gcp:548
|
||||||
|
msgid ""
|
||||||
|
"No saved sources with this name, check existing names with --sources-list"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:525
|
||||||
|
msgid "Saved sources:"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:539
|
||||||
|
msgid ""
|
||||||
|
"There is already a saved sources with this name, skipping --sources-save"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:578
|
||||||
|
msgid "force overwriting of existing files"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:582
|
||||||
|
msgid "always follow symbolic links in sources"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:586
|
||||||
|
msgid "never follow symbolic links in sources"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:590
|
||||||
|
#, python-format
|
||||||
|
msgid "same as --preserve=%s"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:594
|
||||||
|
#, python-format
|
||||||
|
msgid ""
|
||||||
|
"preserve specified attributes; accepted values: 'all', or "
|
||||||
|
"one or more amongst %s"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:599
|
||||||
|
msgid "copy directories recursively"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:603
|
||||||
|
msgid "display what is being done"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:614
|
||||||
|
msgid ""
|
||||||
|
"fix file names incompatible with the destination file "
|
||||||
|
"system (default: auto)"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:619
|
||||||
|
msgid ""
|
||||||
|
"[DEPRECATED] same as --fix-filename=no (overrides --fix-"
|
||||||
|
"filenames)"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:624
|
||||||
|
msgid "disable progress bar"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:631
|
||||||
|
msgid "save sources arguments"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:635
|
||||||
|
msgid "save sources arguments and replace memory if it already exists"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:639
|
||||||
|
msgid "load sources arguments"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:643
|
||||||
|
msgid "delete saved sources list"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:647
|
||||||
|
msgid "list names of saved sources"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:651
|
||||||
|
msgid "list names of saved sources and files in it"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:663
|
||||||
|
msgid "Progress bar is not available, deactivating"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:687
|
||||||
|
#, python-format
|
||||||
|
msgid "Invalid --preserve value '%s'"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:714
|
||||||
|
#, python-format
|
||||||
|
msgid "There is already one instance of %s running, pluging to it"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:720
|
||||||
|
msgid "Wrong number of arguments"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:722
|
||||||
|
#, python-format
|
||||||
|
msgid "adding args to gcp: %s"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
#: gcp:731
|
||||||
|
msgid "User interruption: good bye"
|
||||||
|
msgstr ""
|
Binary file not shown.
73
setup.py
73
setup.py
|
@ -1,31 +1,50 @@
|
||||||
#!/usr/bin/env python
|
#!/usr/bin/env python3
|
||||||
# -*- coding: utf-8 -*-
|
|
||||||
|
|
||||||
from distribute_setup import use_setuptools
|
import setuptools
|
||||||
use_setuptools()
|
|
||||||
|
|
||||||
from setuptools import setup
|
|
||||||
import sys
|
|
||||||
from os import path
|
|
||||||
|
|
||||||
name = 'gcp'
|
name = 'gcp'
|
||||||
|
|
||||||
setup(name=name,
|
setuptools.setup(
|
||||||
version='0.1.3',
|
name=name,
|
||||||
description=u"gcp is an advanced copy tool loosely inspired from cp",
|
version='0.2.1',
|
||||||
long_description=u'gcp is a command-line tool to copy files, loosely inspired from cp, but with high level functionalities such as progress bar, copy continuation on error, journaling to know which files were successfuly copied, name mangling to workaround filesystem limitations (FAT), unique copy queue, copy list managemet, command arguments close to cp',
|
url='https://code.lm7.fr/mcy/gcp',
|
||||||
author='Goffi (Jérôme Poisson)',
|
license='GPL-3+',
|
||||||
author_email='goffi@goffi.org',
|
|
||||||
url='http://wiki.goffi.org/wiki/Gcp',
|
description="An advanced file copy tool loosely inspired from cp",
|
||||||
classifiers=['Environment :: Console',
|
long_description_content_type='text/markdown',
|
||||||
'Intended Audience :: End Users/Desktop',
|
long_description="""
|
||||||
'License :: OSI Approved :: GNU General Public License (GPL)',
|
**%s** is a command-line tool to copy files, loosely inspired from the `cp`
|
||||||
'Operating System :: POSIX :: Linux',
|
command, but with higher-level functionalities such as progress bar, copy
|
||||||
'Programming Language :: Python',
|
continuation on error, logging to know which files were successfully
|
||||||
'Topic :: Utilities'
|
copied, name mangling to workaround filesystem limitations (FAT), unique
|
||||||
],
|
copy queue, copy list management, etc.""" % name,
|
||||||
data_files=[(path.join(sys.prefix,'share/locale/fr/LC_MESSAGES'), ['i18n/fr/LC_MESSAGES/gcp.mo']),
|
keywords='file copy',
|
||||||
('share/man/man1', ["gcp.1"]),
|
|
||||||
('share/doc/%s' % name, ['COPYING','README'])],
|
author='Goffi (Jérôme Poisson)',
|
||||||
scripts=['gcp'],
|
author_email='goffi@goffi.org',
|
||||||
)
|
maintainer='Matteo Cypriani',
|
||||||
|
maintainer_email='mcy@lm7.fr',
|
||||||
|
|
||||||
|
# Cf. https://pypi.org/pypi?%3Aaction=list_classifiers
|
||||||
|
classifiers=[
|
||||||
|
'Development Status :: 4 - Beta',
|
||||||
|
'Environment :: Console',
|
||||||
|
'Intended Audience :: End Users/Desktop',
|
||||||
|
'License :: OSI Approved :: GNU General Public License v3 or later (GPLv3+)',
|
||||||
|
'Operating System :: POSIX :: Linux',
|
||||||
|
'Programming Language :: Python :: 3',
|
||||||
|
'Topic :: Utilities',
|
||||||
|
],
|
||||||
|
|
||||||
|
scripts=['gcp'],
|
||||||
|
# entry_points={
|
||||||
|
# 'console_scripts': ['gcp=gcp:main'],
|
||||||
|
# },
|
||||||
|
data_files=[
|
||||||
|
('man/man1', ["gcp.1"]),
|
||||||
|
('share/locale/fr/LC_MESSAGES', ['i18n/fr/LC_MESSAGES/gcp.mo']),
|
||||||
|
('share/doc/%s' % name, ['CHANGELOG', 'LICENSE', 'README.md']),
|
||||||
|
],
|
||||||
|
install_requires=['PyGObject', 'dbus-python'],
|
||||||
|
python_requires='>=3',
|
||||||
|
)
|
||||||
|
|
|
@ -1,9 +1,9 @@
|
||||||
#!/usr/bin/python
|
#!/usr/bin/env python3
|
||||||
# -*- coding: utf-8 -*-
|
|
||||||
|
|
||||||
"""
|
"""
|
||||||
gcp: Goffi's CoPier
|
gcp: Gcp CoPier -- unit tests
|
||||||
Copyright (C) 2010, 2011 Jérôme Poisson <goffi@goffi.org>
|
Copyright (c) 2010, 2011 Jérôme Poisson <goffi@goffi.org>
|
||||||
|
Copyright (c) 2018 Matteo Cypriani <mcy@lm7.fr>
|
||||||
|
|
||||||
This program is free software: you can redistribute it and/or modify
|
This program is free software: you can redistribute it and/or modify
|
||||||
it under the terms of the GNU General Public License as published by
|
it under the terms of the GNU General Public License as published by
|
||||||
|
@ -19,29 +19,32 @@ You should have received a copy of the GNU General Public License
|
||||||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
from __future__ import with_statement
|
|
||||||
import tempfile
|
import tempfile
|
||||||
import unittest
|
import unittest
|
||||||
from os import getcwd, chdir, system, mkdir, makedirs, listdir
|
from os import getcwd, chdir, system, mkdir, makedirs, listdir
|
||||||
from os.path import join, isdir
|
from os.path import join, isdir
|
||||||
from shutil import rmtree
|
from shutil import rmtree
|
||||||
from random import randrange
|
|
||||||
from hashlib import sha1
|
from hashlib import sha1
|
||||||
|
|
||||||
#size shorcuts
|
# gcp command. This assumes gcp is in the PATH. Alternatively, we could use
|
||||||
|
# something like this to use gcp from the current directory:
|
||||||
|
#GCP = join(getcwd(), "gcp")
|
||||||
|
GCP = 'gcp'
|
||||||
|
|
||||||
|
# Size shorcuts
|
||||||
S10K = 1024 * 10
|
S10K = 1024 * 10
|
||||||
S100K = 1024 * 100
|
S100K = 1024 * 100
|
||||||
S1M = 1024 * 1024
|
S1M = 1024 * 1024
|
||||||
S10M = 1024 * 1024 * 10
|
S10M = 1024 * 1024 * 10
|
||||||
S100M = 1024 * 1024 * 100
|
S100M = 1024 * 1024 * 100
|
||||||
|
|
||||||
|
|
||||||
def sha1sum(filename, buf_size=4096):
|
def sha1sum(filename, buf_size=4096):
|
||||||
"""Return the SHA1 hash of a file
|
"""Returns the SHA1 hash of a file.
|
||||||
@param filename: path to the file
|
@param filename: path to the file
|
||||||
@param buf_size: size of the buffer to use for calculation"""
|
@param buf_size: size of the buffer to use for calculation
|
||||||
|
"""
|
||||||
csum = sha1()
|
csum = sha1()
|
||||||
with open(filename) as fd:
|
with open(filename, 'rb') as fd:
|
||||||
data = fd.read(buf_size)
|
data = fd.read(buf_size)
|
||||||
while data:
|
while data:
|
||||||
csum.update(data)
|
csum.update(data)
|
||||||
|
@ -49,7 +52,7 @@ def sha1sum(filename, buf_size=4096):
|
||||||
return csum.digest()
|
return csum.digest()
|
||||||
|
|
||||||
def dirCheck(dir_path):
|
def dirCheck(dir_path):
|
||||||
"""Recursively calculate SHA1 sum of a dir
|
"""Recursively calculates SHA1 sums in a directory.
|
||||||
@param path: path of the dir to check
|
@param path: path of the dir to check
|
||||||
@return: a dict in the form [{filepath: sum,...}]
|
@return: a dict in the form [{filepath: sum,...}]
|
||||||
"""
|
"""
|
||||||
|
@ -69,11 +72,12 @@ def dirCheck(dir_path):
|
||||||
return result
|
return result
|
||||||
|
|
||||||
#def makeRandomFile(path, size, buf_size=4096):
|
#def makeRandomFile(path, size, buf_size=4096):
|
||||||
# """Create a fake file
|
# """Creates a fake file.
|
||||||
# @param path: where the file is created
|
# @param path: where the file is created
|
||||||
# @param size: size of the file to create in bytes"""
|
# @param size: size of the file to create in bytes
|
||||||
|
# """
|
||||||
# def seq(size):
|
# def seq(size):
|
||||||
# return ''.join(chr(randrange(256)) for i in range(size))
|
# return ''.join(chr(random.randrange(256)) for i in range(size))
|
||||||
# fd = open(path, 'w')
|
# fd = open(path, 'w')
|
||||||
# for byte in range(size//buf_size):
|
# for byte in range(size//buf_size):
|
||||||
# fd.write(seq(buf_size))
|
# fd.write(seq(buf_size))
|
||||||
|
@ -81,20 +85,22 @@ def dirCheck(dir_path):
|
||||||
# fd.close()
|
# fd.close()
|
||||||
|
|
||||||
def makeRandomFile(path, size=S10K, buf_size=4096):
|
def makeRandomFile(path, size=S10K, buf_size=4096):
|
||||||
"""Create a fake file using /dev/urandom
|
"""Creates a fake file using /dev/urandom.
|
||||||
@param path: where the file must be created
|
@param path: where the file must be created
|
||||||
@param size: size of the file to create in bytes"""
|
@param size: size of the file to create in bytes
|
||||||
source = open('/dev/urandom','r')
|
"""
|
||||||
dest = open(path, 'w')
|
source = open('/dev/urandom', 'rb')
|
||||||
for byte in range(size//buf_size):
|
dest = open(path, 'wb')
|
||||||
|
for _ in range(size // buf_size):
|
||||||
dest.write(source.read(buf_size))
|
dest.write(source.read(buf_size))
|
||||||
dest.write(source.read(size%buf_size))
|
dest.write(source.read(size % buf_size))
|
||||||
dest.close()
|
dest.close()
|
||||||
|
source.close()
|
||||||
|
|
||||||
def makeTestDir(path):
|
def makeTestDir(path):
|
||||||
"""Helper method to easily create a test dir
|
"""Helper method to easily create a test directory.
|
||||||
@param path: where the dir must be created"""
|
@param path: where the dir must be created
|
||||||
|
"""
|
||||||
for i in range(2):
|
for i in range(2):
|
||||||
subdir = join(path,'subdir_%d' % i)
|
subdir = join(path,'subdir_%d' % i)
|
||||||
makedirs(subdir)
|
makedirs(subdir)
|
||||||
|
@ -104,8 +110,8 @@ def makeTestDir(path):
|
||||||
makeRandomFile(join(path,'file_%d' % i), S10K)
|
makeRandomFile(join(path,'file_%d' % i), S10K)
|
||||||
|
|
||||||
class TestCopyCases(unittest.TestCase):
|
class TestCopyCases(unittest.TestCase):
|
||||||
"""Test basic copy use cases, using gcp externally
|
"""Test basic copy use cases, using gcp externally.
|
||||||
gcp must be available in the PATH"""
|
"""
|
||||||
#TODO: check journal
|
#TODO: check journal
|
||||||
|
|
||||||
def setUp(self):
|
def setUp(self):
|
||||||
|
@ -118,109 +124,134 @@ class TestCopyCases(unittest.TestCase):
|
||||||
rmtree(self.tmp_dir)
|
rmtree(self.tmp_dir)
|
||||||
|
|
||||||
def test_one_file_copy(self):
|
def test_one_file_copy(self):
|
||||||
"""Copy one file and test the result"""
|
"""Copies one file and tests the result.
|
||||||
|
"""
|
||||||
makeRandomFile('file_1', S10K)
|
makeRandomFile('file_1', S10K)
|
||||||
ori_sum = sha1sum('file_1')
|
ori_sum = sha1sum('file_1')
|
||||||
ret = system("gcp file_1 file_2")
|
ret = system(GCP + " file_1 file_2")
|
||||||
self.assertEqual(ret,0)
|
self.assertEqual(ret,0)
|
||||||
dest_sum = sha1sum('file_2')
|
dest_sum = sha1sum('file_2')
|
||||||
self.assertEqual(ori_sum, dest_sum)
|
self.assertEqual(ori_sum, dest_sum)
|
||||||
|
|
||||||
def test_one_file_copy_already_exists(self):
|
def test_one_file_copy_already_exists(self):
|
||||||
"""Check that an existing file is not overwritten"""
|
"""Checks that an existing file is not overwritten.
|
||||||
|
"""
|
||||||
makeRandomFile('file_1', S10K)
|
makeRandomFile('file_1', S10K)
|
||||||
makeRandomFile('file_2', S10K)
|
makeRandomFile('file_2', S10K)
|
||||||
file_2_sum = sha1sum('file_2')
|
file_2_sum = sha1sum('file_2')
|
||||||
ret = system("gcp file_1 file_2")
|
ret = system(GCP + " file_1 file_2")
|
||||||
self.assertNotEqual(ret,0)
|
self.assertNotEqual(ret,0)
|
||||||
file_2_sum_bis = sha1sum('file_2')
|
file_2_sum_bis = sha1sum('file_2')
|
||||||
self.assertEqual(file_2_sum, file_2_sum_bis)
|
self.assertEqual(file_2_sum, file_2_sum_bis)
|
||||||
|
|
||||||
def test_one_file_copy_already_exists_force(self):
|
def test_one_file_copy_already_exists_force(self):
|
||||||
"""Check that an existing file is overwritten with --force"""
|
"""Checks that an existing file is overwritten with --force.
|
||||||
|
"""
|
||||||
makeRandomFile('file_1', S10K)
|
makeRandomFile('file_1', S10K)
|
||||||
makeRandomFile('file_2', S10K)
|
makeRandomFile('file_2', S10K)
|
||||||
file_1_sum = sha1sum('file_1')
|
file_1_sum = sha1sum('file_1')
|
||||||
ret = system("gcp -f file_1 file_2")
|
ret = system(GCP + " -f file_1 file_2")
|
||||||
self.assertEqual(ret,0)
|
self.assertEqual(ret,0)
|
||||||
file_2_sum_bis = sha1sum('file_2')
|
file_2_sum_bis = sha1sum('file_2')
|
||||||
self.assertEqual(file_1_sum, file_2_sum_bis)
|
self.assertEqual(file_1_sum, file_2_sum_bis)
|
||||||
|
|
||||||
def test_one_dir_copy(self):
|
def test_one_dir_copy(self):
|
||||||
"""Check copy of one dir to a non existant path"""
|
"""Checks copy of a directory to a non-existent path.
|
||||||
|
"""
|
||||||
makeTestDir('dir_1')
|
makeTestDir('dir_1')
|
||||||
check_1 = dirCheck('dir_1')
|
check_1 = dirCheck('dir_1')
|
||||||
ret = system("gcp -r dir_1 dir_2")
|
ret = system(GCP + " -r dir_1 dir_2")
|
||||||
self.assertEqual(ret,0)
|
self.assertEqual(ret,0)
|
||||||
check_2 = dirCheck('dir_2')
|
check_2 = dirCheck('dir_2')
|
||||||
self.assertEqual(check_1, check_2)
|
self.assertEqual(check_1, check_2)
|
||||||
|
|
||||||
def test_one_dir_copy_nocopy(self):
|
def test_one_dir_copy_nocopy(self):
|
||||||
"""Check that a dir is not copied without the recursive option"""
|
"""Checks that a directory is not copied without the recursive option.
|
||||||
|
"""
|
||||||
makeTestDir('dir_1')
|
makeTestDir('dir_1')
|
||||||
check_before = dirCheck('.')
|
check_before = dirCheck('.')
|
||||||
ret = system("gcp dir_1 dir_2")
|
ret = system(GCP + " dir_1 dir_2")
|
||||||
self.assertEqual(ret,0)
|
self.assertEqual(ret,0)
|
||||||
check_after = dirCheck('.')
|
check_after = dirCheck('.')
|
||||||
self.assertEqual(check_before, check_after)
|
self.assertEqual(check_before, check_after)
|
||||||
|
|
||||||
def test_one_dir_copy_existing_dest(self):
|
def test_one_dir_copy_existing_dest(self):
|
||||||
"""Check that a dir is copied inside an existing destination"""
|
"""Checks that a directory is copied inside an existing destination.
|
||||||
|
"""
|
||||||
makeTestDir('dir_1')
|
makeTestDir('dir_1')
|
||||||
mkdir('dir_2')
|
mkdir('dir_2')
|
||||||
check_1 = dirCheck('dir_1')
|
check_1 = dirCheck('dir_1')
|
||||||
ret = system("gcp -r dir_1 dir_2")
|
ret = system(GCP + " -r dir_1 dir_2")
|
||||||
self.assertEqual(ret,0)
|
self.assertEqual(ret,0)
|
||||||
self.assertEqual(listdir('dir_2'), ['dir_1'])
|
self.assertEqual(listdir('dir_2'), ['dir_1'])
|
||||||
check_2 = dirCheck('dir_2/dir_1')
|
check_2 = dirCheck('dir_2/dir_1')
|
||||||
self.assertEqual(check_1, check_2)
|
self.assertEqual(check_1, check_2)
|
||||||
|
|
||||||
def test_mixt_copy_existing_dest(self):
|
def test_mixed_copy_existing_dest(self):
|
||||||
"""Check that a mixt copy (files + dir) to an existing dest work as expected"""
|
"""Mixed copy (files + dir) to an existing destination.
|
||||||
|
"""
|
||||||
for i in range(2):
|
for i in range(2):
|
||||||
makeRandomFile('file_%d' % i, S10K)
|
makeRandomFile('file_%d' % i, S10K)
|
||||||
makeTestDir('dir_%d' % i)
|
makeTestDir('dir_%d' % i)
|
||||||
check_1 = dirCheck('.')
|
check_1 = dirCheck('.')
|
||||||
mkdir('dest_dir')
|
mkdir('dest_dir')
|
||||||
ret = system("gcp -r file_0 file_1 dir_0 dir_1 dest_dir")
|
ret = system(GCP + " -r file_0 file_1 dir_0 dir_1 dest_dir")
|
||||||
self.assertEqual(ret,0)
|
self.assertEqual(ret,0)
|
||||||
check_2 = dirCheck('dest_dir')
|
check_2 = dirCheck('dest_dir')
|
||||||
self.assertEqual(check_1, check_2)
|
self.assertEqual(check_1, check_2)
|
||||||
|
|
||||||
def test_mixt_copy_nonexisting_dest(self):
|
def test_mixed_copy_nonexisting_dest(self):
|
||||||
"""Check that a mixt copy (files + dir) to an non existing dest work as expected
|
"""Mixed copy (files + dir) to an inexistant destination.
|
||||||
/!\\ the behavious is different of the one of cp in this case ! (cp doesn't copy at all, while gcp create the dest)"""
|
/!\\ The behaviour is different than cp's! (cp doesn't copy at all,
|
||||||
|
while gcp create the destintation directory.)
|
||||||
|
"""
|
||||||
for i in range(2):
|
for i in range(2):
|
||||||
makeRandomFile('file_%d' % i, S10K)
|
makeRandomFile('file_%d' % i, S10K)
|
||||||
makeTestDir('dir_%d' % i)
|
makeTestDir('dir_%d' % i)
|
||||||
check_1 = dirCheck('.')
|
check_1 = dirCheck('.')
|
||||||
ret = system("gcp -r file_0 file_1 dir_0 dir_1 dest_dir")
|
ret = system(GCP + " -r file_0 file_1 dir_0 dir_1 dest_dir")
|
||||||
self.assertEqual(ret,0)
|
self.assertEqual(ret,0)
|
||||||
check_2 = dirCheck('dest_dir')
|
check_2 = dirCheck('dest_dir')
|
||||||
self.assertEqual(check_1, check_2)
|
self.assertEqual(check_1, check_2)
|
||||||
|
|
||||||
def test_mixt_copy_existing_dest_nonrecursive(self):
|
def test_mixed_copy_existing_dest_nonrecursive(self):
|
||||||
"""Check that a mixt copy (files + dir) to an existing dest without the recursive option work as expected"""
|
"""Non-recursive mixed copy (files + dir) to an existing destination.
|
||||||
|
"""
|
||||||
for i in range(2):
|
for i in range(2):
|
||||||
makeRandomFile('file_%d' % i, S10K)
|
makeRandomFile('file_%d' % i, S10K)
|
||||||
makeTestDir('dir_%d' % i)
|
makeTestDir('dir_%d' % i)
|
||||||
mkdir('dest_dir')
|
mkdir('dest_dir')
|
||||||
ret = system("gcp file_0 file_1 dir_0 dir_1 dest_dir")
|
ret = system(GCP + " file_0 file_1 dir_0 dir_1 dest_dir")
|
||||||
self.assertEqual(ret,0)
|
self.assertEqual(ret,0)
|
||||||
self.assertEqual(set(listdir('dest_dir')), set(['file_0', 'file_1']))
|
self.assertEqual(set(listdir('dest_dir')), set(['file_0', 'file_1']))
|
||||||
self.assertEqual(sha1sum('file_0'), sha1sum('dest_dir/file_0'))
|
self.assertEqual(sha1sum('file_0'), sha1sum('dest_dir/file_0'))
|
||||||
self.assertEqual(sha1sum('file_1'), sha1sum('dest_dir/file_1'))
|
self.assertEqual(sha1sum('file_1'), sha1sum('dest_dir/file_1'))
|
||||||
|
|
||||||
def test_mixt_copy_nonexisting_dest_nonrecursive(self):
|
def test_mixed_copy_nonexisting_dest_nonrecursive(self):
|
||||||
"""Check that a mixt copy (files + dir) to an existing dest without the recursive option work as expected"""
|
"""Non-recursive mixed copy (files + dir) to an inexistant destination.
|
||||||
|
"""
|
||||||
for i in range(2):
|
for i in range(2):
|
||||||
makeRandomFile('file_%d' % i, S10K)
|
makeRandomFile('file_%d' % i, S10K)
|
||||||
makeTestDir('dir_%d' % i)
|
makeTestDir('dir_%d' % i)
|
||||||
check_before = dirCheck('.')
|
check_before = dirCheck('.')
|
||||||
ret = system("gcp file_0 file_1 dir_0 dir_1 dest_dir")
|
ret = system(GCP + " file_0 file_1 dir_0 dir_1 dest_dir")
|
||||||
self.assertEqual(ret >> 8, 1)
|
self.assertEqual(ret >> 8, 1)
|
||||||
check_after = dirCheck('.')
|
check_after = dirCheck('.')
|
||||||
self.assertEqual(check_before, check_after)
|
self.assertEqual(check_before, check_after)
|
||||||
|
|
||||||
|
def test_fix_filenames(self):
|
||||||
|
"""Checks the --fix-filenames option.
|
||||||
|
"""
|
||||||
|
source = 'wi||fat<like>this:"file*name"?'
|
||||||
|
fixed_name = 'wi!!fat[like]this;\'file+name\'_'
|
||||||
|
makeRandomFile(source, S10K)
|
||||||
|
source_sha = sha1sum(source)
|
||||||
|
mkdir('dest_dir')
|
||||||
|
ret = system(GCP + " --fix-filenames=force '" + source + "' dest_dir")
|
||||||
|
self.assertNotEqual(ret, 0)
|
||||||
|
dest = join('dest_dir', fixed_name)
|
||||||
|
dest_sha = sha1sum(dest)
|
||||||
|
self.assertEqual(source_sha, dest_sha)
|
||||||
|
|
||||||
if __name__ == '__main__':
|
if __name__ == '__main__':
|
||||||
unittest.main()
|
unittest.main()
|
Loading…
Reference in New Issue