@@ -9,82 +9,50 @@ import javax.inject.Inject
9
9
10
10
public open class DependencySyncTask @Inject constructor(
11
11
@Input
12
- public val settings : DependencySyncExtension
12
+ public val settings : DependencySyncExtension
13
13
) : DefaultTask() {
14
14
15
- @TaskAction
16
- public fun action () {
17
- val gradleBuildFile = File (settings.gradleBuildFile.get())
18
- val tomlFile = File (settings.typeSafeFile.get())
15
+ internal companion object {
19
16
20
- var buildText = gradleBuildFile.readText()
21
-
22
- val versionsBlockReg = """ ^[\s\S]*\[versions\]([\S\s]*)\[libraries\]""" .toRegex()
23
- val librariesBlockReg = """ ^[\s\S]*\[libraries\]([\S\s]*)\[\S*\]""" .toRegex()
24
17
val versionReg = """ ((\S*)\s*=\s*"([^"]*)"\s*)""" .toRegex()
18
+
25
19
val simpleReg = """ ((\S*)\s*=\s*"([^":]*):([^":]*):([^"]*)")""" .toRegex()
26
20
val complexReg =
27
21
""" ((\S*)\s*=\s*\{\s*module\s*=\s*"([^:]*):(\S*)"\s*,\s*version\.ref\s*=\s*"([^"]*)"\s*\}\s*)""" .toRegex()
22
+ }
28
23
29
- val dummyConfig = project.configurations.getByName(DependencySyncPlugin .CONFIGURATION_NAME )
24
+ @TaskAction
25
+ public fun action () {
26
+ val gradleBuildFile = File (settings.gradleBuildFile.get())
27
+ val tomlFile = File (settings.typeSafeFile.get())
30
28
31
- val buildFileDeps = dummyConfig.dependencies
32
- .filter { it.version != null }
33
- .map { Dep (it.group!! , it.name, it.version!! ) }
34
- .groupBy { it.group + it.name }
35
- .values
36
- .map { depsByArtifact -> depsByArtifact.maxByOrNull { it.version }!! }
29
+ var buildText = gradleBuildFile.readText()
37
30
38
- val thisFileDeps = buildFileDeps
39
- .groupBy { it.group }
40
- .map { (group, lst) ->
41
- group to lst.associateBy { it.name }
42
- }
43
- .toMap()
31
+ val dependencySyncConfig = project.configurations
32
+ .getByName(DependencySyncPlugin .CONFIGURATION_NAME )
44
33
45
- var tomlText = tomlFile.readText( )
34
+ val parsedBuildFile = ParsedBuildFile .create(dependencySyncConfig )
46
35
47
- val versionBlock = versionsBlockReg.find(tomlText)?.destructured?.component1() ? : " "
48
- val librariesBlock = librariesBlockReg.find(tomlText)?.destructured?.component1() ? : " "
36
+ val parsedToml = ParsedToml .create(tomlFile)
49
37
50
- val tomlVersions = versionReg.findAll(versionBlock)
51
- .map { it.destructured }
52
- .map { (wholeString, name, version) ->
53
- TomlVersion (wholeString, name, version)
54
- }
55
- .associateBy { it.defName }
56
-
57
- val complexEntries = complexReg.findAll(librariesBlock)
58
- .toList()
59
- .map { it.destructured }
60
- .map { (line, defName, group, name, versionRef) ->
61
- val versionDef = tomlVersions.getValue(versionRef)
62
- val version = versionDef.version
63
-
64
- TomlEntry .Complex (
65
- originalText = line,
66
- defName = defName,
67
- versionDef = versionDef,
68
- dep = Dep (group, name, version)
69
- )
70
- }
38
+ var tomlText = parsedToml.text
39
+ val tomlLibrariesBlock = parsedToml.librariesBlock
40
+ val tomlEntries = parsedToml.entries
71
41
72
- val simpleEntries = simpleReg.findAll(librariesBlock)
73
- .map { it.destructured }
74
- .map { (line, defName, group, name, version) ->
75
- TomlEntry .Simple (line, defName, Dep (group, name, version))
76
- }
77
- .toList()
42
+ buildText = addMissingBuildFileDeps(tomlEntries, parsedBuildFile, buildText)
43
+
44
+ val updatedparsedBuildFile = ParsedBuildFile .create(dependencySyncConfig)
78
45
79
- val tomlEntries = (complexEntries + simpleEntries).toMutableList()
46
+ val buildFileDeps = updatedparsedBuildFile.deps
47
+ val groupedBuildFileDeps = updatedparsedBuildFile.groupedDeps
80
48
81
- val leftovers = buildFileDeps
49
+ val missingFromToml = buildFileDeps
82
50
.filterNot { buildFileDep ->
83
51
tomlEntries.any { it.dep.group == buildFileDep.group && it.dep.name == buildFileDep.name }
84
52
}
85
53
.map { it.toSimpleToml() }
86
54
87
- tomlEntries.addAll(leftovers )
55
+ tomlEntries.addAll(missingFromToml )
88
56
89
57
val grouped = tomlEntries.groupBy {
90
58
it.dep
@@ -107,16 +75,23 @@ public open class DependencySyncTask @Inject constructor(
107
75
.sorted()
108
76
.joinToString(" \n " , " \n " , " \n " )
109
77
110
- tomlText = tomlText.replace(librariesBlock , newToml)
78
+ tomlText = tomlText.replace(tomlLibrariesBlock , newToml)
111
79
112
80
tomlEntries
81
+ .filterNot {
82
+ val tomlDep = it.dep
83
+
84
+ val ktsDep = groupedBuildFileDeps[tomlDep.group]?.get(tomlDep.name)
85
+
86
+ ktsDep != null
87
+ }
113
88
.forEach { entry ->
114
89
115
90
val tomlDep = entry.dep
116
91
117
- val ktsDep = thisFileDeps[ tomlDep.group]
92
+ val ktsDep = groupedBuildFileDeps.get( tomlDep.group)
118
93
?.get(tomlDep.name)
119
- ? : throw GradleException ( " Dependency ` $tomlDep ` is declared in $tomlFile but not in $gradleBuildFile " )
94
+ ? : return @forEach
120
95
121
96
if (ktsDep.version != tomlDep.version && entry is TomlEntry .Complex ) {
122
97
val newer = maxOf(ktsDep.version, tomlDep.version)
@@ -153,4 +128,42 @@ public open class DependencySyncTask @Inject constructor(
153
128
gradleBuildFile.writeText(buildText)
154
129
tomlFile.writeText(tomlText)
155
130
}
131
+
132
+ private fun addMissingBuildFileDeps (
133
+ tomlEntries : MutableList <TomlEntry >,
134
+ parsedBuildFile : ParsedBuildFile ,
135
+ originalBuildText : String
136
+ ): String {
137
+ var buildText = originalBuildText
138
+
139
+ val groupedBuildFileDeps = parsedBuildFile.groupedDeps
140
+
141
+ val (missingFromBuildFile, inBuildFile) = tomlEntries
142
+ .map { it.dep }
143
+ .partition { dep -> groupedBuildFileDeps[dep.group]?.get(dep.name) == null }
144
+
145
+ val lastBuildDep = inBuildFile.lastOrNull()
146
+ ? : throw GradleException (" Cannot find any `dependencySync` dependencies in build file" )
147
+
148
+ val lastBuildDepRegex = """ (.*dependencySync.*['"])$lastBuildDep (['"].*)""" .toRegex()
149
+
150
+ val lastBuildDepResult = buildText.lines()
151
+ .asSequence()
152
+ .mapNotNull { lastBuildDepRegex.find(it) }
153
+ .firstOrNull()
154
+ ? : throw GradleException (
155
+ " Cannot find a dependency declaration for $lastBuildDep , " +
156
+ " using the regex ${lastBuildDepRegex.pattern} , in build file"
157
+ )
158
+
159
+ val (prefix, suffix) = lastBuildDepResult.destructured
160
+
161
+ missingFromBuildFile.forEach { missingDep ->
162
+ val new = """ $prefix$missingDep$suffix
163
+ |$prefix$lastBuildDep$suffix
164
+ """ .trimMargin()
165
+ buildText = buildText.replace(lastBuildDepResult.value, new)
166
+ }
167
+ return buildText
168
+ }
156
169
}
0 commit comments