Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Allow error pages with json bodies #1931

Closed
Astavie opened this issue Jul 17, 2021 · 6 comments · Fixed by #6497
Closed

Allow error pages with json bodies #1931

Astavie opened this issue Jul 17, 2021 · 6 comments · Fixed by #6497

Comments

@Astavie
Copy link

Astavie commented Jul 17, 2021

Describe the problem

I'm making a full-stack website in SvelteKit, and am now working on my API. This api is located under /api/*. When a user requests from an unknown URI I want to send a 404 message. As the API responds with json, I would like to respond in the same manner with the error message. However, to my knowledge, __error.svelte files only support html as an output.

Describe the proposed solution

I can see two solutions

  1. implementation of an __error.json file
  2. an export in endpoints that get executed on all undefined http methods

I prefer number 2, as then I could also implement 405 Method Not Allowed errors.

Alternatives considered

For now, I'm using an endpoint like this:

function response() { return {
    status: 404,
    body: {
        "error": "notfound",
        "message": "Not Found"
    }
}}

// Most HTTP request methods
export const get = response;
export const post = response;
export const put = response;
export const patch = response;
export const del = response;
export const options = response;
export const head = response;

this is located under src/routes/api/[...error].ts

Although this works, it's pretty ugly imo and it doesn't cover all infinite custom http methods clients could be using.

Importance

nice to have

Additional Information

No response

@Conduitry
Copy link
Member

Something perhaps tidier that you can already do now is use a handle hook that returns the desired JSON instead of whatever resolve() returns.

I'm not sure how we'd want __error.json (or, more likely, __error.js) to behave. I'm not clear on what its API should be, what the file should export, or what it would be passed. As for what the function returns, presumably it would be the same as other endpoint functions (i.e., an object with status, headers, body, etc.) and if it returned falsy, rendering would fall back to __error.svelte. But it's not clear what should be passed to this function.

@Astavie
Copy link
Author

Astavie commented Jul 20, 2021

The handle hook practically solves my 404 issue (thanks for pointing me to it!)
For my other issue (giving a 405 error when a wrong http method is used) it would be the easiest if there was a generic handle function like the handle hook in endpoints that receives the request before passing it along to the function corresponding to the http method, kinda like this:

export function get() {
    return {
        status: 200,
        body: { "hello": "world" }
    }
}

export async function handle({ request, resolve }) {
    // Try to resolve the request like normal with one of the functions corresponding to an http method
    const response = await resolve(request)

    // If it returns falsy, it means we haven't given the http method a function
    if (!response) {
        const error = { "error": "badmethod", "message": "Method Not Allowed" }
        const body = JSON.stringify(error)
        return {
            status: 405,
            body: body,
            headers: {
                'content-type': 'application/json; charset=utf-8',
                'Allow': 'GET, HEAD'
            }
        }
    }

    // Otherwise, return like normal
    return response
}

Looking through the sveltekit source, I think I might be able to make a pull request for this if this is acceptable.

EDIT:
Through testing I've also seen that HEAD requests must be handled manually in sveltekit, though I suppose that could also be easily fixed within the handle hook.

@benmccann
Copy link
Member

This might also be related to #1857

@xpuu
Copy link

xpuu commented Mar 28, 2022

I have a very same feature request, but different use case and proposed solution.

Describe the problem

I have a complex API consumed by 3rd party programs. I'd like to communicate errors in JSON. So far, I do something like

export const get: RequestHandler = async ({ request, locals }) => {
  try {
    return { body: await getProductsByOwner(locals.ownerId) }
  } catch (e) {
    return { body: { error: e.message } }
  }
}

But it's very repetitive and verbose. I tried to catch the error in hooks.ts, but it doesn't work.

Describe the proposed solution

If handleError hook returns Response use this instead of default or custom error page.

Alternatives considered

Allow creating __error.js|ts endpoint, which would be used instead of __error.svelte if an error comes from endpoint rather than page. But this wouldn't cover 404 errors.

Endpoint would export one default function with the same parameters as handleError hook (error, event).

Importance

Nice to have

@Rich-Harris
Copy link
Member

Nowadays SvelteKit returns 405s automatically (it wasn't possible at the time this issue was raised).

Related: #4477

@Rich-Harris
Copy link
Member

This is basically implemented already — in newer versions of SvelteKit, errors will be turned into JSON if you request with the appropriate accept header — but #6497 finishes the job by making that the default with accept: */*.

See also #6499

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging a pull request may close this issue.

5 participants