| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258 |
- /*
- * 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 thrift
- import (
- "bytes"
- "io"
- "io/ioutil"
- "net/http"
- "net/url"
- "strconv"
- )
- // Default to using the shared http client. Library users are
- // free to change this global client or specify one through
- // THttpClientOptions.
- var DefaultHttpClient *http.Client = http.DefaultClient
- type THttpClient struct {
- client *http.Client
- response *http.Response
- url *url.URL
- requestBuffer *bytes.Buffer
- header http.Header
- nsecConnectTimeout int64
- nsecReadTimeout int64
- }
- type THttpClientTransportFactory struct {
- options THttpClientOptions
- url string
- isPost bool
- }
- func (p *THttpClientTransportFactory) GetTransport(trans TTransport) TTransport {
- if trans != nil {
- t, ok := trans.(*THttpClient)
- if ok && t.url != nil {
- if t.requestBuffer != nil {
- t2, _ := NewTHttpPostClientWithOptions(t.url.String(), p.options)
- return t2
- }
- t2, _ := NewTHttpClientWithOptions(t.url.String(), p.options)
- return t2
- }
- }
- if p.isPost {
- s, _ := NewTHttpPostClientWithOptions(p.url, p.options)
- return s
- }
- s, _ := NewTHttpClientWithOptions(p.url, p.options)
- return s
- }
- type THttpClientOptions struct {
- // If nil, DefaultHttpClient is used
- Client *http.Client
- }
- func NewTHttpClientTransportFactory(url string) *THttpClientTransportFactory {
- return NewTHttpClientTransportFactoryWithOptions(url, THttpClientOptions{})
- }
- func NewTHttpClientTransportFactoryWithOptions(url string, options THttpClientOptions) *THttpClientTransportFactory {
- return &THttpClientTransportFactory{url: url, isPost: false, options: options}
- }
- func NewTHttpPostClientTransportFactory(url string) *THttpClientTransportFactory {
- return NewTHttpPostClientTransportFactoryWithOptions(url, THttpClientOptions{})
- }
- func NewTHttpPostClientTransportFactoryWithOptions(url string, options THttpClientOptions) *THttpClientTransportFactory {
- return &THttpClientTransportFactory{url: url, isPost: true, options: options}
- }
- func NewTHttpClientWithOptions(urlstr string, options THttpClientOptions) (TTransport, error) {
- parsedURL, err := url.Parse(urlstr)
- if err != nil {
- return nil, err
- }
- response, err := http.Get(urlstr)
- if err != nil {
- return nil, err
- }
- client := options.Client
- if client == nil {
- client = DefaultHttpClient
- }
- httpHeader := map[string][]string{"Content-Type": []string{"application/x-thrift"}}
- return &THttpClient{client: client, response: response, url: parsedURL, header: httpHeader}, nil
- }
- func NewTHttpClient(urlstr string) (TTransport, error) {
- return NewTHttpClientWithOptions(urlstr, THttpClientOptions{})
- }
- func NewTHttpPostClientWithOptions(urlstr string, options THttpClientOptions) (TTransport, error) {
- parsedURL, err := url.Parse(urlstr)
- if err != nil {
- return nil, err
- }
- buf := make([]byte, 0, 1024)
- client := options.Client
- if client == nil {
- client = DefaultHttpClient
- }
- httpHeader := map[string][]string{"Content-Type": []string{"application/x-thrift"}}
- return &THttpClient{client: client, url: parsedURL, requestBuffer: bytes.NewBuffer(buf), header: httpHeader}, nil
- }
- func NewTHttpPostClient(urlstr string) (TTransport, error) {
- return NewTHttpPostClientWithOptions(urlstr, THttpClientOptions{})
- }
- // Set the HTTP Header for this specific Thrift Transport
- // It is important that you first assert the TTransport as a THttpClient type
- // like so:
- //
- // httpTrans := trans.(THttpClient)
- // httpTrans.SetHeader("User-Agent","Thrift Client 1.0")
- func (p *THttpClient) SetHeader(key string, value string) {
- p.header.Add(key, value)
- }
- // Get the HTTP Header represented by the supplied Header Key for this specific Thrift Transport
- // It is important that you first assert the TTransport as a THttpClient type
- // like so:
- //
- // httpTrans := trans.(THttpClient)
- // hdrValue := httpTrans.GetHeader("User-Agent")
- func (p *THttpClient) GetHeader(key string) string {
- return p.header.Get(key)
- }
- // Deletes the HTTP Header given a Header Key for this specific Thrift Transport
- // It is important that you first assert the TTransport as a THttpClient type
- // like so:
- //
- // httpTrans := trans.(THttpClient)
- // httpTrans.DelHeader("User-Agent")
- func (p *THttpClient) DelHeader(key string) {
- p.header.Del(key)
- }
- func (p *THttpClient) Open() error {
- // do nothing
- return nil
- }
- func (p *THttpClient) IsOpen() bool {
- return p.response != nil || p.requestBuffer != nil
- }
- func (p *THttpClient) closeResponse() error {
- var err error
- if p.response != nil && p.response.Body != nil {
- // The docs specify that if keepalive is enabled and the response body is not
- // read to completion the connection will never be returned to the pool and
- // reused. Errors are being ignored here because if the connection is invalid
- // and this fails for some reason, the Close() method will do any remaining
- // cleanup.
- io.Copy(ioutil.Discard, p.response.Body)
- err = p.response.Body.Close()
- }
- p.response = nil
- return err
- }
- func (p *THttpClient) Close() error {
- if p.requestBuffer != nil {
- p.requestBuffer.Reset()
- p.requestBuffer = nil
- }
- return p.closeResponse()
- }
- func (p *THttpClient) Read(buf []byte) (int, error) {
- if p.response == nil {
- return 0, NewTTransportException(NOT_OPEN, "Response buffer is empty, no request.")
- }
- n, err := p.response.Body.Read(buf)
- if n > 0 && (err == nil || err == io.EOF) {
- return n, nil
- }
- return n, NewTTransportExceptionFromError(err)
- }
- func (p *THttpClient) ReadByte() (c byte, err error) {
- return readByte(p.response.Body)
- }
- func (p *THttpClient) Write(buf []byte) (int, error) {
- n, err := p.requestBuffer.Write(buf)
- return n, err
- }
- func (p *THttpClient) WriteByte(c byte) error {
- return p.requestBuffer.WriteByte(c)
- }
- func (p *THttpClient) WriteString(s string) (n int, err error) {
- return p.requestBuffer.WriteString(s)
- }
- func (p *THttpClient) Flush() error {
- // Close any previous response body to avoid leaking connections.
- p.closeResponse()
- req, err := http.NewRequest("POST", p.url.String(), p.requestBuffer)
- if err != nil {
- return NewTTransportExceptionFromError(err)
- }
- req.Header = p.header
- response, err := p.client.Do(req)
- if err != nil {
- return NewTTransportExceptionFromError(err)
- }
- if response.StatusCode != http.StatusOK {
- // Close the response to avoid leaking file descriptors. closeResponse does
- // more than just call Close(), so temporarily assign it and reuse the logic.
- p.response = response
- p.closeResponse()
- // TODO(pomack) log bad response
- return NewTTransportException(UNKNOWN_TRANSPORT_EXCEPTION, "HTTP Response code: "+strconv.Itoa(response.StatusCode))
- }
- p.response = response
- return nil
- }
- func (p *THttpClient) RemainingBytes() (num_bytes uint64) {
- len := p.response.ContentLength
- if len >= 0 {
- return uint64(len)
- }
- const maxSize = ^uint64(0)
- return maxSize // the thruth is, we just don't know unless framed is used
- }
|