forked from pgjdbc/r2dbc-postgresql
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathPostgresqlCopyIn.java
98 lines (84 loc) · 3.6 KB
/
PostgresqlCopyIn.java
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
/*
* Copyright 2017 the original author or authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* https://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package io.r2dbc.postgresql;
import io.netty.buffer.ByteBuf;
import io.netty.util.ReferenceCountUtil;
import io.netty.util.ReferenceCounted;
import io.r2dbc.postgresql.client.Client;
import io.r2dbc.postgresql.message.backend.BackendMessage;
import io.r2dbc.postgresql.message.backend.CommandComplete;
import io.r2dbc.postgresql.message.backend.CopyInResponse;
import io.r2dbc.postgresql.message.backend.ReadyForQuery;
import io.r2dbc.postgresql.message.frontend.CopyData;
import io.r2dbc.postgresql.message.frontend.CopyDone;
import io.r2dbc.postgresql.message.frontend.CopyFail;
import io.r2dbc.postgresql.message.frontend.Query;
import io.r2dbc.postgresql.util.Assert;
import io.r2dbc.postgresql.util.Operators;
import org.reactivestreams.Publisher;
import reactor.core.publisher.Flux;
import reactor.core.publisher.Mono;
import static io.r2dbc.postgresql.PostgresqlResult.toResult;
/**
* An implementation for {@link CopyData} PostgreSQL queries.
*/
final class PostgresqlCopyIn {
private final ConnectionResources context;
PostgresqlCopyIn(ConnectionResources context) {
this.context = Assert.requireNonNull(context, "context must not be null");
}
Mono<Long> copy(String sql, Publisher<ByteBuf> stdin) {
return Flux.from(stdin)
.map(CopyData::new)
.as(messages -> copyIn(sql, messages));
}
private Mono<Long> copyIn(String sql, Flux<CopyData> copyDataMessages) {
Client client = context.getClient();
Flux<BackendMessage> backendMessages = copyDataMessages
.doOnNext(client::send)
.doOnError((e) -> sendCopyFail(e.getMessage()))
.doOnDiscard(ReferenceCounted.class, ReferenceCountUtil::release)
.thenMany(client.exchange(Mono.just(CopyDone.INSTANCE)));
return startCopy(sql)
.concatWith(backendMessages)
.doOnCancel(() -> sendCopyFail("Cancelled"))
.as(Operators::discardOnCancel)
.as(messages -> toResult(context, messages, ExceptionFactory.INSTANCE).getRowsUpdated());
}
private Flux<BackendMessage> startCopy(String sql) {
return context.getClient().exchange(
// ReadyForQuery is returned when an invalid query is provided
backendMessage -> backendMessage instanceof CopyInResponse || backendMessage instanceof ReadyForQuery,
Mono.just(new Query(sql))
)
.doOnNext(message -> {
if (message instanceof CommandComplete) {
throw new IllegalArgumentException("Copy from stdin query expected, sql='" + sql + "', message=" + message);
}
});
}
private void sendCopyFail(String message) {
context.getClient().exchange(Mono.just(new CopyFail("Copy operation failed: " + message)))
.as(Operators::discardOnCancel)
.subscribe();
}
@Override
public String toString() {
return "PostgresqlCopyIn{" +
"context=" + this.context +
'}';
}
}