net.identio.server.service.transaction.TransactionService.java Source code

Java tutorial

Introduction

Here is the source code for net.identio.server.service.transaction.TransactionService.java

Source

/*
 * This file is part of Ident.io.
 *
 * Ident.io - A flexible authentication server
 * Copyright (c) 2017 Loeiz TANGUY
 *
 * 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 net.identio.server.service.transaction;

import com.google.common.cache.CacheBuilder;
import com.google.common.cache.CacheLoader;
import com.google.common.cache.LoadingCache;
import net.identio.server.service.orchestration.exceptions.WebSecurityException;
import net.identio.server.service.transaction.model.TransactionData;
import net.identio.server.utils.SecurityUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.context.annotation.Scope;
import org.springframework.stereotype.Service;

import javax.annotation.Nonnull;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.TimeUnit;

@Service
@Scope("singleton")
public class TransactionService {

    private static final Logger LOG = LoggerFactory.getLogger(TransactionService.class);
    private static final int TRANSACTION_IDENTIFIER_LENGTH = 100;

    private LoadingCache<String, TransactionData> transactionCache;

    public TransactionService() {

        transactionCache = CacheBuilder.newBuilder().maximumSize(100000).expireAfterAccess(10, TimeUnit.MINUTES)
                .build(new CacheLoader<String, TransactionData>() {
                    public TransactionData load(@Nonnull String o) {
                        return new TransactionData();
                    }
                });
    }

    public TransactionData createTransaction() {

        LOG.debug("Generating new transaction datas");

        String transactionId = SecurityUtils.generateSecureIdentifier(TRANSACTION_IDENTIFIER_LENGTH);

        TransactionData data = new TransactionData();
        data.setTransactionId(transactionId);
        transactionCache.put(transactionId, data);

        LOG.debug("New transaction generated {}", transactionId);

        return data;
    }

    public void removeTransactionData(TransactionData transactionData) {

        LOG.debug("Destroyed transaction {}", transactionData.getTransactionId());
        transactionCache.invalidate(transactionData.getTransactionId());
    }

    public TransactionData fetchTransaction(String transactionId) {

        if (transactionId == null) {
            return new TransactionData();
        }

        LOG.debug("Fetch transaction {} in cache", transactionId);

        try {
            return transactionCache.get(transactionId);
        } catch (ExecutionException e) {
            return new TransactionData();
        }
    }

    public TransactionData getTransaction(String sessionId, String transactionId) throws WebSecurityException {

        LOG.debug("Security verification of coherence between transaction ID and session ID");

        TransactionData transactionData = fetchTransaction(transactionId);

        if (transactionData.getTransactionId() == null) {
            LOG.error("Could not find a valid transaction");
            throw new WebSecurityException("invalid.transaction");
        }

        if (!sessionId.equals(transactionData.getUserSession().getId())) {
            removeTransactionData(transactionData);

            LOG.error(
                    "Session ID in transaction doesn't match browser session ID. Possible session fixation attack ?");
            throw new WebSecurityException("invalid.transaction");
        }

        LOG.debug("Security verification OK");

        return transactionData;
    }

}