2017-12-16 13:25:59 +01:00
---
id: dev-plugins
title: "Developing Plugins"
---
2017-10-29 07:46:05 +01:00
2018-07-22 20:23:53 +02:00
There are many ways to extend `verdaccio` , the kind of plugins supported are:
* Authentication plugins
* Middleware plugins (since `v2.7.0` )
* Storage plugins since (`v3.x` )
> We recommend developing plugins using our [flow type definitions](https://github.com/verdaccio/flow-types).
2017-10-29 07:46:05 +01:00
2018-05-10 08:48:15 +02:00
## Authentication Plugin
2017-10-29 07:46:05 +01:00
2018-07-22 20:23:53 +02:00
Basically we have to return an object with a single method called `authenticate` that will recieve 3 arguments (`user, password, callback` ).
2017-10-29 07:46:05 +01:00
2018-01-20 23:00:45 +01:00
### API
2018-07-22 20:23:53 +02:00
```flow
interface IPluginAuth extends IPlugin {
login_url?: string;
authenticate(user: string, password: string, cb: Callback): void;
adduser(user: string, password: string, cb: Callback): void;
allow_access(user: RemoteUser, pkg: $Subtype< PackageAccess > , cb: Callback): void;
allow_publish(user: RemoteUser, pkg: $Subtype< PackageAccess > , cb: Callback): void;
2018-01-20 23:00:45 +01:00
}
```
2018-07-22 20:23:53 +02:00
> Only `adduser`, `allow_access` and `allow_publish` are optional, verdaccio provide a fallback in all those cases.
#### Callback
2018-01-20 23:00:45 +01:00
2018-07-22 20:23:53 +02:00
Once the authentication has been executed there is 2 options to give a response to `verdaccio` .
###### OnError
2017-10-29 07:46:05 +01:00
Either something bad happened or auth was unsuccessful.
2018-07-22 20:23:53 +02:00
```flow
2017-10-29 07:46:05 +01:00
callback(null, false)
```
2018-07-22 20:23:53 +02:00
###### OnSuccess
2017-10-29 07:46:05 +01:00
The auth was successful.
`groups` is an array of strings where the user is part of.
```
callback(null, groups);
```
### Example
```javascript
function Auth(config, stuff) {
var self = Object.create(Auth.prototype);
self._users = {};
// config for this module
self._config = config;
// verdaccio logger
self._logger = stuff.logger;
// pass verdaccio logger to ldapauth
self._config.client_options.log = stuff.logger;
return self;
}
Auth.prototype.authenticate = function (user, password, callback) {
var LdapClient = new LdapAuth(self._config.client_options);
....
LdapClient.authenticate(user, password, function (err, ldapUser) {
...
var groups;
...
callback(null, groups);
});
};
module.exports = Auth;
```
2018-07-22 20:23:53 +02:00
And the configuration will looks like:
2017-10-29 07:46:05 +01:00
2018-01-20 23:00:45 +01:00
```yaml
auth:
htpasswd:
file: ./htpasswd
```
Where `htpasswd` is the sufix of the plugin name. eg: `verdaccio-htpasswd` and the rest of the body would be the plugin configuration params.
2017-10-29 07:46:05 +01:00
2018-05-10 08:48:15 +02:00
## Middleware Plugin
2017-10-29 07:46:05 +01:00
2018-05-28 23:47:15 +02:00
Middleware plugins have the capability to modify the API layer, either adding new endpoints or intercepting requests.
2018-07-22 20:23:53 +02:00
```flow
interface verdaccio$IPluginMiddleware extends verdaccio$IPlugin {
register_middlewares(app: any, auth: IBasicAuth, storage: IStorageManager): void;
}
```
### register_middlewares
The method provide full access to the authentification and storage via `auth` and `storage` . `app` is the express application that allows you to add new endpoints.
2018-05-28 23:47:15 +02:00
> A pretty good example
of middleware plugin is the [sinopia-github-oauth ](https://github.com/soundtrackyourbrand/sinopia-github-oauth ) and [verdaccio-audit ](https://github.com/verdaccio/verdaccio-audit ).
2018-01-20 23:00:45 +01:00
### API
```js
2018-05-28 23:47:15 +02:00
function register_middlewares(expressApp, authInstance, storageInstance) {
/* more stuff */
2018-01-20 23:00:45 +01:00
}
```
To register a middleware we need an object with a single method called `register_middlewares` that will recieve 3 arguments (`expressApp, auth, storage` ).
*Auth* is the authentification instance and *storage* is also the main Storage instance that will give you have access to all to the storage actions.
2018-05-10 08:48:15 +02:00
## Storage Plugin
2018-01-20 23:00:45 +01:00
2018-07-22 20:23:53 +02:00
Verdaccio by default uses a file system storage plugin [local-storage ](https://github.com/verdaccio/local-storage ), but, since `verdaccio@3.x` you can plug in a custom storage replacing the default behaviour.
2018-01-20 23:00:45 +01:00
### API
2018-07-22 20:23:53 +02:00
The storage API is a bit more complex, you will need to create a class that return a `IPluginStorage` implementation. Please see details bellow.
2018-01-20 23:00:45 +01:00
2018-07-22 20:23:53 +02:00
```flow
class LocalDatabase< IPluginStorage > {
constructor(config: $Subtype< verdaccio $ Config > , logger: verdaccio$Logger): ILocalData;
2018-01-20 23:00:45 +01:00
}
2018-07-22 20:23:53 +02:00
interface IPluginStorage {
logger: verdaccio$Logger;
config: $Subtype< verdaccio $ Config > ;
2018-05-10 08:48:15 +02:00
add(name: string, callback: verdaccio$Callback): void;
remove(name: string, callback: verdaccio$Callback): void;
get(callback: verdaccio$Callback): void;
getSecret(): Promise< string > ;
setSecret(secret: string): Promise< any > ;
getPackageStorage(packageInfo: string): verdaccio$IPackageStorage;
2018-07-22 20:23:53 +02:00
search(onPackage: verdaccio$Callback, onEnd: verdaccio$Callback, validateName: Function): void;
2018-01-20 23:00:45 +01:00
}
2018-07-22 20:23:53 +02:00
interface IPackageStorageManager {
path: string;
logger: verdaccio$Logger;
2018-05-10 08:48:15 +02:00
writeTarball(name: string): verdaccio$IUploadTarball;
readTarball(name: string): verdaccio$IReadTarball;
readPackage(fileName: string, callback: verdaccio$Callback): void;
createPackage(name: string, value: verdaccio$Package, cb: verdaccio$Callback): void;
deletePackage(fileName: string, callback: verdaccio$Callback): void;
removePackage(callback: verdaccio$Callback): void;
2018-01-20 23:00:45 +01:00
updatePackage(pkgFileName: string,
2018-05-10 08:48:15 +02:00
updateHandler: verdaccio$Callback,
onWrite: verdaccio$Callback,
2018-01-20 23:00:45 +01:00
transformPackage: Function,
2018-05-10 08:48:15 +02:00
onEnd: verdaccio$Callback): void;
savePackage(fileName: string, json: verdaccio$Package, callback: verdaccio$Callback): void;
2018-01-20 23:00:45 +01:00
}
2018-07-22 20:23:53 +02:00
class verdaccio$IUploadTarball extends stream$PassThrough {
abort: Function;
done: Function;
_transform: Function;
2018-01-20 23:00:45 +01:00
abort(): void;
done(): void;
}
2018-07-22 20:23:53 +02:00
class verdaccio$IReadTarball extends stream$PassThrough {
abort: Function;
2018-01-20 23:00:45 +01:00
abort(): void;
}
```
2018-05-10 08:48:15 +02:00
> The Storage API is still experimental and might change in the next minor versions. For further information about Storage API please follow the [types
definitions in our official repository](https://github.com/verdaccio/flow-types).
### Storage Plugins Examples
The following list of plugins are implementing the Storage API and might be used them as example.
* [verdaccio-memory ](https://github.com/verdaccio/verdaccio-memory )
* [local-storage ](https://github.com/verdaccio/local-storage )
* [verdaccio-google-cloud ](https://github.com/verdaccio/verdaccio-google-cloud )
* [verdaccio-s3-storage ](https://github.com/Remitly/verdaccio-s3-storage/tree/s3 )
> Are you willing to contribute with new Storage Plugins? [Click here.](https://github.com/verdaccio/verdaccio/issues/103#issuecomment-357478295)