--- layout: post title: "Firefox custom search engine - search.json.mozlz4" date: 2025-05-03 04:16:49 +0100 --- So I have an ancient custom search engine XML that was broken by remote API changes. In trying to fix it, I inadvertently had it deleted from search.json.mozlz4, and have to figure out how to add it back. This is basically a rewrite of [this post by Frederick Zhang](https://blog.onee3.org/2018/04/manually-add-a-search-engine-to-firefox-quantum/). That post is excellent, but there have been some changes since 2018. The following is tested on the latest stable Firefox (138.0.1). First download [this Python script](https://gist.github.com/kaefer3000/73febe1eec898cd50ce4de1af79a332a/a266410033455d6b4af515d7a9d34f5afd35beec). Use it to decompress search.json.mozlz4 (located in the Firefox profile directory): `python3 mozlz4a.py -d search.json.mozlz4 search.json` Optionally, format it with `python3 -m json.tool`. Add a new object to the `"engines"` array. My example for Startpage with custom params: ``` { "id": "38c37483-6e61-4f86-bfaf-2b99ed7d8464", "_name": "Startpage (Unfiltered)", "_loadPath": "[profile]/searchplugins/startpage-unfiltered.xml", "_iconMapObj": { "16": "" }, "_metaData": { "loadPathHash": "", "alias": "sp", "hideOneOffButton": false, "order": 8 }, "_urls": [ { "params": [ { "name": "query", "value": "{searchTerms}" }, { "name": "cat", "value": "web" }, { "name": "lui", "value": "english" }, { "name": "prfe", "value": "c774daad5dd23ae2db70252aa321b9354508577d11ee4a79664d62c5d56672dd946c593d0daa5e9f3e7d1ebafaf773669d191e7b802605a743232b31a2254feeccf8ccc0f306bf45f8fa73f3" } ], "rels": [], "template": "https://www.startpage.com/sp/search", "method": "POST" } ], "_orderHint": null, "_telemetryId": null, "_filePath": "/searchplugins/startpage-unfiltered.xml", "_definedAliases": [], "_updateInterval": null, "_updateURL": null } ``` I assume the `"id"` is just some UUID. I ended up reusing an existing one that I no longer needed. I'm not sure whether it would be "validated" in any way. The `"loadPathHash"` is calculated using the following function: ``` function getVerificationHash(name, profileDir = PathUtils.profileDir) { let disclaimer = "By modifying this file, I agree that I am doing so " + "only within $appName itself, using official, user-driven search " + "engine selection processes, and in a way which does not circumvent " + "user consent. I acknowledge that any attempt to change this file " + "from outside of $appName is a malicious act, and will be responded " + "to accordingly."; let salt = PathUtils.filename(profileDir) + name + disclaimer.replace(/\$appName/g, Services.appinfo.name); let data = new TextEncoder().encode(salt); let hasher = Cc["@mozilla.org/security/hash;1"].createInstance( Ci.nsICryptoHash ); hasher.init(hasher.SHA256); hasher.update(data, data.length); return hasher.finish(true); } ``` This code is taken from the Firefox source code [here](https://searchfox.org/mozilla-central/rev/4c065f1df299065c305fb48b36cdae571a43d97c/toolkit/components/search/SearchUtils.sys.mjs). It is to be pasted into the Browser Console, then called with `getVerificationHash("", "")`. The second argument is required if you are running on a different profile. The `"order"` is just whichever unused number that comes next. Save this, then compress it again with: `python3 mozlz4a.py search.json search.json.mozlz4`. Finally, create a backup of the original search.json.mozlz4, and replace it with the new one.