Skip to content

Commit

Permalink
Add automated client smoke tests. (#2678)
Browse files Browse the repository at this point in the history
  • Loading branch information
modmuss50 authored Nov 22, 2022
1 parent 8790b57 commit faff3b8
Show file tree
Hide file tree
Showing 12 changed files with 532 additions and 18 deletions.
20 changes: 20 additions & 0 deletions .github/workflows/build.yml
Original file line number Diff line number Diff line change
Expand Up @@ -34,3 +34,23 @@ jobs:
with:
name: Maven Local
path: /root/.m2/repository

client_test:
runs-on: ubuntu-22.04
steps:
- uses: actions/checkout@v3
with:
fetch-depth: 0
- uses: actions/setup-java@v3
with:
distribution: 'microsoft'
java-version: '17'
- name: Run Auto test Client
uses: modmuss50/xvfb-action@v1
with:
run: ./gradlew runProductionAutoTestClient --stacktrace --warning-mode=fail
- uses: actions/upload-artifact@v2
if: always()
with:
name: Test Screenshots
path: run/screenshots
55 changes: 53 additions & 2 deletions build.gradle
Original file line number Diff line number Diff line change
Expand Up @@ -354,15 +354,66 @@ loom {
}
autoTestServer {
inherit testmodServer

name "Auto Test Server"

vmArg "-Dfabric.autoTest"
}
autoTestClient {
inherit testmodClient
name "Auto Test Client"
vmArg "-Dfabric.autoTest"
}
}
}
test.dependsOn runGametest

configurations {
productionRuntime {
extendsFrom configurations.minecraftLibraries
extendsFrom configurations.loaderLibraries
extendsFrom configurations.minecraftRuntimeOnlyLibraries
}
}

dependencies {
productionRuntime "net.fabricmc:fabric-loader:${project.loader_version}"
productionRuntime "net.fabricmc:intermediary:${project.minecraft_version}"
}

import net.fabricmc.loom.util.OperatingSystem

// This is very far beyond loom's API if you copy this, you're on your own.
task runProductionAutoTestClient(type: JavaExec, dependsOn: [remapJar, remapTestmodJar]) {
classpath.from configurations.productionRuntime
mainClass = "net.fabricmc.loader.impl.launch.knot.KnotClient"
workingDir = file("run")

afterEvaluate {
dependsOn downloadAssets
}

doFirst {
classpath.from loom.minecraftProvider.minecraftClientJar
workingDir.mkdirs()

args(
"--assetIndex", loom.minecraftProvider.versionInfo.assetIndex().fabricId(loom.minecraftProvider.minecraftVersion()),
"--assetsDir", new File(loom.files.userCache, "assets").absolutePath,
"--gameDir", workingDir.absolutePath
)

if (OperatingSystem.CURRENT_OS == OperatingSystem.MAC_OS) {
jvmArgs(
"-XstartOnFirstThread"
)
}

jvmArgs(
"-Dfabric.addMods=${remapJar.archiveFile.get().asFile.absolutePath}${File.pathSeparator}${remapTestmodJar.archiveFile.get().asFile.absolutePath}",
"-Dfabric.autoTest"
)
}
}

subprojects {
if (it.name == "deprecated") return

Expand Down
3 changes: 2 additions & 1 deletion fabric-api-base/build.gradle
Original file line number Diff line number Diff line change
Expand Up @@ -3,5 +3,6 @@ version = getSubprojectVersion(project)

testDependencies(project, [
':fabric-command-api-v2',
':fabric-lifecycle-events-v1'
':fabric-lifecycle-events-v1',
':fabric-screen-api-v1'
])
Original file line number Diff line number Diff line change
@@ -0,0 +1,134 @@
/*
* Copyright (c) 2016, 2017, 2018, 2019 FabricMC
*
* 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 net.fabricmc.fabric.test.base;

import static net.fabricmc.fabric.test.base.FabricClientTestHelper.clickScreenButton;
import static net.fabricmc.fabric.test.base.FabricClientTestHelper.closeScreen;
import static net.fabricmc.fabric.test.base.FabricClientTestHelper.enableDebugHud;
import static net.fabricmc.fabric.test.base.FabricClientTestHelper.openGameMenu;
import static net.fabricmc.fabric.test.base.FabricClientTestHelper.openInventory;
import static net.fabricmc.fabric.test.base.FabricClientTestHelper.setPerspective;
import static net.fabricmc.fabric.test.base.FabricClientTestHelper.takeScreenshot;
import static net.fabricmc.fabric.test.base.FabricClientTestHelper.waitForLoadingComplete;
import static net.fabricmc.fabric.test.base.FabricClientTestHelper.waitForScreen;
import static net.fabricmc.fabric.test.base.FabricClientTestHelper.waitForWorldTicks;

import java.io.IOException;
import java.io.UncheckedIOException;
import java.nio.file.DirectoryStream;
import java.nio.file.Files;
import java.nio.file.Path;

import org.spongepowered.asm.mixin.MixinEnvironment;

import net.minecraft.client.gui.screen.ConfirmScreen;
import net.minecraft.client.gui.screen.TitleScreen;
import net.minecraft.client.gui.screen.world.CreateWorldScreen;
import net.minecraft.client.gui.screen.world.SelectWorldScreen;
import net.minecraft.client.option.Perspective;

import net.fabricmc.api.ClientModInitializer;
import net.fabricmc.loader.api.FabricLoader;

public class FabricApiAutoTestClient implements ClientModInitializer {
@Override
public void onInitializeClient() {
if (System.getProperty("fabric.autoTest") == null) {
return;
}

var thread = new Thread(() -> {
try {
runTest();
} catch (Throwable t) {
t.printStackTrace();
System.exit(1);
}
});
thread.setName("Fabric Auto Test");
thread.start();
}

private void runTest() {
waitForLoadingComplete();

{
waitForScreen(TitleScreen.class);
takeScreenshot("title_screen");
clickScreenButton("menu.singleplayer");
}

if (!isDirEmpty(FabricLoader.getInstance().getGameDir().resolve("saves"))) {
waitForScreen(SelectWorldScreen.class);
takeScreenshot("select_world_screen");
clickScreenButton("selectWorld.create");
}

{
waitForScreen(CreateWorldScreen.class);
clickScreenButton("selectWorld.gameMode");
clickScreenButton("selectWorld.gameMode");
takeScreenshot("create_world_screen");
clickScreenButton("selectWorld.create");
}

{
// API test mods use experimental features
waitForScreen(ConfirmScreen.class);
clickScreenButton("gui.yes");
}

{
enableDebugHud();
waitForWorldTicks(200);
takeScreenshot("in_game_overworld");
}

MixinEnvironment.getCurrentEnvironment().audit();

{
// See if the player render events are working.
setPerspective(Perspective.THIRD_PERSON_BACK);
takeScreenshot("in_game_overworld_third_person");
}

{
openInventory();
takeScreenshot("in_game_inventory");
closeScreen();
}

{
openGameMenu();
takeScreenshot("game_menu");
clickScreenButton("menu.returnToMenu");
}

{
waitForScreen(TitleScreen.class);
clickScreenButton("menu.quit");
}
}

private boolean isDirEmpty(Path path) {
try (DirectoryStream<Path> directory = Files.newDirectoryStream(path)) {
return !directory.iterator().hasNext();
} catch (IOException e) {
throw new UncheckedIOException(e);
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,40 @@
/*
* Copyright (c) 2016, 2017, 2018, 2019 FabricMC
*
* 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 net.fabricmc.fabric.test.base;

import org.spongepowered.asm.mixin.MixinEnvironment;

import net.fabricmc.api.DedicatedServerModInitializer;
import net.fabricmc.fabric.api.event.lifecycle.v1.ServerTickEvents;

public class FabricApiAutoTestServer implements DedicatedServerModInitializer {
private int ticks = 0;

@Override
public void onInitializeServer() {
if (System.getProperty("fabric.autoTest") != null) {
ServerTickEvents.END_SERVER_TICK.register(server -> {
ticks++;

if (ticks == 50) {
MixinEnvironment.getCurrentEnvironment().audit();
server.stop(false);
}
});
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -24,24 +24,10 @@

import net.fabricmc.api.ModInitializer;
import net.fabricmc.fabric.api.command.v2.CommandRegistrationCallback;
import net.fabricmc.fabric.api.event.lifecycle.v1.ServerTickEvents;

public class FabricApiBaseTestInit implements ModInitializer {
private int ticks = 0;

@Override
public void onInitialize() {
if (System.getProperty("fabric.autoTest") != null) {
ServerTickEvents.END_SERVER_TICK.register(server -> {
ticks++;

if (ticks == 50) {
MixinEnvironment.getCurrentEnvironment().audit();
server.stop(false);
}
});
}

// Command to call audit the mixin environment
CommandRegistrationCallback.EVENT.register((dispatcher, registryAccess, environment) -> {
dispatcher.register(literal("audit_mixins").executes(context -> {
Expand Down
Loading

0 comments on commit faff3b8

Please sign in to comment.