/*
* Copyright 2014 Red Hat, Inc.
*
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License v1.0
* and Apache License v2.0 which accompanies this distribution.
*
* The Eclipse Public License is available at
* http://www.eclipse.org/legal/epl-v10.html
*
* The Apache License v2.0 is available at
* http://www.opensource.org/licenses/apache2.0.php
*
* You may elect to redistribute this code under either of these licenses.
*/
package io.vertx.ext.auth.htpasswd.impl;
import java.util.HashMap;
import java.util.Map;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import io.vertx.core.AsyncResult;
import io.vertx.core.Future;
import io.vertx.core.Handler;
import io.vertx.core.Vertx;
import io.vertx.core.json.JsonObject;
import io.vertx.ext.auth.HashingStrategy;
import io.vertx.ext.auth.User;
import io.vertx.ext.auth.authentication.CredentialValidationException;
import io.vertx.ext.auth.authentication.Credentials;
import io.vertx.ext.auth.authentication.UsernamePasswordCredentials;
import io.vertx.ext.auth.htpasswd.HtpasswdAuth;
import io.vertx.ext.auth.htpasswd.HtpasswdAuthOptions;
import io.vertx.ext.auth.htpasswd.impl.hash.Plaintext;
import io.vertx.ext.auth.impl.UserImpl;
An implementation of HtpasswdAuth
Author: Neven Radovanović
/**
* An implementation of {@link HtpasswdAuth}
*
* @author Neven Radovanović
*/
public class HtpasswdAuthImpl implements HtpasswdAuth {
private final HashingStrategy strategy = HashingStrategy.load();
private final Map<String, String> htUsers = new HashMap<>();
public HtpasswdAuthImpl(Vertx vertx, HtpasswdAuthOptions options) {
for (String line : vertx.fileSystem().readFileBlocking(options.getHtpasswdFile()).toString().split("\\r?\\n")) {
line = line.trim();
if (line.isEmpty() || line.startsWith("#")) continue;
Pattern entry = Pattern.compile("^([^:]+):(.+)");
Matcher m = entry.matcher(line);
if (m.matches()) {
htUsers.put(m.group(1), m.group(2));
}
}
// handle the plain text vs crypt
if (options.isPlainTextEnabled()) {
// this will show a warning in the log
strategy.put("", new Plaintext());
}
}
@Override
public void authenticate(JsonObject authInfo, Handler<AsyncResult<User>> resultHandler) {
authenticate(new UsernamePasswordCredentials(authInfo), resultHandler);
}
@Override
public void authenticate(Credentials credential, Handler<AsyncResult<User>> resultHandler) {
try {
UsernamePasswordCredentials authInfo = (UsernamePasswordCredentials) credential;
authInfo.checkValid(null);
if (!htUsers.containsKey(authInfo.getUsername())) {
resultHandler.handle((Future.failedFuture("Unknown username.")));
return;
}
if (strategy.verify(htUsers.get(authInfo.getUsername()), authInfo.getPassword())) {
resultHandler.handle(Future.succeededFuture(new UserImpl(new JsonObject().put("username", authInfo.getUsername()))));
} else {
resultHandler.handle(Future.failedFuture("Bad response"));
}
} catch (ClassCastException | CredentialValidationException e) {
resultHandler.handle(Future.failedFuture(e));
}
}
}