All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.hyperfoil.tools.horreum.svc.ConfigServiceImpl Maven / Gradle / Ivy

package io.hyperfoil.tools.horreum.svc;

import java.util.Collections;
import java.util.List;
import java.util.Optional;
import java.util.Set;
import java.util.stream.Collectors;

import jakarta.annotation.security.PermitAll;
import jakarta.annotation.security.RolesAllowed;
import jakarta.enterprise.context.ApplicationScoped;
import jakarta.inject.Inject;
import jakarta.persistence.EntityManager;
import jakarta.transaction.Transactional;

import org.eclipse.microprofile.config.ConfigProvider;
import org.eclipse.microprofile.config.inject.ConfigProperty;
import org.jboss.logging.Logger;

import io.hyperfoil.tools.horreum.api.Version;
import io.hyperfoil.tools.horreum.api.data.Access;
import io.hyperfoil.tools.horreum.api.data.datastore.Datastore;
import io.hyperfoil.tools.horreum.api.services.ConfigService;
import io.hyperfoil.tools.horreum.datastore.BackendResolver;
import io.hyperfoil.tools.horreum.entity.backend.DatastoreConfigDAO;
import io.hyperfoil.tools.horreum.mapper.DatasourceMapper;
import io.hyperfoil.tools.horreum.server.WithRoles;
import io.quarkus.security.identity.SecurityIdentity;

@ApplicationScoped
public class ConfigServiceImpl implements ConfigService {

    private static final Logger log = Logger.getLogger(ConfigServiceImpl.class);

    @ConfigProperty(name = "horreum.privacy")
    Optional privacyStatement;

    @Inject
    SecurityIdentity identity;

    @Inject
    EntityManager em;

    @Inject
    BackendResolver backendResolver;

    @Override
    public KeycloakConfig keycloak() {
        KeycloakConfig config = new KeycloakConfig();
        config.url = getString("horreum.keycloak.url");
        config.realm = getString("horreum.keycloak.realm");
        config.clientId = getString("horreum.keycloak.clientId");
        return config;
    }

    @Override
    public VersionInfo version() {
        VersionInfo info = new VersionInfo();
        info.version = Version.getVersion();
        info.startTimestamp = startTimestamp;
        info.privacyStatement = privacyStatement.orElse(null);
        return info;
    }

    @Override
    @PermitAll
    @Transactional
    public List datastores(String team) {
        String queryWhere = "where access = 0";
        Set roles = identity.getRoles();
        long rolesCount = roles.stream().filter(role -> role.endsWith("-team")).count();
        if (rolesCount != 0) { //user has access to team, retrieve the team datastore as well
            queryWhere = queryWhere.concat(" or owner in ('" + team + "')");
        }
        List backends = DatastoreConfigDAO.list(queryWhere);
        if (backends.size() != 0) {
            return backends.stream().map(DatasourceMapper::from).collect(Collectors.toList());
        } else {
            return Collections.emptyList();
        }
    }

    @Override
    @RolesAllowed(Roles.TESTER)
    @WithRoles
    @Transactional
    public Integer newDatastore(Datastore datastore) {
        DatastoreConfigDAO dao = DatasourceMapper.to(datastore);
        dao.id = null;

        if (dao.owner == null) {
            List uploaders = identity.getRoles().stream().filter(role -> role.endsWith("-uploader"))
                    .collect(Collectors.toList());
            if (uploaders.size() != 1) {
                log.debugf("Failed to create datastore %s: no owner, available uploaders: %s", dao.name, uploaders);
                throw ServiceException.badRequest(
                        "Missing owner and cannot select single default owners; this user has these uploader roles: "
                                + uploaders);
            }
            String uploader = uploaders.get(0);
            dao.owner = uploader.substring(0, uploader.length() - 9) + "-team";
        } else if (!identity.getRoles().contains(dao.owner)) {
            log.debugf("Failed to create datastore %s: requested owner %s, available roles: %s", dao.name, dao.owner,
                    identity.getRoles());
            throw ServiceException.badRequest("This user does not have permissions to upload datastore for owner=" + dao.owner);
        }
        if (dao.access == null) {
            dao.access = Access.PRIVATE;
        }

        io.hyperfoil.tools.horreum.datastore.Datastore datastoreImpl;
        try {
            datastoreImpl = backendResolver.getBackend(datastore.type);
        } catch (IllegalStateException e) {
            throw ServiceException.badRequest("Unknown datastore type: " + datastore.type
                    + ". Please try again, if the problem persists please contact the system administrator.");
        }

        if (datastoreImpl == null) {
            throw ServiceException.badRequest("Unknown datastore type: " + datastore.type);
        }

        String error = datastoreImpl.validateConfig(datastore.config);

        if (error != null) {
            throw ServiceException.badRequest(error);
        }

        log.debugf("Creating new Datastore with owner=%s and access=%s", dao.owner, dao.access);

        try {
            em.persist(dao);
            em.flush();
        } catch (Exception e) {
            log.error("Failed to persist run.", e);
            throw ServiceException.serverError("Failed to persist backend configuration");
        }
        log.debugf("Upload flushed, backendConfig ID %d", dao.id);

        return dao.id;
    }

    @Override
    @RolesAllowed(Roles.TESTER)
    @WithRoles
    @Transactional
    public Integer updateDatastore(Datastore backend) {
        DatastoreConfigDAO dao = DatastoreConfigDAO.findById(backend.id);
        if (dao == null)
            throw ServiceException.notFound("Datastore with id " + backend.id + " does not exist");

        DatastoreConfigDAO newDao = DatasourceMapper.to(backend);

        dao.type = newDao.type;
        dao.name = newDao.name;
        dao.configuration = newDao.configuration;
        dao.access = newDao.access;

        dao.persist();

        return dao.id;

    }

    @Override
    @RolesAllowed(Roles.TESTER)
    @WithRoles
    @Transactional
    public DatastoreTestResponse testDatastore(String datastoreId) {
        return null;
    }

    @Override
    @RolesAllowed(Roles.TESTER)
    @WithRoles
    @Transactional
    public void deleteDatastore(String datastoreId) {
        DatastoreConfigDAO.deleteById(Integer.parseInt(datastoreId));
    }

    private String getString(String propertyName) {
        return ConfigProvider.getConfig().getOptionalValue(propertyName, String.class).orElse("");
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy