McPringle/sportchef

View on GitHub
src/main/java/ch/sportchef/business/user/control/UserRepository.java

Summary

Maintainability
A
0 mins
Test Coverage
/*
 * SportChef – Sports Competition Management Software
 * Copyright (C) 2015, 2016 Marcus Fihlon
 *
 * 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 ch.sportchef.business.user.control;

import ch.sportchef.business.exception.ExpectationFailedException;
import ch.sportchef.business.user.entity.User;

import javax.validation.constraints.NotNull;
import java.io.Serializable;
import java.util.ConcurrentModificationException;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.atomic.AtomicLong;

import static java.util.Comparator.comparingLong;
import static java.util.stream.Collectors.toList;

class UserRepository implements Serializable {

    private static final long serialVersionUID = 1L;

    private final Map<Long, User> users = new ConcurrentHashMap<>();

    private final AtomicLong userSeq = new AtomicLong(0);

    User create(@NotNull final User user) {
        if (findByEmail(user.getEmail()).isPresent()) {
            throw new ExpectationFailedException("Email address has to be unique");
        }
        final Long userId = userSeq.incrementAndGet();
        final Long version = Long.valueOf(user.hashCode());
        final User userToCreate = user.toBuilder()
                .userId(userId)
                .version(version)
                .build();
        this.users.put(userId, userToCreate);
        return userToCreate;
    }

    User update(@NotNull final User user) {
        final User previousUser = users.getOrDefault(user.getUserId(), user);
        if (!previousUser.getVersion().equals(user.getVersion())) {
            throw new ConcurrentModificationException("You tried to update an user that was modified concurrently!");
        }
        final Long version = Long.valueOf(user.hashCode());
        final User userToUpdate = user.toBuilder()
                .version(version)
                .build();
        users.put(userToUpdate.getUserId(), userToUpdate);
        return userToUpdate;
    }

    Optional<User> findByUserId(@NotNull final Long userId) {
        return Optional.ofNullable(this.users.get(userId));
    }

    Optional<User> findByEmail(@NotNull final String email) {
        return this.users.values().stream()
                .filter(user -> email.equals(user.getEmail()))
                .findAny();
    }

    List<User> findAll() {
        return this.users.values().stream()
                .sorted(comparingLong(User::getUserId))
                .collect(toList());
    }

    void delete(final Long userId) {
        this.users.remove(userId);
    }
}