| /* |
| Licensed to the Apache Software Foundation (ASF) under one |
| or more contributor license agreements. See the NOTICE file |
| distributed with this work for additional information |
| regarding copyright ownership. The ASF licenses this file |
| to you 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 |
| |
| http://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 gremlingo |
| |
| import ( |
| "crypto/tls" |
| "sync" |
| "time" |
| ) |
| |
| type connectionState int |
| |
| const ( |
| initialized connectionState = iota + 1 |
| established |
| closed |
| closedDueToError |
| ) |
| |
| type connection struct { |
| logHandler *logHandler |
| protocol protocol |
| results *synchronizedMap |
| state connectionState |
| } |
| |
| type connectionSettings struct { |
| authInfo AuthInfoProvider |
| tlsConfig *tls.Config |
| keepAliveInterval time.Duration |
| writeDeadline time.Duration |
| connectionTimeout time.Duration |
| enableCompression bool |
| readBufferSize int |
| writeBufferSize int |
| enableUserAgentOnConnect bool |
| } |
| |
| func (connection *connection) errorCallback() { |
| connection.logHandler.log(Error, errorCallback) |
| connection.state = closedDueToError |
| |
| // This callback is called from within protocol.readLoop. Therefore, |
| // it cannot wait for it to finish to avoid a deadlock. |
| if err := connection.protocol.close(false); err != nil { |
| connection.logHandler.logf(Error, failedToCloseInErrorCallback, err.Error()) |
| } |
| } |
| |
| func (connection *connection) close() error { |
| if connection.state != established { |
| return newError(err0101ConnectionCloseError) |
| } |
| connection.logHandler.log(Info, closeConnection) |
| var err error |
| if connection.protocol != nil { |
| err = connection.protocol.close(true) |
| } |
| connection.state = closed |
| return err |
| } |
| |
| func (connection *connection) write(request *request) (ResultSet, error) { |
| if connection.state != established { |
| return nil, newError(err0102WriteConnectionClosedError) |
| } |
| connection.logHandler.log(Debug, writeRequest) |
| requestID := request.requestID.String() |
| connection.logHandler.logf(Debug, creatingRequest, requestID) |
| resultSet := newChannelResultSet(requestID, connection.results) |
| connection.results.store(requestID, resultSet) |
| return resultSet, connection.protocol.write(request) |
| } |
| |
| func (connection *connection) activeResults() int { |
| return connection.results.size() |
| } |
| |
| // createConnection establishes a connection with the given parameters. A connection should always be closed to avoid |
| // leaking connections. The connection has the following states: |
| // |
| // initialized: connection struct is created but has not established communication with server |
| // established: connection has established communication established with the server |
| // closed: connection was closed by the user. |
| // closedDueToError: connection was closed internally due to an error. |
| func createConnection(url string, logHandler *logHandler, connSettings *connectionSettings) (*connection, error) { |
| conn := &connection{ |
| logHandler, |
| nil, |
| &synchronizedMap{map[string]ResultSet{}, sync.Mutex{}}, |
| initialized, |
| } |
| logHandler.log(Info, connectConnection) |
| protocol, err := newGremlinServerWSProtocol(logHandler, Gorilla, url, connSettings, conn.results, conn.errorCallback) |
| if err != nil { |
| logHandler.logf(Warning, failedConnection) |
| conn.state = closedDueToError |
| return nil, err |
| } |
| conn.protocol = protocol |
| conn.state = established |
| return conn, err |
| } |
| |
| type synchronizedMap struct { |
| internalMap map[string]ResultSet |
| syncLock sync.Mutex |
| } |
| |
| func (s *synchronizedMap) store(key string, value ResultSet) { |
| s.syncLock.Lock() |
| defer s.syncLock.Unlock() |
| s.internalMap[key] = value |
| } |
| |
| func (s *synchronizedMap) load(key string) ResultSet { |
| s.syncLock.Lock() |
| defer s.syncLock.Unlock() |
| return s.internalMap[key] |
| } |
| |
| func (s *synchronizedMap) delete(key string) { |
| s.syncLock.Lock() |
| defer s.syncLock.Unlock() |
| delete(s.internalMap, key) |
| } |
| |
| func (s *synchronizedMap) size() int { |
| s.syncLock.Lock() |
| defer s.syncLock.Unlock() |
| return len(s.internalMap) |
| } |
| |
| func (s *synchronizedMap) closeAll(err error) { |
| s.syncLock.Lock() |
| defer s.syncLock.Unlock() |
| for _, resultSet := range s.internalMap { |
| resultSet.setError(err) |
| resultSet.unlockedClose() |
| } |
| } |