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

io.resys.hdes.client.spi.decision.ast.CommandMapper Maven / Gradle / Ivy

There is a newer version: 3.130.78
Show newest version
package io.resys.hdes.client.spi.decision.ast;

import java.math.BigDecimal;

/*-
 * #%L
 * hdes-client-api
 * %%
 * Copyright (C) 2020 - 2021 Copyright 2020 ReSys OÜ
 * %%
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 * 
 *      http://www.apache.org/licenses/LICENSE-2.0
 * 
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 * #L%
 */


import java.util.AbstractMap;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.stream.Collectors;

import org.apache.commons.lang3.StringUtils;

import io.resys.hdes.client.api.HdesClient.HdesTypesMapper;
import io.resys.hdes.client.api.ast.AstBody.AstBodyType;
import io.resys.hdes.client.api.ast.AstDecision;
import io.resys.hdes.client.api.ast.AstDecision.AstDecisionRow;
import io.resys.hdes.client.api.ast.AstDecision.ColumnExpressionType;
import io.resys.hdes.client.api.ast.AstDecision.HitPolicy;
import io.resys.hdes.client.api.ast.ImmutableAstDecision;
import io.resys.hdes.client.api.ast.ImmutableAstDecisionCell;
import io.resys.hdes.client.api.ast.ImmutableAstDecisionRow;
import io.resys.hdes.client.api.ast.ImmutableHeaders;
import io.resys.hdes.client.api.ast.TypeDef;
import io.resys.hdes.client.api.ast.TypeDef.Direction;
import io.resys.hdes.client.api.ast.TypeDef.ValueType;
import io.resys.hdes.client.api.exceptions.DecisionAstException;
import io.resys.hdes.client.api.programs.ExpressionProgram;
import io.resys.hdes.client.spi.util.HdesAssert;



public class CommandMapper {
  private final static List headerTypes = Collections.unmodifiableList(
      Arrays.asList(ValueType.STRING,  ValueType.BOOLEAN, ValueType.INTEGER, ValueType.LONG, ValueType.DECIMAL, ValueType.DATE, ValueType.DATE_TIME).stream()
      .map(v -> v.name()).collect(Collectors.toList()));
  
  private final static Map> headerExpressions = Collections.unmodifiableMap(Map.of(
      ValueType.INTEGER, Collections.unmodifiableList(Arrays.asList(ColumnExpressionType.EQUALS.name())),
      ValueType.DECIMAL, Collections.unmodifiableList(Arrays.asList(ColumnExpressionType.EQUALS.name())),
      ValueType.STRING, Collections.unmodifiableList(Arrays.asList(ColumnExpressionType.IN.name()))    
  ));
  private final static List dynamocValueExpressions = Collections.unmodifiableList(Arrays.asList("<=", "<",">=", ">", "="));

  private static Object parseVariable(String expression, ValueType type) {
    Optional comparison = dynamocValueExpressions.stream().filter(v -> expression.startsWith(v)).findFirst();
    if(!comparison.isPresent()) {
      switch(type) {
      case DECIMAL:
        return BigDecimal.ZERO;
      case LONG:
        return 0;
      case INTEGER:
        return 0;
      default: return null;
      }
    }
    String value = expression.substring(comparison.get().length()).trim();
    switch(type) {
    case DECIMAL:
      return new BigDecimal(value);
    case LONG:
      return Long.parseLong(value);
    case INTEGER:
      return Integer.parseInt(value);
    default: return null;
    }
  }
  
  public static Builder builder(HdesTypesMapper dataTypeFactory) {
    return new Builder(dataTypeFactory);
  }

  public static class Builder {
    private final HdesTypesMapper typeDefs; 
    private long idGen = 0;
    private String name;
    private String description;
    private HitPolicy hitPolicy;
    
    private final Map headers = new HashMap<>();
    private final Map cells = new HashMap<>();
    private final Map rows = new HashMap<>();

    public Builder(HdesTypesMapper dataTypeFactory) {
      super();
      this.typeDefs = dataTypeFactory;
    }
    
    private String nextId() {
      return String.valueOf(idGen++);
    }
    private MutableHeader getHeader(String id) {
      HdesAssert.isTrue(headers.containsKey(id), () -> "no header with id: " + id + "!");
      return headers.get(id);
    }
    private MutableCell getCell(String id) {
      HdesAssert.isTrue(cells.containsKey(id), () -> "no cell with id: " + id + "!");
      return cells.get(id);
    }
    private MutableRow getRow(String id) {
      HdesAssert.isTrue(rows.containsKey(id), () -> "no row with id: " + id + "!");
      return rows.get(id);
    }
    private ValueType getValueType(MutableHeader header) {
      return header.getValue();
    }
    public Builder name(String name) {
      this.name = name;
      return this;
    }
    public Builder description(String description) {
      this.description = description;
      return this;
    }
    public Builder hitPolicy(HitPolicy hitPolicy) {
      this.hitPolicy = hitPolicy;
      return this;
    }
    public Map.Entry addHeader(Direction direction, String name) {
      MutableHeader header = new MutableHeader(nextId(), direction, headers.size())
          .setName(name)
          .setValue(ValueType.STRING);
      
      headers.put(header.getId(), header);
      this.rows.values().forEach(row -> {
        MutableCell cell = new MutableCell(nextId(), row.getId());
        header.getCells().add(cell);
        cells.put(cell.getId(), cell);
      });
      return new AbstractMap.SimpleImmutableEntry(header.getId(), this);
    }
    public Builder changeHeaderType(String id, String value) {
      try {
        getHeader(id).setValue(ValueType.valueOf(value));
      } catch(Exception e) {
        getHeader(id).setValue(null);
      }
      
      return this;
    }
    public Builder changeHeaderScript(String id, String value) {
      getHeader(id).setScript(value);
      return this;
    }
    public Builder changeHeaderName(String id, String value) {
      getHeader(id).setName(value);
      return this;
    }
    public Builder changeHeaderDirection(String id, Direction value) {
      MutableHeader header = getHeader(id).setDirection(value);
      ValueType valueType = getValueType(header);

      // Remove expression if cell new direction is out
      if(value == Direction.OUT && valueType != null) {
        header.cells.stream()
        .filter(c -> !StringUtils.isEmpty(c.getValue()))
        .forEach(cell -> {

          try {
            ExpressionProgram expression = typeDefs.expression(valueType, cell.getValue());
            if(expression.getConstants().size() == 1) {
              cell.setValue(expression.getConstants().get(0));
            }
          } catch(DecisionAstException e) {
            cell.setValue(cell.getValue());
          }
        });
      }

      return this;
    }

    private String getExpression(ValueType valueType, ColumnExpressionType value, String columnValue) {
      String constant;
      try {
        ExpressionProgram expression = typeDefs.expression(valueType, columnValue);
        if(expression.getConstants().size() != 1) {
          return null;
        }
        constant = expression.getConstants().get(0);
      } catch(DecisionAstException e) {
        constant = columnValue.trim();
      }
      switch (value) {
      case EQUALS:
        return "= " + constant;
      case IN:
        return "in[\"" + constant + "\"]";
      default:
        return null;
      }
    }

    public Builder setHeaderExpression(String id, ColumnExpressionType value) {
      MutableHeader header = getHeader(id);
      ValueType valueType = getValueType(header);

      if(header.getDirection() == Direction.IN && valueType != null) {
        header.cells.stream()
        .filter(c -> !StringUtils.isEmpty(c.getValue()))
        .forEach(cell -> {
          String operation = getExpression(valueType, value, cell.getValue());
          if(operation != null) {
            cell.setValue(operation);
          }
        });
      }

      return this;
    }
    public Builder changeCell(String id, String value) {
      getCell(id).setValue(value);
      return this;
    }
    public Builder changeCell(String rowId, int columnIndex, String value) {
      MutableHeader column = headers.values().stream().filter(r -> r.order == columnIndex).findFirst().get();
      MutableCell cell = column.getCells().stream().filter(c -> c.getRow().equals(rowId)).findFirst().get();
      cell.setValue(value);
      return this;
    }
    public Builder deleteCell(String id) {
      getCell(id).setValue(null);
      return this;
    }
    public Builder deleteHeader(String id) {
      getHeader(id).getCells().forEach(c -> cells.remove(c.getId()));
      headers.remove(id);
      return this;
    }
    public Map.Entry addRow() {
      MutableRow row = new MutableRow(nextId(), rows.size());
      rows.put(row.getId(), row);
      
      headers.values().forEach(h -> {
        MutableCell cell = new MutableCell(nextId(), row.getId());
        h.getCells().add(cell);
        cells.put(cell.getId(), cell);
      });
      
      return new AbstractMap.SimpleImmutableEntry(row.getId(), this);
    }
    public Builder deleteRow(String id) {
      MutableRow row = getRow(id);
      rows.remove(row.getId());
      int order = row.getOrder();
      rows.values().stream()
      .filter(r -> r.getOrder() > order)
      .forEach(r -> r.setOrder(r.getOrder() - 1));
      
      
      headers.values().forEach(h -> {
        Iterator cell = h.getCells().iterator();
        while(cell.hasNext()) {
          if(id.equals(cell.next().getRow())) {
            cell.remove();
          }
        }
      });
      
      return this;
    }
    public Builder deleteRows() {
      new ArrayList<>(rows.keySet()).forEach(id -> deleteRow(id));
      return this;
    }
    public Builder deleteColumns() {
      new ArrayList<>(headers.keySet()).forEach(id -> deleteHeader(id));
      return this;
    }
    public Builder moveRow(String srcId, String targetId) {
      MutableRow src = getRow(srcId);
      MutableRow target = getRow(targetId);

      int targetOrder = src.getOrder();
      int srcOrder = target.getOrder();
      src.setOrder(srcOrder);
      target.setOrder(targetOrder);
      return this;
    }
    public Builder insertRow(String srcId, String targetId) {
      MutableRow src = getRow(srcId);
      MutableRow target = getRow(targetId);

      // move row from back to front
      if(src.getOrder() > target.getOrder()) {
        int start = target.getOrder();
        int end = src.getOrder();
        
        for(MutableRow row : this.rows.values()) {
          if(row.getOrder() >= start && row.getOrder() < end) {
            row.setOrder(row.getOrder() + 1);
          }
        }
        src.setOrder(start);
      } else {
        // move row from front to back
        int start = src.getOrder();
        int end = target.getOrder();
        
        for(MutableRow row : this.rows.values()) {
          if(row.getOrder() > start && row.getOrder() <= end) {
            row.setOrder(row.getOrder() - 1);
          }
        }
        
        src.setOrder(end);
      }
      return this;
    }
    
    public Builder copyRow(String srcId) {
      MutableRow src = getRow(srcId);
      String targetId = addRow().getKey();
      
      for(MutableHeader header : this.headers.values()) {
        MutableCell from = header.getCells().stream().filter(c -> c.getRow().equals(src.getId())).findFirst().get();
        MutableCell to = header.getCells().stream().filter(c -> c.getRow().equals(targetId)).findFirst().get();
        to.setValue(from.getValue());
      }
      
      return insertRow(targetId, srcId);
    }

    public Builder moveHeader(String srcId, String targetId) {
      MutableHeader src = getHeader(srcId);
      MutableHeader target = getHeader(targetId);

      int targetOrder = src.getOrder();
      int srcOrder = target.getOrder();

      Direction targetDirection = src.getDirection();
      Direction srcDirection = target.getDirection();

      src.setOrder(srcOrder).setDirection(srcDirection);
      target.setOrder(targetOrder).setDirection(targetDirection);
      return this;
    }

    private String resolveScriptValue(MutableHeader header, MutableCell cell) {
      Map context = new HashMap<>();
      for(MutableHeader h : headers.values()) {
        MutableCell value = h.getCells().stream()
            .filter(c -> c.getRow().equals(cell.getRow()))
            .findFirst().get();
        try {
          Object variable = parseVariable(value.getValue(), h.getValue());
          context.put(h.getName(), variable);
        } catch(Exception e) {
        }
      }
      
      try {
        return typeDefs.expression(ValueType.MAP, header.getScript()).run(context) + "";
      } catch(Exception e) {
        return null;
      }
    }

    public AstDecision build() {
      this.headers.values().stream()
      .filter(h -> !StringUtils.isEmpty(h.getScript()))
      .forEach(h -> h.getCells().forEach(c -> c.setValue(resolveScriptValue(h, c))));
      
      List headers = this.headers.values().stream().sorted()
          .map(h -> (TypeDef) typeDefs.dataType()
              .direction(h.getDirection())
              .name(h.getName())
              .valueType(h.getValue())
              .id(h.getId())
              .order(h.getOrder())
              .script(h.getScript())
              .build())
          .collect(Collectors.toList());

      
      List rows = this.rows.values().stream().sorted()
          .map(r -> ImmutableAstDecisionRow.builder()
            .id(r.getId())
            .order(r.getOrder())
            .cells(this.headers.values().stream().sorted()
                .map(h -> {
                  MutableCell c = h.getRowCell(r.getId());
                  return ImmutableAstDecisionCell.builder().id(c.getId()).value(c.getValue()).header(h.getId()).build();
                })
                .collect(Collectors.toList()))
            .build()
          )
          .collect(Collectors.toList());
      
      final HitPolicy hitPolicy = this.hitPolicy == null ? HitPolicy.ALL : this.hitPolicy;
      return ImmutableAstDecision.builder()
          .name(name)
          .bodyType(AstBodyType.DT)
          .description(description)
          .hitPolicy(hitPolicy)
          .headerTypes(headerTypes)
          .headerExpressions(headerExpressions)
          .headers(ImmutableHeaders.builder()
              .acceptDefs(headers.stream().filter(p -> p.getDirection() == Direction.IN).collect(Collectors.toList()))
              .returnDefs(headers.stream().filter(p -> p.getDirection() == Direction.OUT).collect(Collectors.toList()))
              .build())
          .rows(rows)
          .build();
    }
  }

  

  private static class MutableHeader implements Comparable {

    private final String id;
    private Direction direction;

    private String script;
    private String name;
    private ValueType value;
    private int order;
    private final List cells = new ArrayList<>();

    public MutableHeader(String id, Direction direction, int order) {
      super();
      this.id = id;
      this.direction = direction;
      this.order = order;
    }
    public String getScript() {
      return script;
    }
    public void setScript(String script) {
      this.script = script;
    }
    public String getName() {
      return name;
    }
    public MutableHeader setName(String name) {
      this.name = name;
      return this;
    }
    public ValueType getValue() {
      return value;
    }
    public MutableHeader setValue(ValueType value) {
      this.value = value;
      return this;
    }
    public int getOrder() {
      return order;
    }
    public MutableHeader setOrder(int order) {
      this.order = order;
      return this;
    }
    public List getCells() {
      return cells;
    }
    public MutableCell getRowCell(String rowId) {
      return cells.stream().filter(c -> c.getRow().equals(rowId)).findFirst().get();
    }
    public String getId() {
      return id;
    }
    public Direction getDirection() {
      return direction;
    }
    @Override
    public int compareTo(MutableHeader o) {
      int d0 = direction == Direction.IN ? 0 : 1;
      int d1 = o.getDirection() == Direction.IN ? 0 : 1;

      int direction = Integer.compare(d0, d1);
      if(direction == 0) {
        return Integer.compare(order, o.order);
      }
      return direction;
    }
    public MutableHeader setDirection(Direction direction) {
      this.direction = direction;
      return this;
    }
  }

  private static class MutableCell {
    private final String id;
    private final String row;
    private String value;

    public MutableCell(String id, String row) {
      super();
      this.id = id;
      this.row = row;
    }
    public String getValue() {
      return value;
    }
    public void setValue(String value) {
      this.value = value;
    }
    public String getRow() {
      return row;
    }
    public String getId() {
      return id;
    }
  }
  private static class MutableRow implements Comparable {
    private final String id;
    private int order;
    public MutableRow(String id, int order) {
      super();
      this.id = id;
      this.order = order;
    }
    public int getOrder() {
      return order;
    }
    public void setOrder(int order) {
      this.order = order;
    }
    public String getId() {
      return id;
    }
    @Override
    public int compareTo(MutableRow o) {
      return Integer.compare(order, o.order);
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy