2012-10-06 17:57:08 +02:00
|
|
|
#!/usr/bin/env python
|
2012-10-06 16:22:15 +02:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
2012-10-07 16:20:20 +02:00
|
|
|
__credits__ = """
|
|
|
|
Copyright (C) 2012 YunoHost
|
2012-10-06 20:29:00 +02:00
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU Affero General Public License as published
|
|
|
|
by the Free Software Foundation, either version 3 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program 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 Affero General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU Affero General Public License
|
|
|
|
along with this program; if not, see http://www.gnu.org/licenses
|
2012-10-07 16:20:20 +02:00
|
|
|
"""
|
2012-10-07 18:20:06 +02:00
|
|
|
__author__ = 'Kload <kload@kload.fr>'
|
2012-10-07 13:37:59 +02:00
|
|
|
__version__ = '2.0 beta1'
|
2012-10-06 20:29:00 +02:00
|
|
|
|
2012-10-08 18:16:43 +02:00
|
|
|
import os
|
2012-10-06 16:22:15 +02:00
|
|
|
import sys
|
|
|
|
import argparse
|
2012-10-07 16:20:20 +02:00
|
|
|
import gettext
|
2012-10-08 18:16:43 +02:00
|
|
|
import json
|
|
|
|
if not __debug__:
|
|
|
|
import traceback
|
|
|
|
|
2012-10-07 16:20:20 +02:00
|
|
|
gettext.install('YunoHost')
|
2012-10-07 13:16:19 +02:00
|
|
|
|
2012-10-10 16:01:54 +02:00
|
|
|
try:
|
|
|
|
sys.path.append('lib')
|
|
|
|
from yunohost import YunoHostError, YunoHostLDAP, str_to_func, colorize, pretty_print_dict
|
|
|
|
except ImportError:
|
|
|
|
sys.stderr.write('Require YunoHost lib')
|
|
|
|
sys.exit(1)
|
2012-10-08 20:46:52 +02:00
|
|
|
|
2012-10-08 23:02:26 +02:00
|
|
|
"""
|
2012-10-10 12:20:28 +02:00
|
|
|
Category/actions/arguments dictionnary
|
2012-10-08 20:46:52 +02:00
|
|
|
|
|
|
|
|
2012-10-10 12:20:28 +02:00
|
|
|
Except for general_arguments, this dictionary contains 3 levels
|
|
|
|
as in this sample command line :
|
|
|
|
|
|
|
|
yunohost monitor info --cpu --ram
|
|
|
|
^ ^ ^ ^
|
|
|
|
(script) | category | action | parameters
|
2012-10-08 20:46:52 +02:00
|
|
|
|
|
|
|
|
2012-10-10 12:20:28 +02:00
|
|
|
Above example will lead to the function 'monitor_info(args)'
|
|
|
|
in the file 'lib/yunohost_monitor.py' with 'cpu' and 'ram'
|
|
|
|
stored in args dictionnary.
|
|
|
|
|
|
|
|
Usage:
|
|
|
|
You can add a category at the first level, action at the second one,
|
|
|
|
and arguments at the third one.
|
2012-10-10 18:13:39 +02:00
|
|
|
If the action need LDAP connexion, don't forget to add 'ldap' : True
|
|
|
|
to the action parameters.
|
2012-10-10 12:20:28 +02:00
|
|
|
|
|
|
|
Documentation:
|
|
|
|
You can see all arguments settings at the argparse documentation:
|
|
|
|
http://docs.python.org/dev/library/argparse.html
|
|
|
|
#argparse.ArgumentParser.add_argument
|
|
|
|
|
|
|
|
Don't forget to turn them as a dictionnary ('setting' : 'value')
|
2012-10-08 23:02:26 +02:00
|
|
|
|
|
|
|
"""
|
2012-10-10 12:20:28 +02:00
|
|
|
action_dict = {
|
|
|
|
#############################
|
|
|
|
# General args #
|
|
|
|
#############################
|
|
|
|
'general_arguments' : {
|
|
|
|
'-v' : {
|
|
|
|
'full' : '--version',
|
2012-10-10 19:13:38 +02:00
|
|
|
'help' : _("Display %(prog)s version"),
|
2012-10-10 12:20:28 +02:00
|
|
|
'action' : 'version',
|
|
|
|
'version' : '%(prog)s ' + __version__,
|
|
|
|
},
|
2012-10-10 14:47:11 +02:00
|
|
|
},
|
2012-10-10 11:10:57 +02:00
|
|
|
#############################
|
|
|
|
# User #
|
|
|
|
#############################
|
|
|
|
'user' : {
|
2012-10-10 19:13:38 +02:00
|
|
|
'category_help' : _("Manage users"),
|
2012-10-10 16:01:54 +02:00
|
|
|
'actions' : {
|
2012-10-10 11:10:57 +02:00
|
|
|
### user_list()
|
2012-10-10 19:47:57 +02:00
|
|
|
'list' : {
|
2012-10-10 19:13:38 +02:00
|
|
|
'action_help' : _("List users"),
|
2012-10-10 16:01:54 +02:00
|
|
|
'ldap' : True,
|
|
|
|
'arguments' : {
|
2012-10-10 19:13:38 +02:00
|
|
|
'--fields' : {
|
|
|
|
'help' : _("Fields to fetch"),
|
|
|
|
'nargs' : '+',
|
|
|
|
},
|
|
|
|
'-f' : {
|
|
|
|
'full' : '--filter',
|
|
|
|
'help' : _("LDAP filter used to search"),
|
|
|
|
},
|
|
|
|
'-l' : {
|
|
|
|
'full' : '--limit',
|
|
|
|
'help' : _("Maximum number of users fetched"),
|
|
|
|
},
|
|
|
|
'-o' : {
|
|
|
|
'full' : '--offset',
|
|
|
|
'help' : _("Starting number for user fetching"),
|
2012-10-10 11:10:57 +02:00
|
|
|
},
|
|
|
|
}
|
|
|
|
},
|
2012-10-10 19:47:57 +02:00
|
|
|
### user_create()
|
|
|
|
'create' : {
|
2012-10-10 19:13:38 +02:00
|
|
|
'action_help' : _("Create user"),
|
2012-10-10 16:01:54 +02:00
|
|
|
'ldap' : True,
|
|
|
|
'arguments' : {
|
2012-10-10 11:10:57 +02:00
|
|
|
'-u' : {
|
|
|
|
'full' : '--username',
|
2012-10-10 19:13:38 +02:00
|
|
|
'help' : _("Must be unique"),
|
2012-10-10 11:10:57 +02:00
|
|
|
},
|
|
|
|
'-f' : {
|
|
|
|
'full' : '--firstname',
|
|
|
|
},
|
|
|
|
'-l' : {
|
|
|
|
'full' : '--lastname',
|
|
|
|
},
|
|
|
|
'-m' : {
|
|
|
|
'full' : '--mail',
|
2012-10-10 19:13:38 +02:00
|
|
|
'help' : _("Main mail address, must be unique"),
|
2012-10-10 11:10:57 +02:00
|
|
|
},
|
|
|
|
'-p' : {
|
|
|
|
'full' : '--password',
|
|
|
|
},
|
|
|
|
}
|
|
|
|
},
|
2012-10-10 19:13:38 +02:00
|
|
|
### user_delete()
|
2012-10-10 19:47:57 +02:00
|
|
|
'delete' : {
|
2012-10-10 19:13:38 +02:00
|
|
|
'action_help' : _("Delete user"),
|
|
|
|
'ldap' : True,
|
|
|
|
'arguments' : {
|
|
|
|
'users' : {
|
|
|
|
'help' : _("Username of users to delete"),
|
|
|
|
'nargs' : '+',
|
|
|
|
},
|
|
|
|
}
|
|
|
|
},
|
|
|
|
### user_update()
|
2012-10-10 19:47:57 +02:00
|
|
|
'update' : {
|
2012-10-10 19:13:38 +02:00
|
|
|
'action_help' : _("Update user informations"),
|
|
|
|
'ldap' : True,
|
|
|
|
'arguments' : {
|
|
|
|
'user' : {
|
|
|
|
'help' : _("Username of user to update"),
|
|
|
|
},
|
|
|
|
'-f' : {
|
|
|
|
'full' : '--firstname',
|
|
|
|
},
|
|
|
|
'-l' : {
|
|
|
|
'full' : '--lastname',
|
|
|
|
},
|
|
|
|
'-m' : {
|
|
|
|
'full' : '--mail',
|
|
|
|
},
|
|
|
|
'-cp' : {
|
|
|
|
'full' : '--change-password',
|
|
|
|
'help' : _("New password to set"),
|
|
|
|
'metavar' : 'PASSWORD',
|
|
|
|
},
|
|
|
|
'--add-mailforward' : {
|
|
|
|
'help' : _("Mailforward addresses to add"),
|
|
|
|
'nargs' : '+',
|
|
|
|
'metavar' : 'MAIL',
|
|
|
|
},
|
|
|
|
'--remove-mailforward' : {
|
|
|
|
'help' : _("Mailforward addresses to remove"),
|
|
|
|
'nargs' : '+',
|
|
|
|
'metavar' : 'MAIL',
|
|
|
|
},
|
|
|
|
'-add-mailalias' : {
|
|
|
|
'help' : _("Mail aliases to add"),
|
|
|
|
'nargs' : '+',
|
|
|
|
'metavar' : 'MAIL',
|
|
|
|
},
|
|
|
|
'-remove-mailalias' : {
|
|
|
|
'help' : _("Mail aliases to remove"),
|
|
|
|
'nargs' : '+',
|
|
|
|
'metavar' : 'MAIL',
|
|
|
|
},
|
|
|
|
}
|
|
|
|
},
|
|
|
|
### user_info()
|
2012-10-10 14:47:11 +02:00
|
|
|
'info' : {
|
2012-10-10 19:13:38 +02:00
|
|
|
'action_help' : _("Get user informations"),
|
2012-10-10 16:01:54 +02:00
|
|
|
'ldap' : True,
|
2012-10-10 19:13:38 +02:00
|
|
|
'arguments' : {
|
|
|
|
'user' : {
|
|
|
|
'nargs' : '?',
|
|
|
|
},
|
|
|
|
'-m' : {
|
|
|
|
'full' : '--mail',
|
|
|
|
},
|
|
|
|
'-cn' : {
|
|
|
|
'full' : '--fullname',
|
|
|
|
},
|
|
|
|
}
|
2012-10-10 14:47:11 +02:00
|
|
|
},
|
2012-10-10 11:34:36 +02:00
|
|
|
}
|
2012-10-08 18:16:43 +02:00
|
|
|
},
|
2012-10-10 19:47:57 +02:00
|
|
|
#############################
|
|
|
|
# Domain #
|
|
|
|
#############################
|
2012-10-08 18:16:43 +02:00
|
|
|
'domain' : {
|
2012-10-10 19:13:38 +02:00
|
|
|
'category_help' : _("Manage domains"),
|
2012-10-10 19:47:57 +02:00
|
|
|
'actions' : {
|
|
|
|
### domain_list()
|
|
|
|
'list' : {
|
|
|
|
'action_help' : _("List domains"),
|
|
|
|
'ldap' : True,
|
|
|
|
'arguments' : {
|
|
|
|
'-f' : {
|
|
|
|
'full' : '--filter',
|
|
|
|
'help' : _("LDAP filter used to search"),
|
|
|
|
},
|
|
|
|
'-l' : {
|
|
|
|
'full' : '--limit',
|
|
|
|
'help' : _("Maximum number of users fetched"),
|
|
|
|
},
|
|
|
|
'-o' : {
|
|
|
|
'full' : '--offset',
|
|
|
|
'help' : _("Starting number for user fetching"),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
},
|
|
|
|
### domain_create()
|
|
|
|
'create' : {
|
|
|
|
'action_help' : _("Create a custom domain"),
|
|
|
|
'ldap' : True,
|
|
|
|
'arguments' : {
|
|
|
|
'domain name' : {
|
|
|
|
'help' : _("Domain name to create"),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
### domain_delete()
|
|
|
|
'delete' : {
|
|
|
|
'action_help' : _("Delete domains"),
|
|
|
|
'ldap' : True,
|
|
|
|
'arguments' : {
|
|
|
|
'domains' : {
|
|
|
|
'help' : _("Domain names to delete"),
|
|
|
|
'nargs' : '+',
|
|
|
|
},
|
|
|
|
}
|
|
|
|
},
|
|
|
|
### domain_info()
|
|
|
|
'info' : {
|
|
|
|
'action_help' : _("Get domain informations"),
|
|
|
|
'ldap' : True,
|
|
|
|
'arguments' : {
|
|
|
|
'domain-name' : {}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
### domain_renewcert()
|
|
|
|
'renewcert' : {
|
|
|
|
'action_help' : _("Renew domain certificate"),
|
|
|
|
'ldap' : True,
|
|
|
|
'arguments' : {
|
|
|
|
'domain-name' : {}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
}
|
2012-10-08 18:16:43 +02:00
|
|
|
},
|
|
|
|
'app' : {
|
2012-10-10 19:13:38 +02:00
|
|
|
'category_help' : _("Manage apps"),
|
2012-10-10 16:01:54 +02:00
|
|
|
'actions' : {}
|
2012-10-08 18:16:43 +02:00
|
|
|
},
|
|
|
|
'monitor' : {
|
2012-10-10 19:13:38 +02:00
|
|
|
'category_help' : _("Monitoring functions"),
|
2012-10-10 16:01:54 +02:00
|
|
|
'actions' : {}
|
2012-10-08 18:16:43 +02:00
|
|
|
},
|
|
|
|
'tools' : {
|
2012-10-10 19:13:38 +02:00
|
|
|
'category_help' : _("Specific tools"),
|
2012-10-10 16:01:54 +02:00
|
|
|
'actions' : {}
|
2012-10-08 18:16:43 +02:00
|
|
|
}
|
|
|
|
}
|
2012-10-06 16:22:15 +02:00
|
|
|
|
2012-10-10 16:01:54 +02:00
|
|
|
def parse_dict(action_dict):
|
2012-10-08 20:46:52 +02:00
|
|
|
"""
|
2012-10-10 11:10:57 +02:00
|
|
|
Turn action dictionnary to parser, subparsers and arguments
|
2012-10-08 20:46:52 +02:00
|
|
|
|
|
|
|
Keyword arguments:
|
2012-10-10 11:10:57 +02:00
|
|
|
action_dict -- Multi-level dictionnary of categories/actions/arguments list
|
2012-10-08 20:46:52 +02:00
|
|
|
|
|
|
|
Returns:
|
2012-10-10 11:10:57 +02:00
|
|
|
Namespace of args
|
2012-10-08 20:46:52 +02:00
|
|
|
|
|
|
|
"""
|
2012-10-10 16:01:54 +02:00
|
|
|
# Connection required
|
|
|
|
ldap = False
|
|
|
|
|
2012-10-08 20:46:52 +02:00
|
|
|
# Intialize parsers
|
|
|
|
parsers = subparsers_category = subparsers_action = {}
|
|
|
|
parsers['general'] = argparse.ArgumentParser()
|
|
|
|
subparsers = parsers['general'].add_subparsers()
|
|
|
|
|
2012-10-10 12:20:28 +02:00
|
|
|
# Add general arguments
|
|
|
|
for arg_name, arg_params in action_dict['general_arguments'].items():
|
|
|
|
if arg_params['full']:
|
|
|
|
arg_fullname = arg_params['full']
|
|
|
|
del arg_params['full']
|
|
|
|
parsers['general'].add_argument(arg_name, arg_fullname, **arg_params)
|
|
|
|
else:
|
|
|
|
parsers['general'].add_argument(arg_name, **arg_params)
|
|
|
|
|
|
|
|
del action_dict['general_arguments']
|
|
|
|
|
|
|
|
# Split categories into subparsers
|
2012-10-10 16:01:54 +02:00
|
|
|
for category, category_params in action_dict.items():
|
|
|
|
if 'category_help' not in category_params: category_params['category_help'] = ''
|
|
|
|
subparsers_category[category] = subparsers.add_parser(category, help=category_params['category_help'])
|
2012-10-08 20:46:52 +02:00
|
|
|
subparsers_action[category] = subparsers_category[category].add_subparsers()
|
2012-10-10 12:20:28 +02:00
|
|
|
# Split actions
|
2012-10-10 16:01:54 +02:00
|
|
|
if 'actions' in category_params:
|
|
|
|
for action, action_params in category_params['actions'].items():
|
|
|
|
# Service settings
|
|
|
|
if 'ldap' in action_params: ldap = action_params['ldap']
|
|
|
|
if 'action_help' not in action_params: action_params['action_help'] = ''
|
|
|
|
parsers[category + '_' + action] = subparsers_action[category].add_parser(action, help=action_params['action_help'])
|
2012-10-10 16:13:46 +02:00
|
|
|
# Set the action s related function
|
2012-10-10 16:01:54 +02:00
|
|
|
parsers[category + '_' + action].set_defaults(
|
|
|
|
func=str_to_func('yunohost_' + category + '.'
|
|
|
|
+ category + '_' + action))
|
|
|
|
# Add arguments
|
|
|
|
if 'arguments' in action_params:
|
|
|
|
for arg_name, arg_params in action_params['arguments'].items():
|
2012-10-10 19:13:38 +02:00
|
|
|
if 'full' in arg_params:
|
2012-10-10 16:01:54 +02:00
|
|
|
arg_fullname = arg_params['full']
|
|
|
|
del arg_params['full']
|
|
|
|
parsers[category + '_' + action].add_argument(arg_name, arg_fullname, **arg_params)
|
|
|
|
else:
|
|
|
|
parsers[category + '_' + action].add_argument(arg_name, **arg_params)
|
2012-10-08 20:46:52 +02:00
|
|
|
|
2012-10-10 11:10:57 +02:00
|
|
|
args = parsers['general'].parse_args()
|
2012-10-10 16:01:54 +02:00
|
|
|
return { 'ldap' : ldap, 'args' : args }
|
2012-10-08 20:46:52 +02:00
|
|
|
|
2012-10-07 13:16:19 +02:00
|
|
|
def main():
|
2012-10-10 12:20:28 +02:00
|
|
|
"""
|
|
|
|
Main instructions
|
|
|
|
|
2012-10-10 14:19:17 +02:00
|
|
|
Parse the action_dict and execute the action-specific function,
|
|
|
|
then print json or pretty result if executed in a tty :)
|
2012-10-10 12:20:28 +02:00
|
|
|
|
|
|
|
Returns:
|
|
|
|
int -- 0 or error code
|
|
|
|
|
|
|
|
"""
|
2012-10-10 16:12:13 +02:00
|
|
|
action = parse_dict(action_dict)
|
|
|
|
|
|
|
|
# Connect to LDAP if the action is requiring it
|
|
|
|
if action['ldap']:
|
2012-10-10 16:01:54 +02:00
|
|
|
yldap = YunoHostLDAP()
|
|
|
|
|
2012-10-08 18:16:43 +02:00
|
|
|
try:
|
2012-10-10 16:12:13 +02:00
|
|
|
if action['ldap']:
|
|
|
|
result = action['args'].func(vars(action['args']), yldap)
|
2012-10-10 16:01:54 +02:00
|
|
|
else:
|
2012-10-10 16:12:13 +02:00
|
|
|
result = action['args'].func(vars(action['args']))
|
2012-10-10 19:13:38 +02:00
|
|
|
#except TypeError:
|
|
|
|
#print(_("Not (yet) implemented function"))
|
|
|
|
#return 1
|
2012-10-08 18:16:43 +02:00
|
|
|
except YunoHostError, error:
|
|
|
|
if not __debug__ :
|
|
|
|
traceback.print_exc()
|
|
|
|
if os.isatty(1):
|
2012-10-10 19:13:38 +02:00
|
|
|
print('\n' + colorize(_("Error: "), 'red') + error.message)
|
2012-10-08 18:16:43 +02:00
|
|
|
else:
|
|
|
|
print(json.dumps({ 'error' : error.message }))
|
|
|
|
return error.code
|
|
|
|
else:
|
|
|
|
if os.isatty(1):
|
2012-10-10 14:19:17 +02:00
|
|
|
pretty_print_dict(result)
|
2012-10-08 18:16:43 +02:00
|
|
|
else:
|
|
|
|
print(json.dumps(result))
|
2012-10-10 16:01:54 +02:00
|
|
|
finally:
|
2012-10-10 16:12:13 +02:00
|
|
|
if action['ldap']:
|
2012-10-10 16:01:54 +02:00
|
|
|
yldap.disconnect()
|
|
|
|
|
|
|
|
return 0
|
2012-10-07 13:16:19 +02:00
|
|
|
|
2012-10-06 16:22:15 +02:00
|
|
|
if __name__ == '__main__':
|
2012-10-08 18:16:43 +02:00
|
|
|
sys.exit(main())
|