Firefox: Select with over 250 options - InternalError: function nested too deeply
What version of Ajv are you using? Does the issue happen if you use the latest version? I use Ajv Version 8.17.1 and the issue still happens
Ajv options object
const ajv = new Ajv({ strict: false });
JSON Schema
{
"$schema": "http://json-schema.org/draft-07/schema#",
"type": "object",
"properties": {
"select_string_type": {
"type": "string",
"oneOf": [
{
"const": "test_1",
"title": "test_1"
},
{
"const": "test_2",
"title": "test_2"
},{
"const": "test_3",
"title": "test_3"
},{
"const": "test_4",
"title": "test_4"
},{
"const": "test_5",
"title": "test_5"
},
...
{
"const": "test_294",
"title": "test_294"
},{
"const": "test_295",
"title": "test_295"
},{
"const": "test_296",
"title": "test_296"
},{
"const": "test_297",
"title": "test_297"
},{
"const": "test_298",
"title": "test_298"
},{
"const": "test_299",
"title": "test_299"
},{
"const": "test_300",
"title": "test_300"
}
],
"title": "__(SELECT_STRING)",
"x-jsf-presentation": {
"inputType": "select"
}
},
"select_number_type": {
"type": "number",
"oneOf": [
{
"const": 0,
"title": "one"
},
{
"const": 1,
"title": "two"
}
],
"title": "__(SELECT_NUMBER)",
"x-jsf-presentation": {
"inputType": "select"
}
},
"string_type": {
"type": "string",
"title": "__(STRING)",
"maxLength": 64,
"x-jsf-presentation": {
"inputType": "text"
}
}
},
"allOf": [
{
"if": {
"properties": {
"select_number_type": {
"const": 0
}
}
},
"then": {
"properties": {
"string_type": {
"minLength": 1,
"maxLength": 64
}
},
"required": [
"string_type"
]
},
"else": {
"properties": {
"string_type": {
"x-jsf-presentation": {
"inputType": "hidden"
}
}
}
}
}
],
"required": [
"select_string_type",
"select_number_type"
]
}
Sample data
{
"select_string_type": "test_1",
"select_number_type": 0,
"string_type": "string example"
}
Your code
<script setup>
import { ref } from "vue";
import Ajv from "ajv";
import schema from "./../assets/schema/schema.json?raw";
const elements_select_string = ref([
"test_1",
"test_2",
"test_3",
"test_4",
...
"test_297",
"test_298",
"test_299",
"test_300",
]);
const selectStringElement = ref("");
const selectNumberElement = ref(0);
const stringElement = ref("string example");
async function createJSON() {
let config = {
jsonFile: {
version: "0.1.0",
form: {
select_string_type: selectStringElement.value,
select_number_type: selectNumberElement.value,
string_type: stringElement.value,
},
},
};
await validateJSON(config);
}
async function validateJSON(config) {
const ajv = new Ajv({ strict: false });
try {
const validate = ajv.compile(JSON.parse(schema));
const valid = validate(config.jsonFile.form);
if (valid) {
console.log("JSON valid");
} else {
console.log("JSON invalid", validate.errors);
}
} catch (error) {
console.error("Error:", error);
}
}
</script>
<template>
<div>
<h1>Example</h1>
<form>
<label for="select_string_type">Choose Select String Type:</label>
<select v-model="selectStringElement" id="select_string_type">
<option
v-for="select in elements_select_string"
:key="select"
:value="select"
>
{{ select }}
</option>
</select>
<br />
<label for="select_number_type">Choose Select Number Type:</label>
<select v-model="selectNumberElement" id="select_number_type">
<option value="0">one</option>
<option value="1">two</option>
</select>
<br />
<label for="string_type">String Type: </label>
<input
v-model="stringElement"
id="string_type"
name="string_type"
type="text"
/>
<br />
<input
type="button"
name="submit_form"
id="submit_form"
value="Submit Form"
@click="createJSON"
/>
</form>
</div>
</template>
Validation result, data AFTER validation, error messages
Error: InternalError: function nested too deeply
validateJSON HelloWorld.vue:333
createJSON HelloWorld.vue:325
callWithErrorHandling runtime-core.esm-bundler.js:199
callWithAsyncErrorHandling runtime-core.esm-bundler.js:206
invoker runtime-dom.esm-bundler.js:729
addEventListener runtime-dom.esm-bundler.js:680
patchEvent runtime-dom.esm-bundler.js:698
patchProp runtime-dom.esm-bundler.js:775
mountElement runtime-core.esm-bundler.js:4873
processElement runtime-core.esm-bundler.js:4820
patch runtime-core.esm-bundler.js:4688
mountChildren runtime-core.esm-bundler.js:4932
mountElement runtime-core.esm-bundler.js:4855
processElement runtime-core.esm-bundler.js:4820
patch runtime-core.esm-bundler.js:4688
mountChildren runtime-core.esm-bundler.js:4932
mountElement runtime-core.esm-bundler.js:4855
processElement runtime-core.esm-bundler.js:4820
patch runtime-core.esm-bundler.js:4688
componentUpdateFn runtime-core.esm-bundler.js:5326
run reactivity.esm-bundler.js:225
setupRenderEffect runtime-core.esm-bundler.js:5454
mountComponent runtime-core.esm-bundler.js:5229
processComponent runtime-core.esm-bundler.js:5182
patch runtime-core.esm-bundler.js:4700
componentUpdateFn runtime-core.esm-bundler.js:5326
run reactivity.esm-bundler.js:225
setupRenderEffect runtime-core.esm-bundler.js:5454
mountComponent runtime-core.esm-bundler.js:5229
processComponent runtime-core.esm-bundler.js:5182
patch runtime-core.esm-bundler.js:4700
render2 runtime-core.esm-bundler.js:5979
mount runtime-core.esm-bundler.js:3941
mount runtime-dom.esm-bundler.js:1767
<anonymous> main.js:11
HelloWorld.vue:341:12
What results did you expect? When using Chrome, the validation works without any problems. When using Firefox, this issue occurs without any changes made. I tried different approaches with the parameters, but nothing seems to work. As soon as there are more than 250 options, this error occurs.
This issue is happening when using AJV in a Vue.js project with Composition API.
I've added the Vue project below. You can unpack it and run npm run dev in the folder where the entire project is located.
This is likely related to how AJV builds efficient and fast precomputed validation functions using its codegen module. Making any change to avoid this issue would likely impact performance and given this is an edge case I don't think we'll be open to that. There are different ways to do something similar to this schema without creating const options for every possible value.
Hi folks! I work on Firefox, and we got a report of this same issue affecting a site in the wild: https://bugzilla.mozilla.org/show_bug.cgi?id=1965201
I'm interested to see what it'd take to get Firefox to the same experience that Chrome has here, whether that's via changes to Firefox or the server-side components involved here. (Based on another thread, it sounds like Chrome runs into this same issue, but you have to throw more data at them before they'll hit it; I'm curious what the explanation for that delta is.)
In the interests of looking into that, I'm hoping to find a minimal testcase that reproduces the issue. I tried the attached ajv_error.zip in the hopes that it might serve as such a testcase, but I'm not sure what to do with it -- I unpacked it and ran npm run dev, inside of a freshly installed Ubuntu environment (in a VM) where I've installed npm, nodejs, and vite. But that just seems to launch the vite graphical application; no web server gets stood up that I've been able to find, and I'm not sure what to do next.
@mariada-ta could you double-check that the attached zip file is the right one, and (assuming it is) please list whatever additional steps might be necessary (maybe some additional tool/dependency needs to be installed or launched, what local port a web serer might start listening, etc)? (Sorry if it's something that seems trivial/obvious; I don't work with npm/node super often, and have never touched alone vue/ajv/vite.)
Thanks in advance for the help!
(or maybe the code-quotes from comment 0 need to be copypasted into local files inside of the unpacked ajv_error.zip file's directory? not sure. In any case: explicit instructions for how to trigger the issue in Firefox would be great, and maybe that'll help us identify the right team/individuals to look at this on the Firefox side. thanks!)
Ah, never mind - I managed to reproduce the issue with the linked zip testcase after all.
Not sure what happened, but after I deleted node_modules and ran npm install, then I was able to npm run dev and reach a screen that showed a web server running on port 5173. At that point I'm able to trigger the issue by visiting localhost:5173 and checking my web console.
I've filed https://bugzilla.mozilla.org/show_bug.cgi?id=1966196 about changing Firefox's behaviour