/** * Copyright (C) 2013 Open WhisperSystems * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Affero General Public License for more details. * * You should have received a copy of the GNU Affero General Public License * along with this program. If not, see <http://www.gnu.org/licenses/>. */ package org.whispersystems.textsecuregcm.auth; import com.codahale.metrics.Meter; import com.codahale.metrics.MetricRegistry; import com.codahale.metrics.SharedMetricRegistries; import com.google.common.base.Optional; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.whispersystems.textsecuregcm.storage.Account; import org.whispersystems.textsecuregcm.storage.AccountsManager; import org.whispersystems.textsecuregcm.storage.Device; import org.whispersystems.textsecuregcm.util.Constants; import static com.codahale.metrics.MetricRegistry.name; import io.dropwizard.auth.AuthenticationException; import io.dropwizard.auth.Authenticator; import io.dropwizard.auth.basic.BasicCredentials; public class AccountAuthenticator implements Authenticator<BasicCredentials, Account> { private final MetricRegistry metricRegistry = SharedMetricRegistries.getOrCreate(Constants.METRICS_NAME); private final Meter authenticationFailedMeter = metricRegistry.meter(name(getClass(), "authentication", "failed" )); private final Meter authenticationSucceededMeter = metricRegistry.meter(name(getClass(), "authentication", "succeeded")); private final Logger logger = LoggerFactory.getLogger(AccountAuthenticator.class); private final AccountsManager accountsManager; public AccountAuthenticator(AccountsManager accountsManager) { this.accountsManager = accountsManager; } @Override public Optional<Account> authenticate(BasicCredentials basicCredentials) throws AuthenticationException { try { AuthorizationHeader authorizationHeader = AuthorizationHeader.fromUserAndPassword(basicCredentials.getUsername(), basicCredentials.getPassword()); Optional<Account> account = accountsManager.get(authorizationHeader.getNumber()); if (!account.isPresent()) { return Optional.absent(); } Optional<Device> device = account.get().getDevice(authorizationHeader.getDeviceId()); if (!device.isPresent()) { return Optional.absent(); } if (device.get().getAuthenticationCredentials().verify(basicCredentials.getPassword())) { authenticationSucceededMeter.mark(); account.get().setAuthenticatedDevice(device.get()); return account; } authenticationFailedMeter.mark(); return Optional.absent(); } catch (InvalidAuthorizationHeaderException iahe) { return Optional.absent(); } } }