// Copyright (c) 2021, the R8 project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.

package com.android.tools.r8.repackage;

import static com.android.tools.r8.shaking.ProguardConfigurationParser.FLATTEN_PACKAGE_HIERARCHY;
import static com.android.tools.r8.shaking.ProguardConfigurationParser.REPACKAGE_CLASSES;
import static com.android.tools.r8.utils.codeinspector.Matchers.isPresentAndNotRenamed;
import static com.android.tools.r8.utils.codeinspector.Matchers.isPresentAndRenamed;
import static org.hamcrest.MatcherAssert.assertThat;
import static org.junit.Assume.assumeTrue;

import com.android.tools.r8.NeverInline;
import com.android.tools.r8.ProguardVersion;
import com.android.tools.r8.R8TestBuilder;
import com.android.tools.r8.R8TestCompileResult;
import com.android.tools.r8.TestParameters;
import com.android.tools.r8.utils.codeinspector.ClassSubject;
import com.google.common.collect.ImmutableList;
import java.util.List;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.Parameterized;
import org.junit.runners.Parameterized.Parameters;

/**
 * A simple test that checks that R8 behaves the same as PG for repackaging. We do require that
 * classes that are looked up reflectively are kept in some way, however, for easier debugging with
 * -dontobfuscate we should follow PG behavior. *
 */
@RunWith(Parameterized.class)
public class RepackageDontObfuscateTest extends RepackageTestBase {

  private final String[] EXPECTED = new String[] {"A::foo", "A::foo", "A::foo"};

  @Parameters(name = "{1}, kind: {0}, PG-version: {2}")
  public static List<Object[]> data() {
    return buildParameters(
        ImmutableList.of(FLATTEN_PACKAGE_HIERARCHY, REPACKAGE_CLASSES),
        getTestParameters().withAllRuntimesAndApiLevels().build(),
        ProguardVersion.values());
  }

  private final ProguardVersion proguardVersion;

  public RepackageDontObfuscateTest(
      String flattenPackageHierarchyOrRepackageClasses,
      TestParameters parameters,
      ProguardVersion proguardVersion) {
    super(flattenPackageHierarchyOrRepackageClasses, parameters);
    this.proguardVersion = proguardVersion;
  }

  @Test
  public void testPG() throws Exception {
    assumeTrue(parameters.isCfRuntime());
    testForProguard(proguardVersion)
        .addInnerClasses(getClass())
        .addKeepMainRule(Main.class)
        .apply(this::configureRepackaging)
        .addInliningAnnotations()
        .addDontWarn(RepackageDontObfuscateTest.class)
        .noMinification()
        .compile()
        .inspect(
            inspector -> {
              ClassSubject aClass = inspector.clazz(A.class);
              assertThat(aClass, isPresentAndNotRenamed());
            })
        .run(parameters.getRuntime(), Main.class, A.class.getTypeName())
        .assertSuccessWithOutputLines(EXPECTED);
  }

  @Test
  public void testR8Full() throws Exception {
    setup(testForR8(parameters.getBackend()))
        .inspect(
            inspector -> {
              ClassSubject aClass = inspector.clazz(A.class);
              assertThat(aClass, isPresentAndRenamed());
            })
        .run(
            parameters.getRuntime(),
            Main.class,
            getRepackagePackage() + "." + A.class.getTypeName())
        .assertSuccessWithOutputLines(EXPECTED);
  }

  @Test
  public void testR8Compat() throws Exception {
    setup(testForR8Compat(parameters.getBackend()))
        .inspect(
            inspector -> {
              ClassSubject aClass = inspector.clazz(A.class);
              assertThat(aClass, isPresentAndNotRenamed());
            })
        .run(parameters.getRuntime(), Main.class, A.class.getTypeName())
        .assertSuccessWithOutputLines(EXPECTED);
  }

  private R8TestCompileResult setup(R8TestBuilder<?> r8TestBuilder) throws Exception {
    return r8TestBuilder
        .addInnerClasses(getClass())
        .setMinApi(parameters.getApiLevel())
        .addKeepMainRule(Main.class)
        .enableInliningAnnotations()
        .apply(this::configureRepackaging)
        .noMinification()
        .compile();
  }

  public static class A {

    @NeverInline
    public static void foo() {
      System.out.println("A::foo");
    }
  }

  public static class Main {

    public static void main(String[] args) throws Exception {
      A.foo();
      Class.forName("com.android.tools.r8.repackage.RepackageDontObfuscateTest$A")
          .getDeclaredMethod("foo")
          .invoke(null);
      Class.forName(args[0]).getDeclaredMethod("foo").invoke(null);
    }
  }
}
