Cannot determine why request-promise-native fails on a 'POST with a 404'

1.3k Views Asked by At

I'm trying to make this brief. Hopefully, it isn't so brief it makes no sense. I need to read a list of identifiers from an API that will be used to subsequently 'GET' the JSON files associated with the keys in the list. The list array is stored in another JSON file at the same endpoint with an identifier of 'keys.json' In order to prevent the list from being processed twice, I want to immediately write an empty array back to 'keys.json' upon retrieving it.

Here is the successful function that 'GETS' the list.

const getJsonKeys = async () => {
    const options = {
        method: 'GET',
        uri: baseURL + keysID,
        headers: { 'User-Agent': 'Request-Promise' },
        json: true // Automatically parses the JSON string in the response
    };
    return (await rpn(options)).keys;
};

Here is the unsuccessful 'POST' that I try to write with:

const postEmptyJsonKeys = async () => {

    const options = {
        method: 'POST',
        uri: baseURL + keysID,
        body: {
            keys: []
        },
        json: true // Automatically stringifies the body to JSON
    };
    return (await rpn(options)).req.Request.body;
};

Here is the block that calls them both:

module.exports = (rtProcess) => {
    rtProcess.get('/process', async (req, res, next) => {

        const jsonKeysList = await (getJsonKeys());
        console.log("Retrieved Keys", jsonKeysList);
        try {
            const req = await (postEmptyJsonKeys());
            console.log("Wrote", req.Request.body);
        } catch(err) {
            console.log(err.statusCode, err.error);
            console.log(err);
        }

        //
        //  more code here
        //
        jsonKeysList.forEach(courseID => {
            //
            // more code here
            //
        });

        res.render("process");
    }); // end of process route
};  //  end of module exports

I have tried everything I know to do to ferret out the answer in the various docs around but I can find nothing that tells me why the catch block is taken, rather than getting a successful try.

BTW the error.status code is a 404.

The error is a string that looks like HTML, which is also a mystery to me since I am trying to POST a simple:

{
    keys: []
}
1

There are 1 best solutions below

0
On

So this error:

Cannot POST /static/quizdata/keys.json

Makes me think the API endpoint you are POSTing to is not properly defined, and that's why you get a 404 It's telling you there is no POST handler that matches that request.

As the question has the node.js tag and I can see the static part in the URL, that makes me think you might be serving that static content with express.static built-in middleware, and if that's the case, then that's why you can't POST anything there, as that middleware is not meant for that and will only take care of GET requests.

Regarding your comment, it's not static content because the route has static in it or because the content it's in a directory called static (if that's the case).

Take a look at this examples:

  • This will handle GET requests like http.../static/path/inside/public/dir.png:

    app.use('/static', express.static('public'));
    
  • This will handle GET requests like http.../assets/path/inside/public/dir.png:

    app.use('/assets', express.static('public'));
    
  • This will handle GET requests like http.../whatever/something/inside/public/dir.png:

    app.use('/whatever', express.static('public'));
    
  • This will handle GET requests like http.../something/inside/public/dir.png:

    app.use(express.static('public'));
    
  • This will handle GET requests like http.../something/inside/my-static-files/dir.png:

    app.use(express.static('my-static-files'));
    

You can find more examples in the official docs.

In any case, let's assume you are serving static content using this option:

app.use('/static', express.static('public'));

You can still add another middleware to handle POST requests. Something like this:

const express = require('express');
const fs = require('fs');

const app = express();
const port = 3000;

...

app.post('/static', (req, res, next) => {
    // Just an example, you can replace the path and the []
    // with anything you want:
    fs.writeFileSync('path/to/file.json', JSON.stringify([]));
});

app.use('/static', express.static('public'));

app.listen(port, () => console.log(`Listening on port ${port}!`));

You might also find this other question useful: Appending a JSON file from JS file in Express Application