mirror of
https://github.com/samsonjs/vdirsyncer.git
synced 2026-03-29 09:35:50 +00:00
235 lines
6.7 KiB
Python
235 lines
6.7 KiB
Python
# -*- coding: utf-8 -*-
|
|
|
|
import functools
|
|
import logging
|
|
import sys
|
|
|
|
import click
|
|
|
|
import click_log
|
|
|
|
from .. import __version__
|
|
|
|
|
|
cli_logger = logging.getLogger(__name__)
|
|
|
|
|
|
class AppContext(object):
|
|
def __init__(self):
|
|
self.config = None
|
|
self.fetched_params = {}
|
|
self.logger = None
|
|
|
|
|
|
pass_context = click.make_pass_decorator(AppContext, ensure=True)
|
|
|
|
|
|
def catch_errors(f):
|
|
@functools.wraps(f)
|
|
def inner(*a, **kw):
|
|
try:
|
|
f(*a, **kw)
|
|
except BaseException:
|
|
from .utils import handle_cli_error
|
|
handle_cli_error()
|
|
sys.exit(1)
|
|
|
|
return inner
|
|
|
|
|
|
@click.group()
|
|
@click_log.init('vdirsyncer')
|
|
@click_log.simple_verbosity_option()
|
|
@click.version_option(version=__version__)
|
|
@click.option('--config', '-c', metavar='FILE', help='Config file to use.')
|
|
@pass_context
|
|
@catch_errors
|
|
def app(ctx, config):
|
|
'''
|
|
Synchronize calendars and contacts
|
|
'''
|
|
|
|
if not ctx.config:
|
|
from .config import load_config
|
|
ctx.config = load_config(config)
|
|
|
|
|
|
main = app
|
|
|
|
|
|
def max_workers_callback(ctx, param, value):
|
|
if value == 0 and click_log.get_level() == logging.DEBUG:
|
|
value = 1
|
|
|
|
cli_logger.debug('Using {} maximal workers.'.format(value))
|
|
return value
|
|
|
|
|
|
def max_workers_option(default=0):
|
|
help = 'Use at most this many connections. '
|
|
if default == 0:
|
|
help += 'The default is 0, which means "as many as necessary". ' \
|
|
'With -vdebug enabled, the default is 1.'
|
|
else:
|
|
help += 'The default is {}.'.format(default)
|
|
|
|
return click.option(
|
|
'--max-workers', default=default, type=click.IntRange(min=0, max=None),
|
|
callback=max_workers_callback,
|
|
help=help
|
|
)
|
|
|
|
|
|
def collections_arg_callback(ctx, param, value):
|
|
'''
|
|
Expand the various CLI shortforms ("pair, pair/collection") to an iterable
|
|
of (pair, collections).
|
|
'''
|
|
# XXX: Ugly! pass_context should work everywhere.
|
|
config = ctx.find_object(AppContext).config
|
|
rv = {}
|
|
for pair_and_collection in (value or config.pairs):
|
|
pair, collection = pair_and_collection, None
|
|
if '/' in pair:
|
|
pair, collection = pair.split('/')
|
|
|
|
collections = rv.setdefault(pair, set())
|
|
if collection:
|
|
collections.add(collection)
|
|
|
|
return rv.items()
|
|
|
|
|
|
collections_arg = click.argument('collections', nargs=-1,
|
|
callback=collections_arg_callback)
|
|
|
|
|
|
@app.command()
|
|
@collections_arg
|
|
@click.option('--force-delete/--no-force-delete',
|
|
help=('Do/Don\'t abort synchronization when all items are about '
|
|
'to be deleted from both sides.'))
|
|
@max_workers_option()
|
|
@pass_context
|
|
@catch_errors
|
|
def sync(ctx, collections, force_delete, max_workers):
|
|
'''
|
|
Synchronize the given collections or pairs. If no arguments are given, all
|
|
will be synchronized.
|
|
|
|
This command will not synchronize metadata, use `vdirsyncer metasync` for
|
|
that.
|
|
|
|
Examples:
|
|
|
|
`vdirsyncer sync` will sync everything configured.
|
|
|
|
`vdirsyncer sync bob frank` will sync the pairs "bob" and "frank".
|
|
|
|
`vdirsyncer sync bob/first_collection` will sync "first_collection"
|
|
from the pair "bob".
|
|
'''
|
|
from .tasks import prepare_pair, sync_collection
|
|
from .utils import WorkerQueue
|
|
|
|
wq = WorkerQueue(max_workers)
|
|
|
|
with wq.join():
|
|
for pair_name, collections in collections:
|
|
wq.put(functools.partial(prepare_pair, pair_name=pair_name,
|
|
collections=collections,
|
|
config=ctx.config,
|
|
force_delete=force_delete,
|
|
callback=sync_collection))
|
|
wq.spawn_worker()
|
|
|
|
|
|
@app.command()
|
|
@collections_arg
|
|
@max_workers_option()
|
|
@pass_context
|
|
@catch_errors
|
|
def metasync(ctx, collections, max_workers):
|
|
'''
|
|
Synchronize metadata of the given collections or pairs.
|
|
|
|
See the `sync` command for usage.
|
|
'''
|
|
from .tasks import prepare_pair, metasync_collection
|
|
from .utils import WorkerQueue
|
|
|
|
wq = WorkerQueue(max_workers)
|
|
|
|
with wq.join():
|
|
for pair_name, collections in collections:
|
|
wq.put(functools.partial(prepare_pair, pair_name=pair_name,
|
|
collections=collections,
|
|
config=ctx.config,
|
|
callback=metasync_collection))
|
|
wq.spawn_worker()
|
|
|
|
|
|
@app.command()
|
|
@click.argument('pairs', nargs=-1)
|
|
@click.option(
|
|
'--list/--no-list', default=True,
|
|
help=(
|
|
'Whether to list all collections from both sides during discovery, '
|
|
'for debugging. This is slow, but enabled by default for usability '
|
|
'reasons. '
|
|
)
|
|
)
|
|
@max_workers_option(default=1)
|
|
@pass_context
|
|
@catch_errors
|
|
def discover(ctx, pairs, max_workers, list):
|
|
'''
|
|
Refresh collection cache for the given pairs.
|
|
'''
|
|
from .tasks import discover_collections
|
|
from .utils import WorkerQueue
|
|
config = ctx.config
|
|
wq = WorkerQueue(max_workers)
|
|
|
|
with wq.join():
|
|
for pair_name in (pairs or config.pairs):
|
|
pair = config.get_pair(pair_name)
|
|
|
|
wq.put(functools.partial(
|
|
discover_collections,
|
|
status_path=config.general['status_path'],
|
|
pair=pair,
|
|
from_cache=False,
|
|
list_collections=list,
|
|
))
|
|
wq.spawn_worker()
|
|
|
|
|
|
@app.command()
|
|
@click.argument('collection')
|
|
@click.option('--repair-unsafe-uid/--no-repair-unsafe-uid', default=False,
|
|
help=('Some characters in item UIDs and URLs may cause problems '
|
|
'with buggy software. Adding this option will reassign '
|
|
'new UIDs to those items.'))
|
|
@pass_context
|
|
@catch_errors
|
|
def repair(ctx, collection, repair_unsafe_uid):
|
|
'''
|
|
Repair a given collection.
|
|
|
|
Runs a few checks on the collection and applies some fixes to individual
|
|
items that may improve general stability, also with other CalDAV/CardDAV
|
|
clients. In particular, if you encounter URL-encoding-related issues with
|
|
other clients, this command with --repair-unsafe-uid might help.
|
|
|
|
Example: `vdirsyncer repair calendars_local/foo` repairs the `foo`
|
|
collection of the `calendars_local` storage.
|
|
'''
|
|
from .tasks import repair_collection
|
|
|
|
cli_logger.warning('This operation will take a very long time.')
|
|
cli_logger.warning('It\'s recommended to make a backup and '
|
|
'turn off other client\'s synchronization features.')
|
|
click.confirm('Do you want to continue?', abort=True)
|
|
repair_collection(ctx.config, collection,
|
|
repair_unsafe_uid=repair_unsafe_uid)
|