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

Cleanup - Initial Project Refactor #3

Merged
merged 5 commits into from
Jul 31, 2024
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
3 changes: 2 additions & 1 deletion .github/workflows/build.yml
Original file line number Diff line number Diff line change
Expand Up @@ -25,4 +25,5 @@ jobs:
with:
go-version: 1.21
- run: go get -v -t -d ./...
- run: go build -v .
- run: go build -v .
- run: go test ./... --coverprofile=cover.out
174 changes: 21 additions & 153 deletions cmd/root.go
Original file line number Diff line number Diff line change
Expand Up @@ -4,20 +4,19 @@ Copyright © 2024 NAME HERE <EMAIL ADDRESS>
package cmd

import (
"encoding/json"
"fmt"
"log"
"os"
"path/filepath"
"strings"

"github.com/mona-actions/gh-commit-remap/internal/commitremap"
"github.com/spf13/cobra"
)

// Struct to represent a single entry in the commit map
type CommitMapEntry struct {
Old string
New string
func init() {
rootCmd.Flags().StringP("mapping-file", "c", "", "Path to the commit map file Example: /path/to/commit-map")
rootCmd.MarkFlagRequired("mapping-file")

rootCmd.Flags().StringP("migration-archive", "m", "", "Path to the migration archive Example: /path/to/migration-archive.tar.gz")
rootCmd.MarkFlagRequired("migration-archive")
}

// rootCmd represents the base command when called without any subcommands
Expand All @@ -26,9 +25,20 @@ var rootCmd = &cobra.Command{
Short: "remaps commit hashes in a GitHub archive",
Long: `Is a CLI tool that can remap commits hashed
after performing a history re-write when performing a migration For exam`,
// Uncomment the following line if your bare application
// has an action associated with it:
Run: main,
Run: func(cmd *cobra.Command, args []string) {
mapPath, _ := cmd.Flags().GetString("mapping-file")
commitMap, err := commitremap.ParseCommitMap(mapPath)
if err != nil {
log.Fatalf("Error parsing commit map: %v", err)
}

// config to define the types of files to process
types := []string{"pull_requests", "issues", "issue_events"}

archivePath, _ := cmd.Flags().GetString("migration-archive")

commitremap.ProcessFiles(archivePath, types, commitMap)
},
}

// Execute adds all child commands to the root command and sets flags appropriately.
Expand All @@ -39,145 +49,3 @@ func Execute() {
os.Exit(1)
}
}

func init() {
// Here you will define your flags and configuration settings.
// Cobra supports persistent flags, which, if defined here,
// will be global for your application.

// rootCmd.PersistentFlags().StringVar(&cfgFile, "config", "", "config file (default is $HOME/.gh-commit-remap.yaml)")

// Cobra also supports local flags, which will only run
// when this action is called directly.
rootCmd.Flags().StringP("mapping-file", "c", "", "Path to the commit map file Example: /path/to/commit-map")
rootCmd.MarkFlagRequired("mapping-file")

rootCmd.Flags().StringP("migration-archive", "m", "", "Path to the migration archive Example: /path/to/migration-archive.tar.gz")
rootCmd.MarkFlagRequired("migration-archive")
}

func main(cmd *cobra.Command, args []string) {
// leaving this for now to quickly test the code
//mapPath := "test/TestRepo.git/filter-repo/commit-map"
mapPath, _ := cmd.Flags().GetString("mapping-file")
commitMap, err := parseCommitMap(mapPath)
if err != nil {
log.Fatalf("Error parsing commit map: %v", err)
}

// config to define the types of files to process
types := []string{"pull_requests", "issues"}

// leaving this for now to quickly test the code
//archivePath := "test/3723ff5e-4b7e-11ef-9bf5-2aca377420b3"
archivePath, _ := cmd.Flags().GetString("migration-archive")

processFiles(archivePath, types, commitMap)
}

// Parses the file and returns a map of old commit hashes to new commit hashes
func parseCommitMap(filePath string) (*[]CommitMapEntry, error) {
commitMap := []CommitMapEntry{}

// Read the commit-map file
content, err := os.ReadFile(filePath)
if err != nil {
return nil, err
}
// Split the file content into lines
lines := strings.Split(string(content), "\n")

// Iterate over the lines and parse the old and new commit hashes
for _, line := range lines {
if strings.TrimSpace(line) == "" {
continue
}

fields := strings.Fields(line)
if len(fields) != 2 {
return nil, fmt.Errorf("invalid line: %s", line)
}

commitMap = append(commitMap, CommitMapEntry{
Old: fields[0],
New: fields[1],
})
}
return &commitMap, nil
}

func replaceSHA(data interface{}, oldSHA string, newSHA string) {
if data == nil {
return
}

switch v := data.(type) {
case map[string]interface{}:
for key, value := range v {
if str, ok := value.(string); ok && str == oldSHA {
v[key] = newSHA
} else {
replaceSHA(value, oldSHA, newSHA)
}
}
case []interface{}:
for i, value := range v {
if str, ok := value.(string); ok && str == oldSHA {
v[i] = newSHA
} else {
replaceSHA(value, oldSHA, newSHA)
}
}
default:
// Unsupported type, do nothing
}
}

func updateMetadataFile(filePath string, commitMap *[]CommitMapEntry) {
// Read the JSON file
data, err := os.ReadFile(filePath)
if err != nil {
log.Fatalf("Error reading data: %v", err)
}

var dataMap interface{}
err = json.Unmarshal(data, &dataMap)
if err != nil {
log.Fatalf("Error unmarshaling data: %v", err)
}

// Iterate over the commit map and replace the old commit hashes with the new ones
for _, commit := range *commitMap {
replaceSHA(dataMap, commit.Old, commit.New)
}

// Marshal the updated data to JSON and pretty print it
updatedData, err := json.MarshalIndent(dataMap, "", " ")
if err != nil {
log.Fatalf("Error marshaling updated data: %v", err)
}

// Overwrite the original file with the updated data
err = os.WriteFile(filePath, updatedData, 0644)
if err != nil {
log.Fatalf("Error writing updated data: %v", err)
}
}

func processFiles(archiveLocation string, prefixes []string, commitMap *[]CommitMapEntry) {

for _, prefix := range prefixes {
// Get a list of all files that match the pattern
files, err := filepath.Glob(filepath.Join(archiveLocation, prefix+"_*.json"))
if err != nil {
log.Fatalf("Error getting files: %v", err)
}

// Process each file
for _, file := range files {
log.Println("Processing file:", file)

updateMetadataFile(file, commitMap)
}
}
}
2 changes: 1 addition & 1 deletion go.mod
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
module github.com/kuhlman-labs/gh-commit-remap
module github.com/mona-actions/gh-commit-remap

go 1.22.5

Expand Down
123 changes: 123 additions & 0 deletions internal/commitremap/commitremap.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,123 @@
package commitremap

import (
"encoding/json"
"fmt"
"log"
"os"
"path/filepath"
"strings"
)

// Struct to represent a single entry in the commit map
type CommitMapEntry struct {
Old string
New string
}

// Parses the file and returns a map of old commit hashes to new commit hashes
func ParseCommitMap(filePath string) (*[]CommitMapEntry, error) {
commitMap := []CommitMapEntry{}

// Read the commit-map file
content, err := os.ReadFile(filePath)
if err != nil {
return nil, err
}
// Split the file content into lines
lines := strings.Split(string(content), "\n")

// Iterate over the lines and parse the old and new commit hashes
for _, line := range lines {
if strings.TrimSpace(line) == "" {
continue
}

fields := strings.Fields(line)
if len(fields) != 2 {
return nil, fmt.Errorf("invalid line: %s", line)
}

commitMap = append(commitMap, CommitMapEntry{
Old: fields[0],
New: fields[1],
})
}
return &commitMap, nil
}

func ProcessFiles(archiveLocation string, prefixes []string, commitMap *[]CommitMapEntry) {

for _, prefix := range prefixes {
// Get a list of all files that match the pattern
files, err := filepath.Glob(filepath.Join(archiveLocation, prefix+"_*.json"))
if err != nil {
log.Fatalf("Error getting files: %v", err)
}

// Process each file
for _, file := range files {
log.Println("Processing file:", file)

updateMetadataFile(file, commitMap)
}
}
}

func updateMetadataFile(filePath string, commitMap *[]CommitMapEntry) {
// Read the JSON file
data, err := os.ReadFile(filePath)
if err != nil {
log.Fatalf("Error reading data: %v", err)
}

var dataMap interface{}
err = json.Unmarshal(data, &dataMap)
if err != nil {
log.Fatalf("Error unmarshaling data: %v", err)
}

// Iterate over the commit map and replace the old commit hashes with the new ones
for _, commit := range *commitMap {
replaceSHA(dataMap, commit.Old, commit.New)
}

// Marshal the updated data to JSON and pretty print it
updatedData, err := json.MarshalIndent(dataMap, "", " ")
if err != nil {
log.Fatalf("Error marshaling updated data: %v", err)
}

// Overwrite the original file with the updated data
err = os.WriteFile(filePath, updatedData, 0644)
if err != nil {
log.Fatalf("Error writing updated data: %v", err)
}
}

func replaceSHA(data interface{}, oldSHA string, newSHA string) {
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

In a later PR, I would like to rework this function as I'm getting super spooked around the usage of empty interfaces here (e.g., any type)

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@ssulei7 good catch!

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Agreed

if data == nil {
return
}

switch v := data.(type) {
case map[string]interface{}:
for key, value := range v {
if str, ok := value.(string); ok && str == oldSHA {
v[key] = newSHA
} else {
replaceSHA(value, oldSHA, newSHA)
}
}
case []interface{}:
for i, value := range v {
if str, ok := value.(string); ok && str == oldSHA {
v[i] = newSHA
} else {
replaceSHA(value, oldSHA, newSHA)
}
}
default:
// Unsupported type, do nothing
}
}
Loading
Loading