Skip to content

Commit

Permalink
Implement redirecting to IDEAs output again, reduce test overhead on PR
Browse files Browse the repository at this point in the history
  • Loading branch information
marchermans committed Jan 7, 2025
1 parent 5802648 commit 5ebef41
Show file tree
Hide file tree
Showing 6 changed files with 35 additions and 31 deletions.
7 changes: 5 additions & 2 deletions .github/gradle/gradle.gradle
Original file line number Diff line number Diff line change
Expand Up @@ -91,7 +91,7 @@ subprojects.forEach { Project subProject ->

it.group = 'infrastructure'
it.doLast {
subProject.tasks.withType(Test).forEach { Task test ->
subProject.tasks.withType(Test).forEach { Test test ->
def testSourceSetCandidate = test.extensions.findByName('test-source-set')
if (testSourceSetCandidate != null) {
SourceSet testSourceSet = testSourceSetCandidate as SourceSet
Expand Down Expand Up @@ -128,7 +128,10 @@ subprojects.forEach { Project subProject ->

tests.addAll(createTestRuns(test, testClasses))
} else {
tests.addAll(createTestRuns(test, new ArrayList<String>()))
//This test task is not marked as a filterable test task by our central build.gradle file.
//Check if it has source, if so run it, else skip it. No need to setup gradle to run nothing:
if (!test.getCandidateClassFiles().isEmpty())
tests.addAll(createTestRuns(test, new ArrayList<String>()))
}
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -382,22 +382,24 @@ private TaskProvider<?> getOrCreateIdeBeforeRunTask(Project project, RunImpl run
private List<TaskProvider<?>> createIntelliJCopyResourcesTasks(Run run) {
final List<TaskProvider<?>> copyProcessResources = new ArrayList<>();
for (SourceSet sourceSet : run.getModSources().all().get().values()) {
copyProcessResources.add(setupCopyResourcesForIdea(sourceSet));
copyProcessResources.add(setupCopyResourcesForIdea(sourceSet, RunsUtil.IdeaCompileType.Production));
}

if (run.getIsJUnit().get()) {
for (SourceSet sourceSet : run.getUnitTestSources().all().get().values()) {
copyProcessResources.add(setupCopyResourcesForIdea(sourceSet));
copyProcessResources.add(setupCopyResourcesForIdea(sourceSet, RunsUtil.IdeaCompileType.Test));
}
}

return copyProcessResources;
}

private static @NotNull TaskProvider<?> setupCopyResourcesForIdea(SourceSet sourceSet) {
private static @NotNull TaskProvider<?> setupCopyResourcesForIdea(SourceSet sourceSet, RunsUtil.IdeaCompileType compileType) {
final Project sourceSetProject = SourceSetUtils.getProject(sourceSet);

final String taskName = CommonRuntimeUtils.buildTaskName("intelliJCopy", sourceSet.getProcessResourcesTaskName());
//We need a task per compile type, incase the modder configures the task in one run as a mod source and in the other as a unit test source.
final String taskPrefix = CommonRuntimeUtils.buildTaskName("intelliJCopy", compileType.name());
final String taskName = CommonRuntimeUtils.buildTaskName(taskPrefix, sourceSet.getProcessResourcesTaskName());
final TaskProvider<?> intelliJResourcesTask;

if (sourceSetProject.getTasks().findByName(taskName) != null) {
Expand All @@ -407,7 +409,7 @@ private List<TaskProvider<?>> createIntelliJCopyResourcesTasks(Run run) {
intelliJResourcesTask = sourceSetProject.getTasks().register(taskName, Copy.class, task -> {
final TaskProvider<ProcessResources> defaultProcessResources = sourceSetProject.getTasks().named(sourceSet.getProcessResourcesTaskName(), ProcessResources.class);
task.from(defaultProcessResources.map(ProcessResources::getDestinationDir));
task.into(RunsUtil.getRunWithIdeaResourcesDirectory(sourceSet));
task.into(RunsUtil.getRunWithIdeaResourcesDirectory(sourceSet, compileType));

task.dependsOn(defaultProcessResources);
});
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -607,12 +607,13 @@ public static Provider<? extends FileSystemLocation> getRunWithIdeaDirectory(fin
return getIdeaModuleOutDirectory(sourceSet, compileType).map(dir -> dir.dir(name));
}

public static Provider<? extends FileSystemLocation> getRunWithIdeaResourcesDirectory(final SourceSet sourceSet) {
//When running with idea we forcefully redirect all sourcesets to a directory in build, to prevent issues
//with unit tests started from the gutter -> We can only have a single task, that should run always, regardless of run or sourceset:
final Project project = SourceSetUtils.getProject(sourceSet);
final ProjectLayout buildLayout = project.getLayout();
return buildLayout.getBuildDirectory().map(dir -> dir.dir("idea").dir("resources").dir(sourceSet.getName()));
public static Provider<? extends FileSystemLocation> getRunWithIdeaResourcesDirectory(final SourceSet sourceSet, final IdeaCompileType compileType) {
//We previously used this:
//buildLayout.getBuildDirectory().map(dir -> dir.dir("idea").dir("resources").dir(sourceSet.getName()));
//However this has issues at runtime with FML trying to load the old not interpolated mods.toml etc.
//It works smoothly if a user has an excluded templates directory configured for his templates in process resources however.
//To make it work transparently we switched back to this interpolation mechanic where we write into IDEAs output directory.
return getRunWithIdeaDirectory(sourceSet, compileType, "resources");
}

public static Provider<? extends FileSystemLocation> getRunWithIdeaClassesDirectory(final SourceSet sourceSet, final IdeaCompileType compileType) {
Expand All @@ -625,7 +626,7 @@ public static Provider<String> buildRunWithIdeaModClasses(
return buildModClasses(compileSourceSets, sourceSet -> {

if (isRunWithIdea(sourceSet)) {
final File resourcesDir = getRunWithIdeaResourcesDirectory(sourceSet).get().getAsFile();
final File resourcesDir = getRunWithIdeaResourcesDirectory(sourceSet, compileType).get().getAsFile();
final File classesDir = getRunWithIdeaClassesDirectory(sourceSet, compileType).get().getAsFile();
return Stream.of(resourcesDir, classesDir);
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -116,12 +116,12 @@ class ConfigurationCacheTests extends BuilderBasedTestSpecification {
}
runs {
data { }
clientData { }
}
afterEvaluate {
//We don't care for the error here, we just want to run the task so that the config cache is created
tasks.withType(JavaExec).named('runData') {
tasks.withType(JavaExec).named('runClientData') {
ignoreExitValue = true
group = 'run'
}
Expand All @@ -145,11 +145,11 @@ class ConfigurationCacheTests extends BuilderBasedTestSpecification {

when:
def run = project.run {
it.tasks('runData')
it.tasks('runClientData')

}

then:
run.task(':runData').outcome == TaskOutcome.SUCCESS
run.task(':runClientData').outcome == TaskOutcome.SUCCESS
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -93,13 +93,13 @@ class MultiProjectTests extends BuilderBasedTestSpecification {

when:
def run = rootProject.run {
it.tasks(':main:runData')
it.tasks(':main:runClientData')
//We are expecting this test to fail, since there is a mod without any files included so it is fine.
it.shouldFail()
}

then:
run.task(':main:writeMinecraftClasspathData').outcome == TaskOutcome.SUCCESS
run.task(':main:writeMinecraftClasspathClientData').outcome == TaskOutcome.SUCCESS

run.output.contains("Error during pre-loading phase: ERROR: File null is not a valid mod file") ||
run.output.contains("Caused by: net.neoforged.fml.ModLoadingException: Loading errors encountered:")
Expand Down Expand Up @@ -286,13 +286,13 @@ class MultiProjectTests extends BuilderBasedTestSpecification {

when:
def run = rootProject.run {
it.tasks(':main:runData')
it.tasks(':main:runClientData')
//We are expecting this test to fail, since there is a mod without any files included so it is fine.
it.shouldFail()
}

then:
run.task(':main:writeMinecraftClasspathData').outcome == TaskOutcome.SUCCESS
run.task(':main:writeMinecraftClasspathClientData').outcome == TaskOutcome.SUCCESS

run.output.contains("Error during pre-loading phase: ERROR: File null is not a valid mod file") ||
run.output.contains("Caused by: net.neoforged.fml.ModLoadingException: Loading errors encountered:")
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -46,17 +46,16 @@ class RunTests extends BuilderBasedTestSpecification {

when:
def run = project.run {
it.tasks(':runData')
it.tasks(':runClientData')
//We are expecting this test to fail, since there is a mod without any files included so it is fine.
it.shouldFail()
it.stacktrace()
}

then:
true
run.task(':writeMinecraftClasspathData').outcome == TaskOutcome.SUCCESS
run.output.contains("Error during pre-loading phase: ERROR: File null is not a valid mod file") ||
run.output.contains("Caused by: java.io.IOException: Invalid paths argument, contained no existing paths")
run.task(':writeMinecraftClasspathClientData').outcome == TaskOutcome.SUCCESS
run.output.contains("is not a valid mod file")
}

def "configuring of the configurations after the dependencies block should work"() {
Expand Down Expand Up @@ -85,7 +84,7 @@ class RunTests extends BuilderBasedTestSpecification {
}
runs {
data {
clientData {
modSource project.sourceSets.main
}
}
Expand All @@ -100,15 +99,14 @@ class RunTests extends BuilderBasedTestSpecification {

when:
def run = project.run {
it.tasks(':runData')
it.tasks(':runClientData')
//We are expecting this test to fail, since there is a mod without any files included so it is fine.
it.shouldFail()
}

then:
run.task(':writeMinecraftClasspathData').outcome == TaskOutcome.SUCCESS
run.output.contains("Error during pre-loading phase: ERROR: File null is not a valid mod file") ||
run.output.contains("Caused by: java.io.IOException: Invalid paths argument, contained no existing paths")
run.task(':writeMinecraftClasspathClientData').outcome == TaskOutcome.SUCCESS
run.output.contains("is not a valid mod file")
}

def "runs can be declared before the dependencies block"() {
Expand Down

0 comments on commit 5ebef41

Please sign in to comment.