Skip to content

Conversation

@MahdiBM
Copy link
Contributor

@MahdiBM MahdiBM commented Aug 25, 2024

Resolves #499

Checklist

  • Implement basic SimpleQuery handling.
  • Tests.
  • Fix decoding rows from a SimpleQuery response with text format.

Decoding rows from a simple-query response seems to fail for now, although no-row queries do succeed. I think because as described in Postgres docs, the returned format of data for simple-query is text, and PostgresNIO doesn't properly support that.

EDIT: For SimpleQuery only, I removed the line that forced formats to be binary, and things are working now.
I'm not sure how well PostgresNIO supports the text format, but it works for strings and ints.

Not Implemented In This PR

  • Multiple statements in a single SimpleQuery.

@codecov
Copy link

codecov bot commented Aug 25, 2024

Codecov Report

Attention: Patch coverage is 82.83019% with 91 lines in your changes missing coverage. Please review.

Project coverage is 63.05%. Comparing base (ecbc3eb) to head (942a3cb).

Files with missing lines Patch % Lines
...ection State Machine/SimpleQueryStateMachine.swift 80.27% 58 Missing ⚠️
...nection State Machine/ConnectionStateMachine.swift 83.07% 33 Missing ⚠️
Additional details and impacted files
@@            Coverage Diff             @@
##             main     #505      +/-   ##
==========================================
+ Coverage   62.16%   63.05%   +0.88%     
==========================================
  Files         130      131       +1     
  Lines       10422    10919     +497     
==========================================
+ Hits         6479     6885     +406     
- Misses       3943     4034      +91     
Files with missing lines Coverage Δ
...es/PostgresNIO/Connection/PostgresConnection.swift 41.04% <100.00%> (+2.60%) ⬆️
Sources/PostgresNIO/New/PSQLTask.swift 81.39% <100.00%> (+3.61%) ⬆️
...urces/PostgresNIO/New/PostgresChannelHandler.swift 84.70% <100.00%> (+0.19%) ⬆️
...stgresNIO/New/PostgresFrontendMessageEncoder.swift 100.00% <100.00%> (ø)
...nection State Machine/ConnectionStateMachine.swift 66.69% <83.07%> (+2.42%) ⬆️
...ection State Machine/SimpleQueryStateMachine.swift 80.27% <80.27%> (ø)
🚀 New features to boost your workflow:
  • ❄️ Test Analytics: Detect flaky tests, report on failures, and find test suite problems.

@MahdiBM MahdiBM changed the title Implement SimpleQuery Implement Basic SimpleQuery Handling Aug 25, 2024
Comment on lines +441 to +453
/// Run a simple text-only query on the Postgres server the connection is connected to.
/// WARNING: This function is not yet API and is incomplete.
/// The return type will change to another stream.
///
/// - Parameters:
/// - query: The simple query to run
/// - logger: The `Logger` to log into for the query
/// - file: The file, the query was started in. Used for better error reporting.
/// - line: The line, the query was started in. Used for better error reporting.
/// - Returns: A ``PostgresRowSequence`` containing the rows the server sent as the query result.
/// The sequence be discarded.
@discardableResult
public func __simpleQuery(
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not sure what will happen to this PR but the name is underscored for now so we can merge this PR as-is and I can then create follow-up PRs.

@MahdiBM MahdiBM force-pushed the mmbm-simple-query branch from 65b21d6 to 5961b27 Compare August 25, 2024 22:20

mutating func dataRowReceived(_ dataRow: DataRow) -> Action {
switch self.state {
case .rowDescriptionReceived(let queryContext, let columns):
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Happens for queries with actual rows being returned.


mutating func commandCompletedReceived(_ commandTag: String) -> Action {
switch self.state {
case .messagesSent(let context):
Copy link
Contributor Author

@MahdiBM MahdiBM Aug 25, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Happens for queries that do not return any rows.

return .succeedQuery(context.promise, with: result)
}

case .rowDescriptionReceived(let context, _):
Copy link
Contributor Author

@MahdiBM MahdiBM Aug 25, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Happens for queries that do query some rows to be returned, but no rows matched the query.

demandStateMachine.receivedRow(dataRow)
state = .streaming(columns, demandStateMachine)
let result = QueryResult(value: .rowDescription(columns), logger: queryContext.logger)
return .succeedQuery(queryContext.promise, with: result)
Copy link
Contributor Author

@MahdiBM MahdiBM Aug 25, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Need to succeed the query here when we receive the first row and state is RowDescription, since we notice that there are rows to be read.

return self.avoidingStateMachineCoW { state -> Action in
state = .commandComplete(commandTag: commandTag)
let result = QueryResult(value: .noRows(.tag(commandTag)), logger: context.logger)
return .succeedQuery(context.promise, with: result)
Copy link
Contributor Author

@MahdiBM MahdiBM Aug 25, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Need to succeed the query here when we receive the command-complete row and state is RowDescription, since we notice that there are not rows to be read.

import Logging
@testable import PostgresNIO

class SimpleQueryStateMachineTests: XCTestCase {
Copy link
Contributor Author

@MahdiBM MahdiBM Aug 25, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Mostly copy-pasted from extended-query state machine, though properly adjusted to simple-query state machine and the way it should behave.

@MahdiBM MahdiBM force-pushed the mmbm-simple-query branch from 25a69ec to 39de666 Compare August 26, 2024 10:22
@MahdiBM MahdiBM force-pushed the mmbm-simple-query branch from 35f6d6e to 26fe020 Compare August 26, 2024 14:56
@fabianfett
Copy link
Collaborator

I wonder if a better way is to build a dedicated SimpleQueryStateMachine? Wdyt?

@MahdiBM
Copy link
Contributor Author

MahdiBM commented Oct 9, 2024

@fabianfett i moved the stuff to that, mid-PR.
There is a SimpleQueryStateMachine file, though in GitHub UI you might need to chase it and push that "Load diff" button.

@MahdiBM
Copy link
Contributor Author

MahdiBM commented Oct 9, 2024

FWIW we're already using this function on prod and it's been working as it should.
Of course we're not querying rows with multiple queries because that's not implemented.
The tests should also be pretty good 🙂.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

None yet

Projects

None yet

Development

Successfully merging this pull request may close these issues.

Unable to send multiple SQL commands in one simpleQuery

2 participants