org.eolang.jeo.representation.BytecodeRepresentation Maven / Gradle / Ivy
/*
* The MIT License (MIT)
*
* Copyright (c) 2016-2023 Objectionary.com
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included
* in all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE.
*/
package org.eolang.jeo.representation;
import com.jcabi.xml.XML;
import com.jcabi.xml.XMLDocument;
import java.nio.file.Path;
import java.time.ZoneOffset;
import java.time.ZonedDateTime;
import java.time.format.DateTimeFormatter;
import java.util.Arrays;
import java.util.Objects;
import java.util.concurrent.atomic.AtomicReference;
import lombok.ToString;
import org.cactoos.Input;
import org.cactoos.bytes.BytesOf;
import org.cactoos.bytes.UncheckedBytes;
import org.cactoos.io.InputOf;
import org.eolang.jeo.Representation;
import org.objectweb.asm.ClassReader;
import org.objectweb.asm.ClassVisitor;
import org.objectweb.asm.Opcodes;
import org.xembly.Directives;
import org.xembly.ImpossibleModificationException;
import org.xembly.Xembler;
/**
* Intermediate representation of a class files which can be optimized from bytecode.
*
* @since 0.1.0
*/
@ToString
@SuppressWarnings("PMD.UseObjectForClearerAPI")
public final class BytecodeRepresentation implements Representation {
/**
* Input source.
*/
private final byte[] input;
/**
* Constructor.
* @param clazz Path to the class file
*/
public BytecodeRepresentation(final Path clazz) {
this(new InputOf(clazz));
}
/**
* Constructor.
* @param input Input source
*/
BytecodeRepresentation(final Input input) {
this(new UncheckedBytes(new BytesOf(input)).asBytes());
}
/**
* Constructor.
* @param input Input source.
*/
private BytecodeRepresentation(final byte[] input) {
this.input = Arrays.copyOf(input, input.length);
}
@Override
public String name() {
final ClassName name = new ClassName();
new ClassReader(this.input).accept(name, 0);
return name.asString();
}
@Override
public XML toEO() {
try {
final ClassPrinter printer = new ClassPrinter();
new ClassReader(this.input).accept(printer, 0);
final XMLDocument res = new XMLDocument(new Xembler(printer.directives).xml());
new Schema(res).check();
return res;
} catch (final ImpossibleModificationException exception) {
throw new IllegalStateException(
String.format("Can't build XML from %s", this.input),
exception
);
}
}
@Override
public byte[] toBytecode() {
return new UncheckedBytes(new BytesOf(this.input)).asBytes();
}
/**
* Class name.
*
* @since 0.1.0
*/
private class ClassName extends ClassVisitor {
/**
* Atomic reference to store class name.
*/
private final AtomicReference bag;
/**
* Constructor.
*/
ClassName() {
this(new AtomicReference<>());
}
/**
* Constructor.
* @param bag Atomic reference to store class name.
*/
ClassName(final AtomicReference bag) {
this(Opcodes.ASM9, bag);
}
/**
* Constructor.
* @param api ASM API version.
* @param bag Atomic reference to store class name.
*/
private ClassName(final int api, final AtomicReference bag) {
super(api);
this.bag = bag;
}
@Override
public void visit(
final int version,
final int access,
final String name,
final String signature,
final String supername,
final String[] interfaces
) {
this.bag.set(name.replace('/', '.'));
super.visit(version, access, name, signature, supername, interfaces);
}
/**
* Get class name.
* @return Class name.
*/
String asString() {
final String last = this.bag.get();
if (Objects.isNull(last)) {
throw new IllegalStateException(
"Class name is not set, bug is empty. Use #visit() method to set it."
);
}
return last;
}
}
/**
* Class printer.
* ASM class visitor which scans the class and builds Xembly directives.
* You can read more about Xembly right here:
* - https://github.com/yegor256/xembly
* - https://www.xembly.org
* Firther all this directives will be used to build XML representation of the class.
* @since 0.1
*/
private class ClassPrinter extends ClassVisitor {
/**
* Xembly directives.
*/
private final Directives directives;
/**
* Constructor.
*/
ClassPrinter() {
this(Opcodes.ASM9, new Directives());
}
/**
* Constructor.
* @param api ASM API version.
* @param directives Xembly directives.
*/
private ClassPrinter(
final int api,
final Directives directives
) {
super(api);
this.directives = directives;
}
@Override
public void visit(
final int version,
final int access,
final String name,
final String signature,
final String supername,
final String[] interfaces
) {
final String now = ZonedDateTime.now(ZoneOffset.UTC)
.format(DateTimeFormatter.ISO_INSTANT);
this.directives.add("program")
.attr("name", name.replace('/', '.'))
.attr("version", "0.0.0")
.attr("revision", "0.0.0")
.attr("dob", now)
.attr("time", now)
.add("listing")
.set(new Base64Bytecode(BytecodeRepresentation.this.input).asString())
.up()
.add("errors").up()
.add("sheets").up()
.add("license").up()
.add("metas").up()
.attr("ms", System.currentTimeMillis())
.add("objects")
.add("o");
super.visit(version, access, name, signature, supername, interfaces);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy