Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,8 @@
### Added

### Changed
- Upgrading IntelliJ from 2025.1 to 2025.1.1
- Migrate from `StartupActivity` (obsolete) -> `VcsRepositoryMappingListener`

### Deprecated

Expand Down
4 changes: 2 additions & 2 deletions gradle.properties
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,7 @@ pluginUntilBuild = 251.*

# Plugin Verifier integration -> https://github.com/JetBrains/gradle-intellij-plugin#plugin-verifier-dsl
# See https://jb.gg/intellij-platform-builds-list for available build versions
pluginVerifierIdeVersions = 2025.1,LATEST-EAP-SNAPSHOT
pluginVerifierIdeVersions = 2025.1.1,LATEST-EAP-SNAPSHOT
# Failure Levels: https://github.com/JetBrains/gradle-intellij-plugin/blob/master/src/main/kotlin/org/jetbrains/intellij/tasks/RunPluginVerifierTask.kt
pluginVerifierExcludeFailureLevels =
# Mute Plugin Problems -> https://github.com/JetBrains/intellij-plugin-verifier?tab=readme-ov-file#check-plugin
Expand All @@ -34,7 +34,7 @@ platformType = IC
#platformVersion = 2024.1.4 ## 2024.1.4
#platformVersion = 242.20224.91-EAP-SNAPSHOT ## 2024.2 Beta
#platformVersion = 242.20224.159-EAP-SNAPSHOT ## 2024.2 RC1
platformVersion = 2025.1
platformVersion = 2025.1.1

# Plugin Dependencies -> https://plugins.jetbrains.com/docs/intellij/plugin-dependencies.html
# Example: platformPlugins = com.jetbrains.php:203.4449.22, org.intellij.scala:2023.3.27@EAP
Expand Down

This file was deleted.

Original file line number Diff line number Diff line change
@@ -0,0 +1,93 @@
package com.chriscarini.jetbrains.github.issue.navigation;

import com.chriscarini.jetbrains.github.utils.GitHubUri;
import com.chriscarini.jetbrains.github.utils.GitHubUtils;
import com.intellij.dvcs.repo.VcsRepositoryManager;
import com.intellij.dvcs.repo.VcsRepositoryMappingListener;
import com.intellij.openapi.application.ApplicationManager;
import com.intellij.openapi.diagnostic.Logger;
import com.intellij.openapi.project.Project;
import com.intellij.openapi.vcs.IssueNavigationConfiguration;
import com.intellij.openapi.vcs.IssueNavigationLink;
import git4idea.repo.GitRemote;
import git4idea.repo.GitRepositoryImpl;
import org.jetbrains.annotations.NonNls;
import org.jetbrains.annotations.NotNull;

import java.util.Collection;
import java.util.List;
import java.util.stream.Collectors;
import java.util.stream.Stream;


public class GitHubIssueNavigationVcsRepositoryMappingListener implements VcsRepositoryMappingListener {

private static final @NonNls Logger LOG = Logger.getInstance(GitHubIssueNavigationVcsRepositoryMappingListener.class);
private final @NotNull Project project;

public GitHubIssueNavigationVcsRepositoryMappingListener(@NotNull final Project project) {
this.project = project;
}

@Override
public void mappingChanged() {
if (ApplicationManager.getApplication().isUnitTestMode()) {
LOG.debug("Unit test mode; will not attempt to add GitHub Issue Navigation to project");
return;
}

VcsRepositoryManager.getInstance(project).getRepositories().forEach(repository -> {
final Collection<GitRemote> repoRemotes = ((GitRepositoryImpl) repository).getRemotes();
repoRemotes.forEach(
gitRemote -> {
final String url = gitRemote.getFirstUrl();

if (url == null) {
LOG.debug(String.format("Null Git remote URL: %s", gitRemote));
return;
}

if (!GitHubUtils.isKnownGitHubDomain(url)) {
LOG.debug(String.format(
"Git remote URL (%s) not from a known GitHub domain: %s",
gitRemote,
GitHubUtils.knownGitHubDomains().collect(Collectors.joining(", "))
));
return;
}

final String cleanedUrl = GitHubUri.parseUrl(url).asHttpsFormatUrl();

if (existsInIssueNavConfig(project, cleanedUrl)) {
LOG.debug(String.format("%s is already registered.", cleanedUrl));
return;
}

LOG.debug(String.format("Registering [%s].", cleanedUrl));
addNewIssueNavigationConfiguration(project, cleanedUrl);
}
);
});
}

protected static boolean existsInIssueNavConfig(@NotNull final Project project, @NotNull final String url) {
final List<IssueNavigationLink> matching = IssueNavigationConfiguration.getInstance(project).getLinks()
.stream()
.filter(issueNavigationLink -> issueNavigationLink.getLinkRegexp().contains(url))
.toList();

return !matching.isEmpty();
}

protected static void addNewIssueNavigationConfiguration(@NotNull final Project project, @NotNull final String url) {
final IssueNavigationConfiguration issueNavigationConfiguration = IssueNavigationConfiguration.getInstance(project);

final IssueNavigationLink newLink = new IssueNavigationLink("\\(#(\\d+)\\)", String.format("%s/issues/$1", url));

issueNavigationConfiguration.setLinks(
Stream.concat(
issueNavigationConfiguration.getLinks().stream(),
Stream.of(newLink)
).collect(Collectors.toList()));
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -19,8 +19,8 @@ public static Stream<String> knownGitHubDomains() {
public static boolean isKnownGitHubDomain(@NotNull final String url) {
final List<String> result = knownGitHubDomains()
.filter(url::contains)
.collect(Collectors.toList());
.toList();

return result.size() > 0;
return !result.isEmpty();
}
}
13 changes: 7 additions & 6 deletions src/main/resources/META-INF/plugin.xml
Original file line number Diff line number Diff line change
Expand Up @@ -12,10 +12,11 @@
<depends>com.intellij.modules.platform</depends>
<depends>Git4Idea</depends>

<extensions defaultExtensionNs="com.intellij">

<postStartupActivity
implementation="com.chriscarini.jetbrains.github.issue.navigation.GitHubIssueNavigationStartupActivity"/>

</extensions>
<projectListeners>
<listener
class="com.chriscarini.jetbrains.github.issue.navigation.GitHubIssueNavigationVcsRepositoryMappingListener"
topic="com.intellij.dvcs.repo.VcsRepositoryMappingListener"
activeInHeadlessMode="false"
activeInTestMode="false"/>
</projectListeners>
</idea-plugin>
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@
import java.util.stream.Collectors;


public class TestGitHubIssueNavigationStartupActivityTest extends LightPlatformTestCase {
public class TestGitHubIssueNavigationVcsRepositoryMappingListenerTest extends LightPlatformTestCase {
@Override
public void setUp() throws Exception {
super.setUp();
Expand All @@ -26,25 +26,25 @@ public void testExistsInIssueNavConfig() {
IssueNavigationConfiguration.getInstance(getProject()).setLinks(List.of(myIssueNavigationLink));

// expect
assert GitHubIssueNavigationStartupActivity.existsInIssueNavConfig(getProject(), expectedUrl);
assert !GitHubIssueNavigationStartupActivity.existsInIssueNavConfig(getProject(), unexpectedUrl);
assert GitHubIssueNavigationVcsRepositoryMappingListener.existsInIssueNavConfig(getProject(), expectedUrl);
assert !GitHubIssueNavigationVcsRepositoryMappingListener.existsInIssueNavConfig(getProject(), unexpectedUrl);
}

@Test
public void testAddNewIssueNavigationConfiguration() {
// given
final String expectedUrl = "MY_FAKE_URL";
assert IssueNavigationConfiguration.getInstance(getProject()).getLinks().size() == 0;
assert IssueNavigationConfiguration.getInstance(getProject()).getLinks().isEmpty();

// when
GitHubIssueNavigationStartupActivity.addNewIssueNavigationConfiguration(getProject(), expectedUrl);
GitHubIssueNavigationVcsRepositoryMappingListener.addNewIssueNavigationConfiguration(getProject(), expectedUrl);

// then
final List<IssueNavigationLink> issues = IssueNavigationConfiguration.getInstance(getProject())
.getLinks()
.stream()
.filter(issueNavigationLink -> issueNavigationLink.getLinkRegexp().contains(expectedUrl))
.collect(Collectors.toList());
.toList();
assertEquals(1, issues.size());
}
}