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

com.fluxtion.agrona.generation.SpecialisationGenerator Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 2014-2024 Real Logic 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
 *
 * https://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.fluxtion.agrona.generation;

import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.Collections;
import java.util.List;

import static java.nio.charset.StandardCharsets.UTF_8;
import static java.util.stream.Collectors.toList;

/**
 * Specialise classes written for primitive type int for other primitive types by substitution.
 */
public final class SpecialisationGenerator
{
    private static final String COLLECTIONS_PACKAGE = "com/fluxtion/agrona/collections";
    private static final String SRC_DIR = "src/main/java/";
    private static final String DST_DIR = "build/generated-src";
    private static final String SUFFIX = ".java";

    private static final List SUBSTITUTIONS = Collections.singletonList(
        new Substitution("long", "Long", "Long"));

    /**
     * Main method.
     *
     * @param args command line args.
     * @throws IOException in case of I/O error.
     */
    public static void main(final String[] args) throws IOException
    {
        specialise(SUBSTITUTIONS, COLLECTIONS_PACKAGE, "IntIntConsumer", SRC_DIR, DST_DIR);
        specialise(SUBSTITUTIONS, COLLECTIONS_PACKAGE, "IntIntFunction", SRC_DIR, DST_DIR);
        specialise(SUBSTITUTIONS, COLLECTIONS_PACKAGE, "IntIntPredicate", SRC_DIR, DST_DIR);
        specialise(SUBSTITUTIONS, COLLECTIONS_PACKAGE, "IntObjConsumer", SRC_DIR, DST_DIR);
        specialise(SUBSTITUTIONS, COLLECTIONS_PACKAGE, "IntObjPredicate", SRC_DIR, DST_DIR);
        specialise(SUBSTITUTIONS, COLLECTIONS_PACKAGE, "IntObjectToObjectFunction", SRC_DIR, DST_DIR);
        specialise(SUBSTITUTIONS, COLLECTIONS_PACKAGE, "ObjectIntToIntFunction", SRC_DIR, DST_DIR);
        specialise(SUBSTITUTIONS, COLLECTIONS_PACKAGE, "ObjIntConsumer", SRC_DIR, DST_DIR);
        specialise(SUBSTITUTIONS, COLLECTIONS_PACKAGE, "ObjIntPredicate", SRC_DIR, DST_DIR);
        specialise(SUBSTITUTIONS, COLLECTIONS_PACKAGE, "IntArrayList", SRC_DIR, DST_DIR);
        specialise(SUBSTITUTIONS, COLLECTIONS_PACKAGE, "IntArrayQueue", SRC_DIR, DST_DIR);
        specialise(SUBSTITUTIONS, COLLECTIONS_PACKAGE, "Int2IntHashMap", SRC_DIR, DST_DIR);
        specialise(SUBSTITUTIONS, COLLECTIONS_PACKAGE, "Int2IntCounterMap", SRC_DIR, DST_DIR);
        specialise(SUBSTITUTIONS, COLLECTIONS_PACKAGE, "IntHashSet", SRC_DIR, DST_DIR);
        specialise(SUBSTITUTIONS, COLLECTIONS_PACKAGE, "IntLruCache", SRC_DIR, DST_DIR);
        specialise(SUBSTITUTIONS, COLLECTIONS_PACKAGE, "Int2ObjectCache", SRC_DIR, DST_DIR);
        specialise(SUBSTITUTIONS, COLLECTIONS_PACKAGE, "Int2ObjectHashMap", SRC_DIR, DST_DIR);
        specialise(SUBSTITUTIONS, COLLECTIONS_PACKAGE, "Int2NullableObjectHashMap", SRC_DIR, DST_DIR);
        specialise(SUBSTITUTIONS, COLLECTIONS_PACKAGE, "Object2IntHashMap", SRC_DIR, DST_DIR);
        specialise(SUBSTITUTIONS, COLLECTIONS_PACKAGE, "Object2IntCounterMap", SRC_DIR, DST_DIR);
    }

    /**
     * Specialise a class replacing int types based on {@link Substitution}s.
     *
     * @param substitutions to be applied.
     * @param packageName   for the source and destination classes.
     * @param srcClassName  to be specialised.
     * @param srcDirName    containing the source file.
     * @param dstDirName    for where the generated file should be stored.
     * @throws IOException if an error occurs.
     */
    public static void specialise(
        final List substitutions,
        final String packageName,
        final String srcClassName,
        final String srcDirName,
        final String dstDirName)
        throws IOException
    {
        final Path inputPath = Paths.get(srcDirName, packageName, srcClassName + SUFFIX);
        final Path outputDirectory = Paths.get(dstDirName, packageName);
        Files.createDirectories(outputDirectory);

        final List contents = Files.readAllLines(inputPath, UTF_8);
        for (final Substitution substitution : substitutions)
        {
            final String substitutedFileName = substitution.substitute(srcClassName);
            final List substitutedContents = contents
                .stream()
                .map(substitution::conditionalSubstitute)
                .collect(toList());

            final Path outputPath = Paths.get(dstDirName, packageName, substitutedFileName + SUFFIX);

            Files.write(outputPath, substitutedContents, UTF_8);
        }
    }

    /**
     * Substitution to be performed on each code line. Lines with {@link DoNotSub} are ignored.
     */
    public static final class Substitution
    {
        private final String primitiveType;
        private final String boxedType;
        private final String className;

        private Substitution(final String primitiveType, final String boxedType, final String className)
        {
            this.primitiveType = primitiveType;
            this.boxedType = boxedType;
            this.className = className;
        }

        /**
         * Perform code substitutions.
         *
         * @param contents original source code.
         * @return modified source code.
         */
        public String substitute(final String contents)
        {
            return contents
                .replace("int", primitiveType)
                .replace("Integer", boxedType)
                .replace("Int", className);
        }

        /**
         * Perform conditional code substitutions, i.e. only if not disabled.
         *
         * @param contents original source code.
         * @return modified source code.
         */
        public String conditionalSubstitute(final String contents)
        {
            return
                (contents.contains("@DoNotSub") || contents.contains("interface") || contents.contains("Interface")) ?
                    contents : substitute(contents);
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy