package eu.eudat.model.builder; import eu.eudat.authorization.AuthorizationFlags; import eu.eudat.commons.XmlHandlingService; import eu.eudat.convention.ConventionService; import eu.eudat.data.LockEntity; import eu.eudat.model.Lock; import eu.eudat.model.Tenant; import eu.eudat.model.User; import eu.eudat.query.TenantQuery; import eu.eudat.query.UserQuery; import gr.cite.tools.data.builder.BuilderFactory; import gr.cite.tools.data.query.QueryFactory; import gr.cite.tools.exception.MyApplicationException; import gr.cite.tools.fieldset.BaseFieldSet; import gr.cite.tools.fieldset.FieldSet; import gr.cite.tools.logging.DataLogEntry; import gr.cite.tools.logging.LoggerService; import org.slf4j.LoggerFactory; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.beans.factory.config.ConfigurableBeanFactory; import org.springframework.context.annotation.Scope; import org.springframework.stereotype.Component; import java.util.*; import java.util.stream.Collectors; @Component @Scope(value = ConfigurableBeanFactory.SCOPE_PROTOTYPE) public class LockBuilder extends BaseBuilder{ private final BuilderFactory builderFactory; private final QueryFactory queryFactory; private final XmlHandlingService xmlHandlingService; private EnumSet authorize = EnumSet.of(AuthorizationFlags.None); @Autowired public LockBuilder( ConventionService conventionService, BuilderFactory builderFactory, QueryFactory queryFactory, XmlHandlingService xmlHandlingService) { super(conventionService, new LoggerService(LoggerFactory.getLogger(LockBuilder.class))); this.builderFactory = builderFactory; this.queryFactory = queryFactory; this.xmlHandlingService = xmlHandlingService; } public LockBuilder authorize(EnumSet values) { this.authorize = values; return this; } @Override public List build(FieldSet fields, List data) throws MyApplicationException { this.logger.debug("building for {} items requesting {} fields", Optional.ofNullable(data).map(List::size).orElse(0), Optional.ofNullable(fields).map(FieldSet::getFields).map(Set::size).orElse(0)); this.logger.trace(new DataLogEntry("requested fields", fields)); if (fields == null || data == null || fields.isEmpty()) return new ArrayList<>(); FieldSet userFields = fields.extractPrefixed(this.asPrefix(Lock._lockedBy)); Map userMap = this.collectUsers(userFields, data); FieldSet tenantFields = fields.extractPrefixed(this.asPrefix(Lock._tenant)); Map tenantMap = this.collectTenants(tenantFields, data); List models = new ArrayList<>(); for (LockEntity d : data) { Lock m = new Lock(); if (fields.hasField(this.asIndexer(Lock._id))) m.setId(d.getId()); if (fields.hasField(this.asIndexer(Lock._target))) m.setTarget(d.getTarget()); if (fields.hasField(this.asIndexer(Lock._targetType))) m.setTargetType(d.getTargetType()); if (fields.hasField(this.asIndexer(Lock._lockedAt))) m.setLockedAt(d.getLockedAt()); if (fields.hasField(this.asIndexer(Lock._touchedAt))) m.setTouchedAt(d.getTouchedAt()); if (fields.hasField(this.asIndexer(Lock._hash))) m.setHash(this.hashValue(d.getTouchedAt())); if (!userFields.isEmpty() && userMap != null && userMap.containsKey(d.getLockedBy())) m.setLockedBy(userMap.get(d.getLockedBy())); if (!tenantFields.isEmpty() && tenantMap != null && tenantMap.containsKey(d.getTenantId())) m.setTenant(tenantMap.get(d.getTenantId())); models.add(m); } this.logger.debug("build {} items", Optional.of(models).map(List::size).orElse(0)); return models; } private Map collectUsers(FieldSet fields, List data) throws MyApplicationException { if (fields.isEmpty() || data.isEmpty()) return null; this.logger.debug("checking related - {}", User.class.getSimpleName()); Map itemMap; if (!fields.hasOtherField(this.asIndexer(User._id))) { itemMap = this.asEmpty( data.stream().map(LockEntity::getLockedBy).distinct().collect(Collectors.toList()), x -> { User item = new User(); item.setId(x); return item; }, User::getId); } else { FieldSet clone = new BaseFieldSet(fields.getFields()).ensure(User._id); UserQuery q = this.queryFactory.query(UserQuery.class).authorize(this.authorize).ids(data.stream().map(LockEntity::getLockedBy).distinct().collect(Collectors.toList())); itemMap = this.builderFactory.builder(UserBuilder.class).authorize(this.authorize).asForeignKey(q, clone, User::getId); } if (!fields.hasField(User._id)) { itemMap.forEach((id, item) -> { if (item != null) item.setId(null); }); } return itemMap; } private Map collectTenants(FieldSet fields, List datas) throws MyApplicationException { if (fields.isEmpty() || datas.isEmpty()) return null; this.logger.debug("checking related - {}", Tenant.class.getSimpleName()); Map itemMap = null; if (!fields.hasOtherField(this.asIndexer(Tenant._id))) { itemMap = this.asEmpty( datas.stream().map(x -> x.getTenantId()).distinct().collect(Collectors.toList()), x -> { Tenant item = new Tenant(); item.setId(x); return item; }, x -> x.getId()); } else { FieldSet clone = new BaseFieldSet(fields.getFields()).ensure(Tenant._id); TenantQuery q = this.queryFactory.query(TenantQuery.class).authorize(this.authorize).ids(datas.stream().map(x -> x.getTenantId()).distinct().collect(Collectors.toList())); itemMap = this.builderFactory.builder(TenantBuilder.class).authorize(this.authorize).asForeignKey(q, clone, x -> x.getId()); } if (!fields.hasField(Tenant._id)) { itemMap.values().stream().filter(x -> x != null).map(x -> { x.setId(null); return x; }).collect(Collectors.toList()); } return itemMap; } }