بررسی تفاوتهای ریزی که حین کار با URLها باعث به وجود آمدن باگهای غیرمنتظره میشود.
همه چیز از یک باگ شروع شد 🔗
کار با URLها در JavaScript و Node.js باید ساده باشد، اما یک باگ اخیر در پروژه ما، من را به دنیایی از جزئیات پیچیده در APIهای URL و URLSearchParams برد. در این پست، به بررسی این جزئیات و مشکلات احتمالی آنها در کد و نحوه اجتناب از آنها خواهیم پرداخت.
مشکل: مدیریت URL با Axios 🔗
ما این مشکل را هنگام تولید URLها و اضافه کردن امضاهای هش به آنها پیدا کردیم. پارامترهای کوئری به طور یکپارچه percent-encode نمیشدند که منجر به رفتار غیرمنتظره و امضاهای هش اشتباه میشد.
واضح شد که تعامل بین اشیای URL و URLSearchParams نیاز به دقت بیشتری دارد.
مشکل شماره 1: تفاوت بین URL.search و ()URLSearchParams.toString 🔗
اولین شگفتی تفاوت بین URL.search و ()URLSearchParams.toString بود.
هنگام استفاده از searchParams. برای تغییر URL، دقت کنید، زیرا طبق مشخصات WHATWG ، شیء URLSearchParams از قوانین متفاوتی برای تعیین اینکه کدام کاراکترها باید percent-encode شوند استفاده میکند. به عنوان مثال، شیء URL کاراکتر تیلد ASCII (~) را percent-encode نمیکند، در حالی که URLSearchParams همیشه آن را encode میکند.
1// Example 1
2const url = new URL("https://example.com?param=foo bar")
3console.log(url.search) // prints param=foo%20bar
4console.log(url.searchParams.toString()) // prints ?param=foo+bar
5
6// Example 2
7const myURL = new URL("https://example.org/abc?foo=~bar")
8console.log(myURL.search) // prints ?foo=~bar
9// Modify the URL via searchParams...
10myURL.searchParams.sort()
11console.log(myURL.search) // prints ?foo=%7Ebar
در پروژه ما، لازم بود به طور صریح ()url.search = url.searchParams.toString را دوباره اختصاص دهیم تا اطمینان حاصل شود که رشته کوئری به طور یکنواخت encode شده است.
مشکل شماره 2: چالش علامت بعلاوه 🔗
یکی دیگر از نکات ظریف این است که چگونه URLSearchParams با کاراکترهای + برخورد میکند. به طور پیشفرض، URLSearchParams کاراکتر + را به عنوان فضای خالی تفسیر میکند که ممکن است هنگام encode دادههای باینری یا رشتههای Base64 منجر به خرابی دادهها شود.
یک راه حل این است که قبل از افزودن مقادیر به URLSearchParams از encodeURIComponent استفاده کنید:
1params.append("bin", encodeURIComponent("E+AXQB+A"))
جزئیات بیشتر در مستندات MDN موجود است.
مشکل شماره 3: URLSearchParams.get در مقابل ()URLSearchParams.toString 🔗
یکی دیگر از جزئیات ظریف زمانی به وجود میآید که خروجیهای URLSearchParams.get و URLSearchParams.toString را مقایسه میکنید. به عنوان مثال:
1const params = new URLSearchParams("?key=value&key=other")
2console.log(params.get("key")) // "value" (اولین مورد)
3console.log(params.toString()) // "key=value&key=other" (همه موارد سریالایز شده)
در سناریوهای چند مقداری، get فقط اولین مقدار را برمیگرداند، در حالی که toString همه را سریالایز میکند.
راهحل در کد ما 🔗
در پروژه ما، مشکل را با اختصاص صریح search حل کردیم:
1url.search = url.searchParams.toString()
2url.searchParams.set(
3 "hash",
4 cryptography.createSha256HmacBase64UrlSafe(url.href, SECRET_KEY ?? "")
5)
این اطمینان حاصل کرد که تمام پارامترهای کوئری قبل از اضافه کردن مقدار hash به درستی encode شده بودند.
ماژول querystring در Node.js 🔗
رابط کاربری WHATWG URLSearchParams و ماژول querystring هدف مشابهی دارند، اما هدف ماژول querystring عمومیتر است، زیرا امکان سفارشیسازی کاراکترهای جداکننده (& و =) را فراهم میکند. از سوی دیگر، API URLSearchParams به طور خاص برای رشتههای کوئری URL طراحی شده است.
ماژول querystring از URLSearchParams کارآمدتر است اما یک API استاندارد نیست. از URLSearchParams زمانی استفاده کنید که عملکرد بحرانی نیست یا وقتی سازگاری با کد مرورگر مطلوب است.
هنگام استفاده از URLSearchParams برخلاف ماژول querystring، کلیدهای تکراری به صورت آرایه مجاز نیستند. آرایهها با استفاده از ()array.toString سریالایز میشوند که به سادگی همه عناصر آرایه را با کاما جدا میکند.
1const params = new URLSearchParams({
2 user: "abc",
3 query: ["first", "second"],
4})
5console.log(params.getAll("query"))
6// Prints [ 'first,second' ]
7console.log(params.toString())
8// Prints 'user=abc&query=first%2Csecond'
با ماژول querystring، رشته کوئری 'foo=bar&abc=xyz&abc=123' به این صورت پارس میشود:
نکات کلیدی 🔗
هنگام استفاده از
URLSearchParamsبه نحوه مدیریت کاراکترهای خاص (مانند~) و فضاهای خالی توجه کنید. در صورت نیاز ازencodeURIComponentاستفاده کنید.تفاوت بین
URL.search،URLSearchParams.getوURLSearchParams.toStringرا برای جلوگیری از رفتار غیرمنتظره درک کنید.در Node.js از ماژول
querystringاستفاده کنید اگر میخواهید پارامترهای کوئری تکراری را به عنوان یک آرایه پارس کنید.
منبع: Pitfalls of URL and URLSearchParams in JavaScript از وبلاگ Software Alchemist