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