Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Feature history view #105

Open
wants to merge 26 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
26 commits
Select commit Hold shift + click to select a range
bcf48d5
a
RimanHoubbi Sep 26, 2024
3b15ad1
Fix series data
RimanHoubbi Sep 28, 2024
40bc264
Restructured code
RimanHoubbi Oct 12, 2024
733e797
fix
RimanHoubbi Nov 5, 2024
133864f
deleted unneccessary code
RimanHoubbi Nov 20, 2024
3449554
Merge branch 'isselab:main' into FeatureHistoryView
RimanHoubbi Nov 20, 2024
048cc0c
.
RimanHoubbi Nov 20, 2024
f57450d
.
RimanHoubbi Nov 20, 2024
719e6c6
revert changes
RimanHoubbi Nov 20, 2024
3cdaa37
Fixed build workflow
RimanHoubbi Nov 21, 2024
e6fc674
Merge branch 'isselab:main' into FeatureHistoryView
RimanHoubbi Nov 21, 2024
f31d70e
Fixed build workflow
RimanHoubbi Nov 21, 2024
b058923
Update FeatureLocationManager.java
RimanHoubbi Dec 2, 2024
ea58588
Update plugin.xml
RimanHoubbi Dec 2, 2024
865231c
Update FeatureHistoryAnalyzer.java
RimanHoubbi Dec 2, 2024
bb2db49
Update plugin.xml
RimanHoubbi Dec 2, 2024
fff92f9
fix qodana comments
RimanHoubbi Dec 2, 2024
2f62133
Update CommitExtractionTask.java
RimanHoubbi Dec 2, 2024
2844a7a
Update FeatureExtractionTask.java
RimanHoubbi Dec 2, 2024
0875e58
Merge branch 'main' into FeatureHistoryView
RimanHoubbi Dec 2, 2024
7ec520c
Merge remote-tracking branch 'origin/FeatureHistoryView' into Feature…
RimanHoubbi Dec 2, 2024
c68c321
Update CommitExtractionTask.java
RimanHoubbi Dec 2, 2024
4b7bcfb
Update FeatureExtractionTask.java
RimanHoubbi Dec 2, 2024
1ddfacb
Merge remote-tracking branch 'origin/FeatureHistoryView' into Feature…
RimanHoubbi Dec 2, 2024
eb138e1
fix qodana comments
RimanHoubbi Dec 2, 2024
ae8b5ce
remove unused imports
RimanHoubbi Dec 2, 2024
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
1 change: 1 addition & 0 deletions .feature-model
Original file line number Diff line number Diff line change
Expand Up @@ -42,4 +42,5 @@ HAnS
NewFile
Quickfix
SettingsPage
FeatureHistory

5 changes: 3 additions & 2 deletions .github/workflows/build.yml
Original file line number Diff line number Diff line change
Expand Up @@ -115,6 +115,7 @@ jobs:
- name: Setup Gradle
uses: gradle/actions/setup-gradle@v4


# Run tests
- name: Run Tests
run: ./gradlew check
Expand All @@ -129,7 +130,7 @@ jobs:

# Upload the Kover report to CodeCov
- name: Upload Code Coverage Report
uses: codecov/codecov-action@v5
uses: codecov/codecov-action@v4
with:
files: ${{ github.workspace }}/build/reports/kover/report.xml

Expand Down Expand Up @@ -164,7 +165,7 @@ jobs:

# Run Qodana inspections
- name: Qodana - Code Inspection
uses: JetBrains/qodana-action@v2024.2
uses: JetBrains/qodana-action@v2024.1.5
with:
cache-default-branch-only: true

Expand Down
2 changes: 1 addition & 1 deletion .github/workflows/code_quality.yml
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,6 @@ jobs:
ref: ${{ github.event.pull_request.head.sha }} # to check out the actual pull request commit, not the merge commit
fetch-depth: 0 # a full history is required for pull request analysis
- name: 'Qodana Scan'
uses: JetBrains/qodana-action@v2024.2
uses: JetBrains/qodana-action@v2024.1
env:
QODANA_TOKEN: ${{ secrets.QODANA_TOKEN }}
6 changes: 5 additions & 1 deletion build.gradle.kts
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@ plugins {
alias(libs.plugins.changelog) // Gradle Changelog Plugin
alias(libs.plugins.qodana) // Gradle Qodana Plugin
alias(libs.plugins.kover) // Gradle Kover Plugin

}

group = properties("pluginGroup").get()
Expand All @@ -23,6 +24,7 @@ kotlin {
jvmToolchain(21)
}


// Configure project's dependencies
repositories {
mavenCentral()
Expand All @@ -48,6 +50,9 @@ dependencies {
pluginVerifier()
zipSigner()
testFramework(TestFrameworkType.Platform)

// Add Git4Idea as a bundled plugin
bundledPlugins("Git4Idea")
}
}

Expand Down Expand Up @@ -164,7 +169,6 @@ intellijPlatformTesting {
)
}
}

plugins {
robotServerPlugin()
}
Expand Down
6 changes: 3 additions & 3 deletions gradle/libs.versions.toml
Original file line number Diff line number Diff line change
Expand Up @@ -3,10 +3,10 @@
junit = "4.13.2"

# plugins
kotlin = "2.0.21"
kotlin = "2.0.20"
changelog = "2.2.1"
intelliJPlatform = "2.1.0"
qodana = "2024.2.6"
intelliJPlatform = "2.0.1"
qodana = "2024.1.9"
kover = "0.8.3"

[libraries]
Expand Down
Empty file.
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
FeatureHistory
Original file line number Diff line number Diff line change
@@ -0,0 +1,227 @@
package se.isselab.HAnS.featureHistoryView;
import com.intellij.openapi.project.Project;
import git4idea.GitCommit;
import org.jetbrains.annotations.NotNull;
import se.isselab.HAnS.featureHistoryView.backgroundTasks.CommitExtractionTask;
import se.isselab.HAnS.featureHistoryView.backgroundTasks.FeatureExtractionTask;
import java.text.SimpleDateFormat;
import java.util.*;

public class FeatureCommitMapper {
private final List<String> featureNames;
private final List<String> commitTimes;
private final List<String> commitHashes; // For mapping
private final List<Map<String, Object>> seriesData; // List of maps to hold data points
private final List<String> deletedFeatureNames;
private final Map<String, String> deletedFeatureCommits; // Map of deleted features and their last commit times
private final Map<String, String> deletedFeatureCommitHashes; // Map of deleted features to their last commit hash
private final SimpleDateFormat dateFormatter = new SimpleDateFormat("MMM dd HH:mm:ss yyyy"); // Date formatter for human-readable format

public FeatureCommitMapper() {
this.featureNames = new ArrayList<>();
this.commitTimes = new ArrayList<>();
this.commitHashes = new ArrayList<>();
this.seriesData = new ArrayList<>();
this.deletedFeatureNames = new ArrayList<>();
this.deletedFeatureCommits = new HashMap<>();
this.deletedFeatureCommitHashes = new HashMap<>();
}

public void mapFeaturesAndCommits(Project project, Runnable onComplete) {
// First, extract the features in the background
FeatureExtractionTask featureTask = new FeatureExtractionTask(project, allFeatures -> {
this.featureNames.addAll(allFeatures);
//System.out.println("Features extracted: " + featureNames);
// Once features are extracted, proceed to extract commits
CommitExtractionTask commitTask = new CommitExtractionTask(project, (commits, featureExistenceMap, featureFileMap, featureFolderMap) -> {
commits.forEach(commit -> {
commitHashes.add(commit.getId().asString());
commitTimes.add(dateFormatter.format(new Date(commit.getCommitTime())));
});
identifyDeletedFeatures(featureExistenceMap);
createSeriesData(featureExistenceMap, featureFileMap, featureFolderMap);
//System.out.println("FeatureCommitMapper - featureFileMap: " + featureFileMap);
//System.out.println("FeatureCommitMapper - featureFolderMap: " + featureFolderMap);
// Once everything is done, call the onComplete callback
if (onComplete != null) {
onComplete.run();
}
});
commitTask.queue();
});
featureTask.queue();
}

// Method to identify deleted features by comparing extracted features with the current model
private void identifyDeletedFeatures(@NotNull Map<String, Set<GitCommit>> featureExistenceMap) {
featureExistenceMap.keySet().stream()
.filter(feature -> !featureNames.contains(feature))
.forEach(feature -> {
deletedFeatureNames.add(feature);
// Get the last commit for the deleted feature
featureExistenceMap.get(feature).stream()
.max(Comparator.comparingLong(GitCommit::getCommitTime))
.ifPresent(commit -> {
String formattedTime = dateFormatter.format(new Date(commit.getCommitTime()));
deletedFeatureCommits.put(feature, formattedTime);
deletedFeatureCommitHashes.put(feature, commit.getId().asString());
});
});
}

private void createSeriesData(Map<String, Set<GitCommit>> featureExistenceMap,
Map<String, Set<FeatureHistoryAnalyzer.FeatureData>> featureFileMap,
Map<String, Set<FeatureHistoryAnalyzer.FeatureData>> featureFolderMap) {
// Map commit hashes to indices
Map<String, Integer> commitHashToIndex = mapCommitHashesToIndices();

// Map feature names to indices
Map<String, Integer> featureNameToIndex = mapFeatureNamesToIndices();

// Process code annotations
processCodeAnnotations(featureExistenceMap, commitHashToIndex, featureNameToIndex);

// Process feature-to-file mappings
processFeatureFileMappings(featureFileMap, commitHashToIndex, featureNameToIndex);

// Process feature-to-folder mappings
processFeatureFolderMappings(featureFolderMap, commitHashToIndex, featureNameToIndex);
}

private Map<String, Integer> mapCommitHashesToIndices() {
Map<String, Integer> commitHashToIndex = new HashMap<>();
for (int i = 0; i < commitHashes.size(); i++) {
commitHashToIndex.put(commitHashes.get(i), i);
}
return commitHashToIndex;
}

private Map<String, Integer> mapFeatureNamesToIndices() {
Map<String, Integer> featureNameToIndex = new HashMap<>();
for (int i = 0; i < featureNames.size(); i++) {
featureNameToIndex.put(featureNames.get(i), i);
}
return featureNameToIndex;
}

private void processCodeAnnotations(Map<String, Set<GitCommit>> featureExistenceMap,
Map<String, Integer> commitHashToIndex,
Map<String, Integer> featureNameToIndex) {
for (Map.Entry<String, Set<GitCommit>> entry : featureExistenceMap.entrySet()) {
String featureName = entry.getKey();
int featureIndex = featureNameToIndex.getOrDefault(featureName, -1);
if (featureIndex == -1) {
continue; // Feature not found
}
Set<GitCommit> commitsForFeature = entry.getValue();
for (GitCommit commit : commitsForFeature) {
String commitHash = commit.getId().asString();
Integer commitIndex = commitHashToIndex.get(commitHash);

if (commitIndex != null) {
Map<String, Object> dataPoint = new HashMap<>();
dataPoint.put("featureIndex", featureIndex);
dataPoint.put("commitIndex", commitIndex);
dataPoint.put("commitHash", commitHash);
dataPoint.put("commitMessage", commit.getFullMessage());
dataPoint.put("commitAuthor", commit.getAuthor().getName());
dataPoint.put("type", "codeAnnotation");
seriesData.add(dataPoint);
}
}
}
}

private void processFeatureFileMappings(Map<String, Set<FeatureHistoryAnalyzer.FeatureData>> featureFileMap,
Map<String, Integer> commitHashToIndex,
Map<String, Integer> featureNameToIndex) {
for (Map.Entry<String, Set<FeatureHistoryAnalyzer.FeatureData>> entry : featureFileMap.entrySet()) {
String featureName = normalizeFeatureName(entry.getKey());
int featureIndex = featureNameToIndex.getOrDefault(featureName, -1);

if (featureIndex == -1) {
continue; // Feature not found
}
Set<FeatureHistoryAnalyzer.FeatureData> dataSet = entry.getValue();
for (FeatureHistoryAnalyzer.FeatureData data : dataSet) {
String commitHash = data.getCommitHash();
Integer commitIndex = commitHashToIndex.get(commitHash);
if (commitIndex != null) {
Map<String, Object> dataPoint = new HashMap<>();
dataPoint.put("featureIndex", featureIndex);
dataPoint.put("commitIndex", commitIndex);
dataPoint.put("commitHash", commitHash);
dataPoint.put("commitTime", data.getCommitTime());
dataPoint.put("entityName", data.getEntityName()); // File name
dataPoint.put("type", "fileMapping");
seriesData.add(dataPoint);
}
}
}
}

private void processFeatureFolderMappings(Map<String, Set<FeatureHistoryAnalyzer.FeatureData>> featureFolderMap,
Map<String, Integer> commitHashToIndex,
Map<String, Integer> featureNameToIndex) {
for (Map.Entry<String, Set<FeatureHistoryAnalyzer.FeatureData>> entry : featureFolderMap.entrySet()) {
String featureName = normalizeFeatureName(entry.getKey());
int featureIndex = featureNameToIndex.getOrDefault(featureName, -1);
if (featureIndex == -1) {
continue; // Feature not found
}
Set<FeatureHistoryAnalyzer.FeatureData> dataSet = entry.getValue();
for (FeatureHistoryAnalyzer.FeatureData data : dataSet) {
String commitHash = data.getCommitHash();
Integer commitIndex = commitHashToIndex.get(commitHash);
if (commitIndex != null) {
Map<String, Object> dataPoint = new HashMap<>();
dataPoint.put("featureIndex", featureIndex);
dataPoint.put("commitIndex", commitIndex);
dataPoint.put("commitHash", commitHash);
dataPoint.put("commitTime", data.getCommitTime());
dataPoint.put("entityName", data.getEntityName()); // Folder name
dataPoint.put("type", "folderMapping");
seriesData.add(dataPoint);
}
}
}
}
private String normalizeFeatureName(String featureName) {
return featureName.trim().replaceAll(",", "");
}

// Getters for the chart data
public List<String> getFeatureNames() {
// System.out.println("Returning feature names: " + featureNames);
return featureNames;
}

public List<String> getCommitTimes() {
// System.out.println("Returning commit times: " + commitTimes);
return commitTimes;
}

public List<Map<String, Object>> getSeriesData() {
System.out.println("Series Data: " + seriesData);
return seriesData;
}

// Getter for commitHashes
public List<String> getCommitHashes() {
return commitHashes;
}

// Getters for the deleted features and their commit times
public List<String> getDeletedFeatureNames() {
return deletedFeatureNames;
}

public Map<String, String> getDeletedFeatureCommits() {
return deletedFeatureCommits;
}

// Add getter for deletedFeatureCommitHashes
public Map<String, String> getDeletedFeatureCommitHashes() {
return deletedFeatureCommitHashes;
}
}
Loading
Loading