summaryrefslogtreecommitdiffstats
path: root/lib/plugins/index.js
blob: c6f1686a3bdee74c39c862223829d9801404c0cf (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
var _ = require('lodash');
var path = require('path');

var Promise = require('../utils/promise');
var fs = require('../utils/fs');
var BookPlugin = require('./plugin');
var registry = require('./registry');
var pluginsConfig = require('../config/plugins');

/*
PluginsManager is an interface to work with multiple plugins at once:
- Extract assets from plugins
- Call hooks for all plugins, etc
*/

function PluginsManager(book) {
    this.book = book;
    this.log = this.book.log;
    this.plugins = [];

    _.bindAll(this);
}

// Returns the list of plugins
PluginsManager.prototype.list = function() {
    return this.plugins;
};

// Return count of plugins loaded
PluginsManager.prototype.count = function() {
    return _.size(this.plugins);
};

// Returns a plugin by its name
PluginsManager.prototype.get = function(name) {
    return _.find(this.plugins, {
        id: name
    });
};

// Load a plugin (could be a BookPlugin or {name,path})
PluginsManager.prototype.load = function(plugin) {
    var that = this;

    if (_.isArray(plugin)) {
        return Promise.serie(plugin, that.load);
    }

    return Promise()

    // Initiate and load the plugin
    .then(function() {
        if (!(plugin instanceof BookPlugin)) {
            plugin = new BookPlugin(that.book, plugin.name, plugin.path);
        }

        if (that.get(plugin.id)) {
            throw new Error('Plugin "'+plugin.id+'" is already loaded');
        }


        if (plugin.isLoaded()) return plugin;
        else return plugin.load()
            .thenResolve(plugin);
    })

    // Setup the plugin
    .then(this._setup);
};

// Load all plugins from the book's configuration
PluginsManager.prototype.loadAll = function() {
    var that = this;
    var pluginNames = _.pluck(this.book.config.get('plugins'), 'name');

    return registry.list(this.book)
    .then(function(plugins) {
        // Filter out plugins not listed of first level
        // (aka pre-installed plugins)
        plugins = _.filter(plugins, function(plugin) {
            return (
                plugin.depth > 1 ||
                _.contains(pluginNames, plugin.name)
            );
        });

        // Sort plugins to match list in book.json
        plugins.sort(function(a, b){
            return pluginNames.indexOf(a.name) < pluginNames.indexOf(b.name) ? -1 : 1;
        });

        // Log state
        that.log.info.ln(_.size(plugins) + ' are installed');
        if (_.size(pluginNames) != _.size(plugins)) that.log.info.ln(_.size(pluginNames) + ' explicitly listed');

        // Verify that all plugins are present
        var notInstalled = _.filter(pluginNames, function(name) {
            return !_.find(plugins, { name: name });
        });

        if (_.size(notInstalled) > 0) {
            throw new Error('Couldn\'t locate plugins "' + notInstalled.join(', ') + '", Run \'gitbook install\' to install plugins from registry.');
        }

        // Load plugins
        return that.load(plugins);
    });
};

// Setup a plugin
// Register its filter, blocks, etc
PluginsManager.prototype._setup = function(plugin) {
    this.plugins.push(plugin);
};

// Install all plugins for the book
PluginsManager.prototype.install = function() {
    var that = this;
    var plugins = _.filter(this.book.config.get('plugins'), function(plugin) {
        return !pluginsConfig.isDefaultPlugin(plugin.name);
    });

    if (plugins.length == 0) {
        this.log.info.ln('nothing to install!');
        return Promise(0);
    }

    this.log.info.ln('installing', plugins.length, 'plugins');

    return Promise.serie(plugins, function(plugin) {
        return registry.install(that.book, plugin.name, plugin.version);
    })
    .thenResolve(plugins.length);
};

// Call a hook on all plugins to transform an input
PluginsManager.prototype.hook = function(name, input) {
    return Promise.reduce(this.plugins, function(current, plugin) {
        return plugin.hook(name, current);
    }, input);
};

// Extract all resources for a namespace
PluginsManager.prototype.getResources = function(namespace) {
    return Promise.reduce(this.plugins, function(out, plugin) {
        return plugin.getResources(namespace)
        .then(function(pluginResources) {
            _.each(BookPlugin.RESOURCES, function(resourceType) {
                out[resourceType] = (out[resourceType] || []).concat(pluginResources[resourceType] || []);
            });

            return out;
        });
    }, {});
};

// Copy all resources for a plugin
PluginsManager.prototype.copyResources = function(namespace, outputRoot) {
    return Promise.serie(this.plugins, function(plugin) {
        return plugin.getResources(namespace)
        .then(function(resources) {
            if (!resources.assets) return;

            var input = path.resolve(plugin.root, resources.assets);
            var output = path.resolve(outputRoot, plugin.npmId);

            return fs.copyDir(input, output);
        });
    });
};

// Get all filters and blocks
PluginsManager.prototype.getFilters = function() {
    return _.reduce(this.plugins, function(out, plugin) {
        var filters = plugin.getFilters();

        return _.extend(out, filters);
    }, {});
};
PluginsManager.prototype.getBlocks = function() {
    return _.reduce(this.plugins, function(out, plugin) {
        var blocks = plugin.getBlocks();

        return _.extend(out, blocks);
    }, {});
};

module.exports = PluginsManager;