This is inspired by emgee3's Accounts Ldap for meteor package. emgee3's package is a proof of concept - this package is an attempt to move past proof of concept and create something production ready and tested.
You can add this package through Atmosphere by typing:
meteor add typ:accounts-ldap
from the command line.
OR if you'd like to customize it a bit:
Clone this repo and copy it to /packages
in your Meteor project.
The package exposes a global variable called LDAP_DEFAULTS
on the server side. You must specify the LDAP_DEFAULTS.url
at a minimum. Other options for the defaults are as follows:
LDAP_DEFAULTS.port
: Default port is the ldap default of 389.
LDAP_DEFAULTS.dn
: The ldap dn you want to authenticate on and search. Chances are you'll want to set this when calling Meteor.loginWithLdap() from client side. See Client Side Configuration for more details
LDAP_DEFAULTS.createNewUser
: Boolean value with a default of true
. This will create a new Meteor.user if the user has not yet been created with the entered ldap email/username.
LDAP_DFAULTS.defaultDomain
: Specify the email domain to be used when creating a new user on login. Defaults to false
- so if the user has entered xyz@site.com and defaultDomain
is not set, then their email will be saved as xyz@site.com.
LDAP_DEFAULTS.searchResultsProfileMap
: This can be used if there are attributes at your specified dn that you'd like to use to set properties when creating a new user's profile.
For example, if the results had a 'cn' value of the user's name and a 'tn' value of their phone number, you'd set the searchResultsProfileMap
to this:
LDAP_DEFAULTS.searchResultsProfileMap = [{
resultKey: 'cn',
profileProperty: 'name'
}, {
resultKey: 'tn',
profileProperty: 'phoneNumber'
}],
// This would create a user profile object that looks like this:
user.profile = {
name: 'Whatever the cn value was',
phoneNumber: 'Whatever the tn value was'
}
LDAP_DEFAULTS.base
: This is the base dn used for searches if the searchResultsProfileMap is set.
If you want to use ldaps
to implement secure authentication, you also need to provide an SSL certificate
(e.g. in the shape of a ssl.pem
file)
Simply set the following defaults in some server-side code:
LDAP_DEFAULTS.ldapsCertificate = Assets.getText('ldaps/ssl.pem'); // asset location of the SSL certificate
LDAP_DEFAULTS.port = 636; // default port for LDAPS
LDAP_DEFAULTS.url = 'ldaps://my-ldap-host.com'; // ldaps protocol
This example configuration will require the ssl.pem
file to be located in <your-project-root>/private/ldap/ssl.pem
.
The package exposes a new Meteor login method Meteor.loginWithLDAP()
which can be called from the client. The usual user and password are required. The third parameter is for custom LDAP options. You'll most likely want to pass in the customLdapOptions.dn on the options object.
An example login call might look like this:
Meteor.loginWithLDAP(username, password, {
// The dn value depends on what you want to search/auth against
// The structure will depend on how your ldap server
// is configured or structured.
dn: "uid=" + username + ",cn=users,dc=whatever,dc=valuesyouneed",
// The search value is optional. Set it if your search does not
// work with the bind dn.
search: "(objectclass=*)"
}, function(err) {
if (err)
console.log(err.reason);
});
In some scenarios, your login names might not directly correspond to the dn
of the LDAP record.
This requires to search for the dn
, before you can authenticate.
If you don't know the dn
, simply provide the attribute to search for to determine the dn
(e.g. searching by email):
Meteor.loginWithLDAP(email, password, {
// search by email (all other options are configured server side)
searchBeforeBind: {
mail: email
}
}, function(err) {
if (err) {
// login failed
}
else {
// login successful
}
}
);
In above example the LDAP record needs to have an attribute mail
which will be searched for. The first matching record
will be used to determine the dn
of the LDAP record. Then the binding will be executed using the provided password
and the dn
retrieved via the search.
If you provide the dn
as option for the loginWithLDAP
call, the search will NOT be executed. Also don't configure
the LDAP_DEFAULTS.dn
as it has the same effect.
- If your app requires that only LDAP authorized users should be able to login, it is strongly recommended that you include the following server-side code to prevent a user from gaining access through Accounts.createUser() on the client, which will otherwise automatically login a newly created (non-LDAP) user:
//on the server
Meteor.startup(function () {
Accounts.config({
forbidClientAccountCreation: true
});
});
- The LDAP dn is specific to your Active Directory. Talk to whoever manages it to figure out what would work best.
- Because the package binds/authenticates with LDAP server-side, the user/password are sent to the server unencrypted. I still need to figure out a solution for this.
- Right now Node throws a warning on meteor startup:
{ [Error: Cannot find module './build/Debug/DTraceProviderBindings'] code: 'MODULE_NOT_FOUND' }
because optional dependencies are missing. It doesn't seem to affect the ldapjs functionality, but I'm still trying to figure out how to squelch it. See this thread. As a workaround, you can re-install the included dtrace-provider NPM package:<project-root>/.meteor/local/build/programs/server/npm/typ_ldapjs/node_modules/ldapjs$ sudo npm install dtrace-provider
Using AD you can bind using domain\username. This example works for me:
//on the server
LDAP_DEFAULTS.base = 'OU=User,DC=your,DC=company,DC=com';
//on the client
var domain = "yourDomain";
Meteor.loginWithLDAP(user, password,
{ dn: domain + '\\' + user, search: '(sAMAccountName=' + user + ')' } , function(err, result) { ... }
);
Please feel free to fork and help improve the repo. I'm very unfamiliar with LDAP and built this package in a way that is probably really specific to my LDAP server configuration. I'm sure configurations vary for everyone, so any suggestions as to how I can make the package more agnostic are much appreciated.
TODO - need to figure out what features are missing and might make sense to add...