Compare commits

...

22 Commits

Author SHA1 Message Date
a33f28225b runner fix 2023-09-02 22:04:07 -04:00
4aa4e11345 testing runners 2023-09-02 21:53:59 -04:00
922678b223 update readme 2022-12-27 21:50:12 -06:00
e156eef14c Update 'README.md' 2022-12-27 21:25:55 -06:00
d7db51824d Merge pull request 'Directory option' (#4) from dirOption into master
Reviewed-on: #4
2022-12-27 21:20:35 -06:00
434ba07d59 modifications to dirOption 2022-12-24 22:38:53 -05:00
384391add1 working file option 2022-12-24 22:11:12 -05:00
702188bc22 wip: starting work on dir option 2022-12-24 17:46:20 -05:00
73f8e80a23 update version 2022-08-24 19:11:51 -04:00
022f3a8937 Merge pull request 'added PDF output, tag filtering functionality' (#6) from staging into master
Reviewed-on: #6
2022-08-23 18:33:09 -05:00
3582eae47c Merge pull request 'Exception handling' (#7) from exceptionHandling into staging
Reviewed-on: #7
2022-08-23 18:29:05 -05:00
5873bdb2a6 try catch all the things 2022-08-23 19:31:39 -04:00
895f322bd6 Merge branch 'staging' of https://scm.wyattjmiller.com/wymiller/DrillSergeant into staging 2022-08-23 19:13:05 -04:00
7721031d62 added command line arg for tag filtering, added arg logic for tag filtering 2022-08-23 18:57:18 -04:00
e4d8e3f66e added commit filtering by tag 2022-08-23 18:56:56 -04:00
82b8cd5c04 Merge pull request 'Added PDF export functionality' (#4) from pdfDataService into staging
Reviewed-on: #4
2022-08-23 17:55:17 -05:00
908c9cf9dc modified readme 2022-08-23 16:35:31 -04:00
1894b68c5a removed imports 2022-08-23 16:34:55 -04:00
7478f38782 added command line args to export to a PDF 2022-08-23 11:30:01 -04:00
ddabd6080f added functionality to export commit reports to a PDF 2022-08-23 11:29:12 -04:00
7504a5a293 added PdfSharpCore dependency 2022-08-23 11:28:48 -04:00
9e312e9c6c added contrib 2022-08-04 17:47:49 -04:00
7 changed files with 369 additions and 78 deletions

24
.github/workflows/dotnet.yml vendored Normal file
View File

@ -0,0 +1,24 @@
# This workflow will build a .NET project
# For more information see: https://docs.github.com/en/actions/automating-builds-and-tests/building-and-testing-net
name: .NET build
on: [push]
jobs:
build:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v3
- name: Setup .NET
uses: actions/setup-dotnet@v3
with:
dotnet-version: 6.0.x
- name: Restore dependencies
run: dotnet restore
- name: Build
run: dotnet build --no-restore
- name: Test
run: dotnet test --no-build --verbosity normal

View File

@ -5,13 +5,17 @@ public class CommitDetail
private List<string>? _authors;
private SortedList<string, int>? _commitDetails;
private string _currentBranch;
private string? _outputOption;
private string? _branchOption;
private string? _tagOption;
private string? _selectedDir;
public List<string>? Authors
{
public List<string>? Authors
{
get { return _authors; }
set { _authors = value; }
}
public SortedList<string, int>? CommitDetails
{
get { return _commitDetails; }
@ -24,16 +28,28 @@ public class CommitDetail
set { _currentBranch = value; }
}
public CommitDetail()
public string? SelectedDir
{
get { return _selectedDir; }
set { _selectedDir = value; }
}
public CommitDetail(string? dir)
{
_selectedDir = !String.IsNullOrEmpty(dir) ? dir : Directory.GetCurrentDirectory();
_authors = new List<string>();
_commitDetails = new SortedList<string, int>();
_currentBranch = GetCurrentBranch();
// unused, might be used later for a refactor
_outputOption = null;
_branchOption = null;
_tagOption = null;
}
public void GetCurrentCommitsByName()
public void GetCurrentCommitsByName()
{
using (var repo = new Repository(Directory.GetCurrentDirectory()))
using (var repo = new Repository(_selectedDir))
{
foreach (var c in repo.Commits)
{
@ -42,7 +58,7 @@ public class CommitDetail
_authors.Add(c.Author.Name);
}
}
foreach (var a in _authors)
{
int commitCount = repo.Commits.Where(r => r.Author.Name == a).Count();
@ -53,7 +69,7 @@ public class CommitDetail
public void GetCurrentCommitsByEmail()
{
using (var repo = new Repository(Directory.GetCurrentDirectory()))
using (var repo = new Repository(_selectedDir))
{
foreach (var c in repo.Commits)
{
@ -62,7 +78,7 @@ public class CommitDetail
_authors.Add(c.Author.Email);
}
}
foreach (var a in _authors)
{
int commitCount = repo.Commits.Where(r => r.Author.Email == a).Count();
@ -73,7 +89,7 @@ public class CommitDetail
public int GetCommitTotal()
{
using (var repo = new Repository(Directory.GetCurrentDirectory()))
using (var repo = new Repository(_selectedDir))
{
return repo.Commits.Count();
}
@ -81,7 +97,7 @@ public class CommitDetail
public string GetCurrentBranch()
{
using (var repo = new Repository(Directory.GetCurrentDirectory()))
using (var repo = new Repository(_selectedDir))
{
return repo.Head.Reference.TargetIdentifier;
}
@ -89,15 +105,23 @@ public class CommitDetail
public void GetCommitsByBranch(string branchName)
{
using (var repo = new Repository(Directory.GetCurrentDirectory()))
using (var repo = new Repository(_selectedDir))
{
var branchResult = repo.Branches[branchName];
if (branchResult == null)
try
{
branchResult = repo.Branches[$"origin/{branchName}"];
var remoteBranch = repo.CreateBranch(branchName, branchResult.Tip);
repo.Branches.Update(remoteBranch, b => b.UpstreamBranch = $"refs/heads/{branchName}");
if (branchResult == null)
{
branchResult = repo.Branches[$"origin/{branchName}"];
var remoteBranch = repo.CreateBranch(branchName, branchResult.Tip);
repo.Branches.Update(remoteBranch, b => b.UpstreamBranch = $"refs/heads/{branchName}");
}
}
catch (System.Exception)
{
Console.WriteLine($"Cannot fetch {branchName} branch.");
Environment.Exit(1);
}
foreach (var c in branchResult.Commits)
@ -107,7 +131,7 @@ public class CommitDetail
_authors.Add(c.Author.Name);
}
}
foreach (var a in _authors)
{
int commitCount = branchResult.Commits.Where(r => r.Author.Name == a).Count();
@ -115,13 +139,41 @@ public class CommitDetail
}
}
}
public void GetCommitsByTag(string tagName)
{
using (var repo = new Repository(Directory.GetCurrentDirectory()))
using (var repo = new Repository(_selectedDir))
{
var tagResult = repo.Tags[tagName];
System.Console.WriteLine(tagResult);
try
{
var tagResult = repo.Tags[tagName].Target.Sha;
var commitFilter = new CommitFilter
{
IncludeReachableFrom = tagResult,
};
var query = repo.Commits.QueryBy(commitFilter);
foreach (var c in query)
{
if (!_authors.Contains(c.Author.Name))
{
_authors.Add(c.Author.Name);
}
}
foreach (var a in _authors)
{
int commitCount = query.Where(r => r.Author.Name == a).Count();
_commitDetails.Add(a, commitCount);
}
}
catch (System.Exception)
{
Console.WriteLine($"Cannot find the tag {tagName}");
Environment.Exit(3);
}
}
}
}

21
CONTRIBUTING.md Normal file
View File

@ -0,0 +1,21 @@
# Contributing
Thank you so much for your interest for contributing! Hopefully, you'll find something here.
###### Table of contents
- Immeditate help
- Future help
## Immeditate Help
Some of these topics are listed in the issues page. Please post your thoughts there (if you have any).
- [Testing (CI/CD?)](https://scm.wyattjmiller.com/wymiller/DrillSergeant/issues/1)
- [Exception handling](https://scm.wyattjmiller.com/wymiller/DrillSergeant/issues/2)
- [Fiter by tag](https://scm.wyattjmiller.com/wymiller/DrillSergeant/issues/3)
## Future help
Have a great idea of where this project will go? File an issue!

43
DAL/PdfDataService.cs Normal file
View File

@ -0,0 +1,43 @@
using PdfSharpCore.Drawing;
using PdfSharpCore.Pdf;
public class PdfDataService : IDataService
{
private string _fileName;
private string _pathName;
private string _dateTime;
public PdfDataService()
{
_dateTime = DateTime.Now.ToString("yyyyMMddhhmm");
_fileName = $"CommitReport-{_dateTime}.pdf";
_pathName = Directory.GetCurrentDirectory() + "/" + _fileName;
}
public void WriteAll(SortedList<string, int> dictResults)
{
// sort the dictionary passed in
var sortedResults = dictResults.OrderByDescending(d => d.Value).ToList();
// setting the created document
var document = new PdfDocument();
var page = document.AddPage();
// heading
var gfx = XGraphics.FromPdfPage(page);
var fontHeading = new XFont("Times New Roman", 36, XFontStyle.Underline);
gfx.DrawString("Commit Report", fontHeading, XBrushes.Black, new XRect(0, 0, page.Width, page.Height), XStringFormats.TopCenter);
// details
var fontDetails = new XFont("Times New Roman", 16, XFontStyle.Regular);
int positionValue = 50;
foreach (var i in sortedResults)
{
gfx.DrawString($"{i.Key}: {i.Value}", fontDetails, XBrushes.Black, new XRect(0, positionValue, page.Width, page.Width), XStringFormats.TopCenter);
positionValue += 25;
}
document.Save(_fileName);
}
}

View File

@ -7,6 +7,7 @@
<ImplicitUsings>enable</ImplicitUsings>
<Nullable>enable</Nullable>
<AssemblyName>drillsergeant</AssemblyName>
<Version>1.2.0</Version>
</PropertyGroup>
<ItemGroup>
@ -14,6 +15,7 @@
<PackageReference Include="Libgit2sharp" Version="0.27.0-preview-0182" />
<PackageReference Include="LibGit2Sharp.NativeBinaries" Version="2.0.315-alpha.0.9" />
<PackageReference Include="Pastel" Version="3.0.1" />
<PackageReference Include="PdfSharpCore" Version="1.3.32" />
<PackageReference Include="System.CommandLine" Version="2.0.0-beta4.22272.1" />
</ItemGroup>

View File

@ -7,7 +7,7 @@ static class Program
var outputOption = new Option<string>(
"--output",
"Specify the output given to the user"
).FromAmong("stdout", "xlsx");
).FromAmong("stdout", "xlsx", "pdf");
outputOption.AddAlias("-o");
var branchOption = new Option<string>(
@ -16,72 +16,147 @@ static class Program
);
branchOption.AddAlias("-b");
var tagOption = new Option<string>(
"--tag",
"Specify the tag to filter by"
);
tagOption.AddAlias("-t");
var dirOption = new Option<string>(
"--file",
"Specify a git directory"
);
dirOption.AddAlias("-f");
var rootCommand = new RootCommand("Get a tally of contributors' commits")
{
outputOption,
branchOption,
tagOption,
dirOption,
};
rootCommand.SetHandler((outputOptionValue, branchOptionValue) => {
CommitDetail commits = new CommitDetail();
switch (outputOptionValue)
rootCommand.SetHandler((outputOptionValue, branchOptionValue, tagOptionValue, dirOptionValue) =>
{
CommitDetail commits = new CommitDetail(dirOptionValue);
switch (outputOptionValue)
{
case "stdout":
StdOutDataService outDataService = new StdOutDataService();
DataAccess dataAccess = new DataAccess(outDataService);
if (branchOptionValue != null && tagOptionValue != null)
{
case "stdout":
StdOutDataService outDataService = new StdOutDataService();
DataAccess dataAccess = new DataAccess(outDataService);
Console.WriteLine("Please specify either a branch or a tag");
Environment.Exit(2);
}
else if (branchOptionValue != null && tagOptionValue == null)
{
switch (branchOptionValue)
{
case null:
commits.GetCurrentCommitsByName();
dataAccess.WriteData(commits.CommitDetails);
break;
default:
commits.GetCommitsByBranch(branchOptionValue);
dataAccess.WriteData(commits.CommitDetails);
break;
}
}
else if (branchOptionValue == null && tagOptionValue != null)
{
commits.GetCommitsByTag(tagOptionValue);
dataAccess.WriteData(commits.CommitDetails);
}
break;
case "xlsx":
ExcelDataService excelDataService = new ExcelDataService();
DataAccess dataAccessExcelCase = new DataAccess(excelDataService);
switch (branchOptionValue)
{
case null:
commits.GetCurrentCommitsByName();
dataAccess.WriteData(commits.CommitDetails);
break;
default:
commits.GetCommitsByBranch(branchOptionValue);
dataAccess.WriteData(commits.CommitDetails);
break;
}
break;
case "xlsx":
ExcelDataService excelDataService = new ExcelDataService();
DataAccess dataAccessExcelCase = new DataAccess(excelDataService);
if (branchOptionValue != null && tagOptionValue != null)
{
Console.WriteLine("Please specify either a branch or a tag.");
Environment.Exit(2);
}
else if (branchOptionValue != null && tagOptionValue == null)
{
switch (branchOptionValue)
{
case null:
commits.GetCurrentCommitsByName();
dataAccessExcelCase.WriteData(commits.CommitDetails);
break;
default:
commits.GetCommitsByBranch(branchOptionValue);
dataAccessExcelCase.WriteData(commits.CommitDetails);
break;
}
}
else if (branchOptionValue == null && tagOptionValue != null)
{
commits.GetCommitsByTag(tagOptionValue);
dataAccessExcelCase.WriteData(commits.CommitDetails);
}
break;
case "pdf":
PdfDataService pdfDataService = new PdfDataService();
DataAccess dataAccessPdfCase = new DataAccess(pdfDataService);
switch (branchOptionValue)
{
case null:
commits.GetCurrentCommitsByName();
dataAccessExcelCase.WriteData(commits.CommitDetails);
break;
default:
commits.GetCommitsByBranch(branchOptionValue);
dataAccessExcelCase.WriteData(commits.CommitDetails);
break;
}
break;
switch (branchOptionValue)
{
case null:
StdOutDataService stdOutDataService = new StdOutDataService();
DataAccess dataAccessNullCase = new DataAccess(stdOutDataService);
switch (branchOptionValue)
{
case null:
commits.GetCurrentCommitsByName();
dataAccessNullCase.WriteData(commits.CommitDetails);
break;
default:
commits.GetCommitsByBranch(branchOptionValue);
dataAccessNullCase.WriteData(commits.CommitDetails);
break;
}
commits.GetCurrentCommitsByName();
dataAccessPdfCase.WriteData(commits.CommitDetails);
break;
default:
System.Console.WriteLine("This should not happen...");
Environment.Exit(90);
commits.GetCommitsByBranch(branchOptionValue);
dataAccessPdfCase.WriteData(commits.CommitDetails);
break;
}
},
outputOption, branchOption);
break;
case null:
StdOutDataService stdOutDataService = new StdOutDataService();
DataAccess dataAccessNullCase = new DataAccess(stdOutDataService);
if (branchOptionValue != null && tagOptionValue != null)
{
Console.WriteLine("Please specify either a branch or a tag.");
Environment.Exit(2);
}
else if (branchOptionValue != null && tagOptionValue == null)
{
switch (branchOptionValue)
{
case null:
commits.GetCurrentCommitsByName();
dataAccessNullCase.WriteData(commits.CommitDetails);
break;
default:
commits.GetCommitsByBranch(branchOptionValue);
dataAccessNullCase.WriteData(commits.CommitDetails);
break;
}
}
else if (branchOptionValue == null && tagOptionValue != null)
{
commits.GetCommitsByTag(tagOptionValue);
dataAccessNullCase.WriteData(commits.CommitDetails);
}
else
{
commits.GetCurrentCommitsByName();
dataAccessNullCase.WriteData(commits.CommitDetails);
}
break;
default:
System.Console.WriteLine("This should not happen...");
Environment.Exit(4);
break;
}
},
outputOption, branchOption, tagOption, dirOption);
rootCommand.Invoke(args);
}

View File

@ -9,6 +9,7 @@ Licensed by the Mozilla Public License v2.0 (see [LICENSE](./LICENSE.md))
###### Table of Contents
- [Overview](#overview)
- [Usage](#usage)
- [Installation](#installation)
- [Release](#release)
- [Source](#source)
@ -24,17 +25,74 @@ Licensed by the Mozilla Public License v2.0 (see [LICENSE](./LICENSE.md))
When I was in university, I wanted a program that would tally up my git commits in a certain project that I'm working on. Based on search engine ninja skills, I couldn't find a program that did this. I decided to scratch my own itch.
I present DrillSergeant, where you can get a tally of the contributors' commits in a certain project. Additionally, you can filter by branch to get commits by branch.
I present DrillSergeant. This is a command-line application to tally up commits, ordered by commit count per contributor. In other words, DrillSergeant orders the contributors by how many commits they have made in descending order. For example, the first contributor listed will have the most commits in the project. The second contributor will have the second most commits, and so on.
## Usage
Please see the wiki for information.
To use this command-line application, you must be in a git project. Otherwise, a nasty error will appear and you won't get your report. You can filter by branch or by tag (cannot be both, another nasty error appears here). You can get your report in your terminal, in a spreadsheet, or a PDF.
### Command-line arguments
```bash
-o, --output <pdf|stdout|xlsx> Specify the output given to the user
-b, --branch <branch> Specify the branch to filter by
-t, --tag <tag> Specify the tag to filter by
-f, --file <file> Specify a git directory
```
Throw the `-h` flag for quick assistance. Throw the `--version` flag for version information.
### Examples
To get a commit report to your terminal:
```bash
drillsergeant -o stdout
```
Or you can simply:
```bash
drillsergeant
```
To get a commit report to your terminal filtered by the `devel` branch:
```
drillsergeant -b devel
```
To get a commit report to a spreadsheet:
```bash
drillsergeant -o xlsx
```
To get a commit report to a PDF file filtered by the `devel` branch:
```bash
drillsergeant -o pdf -b devel
```
To get a tally based on a directory:
```bash
drillsergeant -f /home/user/drillsergeant
```
This also works on enviroment variables too!
Got too many contributors to fit onto your terminal? Run this by installing [bat](https://github.com/sharkdp/bat):
```bash
drillsergeant | bat
```
## Installation
There's are two forms of installation: releases and building from sources. This readme goes through both.
### Releases
### Release
You may get releases from the [releases page](https://scm.wyattjmiller.com/wymiller/DrillSergeant/releases). This is the recommended way to start using DrillSergeant.
@ -42,6 +100,20 @@ There will be three separate downloads: Windows (x86 64-bit), Linux (x86 64-bit)
Once downloaded and extracted, you can move it to your `$PATH`.
On Windows (Powershell):
```powershell
Copy-Item drillsergeant C:\Windows\
```
On Linux:
```bash
cp drillsergeant /usr/local/bin
```
The following commands assume you have elevated privileges.
If you do not have any of these platforms, read on to source installation as that's the next best option.
### Source
@ -79,7 +151,7 @@ Then, you can find the binary in the `publish/` directory. You can move this exe
## Known Issues
- A user must be in the root of a git project in order for this program to run.
- A user must be in the root of a git project in order for this program to run (unless the user runs the `-f` flag when invoking `drillsergeant`).
## Next Steps/Contributing
@ -97,3 +169,5 @@ Thank you to the developers, engineers, project managers, and contributors of th
- [OpenXML](https://docs.microsoft.com/en-us/office/open-xml/open-xml-sdk)
- [System.CommandLine](https://github.com/dotnet/command-line-api)
- [Pastel](https://github.com/silkfire/Pastel)
- [PdfSharpCore](https://github.com/ststeiger/PdfSharpCore)