You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
Currently Pretender works in two basic and complimentary ways: request stubbing and request proxying. We'd like to add a third, chimeric, methodology: request recording. In this mode the first time a particular verb/path pairing is accessed we'd like to proxy to a real world source of data, store this data in some location, then return the data. Subsequent requests will used the stored version of the data.
The current use case for this feature is performance regression testing JavaScript libraries/frameworks inside real world applications. Performance regression testing involves taking a real application and running a series of operations many hundreds or thousands of times.
A problem arises: if you're making real world requests, it is difficult to isolate the performance characteristics of the UI from the performance characteristics of the server. Stubbing can solve this; however, manually stubbing of large applications can be cumbersome.
Defining Success
Successful PRs on the road to this feature include:
Pretender is a fairly small library, but knowing how it works will help you in this quest.
How Setup works
When an instance of Pretender is created, its constructor captures the global window.XMLHttpRequest object, replacing that global with its own implementation as an "interceptor". This implementation matches the XMLHttpRequest (via a library FakeXMLHttpRequest) and requesting libraries (e.g. 'jQuery') can interact with this implementation without knowing that the "real" XMLHttpRequest object is gone.
The main path into this implementation is the faked XMLHttpRequest.send method. This method returns immediately and the requesting library can listen for specific events.
Pretender, via configuration, allows a developer to control when these events trigger.
After setting up this interceptor, a Pretender instance runs its "maps". Maps are functions that, when executed set up pairings of HTTP methods (GET, POST, etc) and paths ('/some/url'). These are the methods/verbs that Pretender will respond to.
These methods all share the same implementation and are created by the private verbify function. They are: get, put, post, delete, patch, head, and options.
There is also a related functionality called "passthrough".
How Request Stubbing Works
Once Pretender has rudely snatched away the real XMLHttpRequest object and replaced it with a changeling child (FakeXMLHttpRequest) and a set of routes has been mapped, a Pretender instance is ready to start responding to requests.
When a requesting library (e.g "jQuery") makes an XMLHttpRequest, Pretender intercepts that request and responds with the [status, headers, body] data defined in the matching handler from the route map.
XMLHttpRequest.respond is not a thing. However, [FaxeXMLHttpRequest.respond](https://github.com/pretenderjs/FakeXMLHttpRequest/blob/fafac9799ac9989fa4798f2fe23b071c2209e04e/fake_xml_http_request.js#L462-L466) is! This triggers the response and the requesting library is none the wiser that no actual network requests were made!
How Request Proxying Works
Setup for passthrough is roughly the same but instead of going through a verbify method, you simply declare that a route is a passthrough this.get('/photos/:id', this.passthrough).
When a request for a passthrough route is made, well, that's where life gets interesting.
First, we check if the requesting library is asking for a HTTP method/url pair that we said to passthrough. You might think at this juncture "if that's the case, just give let the requesting library use the real XMLHttpRequest object and be done!", but, alas, there's no way to know whether to use FakeXMLHttpRequest or XMLHttpRequest until after a request is made and at point it's too late: the requesting library needs an instance of something.
The enhancement I authored that just shipped with 3.2.0 probably helps in this regard. Fake requests created by pretender now have a .passthrough() method on them that allows you to check the request and decide at runtime if you want to pass it through or not.
Currently Pretender works in two basic and complimentary ways: request stubbing and request proxying. We'd like to add a third, chimeric, methodology: request recording. In this mode the first time a particular verb/path pairing is accessed we'd like to proxy to a real world source of data, store this data in some location, then return the data. Subsequent requests will used the stored version of the data.
The current use case for this feature is performance regression testing JavaScript libraries/frameworks inside real world applications. Performance regression testing involves taking a real application and running a series of operations many hundreds or thousands of times.
A problem arises: if you're making real world requests, it is difficult to isolate the performance characteristics of the UI from the performance characteristics of the server. Stubbing can solve this; however, manually stubbing of large applications can be cumbersome.
Defining Success
Successful PRs on the road to this feature include:
[coming soon]
Understanding Pretender Internals
Pretender is a fairly small library, but knowing how it works will help you in this quest.
How Setup works
When an instance of Pretender is created, its constructor captures the global
window.XMLHttpRequest
object, replacing that global with its own implementation as an "interceptor". This implementation matches the XMLHttpRequest (via a library FakeXMLHttpRequest) and requesting libraries (e.g. 'jQuery') can interact with this implementation without knowing that the "real" XMLHttpRequest object is gone.The main path into this implementation is the faked
XMLHttpRequest.send
method. This method returns immediately and the requesting library can listen for specific events.Pretender, via configuration, allows a developer to control when these events trigger.
After setting up this interceptor, a Pretender instance runs its "maps". Maps are functions that, when executed set up pairings of HTTP methods (
GET
,POST
, etc) and paths ('/some/url'
). These are the methods/verbs that Pretender will respond to.These methods all share the same implementation and are created by the private
verbify
function. They are:get
,put
,post
,delete
,patch
,head
, andoptions
.There is also a related functionality called "passthrough".
How Request Stubbing Works
Once Pretender has rudely snatched away the real XMLHttpRequest object and replaced it with a changeling child (FakeXMLHttpRequest) and a set of routes has been mapped, a Pretender instance is ready to start responding to requests.
When a requesting library (e.g "jQuery") makes an XMLHttpRequest, Pretender intercepts that request and responds with the
[status, headers, body]
data defined in the matching handler from the route map.During setup, we stashed this defined handler in a registry for retrieval later.
When a request occurs,
send
is called out on our faked XMLHttpRequest. This eventually callshandleRequest
handleRequest
looks up the handler and callsrespond
on this handler.XMLHttpRequest.respond
is not a thing. However,[FaxeXMLHttpRequest.respond](https://github.com/pretenderjs/FakeXMLHttpRequest/blob/fafac9799ac9989fa4798f2fe23b071c2209e04e/fake_xml_http_request.js#L462-L466)
is! This triggers the response and the requesting library is none the wiser that no actual network requests were made!How Request Proxying Works
Setup for passthrough is roughly the same but instead of going through a verbify method, you simply declare that a route is a passthrough
this.get('/photos/:id', this.passthrough)
.When a request for a passthrough route is made, well, that's where life gets interesting.
First, we check if the requesting library is asking for a HTTP method/url pair that we said to passthrough. You might think at this juncture "if that's the case, just give let the requesting library use the real XMLHttpRequest object and be done!", but, alas, there's no way to know whether to use
FakeXMLHttpRequest
orXMLHttpRequest
until after a request is made and at point it's too late: the requesting library needs an instance of something.And that something is a FakeXMLHttpRequest. But we also need to trigger real network interactions. This is done by creating a real XMLHttpRequest instance using our stashed copy of the constructor and, as the network interactions occur, triggering the matching method on the
FakeXMLHttpRequest
object that Pretender handed back to the requesting library when the request was made.This means we have to handle all the various things a real network request can do. Uploads? Yep. Timeouts? Yes'm. HTTP Credentials. That too.
The text was updated successfully, but these errors were encountered: