3
3

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 1 year has passed since last update.

YUMEMI Advent CalendarAdvent Calendar 2021

Day 13

Vaporをasync/awaitに書き直してみる(比較する)

Posted at

YUMEMI Advent Calendar13日目の記事です(!?!?!?!?!?!?)
Valorantが面白すぎて...(言い訳)

Swiftで非同期処理

Swift5.5から非同期処理の機能としてasync/awaitが追加されました。が、自分は全くキャッチアップできてないので参考にした記事を載せておきます。

Vaporでの非同期処理

今までのVaporでもSwiftNIOのEventLoopFutureを使った非同期処理は行われていました。

DBアクセスが1番に挙げられると思いますが、中間テーブルなどを使い出すとコールバック地獄がちょっと大変でした。

/*-- 番組情報と出演者情報をupsertし、リレーションを張る --*/
let program = upsertProgram(item.program, app.db)
let personalities = item.personalities.map { upsertPersonality($0, app.db) }
return personalities.map {
    $0.flatMap { personality -> EventLoopFuture<Void> in
        program.flatMap { program -> EventLoopFuture<Void> in
            program.$personalities.isAttached(to: personality, on: app.db).flatMap {
                isAttach -> EventLoopFuture<Void> in
                if !isAttach {
                    return program.$personalities.attach(personality, on: app.db)
                }
                return app.eventLoopGroup.future()
            }
        }
    }
}
.flatten(on: app.eventLoopGroup.next())

これは流石に他にいい書き方絶対あると思うんですけどね......

EventLoopFutureのasync/await対応はSwiftNIO2.33.0で行われており、

それ以外のVaporの一部の機能のasync/await対応はVapor4.50.0で行われました

書き直す

今回もagqrの番組表APIを題材にしてます。公式ドキュメントに移行方法があるのでこれを参考にします

まずSwift5.5にあげます

- FROM swift:5.2-focal-slim
+ FROM swift:5.5-focal-slim

VaporとSwiftNIOを更新します(ビルドすればいいので特にやることはない)

Fluentの書き直し

対応が入っているFluentは結構簡単です。

/// 開始時間と終了時間を基準に(unique)insert or updateを行う.
func upsertProgram(_ program: Program, _ db: Database) -> EventLoopFuture<Program> {
    return
        Program
        .query(on: db)
        .filter(\.$startDatetime == program.startDatetime)
        .filter(\.$endDatetime == program.endDatetime)
        .first()
        .flatMap { dbProgram -> EventLoopFuture<Program> in
            program.id = dbProgram?.id
            // idを挿入するだけだとうまくupdateできなかったため、判定要素を上書きする
            program._$id.exists = dbProgram?.id != nil
            return program.save(on: db).transform(to: program)
        }
}
/// 開始時間と終了時間を基準に(unique)insert or updateを行う.
func upsertProgram(_ program: Program, _ db: Database) async throws -> Program {
    let dbProgram = try? await Program.query(on: db)
        .filter(\.$startDatetime == program.startDatetime)
        .filter(\.$endDatetime == program.endDatetime)
        .first()

    program.id = dbProgram?.id
    // idを挿入するだけだとうまくupdateできなかったため、判定要素を上書きする
    program._$id.exists = dbProgram != nil

    try await program.save(on: db)
    return program
}

単純にネストが減る感じ

ちなみにさっきの大変なやつもスッキリ書けます。

let insertedProgram = try await upsertProgram(programGuide.program, app.db)
var insertedPersonalities: [Personality] = []
for personality in programGuide.personalities {
    let p = try await upsertPersonality(personality, app.db)
    insertedPersonalities.append(p)
}
for personality in insertedPersonalities {
    let isAttached = try? await insertedProgram.$personalities.isAttached(to: personality, on: app.db)
    if isAttached == false {
        try await insertedProgram.$personalities.attach(personality, on: app.db)
    }
}

未対応のVapor定義オブジェクトを書き直す

リリースノートにある通り、

  • Client
  • RoutesBuilder
  • ResponseEncodable
  • Responder
  • Middleware
  • AsyncPasswordHasher
  • Cache
  • View
  • ViewRenderer

についてはasync/awaitに対応しているので簡単に(上のように)書き直せますが、これ以外のは少し手間がかかります。

公式にもありますが、EventLoopFutureからasyncへ・asyncからEventLoopFutureへ書き換える必要があります。

// EventLoopFuture -> async
let futureResult = try await someMethodThatReturnsAFuture().get()
// async -> EventLoopFuture
let promise = request.eventLoop.makePromise(of: String.self)
promise.completeWithTask {
    try await someAsyncFunctionThatGetsAString()
}
let futureString: EventLoopFuture<String> = promise.futureResult

ScheduledJobの場合 func run(context: QueueContext) -> EventLoopFuture<Void> を要求しているので 、async用func asyncRun(context: QueueContext) async -> Void を定義してあげることで書き換えました。

func run(context: QueueContext) -> EventLoopFuture<Void> {
    // fetchWeeklyの返り値はEventLoopFuture<[Data?]>
    return client.fetchWeekly(app: context.application).flatMap { responses -> EventLoopFuture<Void> in
        responses
            .map { response -> EventLoopFuture<Void> in
                guard let response = response else {
                    return context.application.eventLoopGroup.future(error: "番組表データがありませんでした。")
                }
                let programGuide = self.parser.parse(response)
                return self.repository.save(programGuide, app: context.application)
            }
            .flatten(on: context.application.eventLoopGroup.next())
    }
}
func run(context: QueueContext) -> EventLoopFuture<Void> {
    let promise = context.eventLoop.makePromise(of: Void.self)
    promise.completeWithTask {
        await self.asyncRun(context: context)
    }
    return promise.futureResult

}

func asyncRun(context: QueueContext) async -> Void {
    // fetchWeeklyは async -> [Data?]
    let responses = await client.fetchWeekly(app: context.application)
    for response in responses {
        guard let response = response else {
            print("番組表データがありません。")
            continue
        }
        let programGuide = self.parser.parse(response)
        await self.repository.save(programGuide, app: context.application)
    }
}

感想

めちゃ書きやすい、最高。

throwにtryが必須なようにasyncにはawaitが必須になっているため、jsのasyncとはイメージが少し違った。

arrayの各要素に対してasyncな処理を加えるみたいな書き方をしたい時に、単純にmapだとダメでtaskを使わないといけなかったり、forのが綺麗だったりが結構引っかかりポイントだった。

アドベントカレンダー1週間遅れガチ謝罪、アドベントカレンダーは こちら

ちなみに書き直した時のPRはこれです -> https://github.com/sun-yryr/agqr-program-guide/pull/16

3
3
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
3
3

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?