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

Remove qs from list #35

Open
wants to merge 1 commit into
base: main
Choose a base branch
from

Conversation

OiYouYeahYou
Copy link

On the basic KV-pair structure of a query: qs is indeed made redundant by URLSearchParams. But, qs supports a superset of the URL Query Syntax that supports deeply nested data, and acts differently is certain conditions.

The reason a more complicated format become useful in situations where you need to transmit non-path data in HTTP methods like GET. For instance, applications like Strapi strictly adhere to the HTTP method definition but needs additional information for functions like nested data population.

In this changeset, I've removed the whole recommendation, but I accept that another change would be to recommend only using qs if you need the superset support.

Examples

Stringifying

const data = {
  nested: {
    data: {
      is: ["supported", "and", "expressed"],
    },
  },
}

new URLSearchParams(data).toString()
// nested=[object+Object]
qs.stringify(data)
// nested[data][is][0]=supported&nested[data][is][1]=and&nested[data][is][2]=expressed

* I've URI decoded the strings for clarity

Parsing

const q1 =
  "nested[data][is][0]=supported&nested[data][is][1]=and&nested[data][is][2]=expressed";

Object.fromEntries(new URLSearchParams(q1))
/* {
  'nested[data][is][0]': 'supported',
  'nested[data][is][1]': 'and',
  'nested[data][is][2]': 'expressed'
} */
qs.parse(q1)
// { nested: { data: { is: ['supported', 'and', 'expressed] } } }
const q2 =
  "repeated=keys&repeated=are&repeated=supported&repeated=differently&repeated="

new URLSearchParams(q2)
/* URLSearchParams {
  'repeated' => 'keys',
  'repeated' => 'are',
  'repeated' => 'supported',
  'repeated' => 'differently',
  'repeated' => '' } */
qs.parse(q2)
// { repeated: [ 'keys', 'are', 'supported', 'differently', '' ] }

Aside about URL

Tangential FYI: URLSearchParams is fine for most applications, but URL is finicky. Depending on environment (Node vs Browser), it may or may-not parse the string as a URL or URL-like. For instance:

  • mongodb://... URL will not be treated as a URL in the browser, but will in Node.
    A demonstraction is available here.
  • chrome:... ends up getting special treatment and will get treated as if it is a URL, while being out of spec.

Most of the time people use HTTP(S) URLs and will never face this problem, but sometimes this can require avoiding the built-in.

@43081j
Copy link
Collaborator

43081j commented Apr 12, 2024

You're right that qs isn't directly replaced by URLSearchParams.

However, i think this is more that the wording is wrong rather than us needing to remove it from the list

qs has a large footprint thanks to the deep layers of node compat it has under the hood (for very old versions of node). We should still be pushing people to lighter alternatives because of that, but making those clear rather than thinking URLSearchParams is enough

for example, i've already moved many packages to use fast-querystring, or node's own querystring module

though there's also a piece of work (unfinished) to create a lighter qs-like alternative which uses fast-querystring under the hood (or uses native functionality, either way).

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

Successfully merging this pull request may close these issues.

2 participants