Compare commits
14 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| c8c0966c63 | |||
| efaad2a3c7 | |||
| 9059849673 | |||
| aeabb2cc70 | |||
| 588e5874f3 | |||
| 5fc4858d6d | |||
| 22b3cc7acb | |||
| 5ffc869490 | |||
| cc74bc83c8 | |||
| 5e81240a05 | |||
| 29a2ad2032 | |||
| 6b544c54a3 | |||
| 734974d279 | |||
| 57ce59c207 |
@@ -1,6 +1,6 @@
|
||||
# stream-http [](https://travis-ci.org/jhiesey/stream-http)
|
||||
|
||||
[](https://saucelabs.com/u/jhiesey)
|
||||
[](https://saucelabs.com/u/stream-http)
|
||||
|
||||
This module is an implementation of node's native `http` module for the browser.
|
||||
It tries to match node's api and behavior as closely as possible, but some features
|
||||
|
||||
@@ -25,9 +25,6 @@ http.request = function (opts, cb) {
|
||||
opts.hostname = opts.hostname || hostHostname || window.location.hostname
|
||||
opts.port = opts.port || hostPort || defaultPort
|
||||
|
||||
if (opts.withCredentials === undefined)
|
||||
opts.withCredentials = true
|
||||
|
||||
// Also valid opts.auth, opts.mode
|
||||
|
||||
var req = new ClientRequest(opts)
|
||||
|
||||
+16
-4
@@ -90,6 +90,8 @@ ClientRequest.prototype.removeHeader = function (name) {
|
||||
ClientRequest.prototype._onFinish = function () {
|
||||
var self = this
|
||||
|
||||
if (self._destroyed)
|
||||
return
|
||||
var opts = self._opts
|
||||
|
||||
var headersObj = self._headers
|
||||
@@ -117,7 +119,7 @@ ClientRequest.prototype._onFinish = function () {
|
||||
headers: headers,
|
||||
body: body,
|
||||
mode: 'cors',
|
||||
credentials: opts.withCredentials ? 'include' : 'omit'
|
||||
credentials: opts.withCredentials ? 'include' : 'same-origin'
|
||||
}).then(function (response) {
|
||||
self._fetchResponse = response
|
||||
self._connect()
|
||||
@@ -167,6 +169,8 @@ ClientRequest.prototype._onFinish = function () {
|
||||
}
|
||||
|
||||
xhr.onerror = function () {
|
||||
if (self._destroyed)
|
||||
return
|
||||
self.emit('error', new Error('XHR error'))
|
||||
}
|
||||
|
||||
@@ -196,7 +200,7 @@ function statusValid (xhr) {
|
||||
ClientRequest.prototype._onXHRProgress = function () {
|
||||
var self = this
|
||||
|
||||
if (!statusValid(self._xhr) || self._failed)
|
||||
if (!statusValid(self._xhr) || self._destroyed)
|
||||
return
|
||||
|
||||
if (!self._response)
|
||||
@@ -208,6 +212,9 @@ ClientRequest.prototype._onXHRProgress = function () {
|
||||
ClientRequest.prototype._connect = function () {
|
||||
var self = this
|
||||
|
||||
if (self._destroyed)
|
||||
return
|
||||
|
||||
self._response = new IncomingMessage(self._xhr, self._fetchResponse, self._mode)
|
||||
self.emit('response', self._response)
|
||||
}
|
||||
@@ -219,10 +226,15 @@ ClientRequest.prototype._write = function (chunk, encoding, cb) {
|
||||
cb()
|
||||
}
|
||||
|
||||
ClientRequest.prototype.abort = function () {
|
||||
ClientRequest.prototype.abort = ClientRequest.prototype.destroy = function () {
|
||||
var self = this
|
||||
self._destroyed = true
|
||||
if (self._response)
|
||||
self._response._destroyed = true
|
||||
if (self._xhr)
|
||||
self._xhr.abort()
|
||||
// Currently, there isn't a way to truly abort a fetch.
|
||||
// If you like bikeshedding, see https://github.com/whatwg/fetch/issues/27
|
||||
}
|
||||
|
||||
ClientRequest.prototype.end = function (data, encoding, cb) {
|
||||
@@ -266,4 +278,4 @@ var unsafeHeaders = [
|
||||
'upgrade',
|
||||
'user-agent',
|
||||
'via'
|
||||
];
|
||||
]
|
||||
|
||||
@@ -21,6 +21,14 @@ var IncomingMessage = exports.IncomingMessage = function (xhr, response, mode) {
|
||||
self.trailers = {}
|
||||
self.rawTrailers = []
|
||||
|
||||
// Fake the 'close' event, but only once 'end' fires
|
||||
self.on('end', function () {
|
||||
// The nextTick is necessary to prevent the 'request' module from causing an infinite loop
|
||||
process.nextTick(function () {
|
||||
self.emit('close')
|
||||
})
|
||||
})
|
||||
|
||||
if (mode === 'fetch') {
|
||||
self._fetchResponse = response
|
||||
|
||||
@@ -37,6 +45,8 @@ var IncomingMessage = exports.IncomingMessage = function (xhr, response, mode) {
|
||||
var reader = response.body.getReader()
|
||||
function read () {
|
||||
reader.read().then(function (result) {
|
||||
if (self._destroyed)
|
||||
return
|
||||
if (result.done) {
|
||||
self.push(null)
|
||||
return
|
||||
|
||||
+9
-1
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "stream-http",
|
||||
"version": "1.1.1",
|
||||
"version": "1.3.0",
|
||||
"description": "Streaming http in the browser",
|
||||
"main": "index.js",
|
||||
"repository": {
|
||||
@@ -15,6 +15,13 @@
|
||||
},
|
||||
"author": "John Hiesey",
|
||||
"license": "MIT",
|
||||
"keywords": [
|
||||
"http",
|
||||
"stream",
|
||||
"streaming",
|
||||
"xhr",
|
||||
"http-browserify"
|
||||
],
|
||||
"dependencies": {
|
||||
"builtin-status-codes": "~1.0.0",
|
||||
"foreach": "^2.0.5",
|
||||
@@ -26,6 +33,7 @@
|
||||
"devDependencies": {
|
||||
"basic-auth": "^1.0.3",
|
||||
"brfs": "^1.4.0",
|
||||
"cookie-parser": "^1.3.5",
|
||||
"express": "^4.13.0",
|
||||
"tape": "^4.0.0",
|
||||
"ua-parser-js": "^0.7.7",
|
||||
|
||||
@@ -0,0 +1,48 @@
|
||||
var Buffer = require('buffer').Buffer
|
||||
var fs = require('fs')
|
||||
var test = require('tape')
|
||||
|
||||
var http = require('../..')
|
||||
|
||||
test('abort before response', function (t) {
|
||||
var req = http.get('/basic.txt', function (res) {
|
||||
t.fail('unexpected response')
|
||||
})
|
||||
req.abort()
|
||||
t.end()
|
||||
})
|
||||
|
||||
test('abort on response', function (t) {
|
||||
var req = http.get('/basic.txt', function (res) {
|
||||
req.abort()
|
||||
t.end()
|
||||
|
||||
res.on('end', function () {
|
||||
t.fail('unexpected end')
|
||||
})
|
||||
|
||||
res.on('data', function (data) {
|
||||
t.fail('unexpected data')
|
||||
})
|
||||
})
|
||||
})
|
||||
|
||||
test('abort on data', function (t) {
|
||||
var req = http.get('/browserify.png?copies=5', function (res) {
|
||||
var firstData = true
|
||||
|
||||
res.on('end', function () {
|
||||
t.fail('unexpected end')
|
||||
})
|
||||
|
||||
res.on('data', function (data) {
|
||||
if (firstData) {
|
||||
firstData = false
|
||||
req.abort()
|
||||
t.end()
|
||||
} else {
|
||||
t.fail('unexpected data')
|
||||
}
|
||||
})
|
||||
})
|
||||
})
|
||||
@@ -0,0 +1,25 @@
|
||||
var Buffer = require('buffer').Buffer
|
||||
var test = require('tape')
|
||||
|
||||
var http = require('../..')
|
||||
|
||||
test('cookie', function (t) {
|
||||
var cookie = 'hello=world'
|
||||
window.document.cookie = cookie
|
||||
|
||||
http.get({
|
||||
path: '/cookie',
|
||||
withCredentials: false
|
||||
}, function (res) {
|
||||
var buffers = []
|
||||
|
||||
res.on('end', function () {
|
||||
t.ok(new Buffer(cookie).equals(Buffer.concat(buffers)), 'hello cookie echoed')
|
||||
t.end()
|
||||
})
|
||||
|
||||
res.on('data', function (data) {
|
||||
buffers.push(data)
|
||||
})
|
||||
})
|
||||
})
|
||||
@@ -85,7 +85,7 @@ test('Test withCredentials param', function(t) {
|
||||
t.equal( request._xhr.withCredentials, true, 'xhr.withCredentials should be true')
|
||||
|
||||
var request = http.get({ url: url }, noop)
|
||||
t.equal( request._xhr.withCredentials, true, 'xhr.withCredentials should be true')
|
||||
t.equal( request._xhr.withCredentials, false, 'xhr.withCredentials should be false')
|
||||
|
||||
t.end()
|
||||
})
|
||||
|
||||
@@ -1,3 +1,4 @@
|
||||
var cookieParser = require('cookie-parser')
|
||||
var basicAuth = require('basic-auth')
|
||||
var express = require('express')
|
||||
var fs = require('fs')
|
||||
@@ -46,6 +47,12 @@ app.get('/testHeaders', function (req, res) {
|
||||
res.end()
|
||||
})
|
||||
|
||||
app.get('/cookie', cookieParser(), function (req, res) {
|
||||
res.setHeader('Content-Type', 'text/plain')
|
||||
res.write('hello=' + req.cookies.hello)
|
||||
res.end()
|
||||
})
|
||||
|
||||
app.get('/auth', function (req, res) {
|
||||
var user = basicAuth(req)
|
||||
|
||||
|
||||
Reference in New Issue
Block a user