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

com.exactpro.sf.aml.iomatrix.AdvancedMatrixWriter Maven / Gradle / Ivy

There is a newer version: 3.4.260
Show newest version
/******************************************************************************
 * Copyright 2009-2022 Exactpro (Exactpro Systems Limited)
 *
 * 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.
 ******************************************************************************/
package com.exactpro.sf.aml.iomatrix;

import java.io.File;
import java.io.IOException;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import com.exactpro.sf.aml.generator.matrix.Column;
import com.exactpro.sf.aml.generator.matrix.JavaStatement;
import com.exactpro.sf.common.util.EPSCommonException;

public class AdvancedMatrixWriter implements AutoCloseable {

    private final Logger logger = LoggerFactory.getLogger(getClass());
    private static final String SYSTEM_FIELD_PREFIX = Column.getSystemPrefix();
    private static final String ACTION_FIELD = Column.Action.getName();
    private static final String DEFINE_HEADER = JavaStatement.DEFINE_HEADER.getValue();
    private final IMatrixWriter writer;
    private final List header;
    private final Map convertedHeaders;

    private boolean isHeaderWrited;
    private int rowNumber = 1;

    public static IMatrixWriter getWriter(File matrixPath) throws IOException {
        MatrixFileTypes matrixType = MatrixFileTypes.detectFileType(matrixPath.getName());
        switch (matrixType) {
        case XLS:
            return new ExcelMatrixWriter(matrixPath.getAbsolutePath(), false);
        case XLSX:
            return new ExcelMatrixWriter(matrixPath.getAbsolutePath(), true);
        case CSV:
            return new CSVMatrixWriter(matrixPath.getAbsolutePath());
        case JSON:
        case YAML:
            return new JSONMatrixWriter(matrixPath, matrixType);
        default:
            throw new IllegalStateException("Unknown matrix type: " + matrixType);
        }
    }

    public static IMatrixWriter getWriter(File matrixPath, String encoding) throws IOException {
        MatrixFileTypes matrixType = MatrixFileTypes.detectFileType(matrixPath.getName());
        switch (matrixType) {
        case XLS:
            return new ExcelMatrixWriter(matrixPath.getAbsolutePath(), false);
        case XLSX:
            return new ExcelMatrixWriter(matrixPath.getAbsolutePath(), true);
        case CSV:
            return new CSVMatrixWriter(matrixPath.getAbsolutePath(), encoding);
        default:
            throw new IllegalStateException("Unknown matrix type: " + matrixType);
        }
    }

    public AdvancedMatrixWriter(File file, List mainHeaders) throws IOException {
        writer = getWriter(file);
        header = new ArrayList<>(mainHeaders.size());
        header.addAll(mainHeaders);
        convertedHeaders = new HashMap(header.size());
        checkHeaders(header);
        rowNumber++;
    }

    @Override
    public void close() throws Exception {
        writer.close();
    }
    
    public void writeCells(Map line) throws IOException {

        if (!isHeaderWrited) {
            isHeaderWrited = true;
            writeHeaders();
        }

        int unknownFieldsCount = 0;

        for(Entry cell : line.entrySet()) {
            if (!convertedHeaders.containsKey(cell.getKey())) {
                unknownFieldsCount++;
            }
        }
        if (unknownFieldsCount > 0) {
            logger.warn("Detected {} unknown fields.", unknownFieldsCount);
        }

        SimpleCell[] cells = new SimpleCell[header.size() + unknownFieldsCount];
        int additionalIndx = header.size();

        for(Entry entry : line.entrySet()) {
            if (!convertedHeaders.containsKey(entry.getKey())) {
                convertedHeaders.put(entry.getKey(), additionalIndx);
                cells[additionalIndx++] = entry.getValue();
                header.add(new SimpleCell(entry.getKey()));
                continue;
            }
            int position = convertedHeaders.get(entry.getKey());
            cells[position] = entry.getValue();
        }

        if (unknownFieldsCount != 0) {
            writeDefineHeaderCommand(header);
        }

        writer.writeCells(cells);

        rowNumber++;
    }

    public void writeDefineHeader(List header) throws IOException {

        writeDefineHeader(header, false);
    }
    
    public void writeDefineHeader(List header, boolean onlyRedefineInternalHeader) throws IOException {
        
        SimpleCell[] tmp = header.toArray(new SimpleCell[] {});
        redefineHeaders(tmp);
        if (!onlyRedefineInternalHeader) {
            writeDefineHeaderCommand(this.header);
        } 
    }

    private void writeDefineHeaderCommand(List header) throws IOException {
        Map newHeader = new HashMap<>();

        for (SimpleCell sc : header) {
            newHeader.put(sc.getValue(), sc);
        }
        newHeader.put(ACTION_FIELD, new SimpleCell(DEFINE_HEADER));
        newHeader.remove(""); //if new header smaller then current, then after it's redefining we got empty cells, because we attempt write this.header. This leads to writing DefineHeader command again.
        writeCells(newHeader);
    }

    private void writeHeaders() throws IOException {
        writer.writeCells(header.toArray(new SimpleCell[] {}));
    }

    private void redefineHeaders(SimpleCell[] readedSimpleCells) throws IOException {

        int length = readedSimpleCells.length;
        int totalLength = Math.max(readedSimpleCells.length, header.size());

        if (header.size() < totalLength) {
            int count = totalLength - header.size();
            for (int n = 0; n < count; n++) {
                header.add(new SimpleCell());
            }
        }

        for (int i = 0; i < totalLength; i++) {
            SimpleCell cell = i < length ? readedSimpleCells[i] : new SimpleCell();
            // Insert system field
            if (i < header.size() && header.get(i).getValue().startsWith(SYSTEM_FIELD_PREFIX)) {
                // Keep system fields from previous header
                if (!header.get(i).getValue().equals(cell.getValue())) {
                    throw new EPSCommonException("Redefining error. Try redefine system field " + header.get(i).getValue() + " to "+ cell.getValue());
                }
            } else {
                header.set(i, cell);
            }
        }

        checkHeaders(header);
    }

    private boolean checkHeaders(List headers) throws IOException {

        SimpleCell defaultCell = new SimpleCell();
        int counter = -1;

        convertedHeaders.clear();

        for (SimpleCell item : headers) {
            counter++;
            // Skip commented fields

            if (item.getValue().startsWith("~")) {
                convertedHeaders.put(headers.get(counter).getValue(), counter);
                continue;
            }

            if (!item.equals(defaultCell) && convertedHeaders.put(headers.get(counter).getValue(), counter) != null) {

                List positions = new ArrayList<>();
                for (int j = 0; j < headers.size(); j++) {
                    if (headers.get(j).getValue().equals(item.getValue())) {
                        positions.add(j + 1);
                    }
                }

                StringBuilder letterPositions = new StringBuilder();

                for (int pos : positions) {
                    letterPositions.append(TableUtils.columnAdress(pos));
                    letterPositions.append(rowNumber);
                    letterPositions.append(", ");
                }

                throw new IOException("Invalid matrix structure. Detected duplicated fields at "
                        + letterPositions + " positions. Field name is " + headers.get(counter).getValue());
            }            
        }

        return true;
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy