-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathindex.html
310 lines (243 loc) · 16.8 KB
/
index.html
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
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
<!DOCTYPE html>
<html xmlns="http://www.w3.org/1999/xhtml" lang="en" xml:lang="en">
<head>
<meta charset="utf-8"/>
<title>Jmim</title>
<link rel="canonical" href="https://ijmo.github.io/">
<meta name="viewport" content="width=device-width, initial-scale=1">
<link rel="stylesheet" href="//maxcdn.bootstrapcdn.com/bootstrap/3.3.0/css/bootstrap.min.css">
<link href="//maxcdn.bootstrapcdn.com/font-awesome/4.2.0/css/font-awesome.min.css" rel="stylesheet">
<link rel="stylesheet" href="//cdnjs.cloudflare.com/ajax/libs/highlight.js/9.7.0/styles/default.min.css">
<link rel="icon" href="/img/favicon.ico">
<link href="/css/screen.css" rel="stylesheet" type="text/css" />
<script async src="https://www.googletagmanager.com/gtag/js?id=UA-143606102-1"></script>
<script>
window.dataLayer = window.dataLayer || [];
function gtag(){dataLayer.push(arguments);}
gtag('js', new Date());
gtag('config', 'UA-143606102-1');
</script>
</head>
<body>
<nav class="navbar navbar-default">
<div class="container">
<div class="navbar-header">
<button type="button" class="navbar-toggle collapsed" data-toggle="collapse" data-target="#navbar" aria-expanded="false" aria-controls="navbar">
<span class="sr-only">Toggle navigation</span>
<span class="icon-bar"></span>
<span class="icon-bar"></span>
<span class="icon-bar"></span>
</button>
<a class="navbar-brand" href="/">🌵 Jmim</a>
</div>
<div id="navbar" class="navbar-collapse collapse">
<ul class="nav navbar-nav navbar-right">
<li class="active" ><a href="/">Home</a></li>
<li
><a href="/archives/">Archives</a></li>
<li
>
<a href="/pages/about/">About</a>
</li>
<li><a href="/feed.xml">RSS</a></li>
<li class="dropdown">
<a href="#" class="dropdown-toggle" data-toggle="dropdown" role="button" aria-expanded="false">
More <span class="caret"></span></a>
<ul class="dropdown-menu" role="menu">
<li class="divider"></li>
<li class="dropdown-header">Recent Posts</li>
<li><a href="/posts/2023-12-07-manage-dependencies-with-gradle-convention-plugin/">Gradle 멀티 프로젝트에서 컨벤션 플러그인으로 의존성 관리</a></li>
<li><a href="/posts/2023-11-19-gradle-8.4-multiproject/">Gradle 8.4 멀티 프로젝트</a></li>
<li><a href="/posts/2019-07-04-solving-problems-the-clojure-way/">함수형으로 문제 해결하기 (스크린샷)</a></li>
<li class="divider"></li>
<li class="dropdown-header">Tags</li>
<li><a href="/tags/gradle%20plugin/">gradle plugin</a></li>
<li><a href="/tags/multi-project/">multi-project</a></li>
<li><a href="/tags/Refactoring/">Refactoring</a></li>
<li><a href="/tags/Data-driven/">Data-driven</a></li>
<li><a href="/tags/convention%20plugin/">convention plugin</a></li>
<li><a href="/tags/dependency/">dependency</a></li>
<li><a href="/tags/Clojure/">Clojure</a></li>
<li><a href="/tags/gradle/">gradle</a></li>
<li><a href="/tags/gradle-8.4/">gradle-8.4</a></li>
<li><a href="/tags/Functional/">Functional</a></li>
</ul>
</li>
</ul>
</div><!--/.nav-collapse -->
</div><!--/.container-fluid -->
</nav>
<button onclick="scrollToTop()" id="btn-top" title="Go to top">Top</button>
<div class="container">
<div class="row">
<div class="col-lg-12">
<div id="content">
<div id="post">
<div class="post-header">
<div id="post-meta" class="row">
<div class="col-lg-6">December 7, 2023</div>
</div>
<h2>Gradle 멀티 프로젝트에서 컨벤션 플러그인으로 의존성 관리</h2>
</div>
<div>
<div class="paragraph"><p><br /></p></div><div class="sect2"><h3 id="_multi_project_structure">Multi-project Structure</h3><div class="paragraph"><p>빠른 이해를 위해 아래처럼 구성해 보았습니다.</p></div><div class="paragraph"><p>예제 소스는 <a href="https://github.com/ijmo/gradle-multiprojects/blob/main/multiproject-with-convention-plugin">Github</a>에 올려두었습니다.</p></div><div class="listingblock"><div class="content"><pre>├── gradle
│ ├── wrapper
│ │ ├── gradle-wrapper.jar
│ │ └── gradle-wrapper.properties
│ └── libs.versions.toml // 1. 버전 카탈로그
│
├── build-logic // 2. 빌드 플러그인을 위한 프로젝트
│ ├── src
│ │ └── main
│ │ └── kotlin
│ │ ├── io.github.ijmo.kotlin-application-conventions.gradle.kts
│ │ ├── io.github.ijmo.kotlin-common-conventions.gradle.kts
│ │ └── io.github.ijmo.kotlin-library-conventions.gradle.kts
│ ├── build.gradle.kts
│ └── settings.gradle.kts
│
├── app // 3. 서브 프로젝트
│ └── build.gradle.kts
├── library // 3. 서브 프로젝트
│ └── build.gradle.kts
│
├── gradlew
├── gradlew.bat
└── settings.gradle.kts // 4. 빌드 플러그인을 include</pre></div></div><div class="paragraph"><p><br /></p></div><div class="sect3"><h4 id="_1_버전_카탈로그">1. 버전 카탈로그</h4><div class="listingblock"><div class="content"><pre>├── gradle
│ ├── wrapper
│ │ └── ...
│ └── libs.versions.toml
...</pre></div></div><div class="ulist"><ul><li><p>버전을 관리하기 위한 <code>libs.versions.toml</code> 을 추가했습니다.</p></li><li><p>의존 라이브러리 목록도 여기서 관리를 할 수 있으나, 저는 버전만 넣어보았습니다.</p></li></ul></div><div class="listingblock"><div class="title">gradle/libs.versions.toml</div><div class="content"><pre class="highlightjs highlight"><code class="language-kotlin hljs" data-lang="kotlin">[versions]
kotlin = "1.9.20"
springBoot = "3.2.0"
springCloud = "2023.0.0"</code></pre></div></div><div class="paragraph"><p><br /></p></div></div><div class="sect3"><h4 id="_2_빌드_플러그인을_위한_프로젝트">2. 빌드 플러그인을 위한 프로젝트</h4><div class="listingblock"><div class="content"><pre>├── build-logic
│ ├── src
│ │ └── main
│ │ └── kotlin
│ │ ├── io.github.ijmo.kotlin-application-conventions.gradle.kts
│ │ ├── io.github.ijmo.kotlin-common-conventions.gradle.kts
│ │ └── io.github.ijmo.kotlin-library-conventions.gradle.kts
│ ├── build.gradle.kts
│ └── settings.gradle.kts
...</pre></div></div><div class="paragraph"><p><br /></p></div><div class="ulist"><ul><li><p><code>libs.versions.toml</code> 을 읽어오도록 합니다. settings 파일 기준으로 상대경로를 입력했습니다.</p></li></ul></div><div class="listingblock"><div class="title">build-logic/settings.gradle.kts</div><div class="content"><pre class="highlightjs highlight"><code class="language-kotlin hljs" data-lang="kotlin">dependencyResolutionManagement {
versionCatalogs {
create("libs") {
from(files("../gradle/libs.versions.toml"))
}
}
}
rootProject.name = "my-example-app-build-logic"</code></pre></div></div><div class="paragraph"><p><br /></p></div><div class="ulist"><ul><li><p>만들려는 플러그인에서 외부 플러그인을 사용하고 싶다면 빌드 스크립트에서 의존성을 추가해주어야 합니다. Spring Boot을 사용하기 위해 <code>gradlePluginPortal</code> 에 있는 <code>spring-boot-gradle-plugin</code> 을 추가해 주었습니다.</p></li><li><p><code>libs.versions.toml</code> 에서 지정한 버전을 가져왔습니다.</p></li></ul></div><div class="listingblock"><div class="title">build-logic/build.gradle.kts</div><div class="content"><pre class="highlightjs highlight"><code class="language-kotlin hljs" data-lang="kotlin">plugins {
`kotlin-dsl`
}
repositories {
gradlePluginPortal()
}
val versionCatalog = extensions.getByType<VersionCatalogsExtension>().named("libs")
dependencies {
implementation("org.jetbrains.kotlin:kotlin-gradle-plugin:${versionCatalog.findVersion("kotlin").get()}")
implementation("org.springframework.boot:spring-boot-gradle-plugin:${versionCatalog.findVersion("springBoot").get()}")
}</code></pre></div></div><div class="paragraph"><p><br /></p></div><div class="ulist"><ul><li><p><code>gradle init</code> 을 통해 프로젝트를 만들면 3가지 모델의 플러그인을 스캐폴딩 해줍니다. 파일 이름에서도 알 수 있듯이 프로젝트 공통으로 쓰이는 'common', 앱에서 쓰기 위한 'application', 라이브러리에서 쓰기 위한 'library' 로 나누어진 구조입니다.</p></li><li><p>서브 프로젝트의 유형이 한 종류(앱 또는 라이브러리)이거나 모든 프로젝트가 같은 의존성을 가진다면 굳이 여러 파일로 나눌 필요없이 한 파일에 다 집어넣으면 되겠습니다.</p></li><li><p>파일이름은 제 기준으로 지었으니 적절하게 바꿔서 쓰시면 되겠습니다.</p></li></ul></div><div class="listingblock"><div class="title">build-logic/src/main/kotlin/io.github.ijmo.kotlin-common-conventions.gradle.kts</div><div class="content"><pre class="highlightjs highlight"><code class="language-kotlin hljs" data-lang="kotlin">plugins {
id("org.jetbrains.kotlin.jvm")
}
repositories {
mavenCentral()
}
dependencies {
constraints {
implementation("org.apache.commons:commons-text:1.10.0")
}
}
testing {
suites {
val test by getting(JvmTestSuite::class) {
useJUnitJupiter("5.9.3")
}
}
}
java {
toolchain {
languageVersion.set(JavaLanguageVersion.of(17))
}
}</code></pre></div></div><div class="paragraph"><p><br /></p></div><div class="ulist"><ul><li><p>빌드 스크립트에서 의존성을 추가했었던 Spring Boot 플러그인을 <code>plugins {}</code> 안에서 <code>apply</code> 해주었습니다.</p></li><li><p>Spring BOM(Bill of Material)을 적용하기 위해 <code>implementation(platform())</code> 을 사용했습니다.</p></li><li><p>여기서 <code>dependencies {}</code> 에 추가한 라이브러리는 <code>resolve</code> 되어 classpath에 추가됩니다. 서브 프로젝트에서 사용하려면 해당 프로젝트의 빌드 스크립트에서 <code>dependencies {}</code> 에 추가해야 합니다.</p></li></ul></div><div class="listingblock"><div class="title">build-logic/src/main/kotlin/io.github.ijmo.kotlin-application-conventions.gradle.kts</div><div class="content"><pre class="highlightjs highlight"><code class="language-kotlin hljs" data-lang="kotlin">plugins {
id("io.github.ijmo.kotlin-common-conventions")
id("org.springframework.boot")
application
}
val versionCatalog: VersionCatalog = extensions.getByType<VersionCatalogsExtension>().named("libs")
val springBootVersion = versionCatalog.findVersion("springBoot").get()
val springCloudVersion = versionCatalog.findVersion("springCloud").get()
dependencies {
implementation(platform("org.springframework.boot:spring-boot-dependencies:$springBootVersion"))
implementation(platform("org.springframework.cloud:spring-cloud-dependencies:$springCloudVersion"))
implementation("com.fasterxml.jackson.module:jackson-module-kotlin")
implementation("io.projectreactor.kotlin:reactor-kotlin-extensions:1.2.2")
implementation("org.jetbrains.kotlinx:kotlinx-coroutines-reactor:1.7.3")
implementation("org.springframework.boot:spring-boot-starter-actuator")
implementation("org.springframework.boot:spring-boot-starter-webflux")
testImplementation("io.projectreactor:reactor-test:3.6.0")
}</code></pre></div></div><div class="paragraph"><p><br /></p></div><div class="ulist"><ul><li><p>라이브러리 프로젝트를 위한 플러그인 입니다.</p></li><li><p>필요한 라이브러리를 <code>dependencies {}</code> 에 추가해주세요.</p></li></ul></div><div class="listingblock"><div class="title">build-logic/src/main/kotlin/io.github.ijmo.kotlin-library-conventions.gradle.kts</div><div class="content"><pre class="highlightjs highlight"><code class="language-kotlin hljs" data-lang="kotlin">plugins {
id("io.github.ijmo.kotlin-common-conventions")
`java-library`
}</code></pre></div></div><div class="paragraph"><p><br /></p></div></div><div class="sect3"><h4 id="_3_빌드_플러그인을_include">3. 빌드 플러그인을 include</h4><div class="listingblock"><div class="content"><pre>├── settings.gradle.kts
...</pre></div></div><div class="ulist"><ul><li><p>만든 플러그인을 <code>pluginManagement {}</code> 을 통해 포함시켜 줍니다.</p></li><li><p>서브 프로젝트도 <code>include()</code> 해줍니다.</p></li></ul></div><div class="listingblock"><div class="title">settings.gradle.kts</div><div class="content"><pre class="highlightjs highlight"><code class="language-kotlin hljs" data-lang="kotlin">pluginManagement {
includeBuild("build-logic")
}
rootProject.name = "my-example-app"
include("app", "library")</code></pre></div></div><div class="paragraph"><p><br /></p></div></div><div class="sect3"><h4 id="_4_서브_프로젝트">4. 서브 프로젝트</h4><div class="listingblock"><div class="content"><pre>├── app
│ └── build.gradle.kts
├── library
│ └── build.gradle.kts
...</pre></div></div><div class="ulist"><ul><li><p>classpath에 추가된 패키지(라이브러리) 중에 각 프로젝트에 사용할 패키지를 빌드 스크립트에 추가합니다.</p></li></ul></div><div class="listingblock"><div class="title">app/build.gradle.kts</div><div class="content"><pre class="highlightjs highlight"><code class="language-kotlin hljs" data-lang="kotlin">plugins {
id("io.github.ijmo.kotlin-application-conventions")
}
dependencies {
implementation("org.springframework.boot:spring-boot-starter-actuator")
implementation("org.springframework.boot:spring-boot-starter-webflux")
testImplementation("io.projectreactor:reactor-test")
implementation(project(":library"))
}</code></pre></div></div><div class="paragraph"><p><br /></p></div></div></div>
</div>
<div id="post-tags">
<b>Tags: </b>
<a href="/tags/gradle%20plugin/">gradle plugin</a>
<a href="/tags/multi-project/">multi-project</a>
<a href="/tags/convention%20plugin/">convention plugin</a>
<a href="/tags/dependency/">dependency</a>
</div>
<div id="prev-next">
<a class="right" href="/posts/2023-11-19-gradle-8.4-multiproject/">Gradle 8.4 멀티 프로젝트 »</a>
</div>
<script src="https://utteranc.es/client.js"
repo="ijmo/ijmo.github.io"
issue-term="/posts/2023-12-07-manage-dependencies-with-gradle-convention-plugin/"
theme="github-light"
crossorigin="anonymous"
async>
</script>
</div>
</div>
</div>
</div>
<footer>Copyleft © 2023 Jinmo Im
<p style="text-align: center;">Powered by <a href="http://cryogenweb.org">Cryogen</a></p></footer>
</div>
<script src="//code.jquery.com/jquery-1.11.0.min.js"></script>
<script src="//maxcdn.bootstrapcdn.com/bootstrap/3.3.0/js/bootstrap.min.js"></script>
<script src="/js/highlight.pack.js" type="application/javascript"></script>
<script>hljs.initHighlightingOnLoad();</script>
<script>
window.onscroll = function() {scrollFunction()};
function scrollFunction() {
if (document.body.scrollTop > 500 || document.documentElement.scrollTop > 500) {
document.getElementById("btn-top").style.display = "block";
} else {
document.getElementById("btn-top").style.display = "none";
}
}
function scrollToTop() {
document.body.scrollTop = 0;
document.documentElement.scrollTop = 0;
}
</script>
</body>
</html>