Upgrade to Pro
— share decks privately, control downloads, hide ads and more …
Speaker Deck
Features
Speaker Deck
PRO
Sign in
Sign up for free
Search
Search
サービス間をテストするフレームワーク集
Search
Sponsored
·
SiteGround - Reliable hosting with speed, security, and support you can count on.
→
Shin'ya Ueoka
December 15, 2021
Technology
380
0
Share
サービス間をテストするフレームワーク集
2021-12-15 開催 JJUGナイトセミナー「おうちで!ビール片手にLT大会!」でお話した資料です
https://jjug.doorkeeper.jp/events/129543
Shin'ya Ueoka
December 15, 2021
More Decks by Shin'ya Ueoka
See All by Shin'ya Ueoka
開発インパクトを最大化!エンジニアが主導する組織づくりの実例
ueokande
0
68
エンジニアが主導できる組織づくり ー 製品と事業を進化させる体制へのシフト
ueokande
1
1.6k
どこで動かすか、誰が動かすか 〜 kintoneのインフラ基盤刷新と運用体制のシフト 〜
ueokande
0
330
kintone開発組織のDevOpsへの移り変わりと実践
ueokande
3
1.3k
運用できる開発組織の作り方 ― kintone開発組織のストーリー
ueokande
0
210
英語ができなかった自分達が、グローバルチーム立ち上げに挑戦!?
ueokande
1
1k
技術書典12協賛企業サイボウズゲストトーク
ueokande
0
320
kintone.comを支える技術
ueokande
0
240
SLO策定とアラート設定までの長い道のり
ueokande
6
5k
Other Decks in Technology
See All in Technology
試されDATA SAPPORO [LT]Claude Codeで「ゆっくりデータ分析」
ishikawa_satoru
0
330
バックオフィスPJのPjMをコーポレートITが担うとうまくいく3つの理由
yueda256
1
290
Hooks, Filters & Now Context: Why MCPs Are the “Hooks” of the AI Era
miriamschwab
0
130
マルチモーダル非構造データとの闘い
shibuiwilliam
2
310
Webアクセシビリティは“もしも”に備える設計
tomokusaba
0
170
TanStack Start エコシステムの現在地 / TanStack Start Ecosystem 2026
iktakahiro
1
360
Zero Data Loss Autonomous Recovery Service サービス概要
oracle4engineer
PRO
4
14k
新規サービス開発におけるReact Nativeのリアル〜技術選定の裏側と実践的OSS活用〜
grandbig
2
160
AI環境整備はどのくらい開発生産性を変えうるか? #AI駆動開発 #AI自走環境
ucchi0909
0
110
Podcast配信で広がったアウトプットの輪~70人と音声発信してきた7年間~/outputconf_01
fortegp05
0
240
主催・運営として"場をつくる”というアウトプットのススメ
_mossann_t
0
130
BIツール「Omni」の紹介 @Snowflake中部UG
sagara
0
250
Featured
See All Featured
30 Presentation Tips
portentint
PRO
1
270
Building Adaptive Systems
keathley
44
3k
Principles of Awesome APIs and How to Build Them.
keavy
128
17k
How to Create Impact in a Changing Tech Landscape [PerfNow 2023]
tammyeverts
55
3.3k
Amusing Abliteration
ianozsvald
1
150
A Modern Web Designer's Workflow
chriscoyier
698
190k
JavaScript: Past, Present, and Future - NDC Porto 2020
reverentgeek
52
5.9k
How to build an LLM SEO readiness audit: a practical framework
nmsamuel
1
710
More Than Pixels: Becoming A User Experience Designer
marktimemedia
3
370
How Software Deployment tools have changed in the past 20 years
geshan
0
33k
What does AI have to do with Human Rights?
axbom
PRO
1
2.1k
What the history of the web can teach us about the future of AI
inesmontani
PRO
1
510
Transcript
αʔϏεؒΛςετ͢Δ ϑϨʔϜϫʔΫू αΠϘζגࣜձࣾ ্Ԭ ਅ (@ueokande) 2021-12-15. JJUGφΠτηϛφʔʮ͓͏ͪͰʂϏʔϧยखʹLTେձʂʯ
Me • αΠϘζגࣜձࣾ • άϩʔόϧ͚B2BαʔϏεͷ όοΫΤϯυΛ࡞͍ͬͯ·͢ • ݱͰKotlin +
Spring Boot͕ϝΠϯ 2
ຊͷ͓ • ෳαʔϏε͕ϓϩμΫτΛߏ͢Δ࣌ʢMicroservice, SOA, …ʣ • αʔϏεؒͷΓऔΓΛͲ͏ͬͯςετ͢Δ͔ʁ • REST API௨৴ͷݕূʹཱͭϑϨʔϜϫʔΫΛհ
3 αʔϏε" αʔϏε# )551ϦΫΤετ +40/
հϑϨʔϜϫʔΫ • Mockito … ΫϥΠΞϯτͷ࣮ͱϏδωεϩδοΫΛ • WireMock … ϞοΫαʔόʔΛཱͯͯΫϥΠΞϯτ࣮Λςετ •
Pact … αʔϏεؒ௨৴ͷAPIͷมߋΛݕ 4
Mockito • JavaͷΠϯλʔϑΣΠεɺΫϥεΛϞοΫ • ΫϥΠΞϯτ࣮ΛϞοΫͯ͠ϏδωεϩδοΫ୯ମΛςετ 5 αʔϏε" αʔϏε# )551ϦΫΤετ +40/
͜͜ΒΜͷςετ
Mockito͔͍͔ͭͨ • MockitoΛͬͯΫϥΠΞϯτ࣮ͷϞοΫΛهड़ • ϩδοΫ୯ମΛςε͠ɺΫϥΠΞϯτར༻͞Ε͔ͨݕূ 6 val client = mock<BClient>()
whenever(client.getItems()) .thenReturn(Arrays.asList(item1, …)) val logic = ItemLogic(bClient) val actual = sut.calc() assertEquals("...", actual) verify(bClient, times(1)).getItems() ϞοΫΫϥΠΞϯτͷ࡞ ϞοΫΫϥΠΞϯτΛͯ͠ ରΫϥεʢϏδωεϩδοΫʣͷݕূ MockΫϥΠΞϯτ͕ར༻͞Ε͔ͨݕূ ※ Kotlinͷίʔυྫ
WireMock • ϞοΫͷϨεϙϯεΛฦ͢HTTPαʔόʔΛཱͯΔ • HTTPαʔόʔϦΫΤετΛهͯ͠ظ͢ΔϦΫΤετ͔ݕূ 7 αʔϏε" αʔϏε# )551ϦΫΤετ +40/
͜͜ΒΜͷςετ
WireMock͔͍͔ͭͨ • ϦΫΤετʹର͢ΔϨεϙϯεΛఆٛͯ͠ϞοΫαʔόʔΛىಈ • ΫϥΠΞϯτ࣮ͱɺൃߦ͞ΕͨϦΫΤετ͕ਖ਼͍͔͠ݕূ 8 val server = WireMockServer(12345)
server.stubFor( WireMock.post(“/v1/item/1“) .willReturn(aResponse().withBody("{ ... }”))) server.start() val client = BClientImpl("127.0.0.1", 12345) List<Items> item = bClient.getItem(1L) assertEquals(10, actual.size) server.verify( postRequestedFor(urlPathEqualTo("/v1/items")) .withRequestBody(equalToJson("{ ... }")) ) ϞοΫαʔόʔͷ࡞ͱىಈ ϦΫΤετͷൃߦͱΓͷݕূ ࣮ߦ͞ΕͨϦΫΤετͷݕূ
Pact/Pact JVM • αʔϏεؒͷAPIͷৼΔ͍Λ୲อ͢ΔϑϨʔϜϫʔΫ 1. ίϯγϡʔϚʔɺར༻APIͷظ͢ΔϦΫΤετɾϨεϙϯεʢܖʣΛఆٛ 2. ίϯγϡʔϚʔɺΫϥΠΞϯτ࣮͕ܖͷ௨ΓʹৼΔ͏͔ݕূ 3. ϓϩόΠμʔɺܖͷϦΫΤετʹର͠ظͷϨεϙϯεΛฦ͔͢ݕূ
9 αʔϏε" $POTVNFS αʔϏε# 1SPWJFS ܖ ϞοΫ ϞοΫ ϦΫΤετ Ϩεϙϯε ϦΫΤετ Ϩεϙϯε ࡞ ݕূ ᶃ ᶄ ᶅ json
Pact JVM͍ํʛܖͷఆٛ • ίϯγϡʔϚʔଆ͕ར༻͢ΔAPIͷಈ࡞ΛDLSͰఆٛ 10 @Pact(provider = “service-b", consumer =
“service-a") fun getItemPact(builder: PactDslWithProvider): V4Pact { builder .given("item1 and item2 exist") .uponReceiving("get all items") .method(“GET”).matchPath(“/v1/item/1“) .willRespondWith().status(200).body( LambdaDsl.newJsonBody { it -> it.`object` { it.stringType(“name", “pen") it.numberType(“cost", 100) } } ) } ϦΫΤετʹର͢Δظ͢ΔϨεϙϯεΛఆٛ
Pact JVM͍ํʛίϯγϡʔϚʔͷݕূ • هड़ͨ͠ܖʹରͯ͠ΫϥΠΞϯτ࣮͕ਖ਼͘͠ৼΔ͏͔ݕূ • ϞοΫαʔόʔ͕ىಈͯ͠HTTPϦΫΤετΛهɾݕূ 11 @Test @PactTestFor(providerName =
“service-b", pactMethod = "getItemPact") fun getItem(mockServer: MockServer) { val client = BClientImpl(mockServer.getUrl()) val item = client.getItems(1L) assertEquals("pen", item.name) assertEquals(100, item.cost) }
Pact JVM͍ํʛϓϩόΠμʔͷݕূ • ϓϩόΠμʔ͕ܖ௨ΓͷϨεϙϯεΛฦ͔͢ݕূ • ܖΛͱʹPactϑϨʔϜϫʔΫ͕ࣗಈͰςετ 12 @ExtendWith(SpringExtension::class) @Provider(“service-b") @SpringBootTest(webEnvironment
= SpringBootTest.WebEnvironment.RANDOM_PORT) class ContractTest { @TestTemplate @ExtendWith(PactVerificationInvocationContextProvider::class) fun testTemplate( pact: Pact, interaction: Interaction, context: PactVerificationContext ) { context.verifyInteraction() } }
·ͱΊ • ਖ਼͍͠ϨΠϠʔͰϞοΫͯ͠ਖ਼͘͠ςετ • Mockito … αʔϏεͷΠϯλʔϑΣΠεͱ࣮ΫϥεΛ • WireMock …
ϦΫΤετɾϨεϙϯεΛϞοΫ • Pact … ৗʹมԽ͠ଓ͚ΔαʔϏεؒͷAPIͰഁյతมߋ͕ͳ͍͔νΣοΫ • ͍͠ϨΠϠʔςετɾࣗಈԽͯ҆͠৺ײΛಘΑ͏ 13