-
-
Notifications
You must be signed in to change notification settings - Fork 77
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Expose query metadata
in PostgresRowSequence
#504
base: main
Are you sure you want to change the base?
Changes from all commits
2e98065
bedc86a
a6dc894
d299741
68084f9
3933fc4
e70add9
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -9,14 +9,18 @@ public struct PostgresRowSequence: AsyncSequence, Sendable { | |
|
||
typealias BackingSequence = NIOThrowingAsyncSequenceProducer<DataRow, Error, AdaptiveRowBuffer, PSQLRowStream> | ||
|
||
let backing: BackingSequence | ||
let lookupTable: [String: Int] | ||
let columns: [RowDescription.Column] | ||
private let backing: BackingSequence | ||
private let rowStream: PSQLRowStream | ||
var lookupTable: [String: Int] { | ||
self.rowStream.lookupTable | ||
} | ||
var columns: [RowDescription.Column] { | ||
self.rowStream.rowDescription | ||
} | ||
|
||
init(_ backing: BackingSequence, lookupTable: [String: Int], columns: [RowDescription.Column]) { | ||
init(_ backing: BackingSequence, rowStream: PSQLRowStream) { | ||
self.backing = backing | ||
self.lookupTable = lookupTable | ||
self.columns = columns | ||
self.rowStream = rowStream | ||
} | ||
|
||
public func makeAsyncIterator() -> AsyncIterator { | ||
|
@@ -60,13 +64,48 @@ extension PostgresRowSequence { | |
extension PostgresRowSequence.AsyncIterator: Sendable {} | ||
|
||
extension PostgresRowSequence { | ||
/// Collect and return all rows. | ||
/// - Returns: The rows. | ||
public func collect() async throws -> [PostgresRow] { | ||
var result = [PostgresRow]() | ||
for try await row in self { | ||
result.append(row) | ||
} | ||
return result | ||
} | ||
|
||
/// Collect and return all rows, alongside the query metadata. | ||
/// - Returns: The query metadata and the rows. | ||
public func collectWithMetadata() async throws -> (metadata: PostgresQueryMetadata, rows: [PostgresRow]) { | ||
let rows = try await self.collect() | ||
guard let metadata = PostgresQueryMetadata(string: self.rowStream.commandTag) else { | ||
throw PSQLError.invalidCommandTag(self.rowStream.commandTag) | ||
} | ||
return (metadata, rows) | ||
} | ||
|
||
/// Consumes all rows and returns the query metadata. | ||
/// | ||
/// If you don't need the returned query metadata, just use the for-try-await-loop syntax: | ||
/// ```swift | ||
/// for try await row in myRowSequence { | ||
/// /// Process each row | ||
/// } | ||
/// ``` | ||
/// | ||
/// - Parameter onRow: Processes each row. | ||
/// - Returns: The query metadata. | ||
public func consume( | ||
onRow: @Sendable (PostgresRow) throws -> () | ||
) async throws -> PostgresQueryMetadata { | ||
Comment on lines
+98
to
+100
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I'm open to better names. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Also the closure is intentionally not I could change that if you think with |
||
for try await row in self { | ||
try onRow(row) | ||
} | ||
guard let metadata = PostgresQueryMetadata(string: self.rowStream.commandTag) else { | ||
throw PSQLError.invalidCommandTag(self.rowStream.commandTag) | ||
} | ||
return metadata | ||
} | ||
} | ||
|
||
struct AdaptiveRowBuffer: NIOAsyncSequenceProducerBackPressureStrategy { | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,3 +1,4 @@ | ||
import Atomics | ||
import Logging | ||
import XCTest | ||
import PostgresNIO | ||
|
@@ -46,6 +47,58 @@ final class AsyncPostgresConnectionTests: XCTestCase { | |
} | ||
} | ||
|
||
func testSelect10kRowsAndConsume() async throws { | ||
let eventLoopGroup = MultiThreadedEventLoopGroup(numberOfThreads: 1) | ||
defer { XCTAssertNoThrow(try eventLoopGroup.syncShutdownGracefully()) } | ||
let eventLoop = eventLoopGroup.next() | ||
|
||
let start = 1 | ||
let end = 10000 | ||
|
||
try await withTestConnection(on: eventLoop) { connection in | ||
let rows = try await connection.query("SELECT generate_series(\(start), \(end));", logger: .psqlTest) | ||
|
||
let counter = ManagedAtomic(0) | ||
let metadata = try await rows.consume { row in | ||
let element = try row.decode(Int.self) | ||
let newCounter = counter.wrappingIncrementThenLoad(ordering: .relaxed) | ||
XCTAssertEqual(element, newCounter) | ||
} | ||
|
||
XCTAssertEqual(metadata.command, "SELECT") | ||
XCTAssertEqual(metadata.oid, nil) | ||
XCTAssertEqual(metadata.rows, 10000) | ||
|
||
XCTAssertEqual(counter.load(ordering: .relaxed), end) | ||
} | ||
} | ||
|
||
func testSelect10kRowsAndCollect() async throws { | ||
let eventLoopGroup = MultiThreadedEventLoopGroup(numberOfThreads: 1) | ||
defer { XCTAssertNoThrow(try eventLoopGroup.syncShutdownGracefully()) } | ||
let eventLoop = eventLoopGroup.next() | ||
|
||
let start = 1 | ||
let end = 10000 | ||
|
||
try await withTestConnection(on: eventLoop) { connection in | ||
let rows = try await connection.query("SELECT generate_series(\(start), \(end));", logger: .psqlTest) | ||
let (metadata, elements) = try await rows.collectWithMetadata() | ||
var counter = 0 | ||
for row in elements { | ||
let element = try row.decode(Int.self) | ||
XCTAssertEqual(element, counter + 1) | ||
counter += 1 | ||
} | ||
|
||
XCTAssertEqual(metadata.command, "SELECT") | ||
XCTAssertEqual(metadata.oid, nil) | ||
XCTAssertEqual(metadata.rows, 10000) | ||
|
||
XCTAssertEqual(counter, end) | ||
} | ||
} | ||
|
||
func testSelectActiveConnection() async throws { | ||
let eventLoopGroup = MultiThreadedEventLoopGroup(numberOfThreads: 1) | ||
defer { XCTAssertNoThrow(try eventLoopGroup.syncShutdownGracefully()) } | ||
|
@@ -207,7 +260,7 @@ final class AsyncPostgresConnectionTests: XCTestCase { | |
|
||
try await withTestConnection(on: eventLoop) { connection in | ||
// Max binds limit is UInt16.max which is 65535 which is 3 * 5 * 17 * 257 | ||
// Max columns limit is 1664, so we will only make 5 * 257 columns which is less | ||
// Max columns limit appears to be ~1600, so we will only make 5 * 257 columns which is less | ||
// Then we will insert 3 * 17 rows | ||
// In the insertion, there will be a total of 3 * 17 * 5 * 257 == UInt16.max bindings | ||
// If the test is successful, it means Postgres supports UInt16.max bindings | ||
|
@@ -241,13 +294,9 @@ final class AsyncPostgresConnectionTests: XCTestCase { | |
unsafeSQL: "INSERT INTO table1 VALUES \(insertionValues)", | ||
binds: binds | ||
) | ||
try await connection.query(insertionQuery, logger: .psqlTest) | ||
|
||
let countQuery = PostgresQuery(unsafeSQL: "SELECT COUNT(*) FROM table1") | ||
let countRows = try await connection.query(countQuery, logger: .psqlTest) | ||
var countIterator = countRows.makeAsyncIterator() | ||
let insertedRowsCount = try await countIterator.next()?.decode(Int.self, context: .default) | ||
XCTAssertEqual(rowsCount, insertedRowsCount) | ||
let result = try await connection.query(insertionQuery, logger: .psqlTest) | ||
let metadata = try await result.collectWithMetadata().metadata | ||
XCTAssertEqual(metadata.rows, rowsCount) | ||
Comment on lines
-246
to
+299
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. The metadata contains the count of the inserted rows so no need to do another query. |
||
|
||
let dropQuery = PostgresQuery(unsafeSQL: "DROP TABLE table1") | ||
try await connection.query(dropQuery, logger: .psqlTest) | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm open to better names.