rt/vmtest/src/main/java/org/apidesign/bck2brwsr/vmtest/impl/CompareCase.java
author Jaroslav Tulach <jaroslav.tulach@apidesign.org>
Tue, 26 Jan 2016 04:36:23 +0100
changeset 1860 4ce38f21f4cd
parent 1787 ea12a3bb4b33
child 1861 433856f897dd
permissions -rw-r--r--
JavaScript in a browser can be at most three times slower than HotSpot when computing five or ten thousands of prime numbers
jaroslav@273
     1
/**
jaroslav@273
     2
 * Back 2 Browser Bytecode Translator
jaroslav@1787
     3
 * Copyright (C) 2012-2015 Jaroslav Tulach <jaroslav.tulach@apidesign.org>
jaroslav@273
     4
 *
jaroslav@273
     5
 * This program is free software: you can redistribute it and/or modify
jaroslav@273
     6
 * it under the terms of the GNU General Public License as published by
jaroslav@273
     7
 * the Free Software Foundation, version 2 of the License.
jaroslav@273
     8
 *
jaroslav@273
     9
 * This program is distributed in the hope that it will be useful,
jaroslav@273
    10
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
jaroslav@273
    11
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
jaroslav@273
    12
 * GNU General Public License for more details.
jaroslav@273
    13
 *
jaroslav@273
    14
 * You should have received a copy of the GNU General Public License
jaroslav@273
    15
 * along with this program. Look for COPYING file in the top folder.
jaroslav@273
    16
 * If not, see http://opensource.org/licenses/GPL-2.0.
jaroslav@273
    17
 */
jaroslav@372
    18
package org.apidesign.bck2brwsr.vmtest.impl;
jaroslav@273
    19
jaroslav@1185
    20
import java.lang.annotation.Annotation;
jaroslav@372
    21
import org.apidesign.bck2brwsr.vmtest.*;
jaroslav@273
    22
import java.lang.reflect.Method;
jaroslav@372
    23
import java.util.ArrayList;
jaroslav@372
    24
import java.util.List;
jaroslav@384
    25
import org.apidesign.bck2brwsr.launcher.Launcher;
jaroslav@273
    26
import org.testng.Assert;
jaroslav@273
    27
import org.testng.ITest;
jaroslav@273
    28
import org.testng.annotations.Factory;
jaroslav@273
    29
import org.testng.annotations.Test;
jaroslav@273
    30
jaroslav@273
    31
/** A TestNG {@link Factory} that seeks for {@link Compare} annotations
jaroslav@273
    32
 * in provided class and builds set of tests that compare the computations
jaroslav@273
    33
 * in real as well as JavaScript virtual machines. Use as:<pre>
jaroslav@273
    34
 * {@code @}{@link Factory} public static create() {
jaroslav@346
    35
 *   return @{link VMTest}.{@link #create(YourClass.class);
jaroslav@273
    36
 * }</pre>
jaroslav@273
    37
 *
jaroslav@273
    38
 * @author Jaroslav Tulach <jtulach@netbeans.org>
jaroslav@273
    39
 */
jaroslav@372
    40
public final class CompareCase implements ITest {
jaroslav@372
    41
    private final Bck2BrwsrCase first, second;
jaroslav@273
    42
    private final Method m;
jaroslav@1860
    43
    private final double slowdown;
jaroslav@273
    44
    
jaroslav@1860
    45
    private CompareCase(Method m, Bck2BrwsrCase first, Bck2BrwsrCase second, double slowdown) {
jaroslav@273
    46
        this.first = first;
jaroslav@273
    47
        this.second = second;
jaroslav@273
    48
        this.m = m;
jaroslav@1860
    49
        this.slowdown = slowdown;
jaroslav@273
    50
    }
jaroslav@273
    51
jaroslav@346
    52
    /** Inspects <code>clazz</code> and for each {@lik Compare} method creates
jaroslav@347
    53
     * instances of tests. Each instance runs the test in different virtual
jaroslav@346
    54
     * machine and at the end they compare the results.
jaroslav@346
    55
     * 
jaroslav@346
    56
     * @param clazz the class to inspect
jaroslav@346
    57
     * @return the set of created tests
jaroslav@346
    58
     */
jaroslav@1185
    59
    public static Object[] create(String[] brwsr, Class[] classes, Class<? extends Annotation> brwsrTest) {
jaroslav@372
    60
        List<Object> ret = new ArrayList<>();
jaroslav@372
    61
        
jaroslav@384
    62
        final LaunchSetup l = LaunchSetup.INSTANCE;
jaroslav@372
    63
        ret.add(l);
jaroslav@372
    64
        
jaroslav@383
    65
        {
jaroslav@383
    66
            String p = System.getProperty("vmtest.brwsrs");
jaroslav@383
    67
            if (p != null) {
jaroslav@383
    68
                brwsr = p.split(",");
jaroslav@383
    69
            }
jaroslav@383
    70
        }
jaroslav@383
    71
        
jaroslav@1018
    72
        for (Class clazz : classes) {
jaroslav@1018
    73
            Method[] arr = clazz.getMethods();
jaroslav@1018
    74
            for (Method m : arr) {
jaroslav@1018
    75
                registerCompareCases(m, l, ret, brwsr);
jaroslav@1185
    76
                registerBrwsrCases(brwsrTest, m, l, ret, brwsr);
jaroslav@1018
    77
            }
jaroslav@273
    78
        }
jaroslav@372
    79
        return ret.toArray();
jaroslav@273
    80
    }
jaroslav@346
    81
jaroslav@346
    82
    /** Test that compares the previous results.
jaroslav@346
    83
     * @throws Throwable 
jaroslav@346
    84
     */
jaroslav@280
    85
    @Test(dependsOnGroups = "run") public void compareResults() throws Throwable {
jaroslav@280
    86
        Object v1 = first.value;
jaroslav@280
    87
        Object v2 = second.value;
jaroslav@825
    88
        if (v1 instanceof Integer || v1 instanceof Long || v1 instanceof Byte || v1 instanceof Short) {
jaroslav@825
    89
            try {
jaroslav@825
    90
                v1 = Long.parseLong(v1.toString());
jaroslav@825
    91
            } catch (NumberFormatException nfe) {
jaroslav@825
    92
                v1 = "Can't parse " + v1.toString();
jaroslav@825
    93
            }
jaroslav@825
    94
            try {
jaroslav@825
    95
                v2 = Long.parseLong(v2.toString());
jaroslav@825
    96
            } catch (NumberFormatException nfe) {
jaroslav@825
    97
                v2 = "Can't parse " + v2.toString();
jaroslav@825
    98
            }
jaroslav@825
    99
        } else if (v1 instanceof Number) {
jaroslav@824
   100
            try {
jaroslav@824
   101
                v1 = Double.parseDouble(v1.toString());
jaroslav@824
   102
            } catch (NumberFormatException nfe) {
jaroslav@824
   103
                v1 = "Can't parse " + v1.toString();
jaroslav@824
   104
            }
jaroslav@824
   105
            try {
jaroslav@824
   106
                v2 = Double.parseDouble(v2.toString());
jaroslav@824
   107
            } catch (NumberFormatException nfe) {
jaroslav@824
   108
                v2 = "Can't parse " + v2.toString();
jaroslav@824
   109
            }
jaroslav@356
   110
        } else {
jaroslav@824
   111
            if (v1 != null) {
jaroslav@824
   112
                v1 = v1.toString();
jaroslav@824
   113
            } else {
jaroslav@824
   114
                v1 = "null";
jaroslav@824
   115
            }
jaroslav@280
   116
        }
jaroslav@393
   117
        try {
jaroslav@393
   118
            Assert.assertEquals(v2, v1, "Comparing results");
jaroslav@393
   119
        } catch (AssertionError e) {
jaroslav@393
   120
            StringBuilder sb = new StringBuilder();
jaroslav@393
   121
            sb.append(e.getMessage());
jaroslav@393
   122
            Bck2BrwsrCase.dumpJS(sb, second);
jaroslav@393
   123
            throw new AssertionError(sb.toString());
jaroslav@393
   124
        }
jaroslav@1860
   125
        if (slowdown > 0.0) {
jaroslav@1860
   126
            Bck2BrwsrCase slow;
jaroslav@1860
   127
            Bck2BrwsrCase fast;
jaroslav@1860
   128
            if (first.time >= second.time) {
jaroslav@1860
   129
                slow = second;
jaroslav@1860
   130
                fast = first;
jaroslav@1860
   131
            } else {
jaroslav@1860
   132
                fast = second;
jaroslav@1860
   133
                slow = first;
jaroslav@1860
   134
            }
jaroslav@1860
   135
            if (slow.time * slowdown < fast.time) {
jaroslav@1860
   136
                Assert.fail("Too slow " + slow.getTestName() + " took " + slow.time + " ms vs. " + fast.time + " ms of " + fast.getTestName());
jaroslav@1860
   137
            }
jaroslav@1860
   138
        }
jaroslav@280
   139
    }
jaroslav@280
   140
    
jaroslav@346
   141
    /** Test name.
jaroslav@346
   142
     * @return name of the tested method followed by a suffix
jaroslav@346
   143
     */
jaroslav@280
   144
    @Override
jaroslav@280
   145
    public String getTestName() {
jaroslav@349
   146
        return m.getName() + "[Compare " + second.typeName() + "]";
jaroslav@280
   147
    }
jaroslav@518
   148
    
jaroslav@518
   149
    private static void registerCompareCases(Method m, final LaunchSetup l, List<Object> ret, String[] brwsr) {
jaroslav@518
   150
        Compare c = m.getAnnotation(Compare.class);
jaroslav@518
   151
        if (c == null) {
jaroslav@518
   152
            return;
jaroslav@518
   153
        }
jaroslav@1860
   154
        String slowdownOverride = System.getProperty("vmtest.slowdown");
jaroslav@623
   155
        final Bck2BrwsrCase real = new Bck2BrwsrCase(m, "Java", null, false, null, null);
jaroslav@518
   156
        ret.add(real);
jaroslav@1860
   157
        double slowdown = c.slowdown();
jaroslav@1860
   158
        if (slowdown > 0.0 && slowdownOverride != null) {
jaroslav@1860
   159
            slowdown = Double.parseDouble(slowdownOverride);
jaroslav@1860
   160
        }
jaroslav@537
   161
        if (c.scripting()) {
jaroslav@623
   162
            final Bck2BrwsrCase js = new Bck2BrwsrCase(m, "JavaScript", l.javaScript(), false, null, null);
jaroslav@537
   163
            ret.add(js);
jaroslav@1860
   164
            ret.add(new CompareCase(m, real, js, slowdown));
jaroslav@537
   165
        }
jaroslav@518
   166
        for (String b : brwsr) {
jaroslav@518
   167
            final Launcher s = l.brwsr(b);
jaroslav@518
   168
            ret.add(s);
jaroslav@623
   169
            final Bck2BrwsrCase cse = new Bck2BrwsrCase(m, b, s, false, null, null);
jaroslav@518
   170
            ret.add(cse);
jaroslav@1860
   171
            ret.add(new CompareCase(m, real, cse, slowdown));
jaroslav@518
   172
        }
jaroslav@518
   173
    }
jaroslav@1185
   174
    private static void registerBrwsrCases(Class<? extends Annotation> brwsrTest, Method m, final LaunchSetup l, List<Object> ret, String[] brwsr) {
jaroslav@1185
   175
        Object c = m.getAnnotation(brwsrTest);
jaroslav@518
   176
        if (c == null) {
jaroslav@518
   177
            return;
jaroslav@518
   178
        }
jaroslav@526
   179
        HtmlFragment f = m.getAnnotation(HtmlFragment.class);
jaroslav@526
   180
        if (f == null) {
jaroslav@526
   181
            f = m.getDeclaringClass().getAnnotation(HtmlFragment.class);
jaroslav@526
   182
        }
jaroslav@667
   183
        Http.Resource[] r = {};
jaroslav@667
   184
        Http h = m.getAnnotation(Http.class);
jaroslav@667
   185
        if (h == null) {
jaroslav@667
   186
            h = m.getDeclaringClass().getAnnotation(Http.class);
jaroslav@667
   187
            if (h != null) {
jaroslav@667
   188
                r = h.value();
jaroslav@667
   189
            }
jaroslav@667
   190
        } else {
jaroslav@667
   191
            r = h.value();
jaroslav@623
   192
        }
jaroslav@518
   193
        if (brwsr.length == 0) {
jaroslav@518
   194
            final Launcher s = l.brwsr(null);
jaroslav@518
   195
            ret.add(s);
jaroslav@623
   196
            ret.add(new Bck2BrwsrCase(m, "Brwsr", s, true, f, r));
jaroslav@518
   197
        } else {
jaroslav@518
   198
            for (String b : brwsr) {
jaroslav@518
   199
                final Launcher s = l.brwsr(b);
jaroslav@518
   200
                ret.add(s);
jaroslav@623
   201
                ret.add(new Bck2BrwsrCase(m, b, s, true, f, r));
jaroslav@518
   202
            }
jaroslav@518
   203
        }
jaroslav@518
   204
    }
jaroslav@273
   205
}