forked from exercism/cli
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathexercise.go
121 lines (107 loc) · 3.29 KB
/
exercise.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
package workspace
import (
"os"
"path"
"path/filepath"
)
// Exercise is an implementation of a problem in a track.
type Exercise struct {
Root string
Track string
Slug string
}
// NewExerciseFromDir constructs an exercise given the exercise directory.
func NewExerciseFromDir(dir string) Exercise {
slug := filepath.Base(dir)
dir = filepath.Dir(dir)
track := filepath.Base(dir)
root := filepath.Dir(dir)
return Exercise{Root: root, Track: track, Slug: slug}
}
// Path is the normalized relative path.
// It always has forward slashes, regardless
// of the operating system.
func (e Exercise) Path() string {
return path.Join(e.Track, e.Slug)
}
// Filepath is the absolute path on the filesystem.
func (e Exercise) Filepath() string {
return filepath.Join(e.Root, e.Track, e.Slug)
}
// MetadataFilepath is the absolute path to the exercise metadata.
func (e Exercise) MetadataFilepath() string {
return filepath.Join(e.Filepath(), metadataFilepath)
}
// LegacyMetadataFilepath is the absolute path to the legacy exercise metadata.
func (e Exercise) LegacyMetadataFilepath() string {
return filepath.Join(e.Filepath(), legacyMetadataFilename)
}
// MetadataDir returns the directory that the exercise metadata lives in.
// For now this is the exercise directory.
func (e Exercise) MetadataDir() string {
return e.Filepath()
}
// HasMetadata checks for the presence of an exercise metadata file.
// If there is no such file, this may be a legacy exercise.
// It could also be an unrelated directory.
func (e Exercise) HasMetadata() (bool, error) {
_, err := os.Lstat(e.MetadataFilepath())
if os.IsNotExist(err) {
return false, nil
}
if err == nil {
return true, nil
}
return false, err
}
// HasLegacyMetadata checks for the presence of a legacy exercise metadata file.
// If there is no such file, it could also be an unrelated directory.
func (e Exercise) HasLegacyMetadata() (bool, error) {
_, err := os.Lstat(e.LegacyMetadataFilepath())
if os.IsNotExist(err) {
return false, nil
}
if err == nil {
return true, nil
}
return false, err
}
// MigrationStatus represents the result of migrating a legacy metadata file.
type MigrationStatus int
// MigrationStatus
const (
MigrationStatusNoop MigrationStatus = iota
MigrationStatusMigrated
MigrationStatusRemoved
)
func (m MigrationStatus) String() string {
switch m {
case MigrationStatusMigrated:
return "\nMigrated metadata\n"
case MigrationStatusRemoved:
return "\nRemoved legacy metadata\n"
default:
return ""
}
}
// MigrateLegacyMetadataFile migrates a legacy metadata file to the modern location.
// This is a noop if the metadata file isn't legacy.
// If both legacy and modern metadata files exist, the legacy file will be deleted.
func (e Exercise) MigrateLegacyMetadataFile() (MigrationStatus, error) {
if ok, _ := e.HasLegacyMetadata(); !ok {
return MigrationStatusNoop, nil
}
if err := os.MkdirAll(filepath.Dir(e.MetadataFilepath()), os.FileMode(0755)); err != nil {
return MigrationStatusNoop, err
}
if ok, _ := e.HasMetadata(); !ok {
if err := os.Rename(e.LegacyMetadataFilepath(), e.MetadataFilepath()); err != nil {
return MigrationStatusNoop, err
}
return MigrationStatusMigrated, nil
}
if err := os.Remove(e.LegacyMetadataFilepath()); err != nil {
return MigrationStatusNoop, err
}
return MigrationStatusRemoved, nil
}