Browse Source

Add an action allowing to install plugins in lodel_admin

the action is named plugins
Yann Weber 8 years ago
parent
commit
0fc2b6ad0f
2 changed files with 115 additions and 6 deletions
  1. 106
    1
      lodel/plugin/core_scripts.py
  2. 9
    5
      lodel/plugin/plugins.py

+ 106
- 1
lodel/plugin/core_scripts.py View File

@@ -1,7 +1,11 @@
1 1
 import operator
2
+import shutil
3
+import tempfile
4
+import os, os.path
2 5
 from lodel.context import LodelContext
3 6
 LodelContext.expose_modules(globals(), {
4
-    'lodel.plugin.scripts': 'lodel_script'})
7
+    'lodel.plugin.scripts': 'lodel_script',
8
+    'lodel.logger': 'logger'})
5 9
 
6 10
 ##@package lodel.plugin.core_scripts
7 11
 #@brief Lodel2 internal scripts declaration
@@ -52,6 +56,107 @@ class ListPlugins(lodel_script.LodelScript):
52 56
                 res += fmt % (pinfos['name'], pinfos['version'])
53 57
         print(res)
54 58
 
59
+
60
+##@brief Handle install & uninstall of lodel plugins
61
+class PluginManager(lodel_script.LodelScript):
62
+    _action = 'plugins'
63
+    _description = "Install/Uninstall plugins"
64
+
65
+    @classmethod
66
+    def argparser_config(cls, parser):
67
+        parser.add_argument('-u', '--uninstall',
68
+            help="Uninstall specified plugin",
69
+            action='store_true')
70
+        parser.add_argument('-n', '--plugin-name', nargs='*',
71
+            default = list(),
72
+            help="Indicate a plugin name to uninstall",
73
+            type=str)
74
+        parser.add_argument('-f', '--file', nargs='*',
75
+            default = list(),
76
+            help="Specify a tarball containing a plugin to install",
77
+            type=str)
78
+        parser.add_argument('-d', '--directory', nargs='*',
79
+            default = list(),
80
+            help="Specify a plugin by its directory",
81
+            type=str)
82
+
83
+    @classmethod
84
+    def run(cls, args):
85
+        if args.uninstall:
86
+            return cls.uninstall(args)
87
+        return cls.install(args)
88
+    
89
+    ##@brief Handles plugins install
90
+    @classmethod
91
+    def install(cls, args):
92
+        import lodel.plugin.plugins
93
+        from lodel.plugin.plugins import Plugin
94
+        from lodel.plugin.exceptions import PluginError
95
+        if len(args.plugin_name) > 0:
96
+            raise RuntimeError("Unable to install a plugin from its name !\
97
+We do not know where to find it...")
98
+        plist = Plugin.discover()
99
+        errors = dict()
100
+        if len(args.file) > 0:
101
+            raise NotImplementedError("Not supported yet")
102
+        
103
+        plugins_infos = {}
104
+        for cur_dir in args.directory:
105
+            try:
106
+                res = Plugin.dir_is_plugin(cur_dir, assert_in_package = False)
107
+                if res is False:
108
+                    errors[cur_dir] = PluginError("Not a plugin")
109
+                else:
110
+                    plugins_infos[res['name']] = res
111
+            except Exception as e:
112
+                errors[cur_dir] = e
113
+        #Abording because of previous errors
114
+        if len(errors) > 0:
115
+            msg = "Abording installation because of following errors :\n"
116
+            for path, expt in errors.items():
117
+                msg += ("\t- For path '%s' : %s\n" % (path, expt))
118
+            raise RuntimeError(msg)
119
+        #No errors continuing to install
120
+        for pname, pinfos in plugins_infos.items():
121
+            if pname in plist:
122
+                #Found an installed plugin with the same name
123
+                #Cehcking both versions
124
+                if plist[pname]['version'] == pinfos['version']:
125
+                    errors[pinfos['path']] = 'Abording installation of %s\
126
+found in %s because it seems to be allready installed in %s' % (
127
+    pname, pinfos['path'], plist[pname]['path'])
128
+                    continue
129
+                if plist[pname]['version'] > pinfos['version']:
130
+                    errors[pinfos['path']] = 'Abording installation of %s \
131
+found in %s because the same plugins with a greater version seems to be \
132
+installed in %s' % (pname, pinfos['path'], plist[pname]['path'])
133
+                    continue
134
+                logger.info("Found a plugin with the same name but with an \
135
+inferior version. Continuing to install")
136
+            #Checking that we can safely copy our plugin
137
+            dst_path = os.path.join(lodel.plugin.plugins.PLUGINS_PATH,
138
+                os.path.basename(os.path.dirname(pinfos['path'])))
139
+            orig_path = dst_path
140
+            if os.path.isdir(dst_path):
141
+                dst_path = tempfile.mkdtemp(
142
+                    prefix = os.path.basename(dst_path)+'_',
143
+                    dir = lodel.plugin.plugins.PLUGINS_PATH)
144
+                logger.warning("A plugin allready exists in %s. Installing \
145
+in %s" % (orig_path, dst_path))
146
+                shutil.rmtree(dst_path)
147
+                
148
+            #Installing the plugin
149
+            shutil.copytree(pinfos['path'], dst_path, symlinks = False)
150
+            print("%s(%s) installed in %s" % (
151
+                pname, pinfos['version'], dst_path))
152
+    
153
+    ##@brief Handles plugin uninstall
154
+    @classmethod
155
+    def uninstall(cls, args):
156
+        pass
157
+        
158
+
159
+
55 160
 ##@brief Implements lodel_admin.py **hooks-list** action
56 161
 #@ingroup lodel2_script
57 162
 #@ingroup lodel2_hooks

+ 9
- 5
lodel/plugin/plugins.py View File

@@ -709,14 +709,18 @@ name differ from the one found in plugin's init file"
709 709
 
710 710
     ##@brief Check if a directory is a plugin module
711 711
     #@param path str : path to check
712
+    #@param assert_in_package bool : if False didn't check that path is
713
+    #a subdir of PLUGINS_PATH
712 714
     #@return a dict with name, version and path if path is a plugin module, else False
713 715
     @classmethod
714
-    def dir_is_plugin(cls, path):
716
+    def dir_is_plugin(cls, path, assert_in_package = True):
715 717
         log_msg = "%s is not a plugin directory because : " % path
716
-        #Check that path is a subdir of PLUGINS_PATH
717
-        abspath = os.path.abspath(path)
718
-        if not abspath.startswith(os.path.abspath(PLUGINS_PATH)):
719
-            raise PluginError("%s is not a subdir of %s" % log_msg, PLUGINS_PATH)
718
+        if assert_in_package:
719
+            #Check that path is a subdir of PLUGINS_PATH
720
+            abspath = os.path.abspath(path)
721
+            if not abspath.startswith(os.path.abspath(PLUGINS_PATH)):
722
+                raise PluginError(
723
+                    "%s is not a subdir of %s" % log_msg, PLUGINS_PATH)
720 724
         #Checks that path exists
721 725
         if not os.path.isdir(path):
722 726
             raise ValueError(

Loading…
Cancel
Save