Generating a file without blocking form submission
Last week I was discussing with one teammate at my current client about an issue they had.
The application generates a document based on some data provided by the user.
He explained to me that we needed to wait for the document to be generated to be able to redirect to the next page, just after form submission, because there is a link to download the document there, which means we need to know where the document is, which means we need to have already created the document.
At first, it may seem like an intractable issue, but it’s actually not the case.
There is a solution that doesn’t require the document to be generated before redirecting the user to the confirmation page.
That solution, in its most basic form, is relatively simple and doesn’t require a lot of code.
As we’ll see, we can add some complexity to the solution to improve the user experience.
You can have access to the code in the code repository.
Background story
Here is what the story looks like:
- The user enters some information
- The server starts generating a file based on that information asynchronously.
- The user can click a link to download the file.
We want the user to have a pleasant experience, so we don’t want to block the form submission until the file is created.
To provide that nice experience, when the form is submitted, the information is stored, an asynchronous process is started for file generation, and the user is redirected to a confirmation page containing a download link.
If we’re lucky, the file was created by the time the user gets to the confirmation page, and we can show a download link pointing directly to it.
But what if the file was not created yet?
We still want to display the confirmation page as soon as possible. The first trick is that we actually don’t need the file until the user clicks on the download link.
Not needing the file until the user requests the download gives us some extra time to generate it in the background.
“That’s great, you may say, but we still need a URL to download the file. How can we know the file URL if the file hasn’t been generated?”
This is a good question, and the answer is to use a second trick: You don’t need the file URL; you need a promise to be able to get that URL at some point.
You need a way to ask the server if the file is available and how to get it when it’s ready.
This repository aims to show how to implement that, evolving the solution from a super simple solution with pure HTML pages to a click-to-download link with a spinner.
Seeing the iterations from the first to the final one helps to understand the pattern to solve that problem.
Installation and run
The demo is a node application using the express framework.
Start with a git clone from the code repository.
Everything should be properly installed with:
npm install
Run the application with:
node index.js
Using the demo app
You are the worker
⚠️ The background process is faked; you need to manually create the file.
Because implementing an asynchronous job is not in the scope of this demo, there is no worker.
To make things convenient, this repo includes two commands to create and delete the file.
Said differently, you are the worker.
This gives you the advantage of experimenting at each step and observing the behavior of each click when the file is there or not.
You decide.
To create the file, run:
npm run create-receipt
To remove the file, run:
npm run clean-receipt
There is no form
Because there is no worker, there is no need to trigger the worker, which allows us to skip building the form.
The confirmation page is the first page, served at the /
route.
In a real application, you are expected to see this page right after submitting the form.
It displays a link to download a file.
Iterations
I’ve created this demo iterating on the code, starting with the simplest solution.
The cool thing is that it demonstrates how the solution works without JS.
Even better, because it was built gradually, it continues working even if the user browser deactivates JS.
JS is only there to improve the UX, but the application is still functional without it.
Play with the demo
Once you’ve cloned the repository, you can see the code for each iteration with a checkout of the correct tag.
I’ve provided a list of things you can experiment with to see how the system behaves for each iteration in experiment block such as this one.
To get things more interesting, look at the network tab of your browser development tools.
Aside discussions
In blocks such as this one, I’ve included aside discussions to extend on some of the ideas.
Iteration 1: User polling
This iteration is super raw but really helps understand how this works as it requires the user to manually ask (and ask again, if needed) for the file.
Future iterations are just UX improvements on top of it.
app.get('/', (req, res) => {
const htmlContent = `
<!DOCTYPE html>
<html lang="en">
<head>
<title>Form submitted</title>
</head>
<body>
<h1>Your form was submitted !</h1>
<p>Thank you for your information.</p>
<a href="/receipt">Get your receipt</a>
</body>
</html>
`;
res.send(htmlContent);
})
app.get('/receipt', (req, res) => {
fs.access(filePath, fs.constants.F_OK, (err) => {
if (!err) { // File exists
res.redirect('/download');
return;
}
// File doesn't exist yet
const htmlContent = `
<!DOCTYPE html>
<html lang="en">
<head>
<title>File Download</title>
</head>
<body>
<h1>File download</h1>
<p>We are creating your file.</p>
<p>Please wait 5 seconds and click the download link below:</p>
<a href="/receipt">Download</a>
</body>
</html>
`;
res.send(htmlContent);
});
});
app.get('/download', (req, res) => {
res.download(filePath, (err) => {
if (err) {
console.error(`Error during download: ${err}`);
res.status(500).send('Error downloading the file.');
} else {
console.log(`File downloaded: ${filePath}`);
}
});
});
The confirmation page now displays a download link to the /receipt
route.
The /receipt
route controller acts as a traffic cop.
If the file is available, it will redirect to the real file download URL.
Otherwise, it returns a page explaining that the file is still being created and asking the user to wait a few seconds before clicking on a link to download it.
How can we deal with the file not being available when the user clicks on that second link?
The solution is to repeat that same process and show a page asking to wait and a link to download the file.
Fortunately, we already have a route that behaves this way, the /receipt
one.
Here the solution is that the link on the waiting page is a link to the waiting page.
Users are manully polling to see if the file is available when they click on the link.
Experiment
- Create the file and click on the link on the confirmation page
- Create the file once you are on the waiting page and click the link
- Go to the waiting page, click the link, create the file, and click the link again
Why redirect and not send the file immediately from the receipt route?
You might think that the redirection overcomplicates the flow, and you want to send the file directly from the receipt route.
Sure, it saves from creating a new route and removes one HTTP request, but you should probably keep that as it is.
First, the file can be stored elsewhere, on a file storage like AWS S3.
In that case, you don’t want the file to pass through your server and want the file storage to serve it.
A redirection makes a lot of sense here.
Second, using the same URL for the traffic cop and serving the file will make caching harder.
Having a dedicated URL for the file will help the caching mechanisms detect that they can avoid fetching the file from the server again and again.
Iteration 2: Blinking polling
The next improvement introduced is to avoid the need for the user to click on the second link on the waiting page.
app.get('/receipt', (req, res) => {
fs.access(filePath, fs.constants.F_OK, (err) => {
if (!err) { // File exists
res.redirect('/download');
return;
}
// File doesn't exist yet
const htmlContent = `
<!DOCTYPE html>
<html lang="en">
<head>
<title>File Download</title>
<meta http-equiv="refresh" content="5">
</head>
<body>
<h1>File download</h1>
<p>We are creating your file.</p>
<p>Please wait 5 seconds and click the download link below:</p>
<a href="/receipt">Download</a>
</body>
</html>
`;
res.send(htmlContent);
});
});
The simplest solution is to reload the waiting page every N seconds, which is what the user was manually doing by clicking on the link.
Once the file is ready, the download starts.
This is achieved by using a meta tag to the header:
<meta http-equiv="refresh" content="5">
Still no JS.
An information for the HTTP client
One interesting thing to note with the introduction of that meta tag is that we start differentiating information for the client and the user.
The meta tags is for the browser, the HTTP client, which knows how to interpret it.
The text is for the user, who knows how to interpret that as well.
The two pieces of information don’t have to match.
Experiment
- Go to the waiting page and wait until the page reloads, then create the file
Iteration 3: Polling without blinking
The next improvement is to poll the server to see if the file has been created without reloading the page.
Now is the time for JS to come into play.
Once the waiting page is displayed, it starts polling the server to get information about the file.
It keeps polling the same route as before using JS fetch
function. We are moving from a space where the browser is the HTTP client to a world where we are creating our own client.
app.get('/receipt', (req, res) => {
fs.access(filePath, fs.constants.F_OK, (err) => {
if (!err) { // File exists
res.redirect('/download');
return;
}
// File doesn't exist yet
const htmlContent = `
<!DOCTYPE html>
<html lang="en">
<head>
<title>File Download</title>
</head>
<body>
<h1>File download</h1>
<p>We are creating your file.</p>
<p>Please wait 5 seconds and click the download link below:</p>
<a href="/receipt">Download</a>
</body>
<script>
async function retryWithDelay(taskFn, maxRetries = 5) {
let attempt = 0;
let delayMs = 0;
while (attempt < maxRetries) {
console.log('Attempt');
// Attempt to run the task function
const result = await taskFn();
attempt++;
if (attempt >= maxRetries) {
throw new Error('Max attempts')
}
if(result.type !== 'retry-after') {
return result.value;
}
delayMs = result.value * 1000;
await new Promise(resolve => setTimeout(resolve, delayMs));
}
}
// Function to perform a GET request to the current page
async function pollForFile() {
const response = await fetch(window.location.href);
if(response.headers.get('Retry-After')) {
return {
type: 'retry-after',
value: response.headers.get('Retry-After')
};
}
const contentDisposition = response.headers.get('Content-Disposition');
if(contentDisposition && contentDisposition.includes('attachment')) {
console.log('The content is marked for attachment download.');
// Extract filename (if any) from Content-Disposition
const filenameMatch = contentDisposition.match(/filename="?(.+)"?/);
const filename = filenameMatch ? filenameMatch[1] : 'default-filename';
// Process the response as a Blob and initiate the download
const data = await response.blob();
const downloadUrl = window.URL.createObjectURL(data);
const a = document.createElement('a');
a.href = downloadUrl;
a.download = filename;
document.body.appendChild(a);
a.click();
a.remove();
window.URL.revokeObjectURL(downloadUrl);
}
return {
type: 'success',
value: true
};
}
retryWithDelay(pollForFile, 10, 5000)
.then(result => console.log(result))
.catch(error => console.error(error.message));
</script>
</html>
`;
res.header('Retry-After', 5);
res.send(htmlContent);
});
});
Until the file is ready, a Retry-after
header is added to the response.
It indicates to the client how long it should wait before asking again if the file is ready.
This is the information for the HTTP client.
The JS code interprets that header and retries after the delay proposed by the server.
Once the file is ready, the waiting route will issue a redirect to the file’s location.
The fetch
function follows the redirection and gets a response containing Content-Disposition: attachment;
header.
This is the trigger for doing some JS vaudoo to download the file.
That’s sad because we need to use some clever tricks to emulate something that was working natively until that point when the browser was the HTTP client.
Experiment
- Go to the waiting page, wait a little, then create the file
Why not hard-code the retry delay?
The server provides the delay for each retry using the Retry-After
header.
An alternative could be to hardcode the delay in the client.
Using a header provided by the server adds some interesting possibilities, though.
First, it’s our mechanism to tell the client that it still needs to wait.
Without that we would need another way to communicate that information.
Alternatively, we could have a longer hardcoded delay, but that would mean potentially asking the user to wait longer than necessary.
Secondly, because the server can communicate how long the client should retry, it can change that waiting time.
Maybe the server can predict how long it will take until the file is ready and ask the client to wait just that long.
Or the server is serving many requests at the moment and would like every client to slow down with the amount of requests they send.
In that case, it can increase the delay, using that tool as a back pressure mechanism.
Is there a less hacky way to start the download ?
Hi, JS folks. Is there a better way to start the download of a file when the request is made via fetch
?
This is one of the solutions I found online, but I feel like it should be more straightforward.
Iteration 4: Clean and get a clever HTTP client
This iteration doesn’t bring any functional improvements, but cleans the code by bringing some functional composition.
app.get('/receipt', (req, res) => {
fs.access(filePath, fs.constants.F_OK, (err) => {
if (!err) { // File exists
res.redirect('/download');
return;
}
// File doesn't exist yet
const htmlContent = `
<!DOCTYPE html>
<html lang="en">
<head>
<title>File Download</title>
</head>
<body>
<h1>File download</h1>
<p>We are creating your file.</p>
<p>Please wait 5 seconds and click the download link below:</p>
<a href="/receipt">Download</a>
</body>
<script>
async function fetchWithRetryAfter(fetch, maxRetries = 5) {
let attempt = 0;
let response;
while (attempt < maxRetries) {
attempt++;
response = await fetch();
if(response.headers.get('Retry-After')) {
let delayMs = response.headers.get('Retry-After') * 1000;
await new Promise(resolve => setTimeout(resolve, delayMs));
continue;
}
return response;
}
return response;
}
// Function to perform a GET request to the current page
async function fetchDownloadAttachment(fetch) {
const response = await fetch;
const contentDisposition = response.headers.get('Content-Disposition');
if(contentDisposition && contentDisposition.includes('attachment')) {
// Extract filename (if any) from Content-Disposition
const filenameMatch = contentDisposition.match(/filename="?(.+)"?/);
const filename = filenameMatch ? filenameMatch[1] : 'default-filename';
// Process the response as a Blob and initiate the download
const data = await response.blob();
const downloadUrl = window.URL.createObjectURL(data);
const a = document.createElement('a');
a.href = downloadUrl;
a.download = filename;
document.body.appendChild(a);
a.click();
a.remove();
window.URL.revokeObjectURL(downloadUrl);
}
return response;
}
fetchWithRetryAfter(() => fetchDownloadAttachment(fetch(window.location.href)), 10)
.then(result => console.log(result))
.catch(error => console.error(error.message));
</script>
</html>
`;
res.header('Retry-After', 5);
res.send(htmlContent);
});
});
The JS code is refactored to have specific functions that are able to interpret an HTTP response and act accordingly.
fetchWithRetryAfter
looks for a Retry-After
header contained in a HTTP response and performs the retry logic.fetchDownloadAttachment
looks for an attachment and does the download trick.
You can create a clever HTTP client by combining all functions and reusing that for all your HTTP requests:
const cleverFetch = (fetchParams) => fetchWithRetryAfter(() => fetchDownloadAttachment(fetch(fetchParams)))
The HTTP response handling logic is now contained in one place, and these functions can be tested independently.
Iteration 5: Spinner
Let’s now move to the modern world of SPA.
In a SPA, you probably wouldn’t want the user to move from the confirmation page to the waiting page if the file isn’t ready but displays a spinner.
This is what this iteration is about. We want to stay on the confirmation page and avoid moving to the waiting page, even if the file is not there.
app.get('/', (req, res) => {
const htmlContent = `
<!DOCTYPE html>
<html lang="en">
<head>
<title>Form submitted</title>
<style>
.spinner {
display: none;
border: 4px solid rgba(0, 0, 0, 0.1);
width: 16px;
height: 16px;
border-radius: 50%;
border-left-color: #09f;
animation: spin 1s linear infinite;
}
@keyframes spin {
0% {
transform: rotate(0deg);
}
100% {
transform: rotate(360deg);
}
}
.hidden {
display: none;
}
</style>
</head>
<body>
<h1>Your form was submitted !</h1>
<p>Thank you for your information.</p>
<a href="/receipt" class="download">Get your receipt<div class="spinner"></div></a>
</body>
<script>
const downloadLinks = document.getElementsByClassName('download');
for(let downloadLink of downloadLinks) {
downloadLink.addEventListener('click', function(event) {
event.preventDefault();
const spinner = event.currentTarget.querySelector('.spinner');
spinner.style.display = 'inline-block';
fetchWithRetryAfter(() => fetchDownloadAttachment(fetch(this.getAttribute('href'))), 10)
.then(() => {
spinner.style.display = 'none'
})
.catch(error => console.error(error))
})
}
async function fetchWithRetryAfter(fetch, maxRetries = 5) {
let attempt = 0;
let response;
while (attempt < maxRetries) {
attempt++;
response = await fetch();
if(response.headers.get('Retry-After')) {
let delayMs = response.headers.get('Retry-After') * 1000;
await new Promise(resolve => setTimeout(resolve, delayMs));
continue;
}
return response;
}
return response;
}
// Function to perform a GET request to the current page
async function fetchDownloadAttachment(fetch) {
const response = await fetch;
const contentDisposition = response.headers.get('Content-Disposition');
if(contentDisposition && contentDisposition.includes('attachment')) {
// Extract filename (if any) from Content-Disposition
const filenameMatch = contentDisposition.match(/filename="?(.+)"?/);
const filename = filenameMatch ? filenameMatch[1] : 'default-filename';
// Process the response as a Blob and initiate the download
const data = await response.blob();
const downloadUrl = window.URL.createObjectURL(data);
const a = document.createElement('a');
a.href = downloadUrl;
a.download = filename;
document.body.appendChild(a);
a.click();
a.remove();
window.URL.revokeObjectURL(downloadUrl);
}
return response;
}
</script>
</html>
`;
res.send(htmlContent);
})
For this, we attach an event listener to the link, prevent it from moving to the next page, display a spinner, and start polling the receipt
route.
We keep the same logic as before and mostly the same JS code.
If the response contains a Retry-After
header, the file is not there yet, and a future request is scheduled.
If the response contains a Content-Disposition: attachment;
header, we do the JS download trick and hide the spinner.
Also, the URL used for polling is the one specified in the href
attribute of the link.
We didn’t need to change that template part to make it work.
Even better, it means that if JS is disabled, the default behavior will take over from our code and will issue a request to /receipt
, which will display the waiting page or redirect to the file, depending on the file’s existence.
That’s progressive enhancement.
Experiment
- On the confirmation page, click on the link, see the spinner and create the file
- On the confirmation page, create the file, then click on the link
- On the confirmation page, click on the link, and do nothing. (I agree, I should have included an error message here.)
- Disable JS, and try whatever you want to.
Iteration 6: Cleaning
The next iteration is cleaning the code, reducing page duplication, using a templating system.
Help me with Handlebars inlined partials and ExpressJs
I want to use two regions in the layout, body
and scripts
.
The idea is that every page template is able to provide its content and its own script.
At the moment, the script is pushed right into the <body>
.
It works, but I’m curious how to do that with Handlebars and ExpressJs.
I actually spent more time on this issue than on the rest of that demo...
That looks like a lot of work, should I do that?
Maybe, I don’t know, it depends.
Maybe your users are fine with waiting for the file to be generated, and you don’t need to do that at all.
As we’ve seen, if you take the asynchronous road, you have multiple stops.
Do you really need to stop at the spinner stop?
You can start with the first solution, without any JS, and track how many times the file is not available at the moment the user clicks the link to download the file.
If that doesn’t occur very often, because the system is faster at creating the file than user is at clicking the link in the majority of case, that solution may be good enough.
If this is something common you can start iterating on the solution as we did here.
Alternatively, you can also try to make file generation faster.
Alternatives
That solution based on an intermediary status resource is not the only solution to that problem.
Instead of polling to get the status of the file creation, you could use sockets to push the information to the client.
If you’re working in a server-to-server mode, you could also use a webhook mechanism to notify the other server that the file is ready.
Resources
Derek Comartin from the Code Opinion YouTube channel, made a nice video on that subject:
📺 Avoiding long running HTTP API requests. - Code Opinion.
The "RESTful Web Services Cookbook" covers the solution of the status resource as well.
So, here we are.
We are now able to display a confirmation page even when a file is still being generated.
This improves users experience as they don’t have to wait.
As we’ve seen, the solution, in its simplest form, doesn’t require a lot of code.
- Improve your automated testing : You will learn how to fix your tests and make them pass from things that slow you down to things that save you time. This is a self-paced video course in French.
- Helping your teams: I help software teams deliver better software sooner. We'll work on technical issues with code, test or architecture, or the process and organization depending on your needs. Book a free call where we'll discuss how things are going on your side and how I can help you.
- Deliver a talk in your organization: I have a few talks that I enjoy presenting, and I can share with your organization(meetup, conference, company, BBL). If you feel that we could work on a new topic together, let's discuss that.