Skip to content

Commit 745c6d9

Browse files
authored
Implement a .setHeaders() method for WPAPI and WPRequest objects (#315)
* Implement a `.setHeaders()` method for WPAPI and WPRequest objects In response to #299, #312, and #314, this introduces a new .setHeaders() prototype method for both WPAPI and WPRequest. `.setHeaders()` allows a single WPRequest, or all requests generated from a WPAPI site client instance, to be augmented with one or more custom HTTP headers. Providing this functionality allows consumers of the wpapi library to utilize custom authentication schemes (such as JWT) with the Authentication header, to specify the preferred language of the response endpoints with the Accept-Language header, and so on. Providing a general-purpose interface is preferred in place of specifically implementing individual methods for JWT auth, language preferences, or other specific headers that a consumer of this library may want to send. Closes #299, fixes #312, fixes #314 **Usage:** If you need to send additional HTTP headers along with your request (for example to provide a specific `Authorization` header for use with alternative authentication schemes), you can use the `.setHeaders()` method to specify one or more headers to send with the dispatched request: ```js // Specify a single header to send with the outgoing request wp.posts().setHeaders( 'Authorization', 'Bearer xxxxx.yyyyy.zzzzz' )... // Specify multiple headers to send with the outgoing request wp.posts().setHeaders({ Authorization: 'Bearer xxxxx.yyyyy.zzzzz', 'Accept-Language': 'pt-BR' })... ``` You can also set headers on the WPAPI instance itself, which will then be used for all subsequent requests created from that site instance: ```js wp.setHeaders( 'Authorization', 'Bearer xxxxx.yyyyy.zzzzz' ); wp.users().me()... wp.posts().id( unpublishedPostId )... ``` props to @anagio, @andreasvirkus, @gnarf, @Matthewnie, @mnivoliez, and @mzalewski for the feature request, feedback & discussion
1 parent b50f711 commit 745c6d9

File tree

7 files changed

+314
-1
lines changed

7 files changed

+314
-1
lines changed

README.md

+30
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,7 @@ This library is an isomorphic client for the [WordPress REST API](http://develop
2525
- [Embedding Data](#embedding-data)
2626
- [Collection Pagination](#collection-pagination)
2727
- [Customizing HTTP Request Behavior](#customizing-http-request-behavior)
28+
- [Specifying HTTP Headers](#specifying-http-headers)
2829
- [Authentication](#authentication)
2930
- [API Documentation](#api-documentation)
3031
- [Issues](#issues)
@@ -809,6 +810,35 @@ site.transport({
809810
```
810811

811812
Note that these transport methods are the internal methods used by `create` and `.update`, so the names of these methods therefore map to the HTTP verbs "get", "post", "put", "head" and "delete"; name your transport methods accordingly or they will not be used.
813+
### Specifying HTTP Headers
814+
815+
If you need to send additional HTTP headers along with your request (for example to provide a specific `Authorization` header for use with alternative authentication schemes), you can use the `.setHeaders()` method to specify one or more headers to send with the dispatched request:
816+
817+
#### Set headers for a single request
818+
819+
```js
820+
// Specify a single header to send with the outgoing request
821+
wp.posts().setHeaders( 'Authorization', 'Bearer xxxxx.yyyyy.zzzzz' )...
822+
823+
// Specify multiple headers to send with the outgoing request
824+
wp.posts().setHeaders({
825+
Authorization: 'Bearer xxxxx.yyyyy.zzzzz',
826+
'Accept-Language': 'pt-BR'
827+
})...
828+
```
829+
830+
#### Set headers globally
831+
832+
You can also set headers globally on the WPAPI instance itself, which will then be used for all subsequent requests created from that site instance:
833+
834+
```js
835+
// Specify a header to be used by all subsequent requests
836+
wp.setHeaders( 'Authorization', 'Bearer xxxxx.yyyyy.zzzzz' );
837+
838+
// These will now be sent with an Authorization header
839+
wp.users().me()...
840+
wp.posts().id( unpublishedPostId )...
841+
```
812842

813843
## Authentication
814844

lib/constructors/wp-request.js

+34
Original file line numberDiff line numberDiff line change
@@ -38,6 +38,7 @@ function WPRequest( options ) {
3838
// Whitelisted options keys
3939
'auth',
4040
'endpoint',
41+
'headers',
4142
'username',
4243
'password',
4344
'nonce'
@@ -652,6 +653,39 @@ WPRequest.prototype.file = function( file, name ) {
652653
// HTTP Methods: Public Interface
653654
// ==============================
654655

656+
/**
657+
* Specify one or more headers to send with the dispatched HTTP request.
658+
*
659+
* @example Set a single header to be used on this request
660+
*
661+
* request.setHeaders( 'Authorization', 'Bearer trustme' )...
662+
*
663+
* @example Set multiple headers to be used by this request
664+
*
665+
* request.setHeaders({
666+
* Authorization: 'Bearer comeonwereoldfriendsright',
667+
* 'Accept-Language': 'en-CA'
668+
* })...
669+
*
670+
* @method setHeaders
671+
* @chainable
672+
* @param {String|Object} headers The name of the header to set, or an object of
673+
* header names and their associated string values
674+
* @param {String} [value] The value of the header being set
675+
* @return {WPRequest} The WPRequest instance (for chaining)
676+
*/
677+
WPRequest.prototype.setHeaders = function( headers, value ) {
678+
// We can use the same iterator function below to handle explicit key-value
679+
// pairs if we convert them into to an object we can iterate over:
680+
if ( typeof headers === 'string' ) {
681+
headers = keyValToObj( headers, value );
682+
}
683+
684+
this._options.headers = Object.assign( {}, this._options.headers || {}, headers );
685+
686+
return this;
687+
};
688+
655689
/**
656690
* Get (download the data for) the specified resource
657691
*

lib/http-transport.js

+26-1
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,27 @@ var objectReduce = require( './util/object-reduce' );
1616
var isEmptyObject = require( './util/is-empty-object' );
1717

1818
/**
19-
* Conditionally set basic authentication on a server request object
19+
* Set any provided headers on the outgoing request object. Runs after _auth.
20+
*
21+
* @method _setHeaders
22+
* @private
23+
* @param {Object} request A superagent request object
24+
* @param {Object} options A WPRequest _options object
25+
* @param {Object} A superagent request object, with any available headers set
26+
*/
27+
function _setHeaders( request, options ) {
28+
// If there's no headers, do nothing
29+
if ( ! options.headers ) {
30+
return request;
31+
}
32+
33+
return objectReduce( options.headers, function( request, value, key ) {
34+
return request.set( key, value );
35+
}, request );
36+
}
37+
38+
/**
39+
* Conditionally set basic authentication on a server request object.
2040
*
2141
* @method _auth
2242
* @private
@@ -265,6 +285,7 @@ function _httpGet( wpreq, callback ) {
265285
var url = wpreq.toString();
266286

267287
var request = _auth( agent.get( url ), wpreq._options );
288+
request = _setHeaders( request, wpreq._options );
268289

269290
return invokeAndPromisify( request, callback, returnBody.bind( null, wpreq ) );
270291
}
@@ -284,6 +305,7 @@ function _httpPost( wpreq, data, callback ) {
284305
var url = wpreq.toString();
285306
data = data || {};
286307
var request = _auth( agent.post( url ), wpreq._options, true );
308+
request = _setHeaders( request, wpreq._options );
287309

288310
if ( wpreq._attachment ) {
289311
// Data must be form-encoded alongside image attachment
@@ -312,6 +334,7 @@ function _httpPut( wpreq, data, callback ) {
312334
data = data || {};
313335

314336
var request = _auth( agent.put( url ), wpreq._options, true ).send( data );
337+
request = _setHeaders( request, wpreq._options );
315338

316339
return invokeAndPromisify( request, callback, returnBody.bind( null, wpreq ) );
317340
}
@@ -333,6 +356,7 @@ function _httpDelete( wpreq, data, callback ) {
333356
checkMethodSupport( 'delete', wpreq );
334357
var url = wpreq.toString();
335358
var request = _auth( agent.del( url ), wpreq._options, true ).send( data );
359+
request = _setHeaders( request, wpreq._options );
336360

337361
return invokeAndPromisify( request, callback, returnBody.bind( null, wpreq ) );
338362
}
@@ -349,6 +373,7 @@ function _httpHead( wpreq, callback ) {
349373
checkMethodSupport( 'head', wpreq );
350374
var url = wpreq.toString();
351375
var request = _auth( agent.head( url ), wpreq._options );
376+
request = _setHeaders( request, wpreq._options );
352377

353378
return invokeAndPromisify( request, callback, returnHeaders );
354379
}
+65
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,65 @@
1+
'use strict';
2+
var chai = require( 'chai' );
3+
// Variable to use as our "success token" in promise assertions
4+
var SUCCESS = 'success';
5+
// Chai-as-promised and the `expect( prom ).to.eventually.equal( SUCCESS ) is
6+
// used to ensure that the assertions running within the promise chains are
7+
// actually run.
8+
chai.use( require( 'chai-as-promised' ) );
9+
var expect = chai.expect;
10+
11+
var WPAPI = require( '../../' );
12+
13+
// Inspecting the titles of the returned posts arrays is an easy way to
14+
// validate that the right page of results was returned
15+
var getTitles = require( './helpers/get-rendered-prop' ).bind( null, 'title' );
16+
var base64credentials = new Buffer( 'apiuser:password' ).toString( 'base64' );
17+
18+
describe( 'integration: custom HTTP Headers', function() {
19+
var wp;
20+
21+
beforeEach(function() {
22+
wp = new WPAPI({
23+
endpoint: 'http://wpapi.loc/wp-json'
24+
});
25+
});
26+
27+
// Testing basic authentication is an acceptable proxy for whether a header
28+
// value (Authentication:, in this case) is being set
29+
it( 'can be provided using WPRequest#setHeaders()', function() {
30+
var prom = wp.posts()
31+
.setHeaders( 'Authorization', 'Basic ' + base64credentials )
32+
.status([ 'future', 'draft' ])
33+
.get()
34+
.then(function( posts ) {
35+
expect( getTitles( posts ) ).to.deep.equal([
36+
'Scheduled',
37+
'Draft'
38+
]);
39+
return SUCCESS;
40+
});
41+
return expect( prom ).to.eventually.equal( SUCCESS );
42+
});
43+
44+
it( 'can be provided at the WPAPI instance level using WPAPI#setHeaders()', function() {
45+
var authenticated = WPAPI
46+
.site( 'http://wpapi.loc/wp-json' )
47+
.setHeaders( 'Authorization', 'Basic ' + base64credentials );
48+
var prom = authenticated.posts()
49+
.status([ 'future', 'draft' ])
50+
.get()
51+
.then(function( posts ) {
52+
expect( getTitles( posts ) ).to.deep.equal([
53+
'Scheduled',
54+
'Draft'
55+
]);
56+
return authenticated.users().me();
57+
})
58+
.then(function( me ) {
59+
expect( me.slug ).to.equal( 'apiuser' );
60+
return SUCCESS;
61+
});
62+
return expect( prom ).to.eventually.equal( SUCCESS );
63+
});
64+
65+
});

tests/unit/lib/constructors/wp-request.js

+79
Original file line numberDiff line numberDiff line change
@@ -671,6 +671,85 @@ describe( 'WPRequest', function() {
671671

672672
});
673673

674+
describe( '.setHeaders()', function() {
675+
676+
it( 'method exists', function() {
677+
expect( request ).to.have.property( 'setHeaders' );
678+
expect( request.setHeaders ).to.be.a( 'function' );
679+
});
680+
681+
it( 'will have no effect if called without any arguments', function() {
682+
request.setHeaders();
683+
expect( request._options.headers ).to.deep.equal({});
684+
});
685+
686+
it( 'will set a header key/value pair', function() {
687+
request.setHeaders( 'Authorization', 'Bearer sometoken' );
688+
expect( request._options.headers ).to.deep.equal({
689+
Authorization: 'Bearer sometoken'
690+
});
691+
});
692+
693+
it( 'will replace an existing header key/value pair', function() {
694+
request
695+
.setHeaders( 'Authorization', 'Bearer sometoken' )
696+
.setHeaders( 'Authorization', 'Basic QWxhZGRpbjpvcGVuIHNlc2FtZQ==' );
697+
expect( request._options.headers ).to.deep.equal({
698+
Authorization: 'Basic QWxhZGRpbjpvcGVuIHNlc2FtZQ=='
699+
});
700+
});
701+
702+
it( 'will set multiple header key/value pairs with chained calls', function() {
703+
request
704+
.setHeaders( 'Accept-Language', 'en-US' )
705+
.setHeaders( 'Authorization', 'Bearer sometoken' );
706+
expect( request._options.headers ).to.deep.equal({
707+
'Accept-Language': 'en-US',
708+
Authorization: 'Bearer sometoken'
709+
});
710+
});
711+
712+
it( 'will set multiple header key/value pairs when passed an object', function() {
713+
request.setHeaders({
714+
'Accept-Language': 'en-US',
715+
Authorization: 'Bearer sometoken'
716+
});
717+
expect( request._options.headers ).to.deep.equal({
718+
'Accept-Language': 'en-US',
719+
Authorization: 'Bearer sometoken'
720+
});
721+
});
722+
723+
it( 'will replace multiple existing header key/value pairs when passed an object', function() {
724+
request
725+
.setHeaders({
726+
'Accept-Language': 'en-US',
727+
Authorization: 'Bearer sometoken'
728+
})
729+
.setHeaders({
730+
'Accept-Language': 'pt-BR',
731+
Authorization: 'Basic QWxhZGRpbjpvcGVuIHNlc2FtZQ=='
732+
});
733+
expect( request._options.headers ).to.deep.equal({
734+
'Accept-Language': 'pt-BR',
735+
Authorization: 'Basic QWxhZGRpbjpvcGVuIHNlc2FtZQ=='
736+
});
737+
});
738+
739+
it( 'inherits headers from the constructor options object', function() {
740+
request = new WPRequest({
741+
endpoint: '/',
742+
headers: {
743+
'Accept-Language': 'pt-BR'
744+
}
745+
});
746+
expect( request._options.headers ).to.deep.equal({
747+
'Accept-Language': 'pt-BR'
748+
});
749+
});
750+
751+
});
752+
674753
describe( '.toString()', function() {
675754

676755
beforeEach(function() {

tests/unit/wpapi.js

+55
Original file line numberDiff line numberDiff line change
@@ -772,6 +772,61 @@ describe( 'WPAPI', function() {
772772

773773
});
774774

775+
describe( '.setHeaders()', function() {
776+
777+
beforeEach(function() {
778+
site = new WPAPI({ endpoint: 'http://my.site.com/wp-json' });
779+
});
780+
781+
it( 'is defined', function() {
782+
expect( site ).to.have.property( 'setHeaders' );
783+
expect( site.setHeaders ).to.be.a( 'function' );
784+
});
785+
786+
it( 'initializes site-wide headers object if called with no arguments', function() {
787+
expect( site._options ).not.to.have.property( 'headers' );
788+
site.setHeaders();
789+
expect( site._options ).to.have.property( 'headers' );
790+
expect( site._options.headers ).to.deep.equal({});
791+
});
792+
793+
it( 'sets site-wide headers when provided a name-value pair', function() {
794+
site.setHeaders( 'Accept-Language', 'en-US' );
795+
expect( site._options ).to.have.property( 'headers' );
796+
expect( site._options.headers ).to.deep.equal({
797+
'Accept-Language': 'en-US'
798+
});
799+
});
800+
801+
it( 'sets site-wide headers when provided an object of header name-value pairs', function() {
802+
site.setHeaders({
803+
'Accept-Language': 'en-CA',
804+
Authorization: 'Bearer sometoken'
805+
});
806+
expect( site._options ).to.have.property( 'headers' );
807+
expect( site._options.headers ).to.deep.equal({
808+
'Accept-Language': 'en-CA',
809+
Authorization: 'Bearer sometoken'
810+
});
811+
});
812+
813+
it( 'passes headers to all subsequently-instantiated handlers', function() {
814+
site.setHeaders({
815+
'Accept-Language': 'en-IL',
816+
Authorization: 'Bearer chicagostylepizza'
817+
});
818+
var req = site.root( '' );
819+
expect( req ).to.have.property( '_options' );
820+
expect( req._options ).to.be.an( 'object' );
821+
expect( req._options ).to.have.property( 'headers' );
822+
expect( req._options.headers ).to.deep.equal({
823+
'Accept-Language': 'en-IL',
824+
Authorization: 'Bearer chicagostylepizza'
825+
});
826+
});
827+
828+
});
829+
775830
describe( '.registerRoute()', function() {
776831

777832
it( 'is a function', function() {

0 commit comments

Comments
 (0)