forked from MapComplete/MapComplete
		
	Fix: correctly optimize 'key=A&key!=B' into 'key=A'
This commit is contained in:
		
							parent
							
								
									201f5446f8
								
							
						
					
					
						commit
						d8a5f3c2be
					
				
					 2 changed files with 36 additions and 10 deletions
				
			
		| 
						 | 
					@ -243,7 +243,9 @@ export class And extends TagsFilter {
 | 
				
			||||||
                    properties[opt.key] = opt.value
 | 
					                    properties[opt.key] = opt.value
 | 
				
			||||||
                }
 | 
					                }
 | 
				
			||||||
            }
 | 
					            }
 | 
				
			||||||
            for (const opt of optimized) {
 | 
					
 | 
				
			||||||
 | 
					            for (let i = 0; i < optimized.length; i++){
 | 
				
			||||||
 | 
					                const opt = optimized[i];
 | 
				
			||||||
                if (opt instanceof Tag) {
 | 
					                if (opt instanceof Tag) {
 | 
				
			||||||
                    const k = opt.key
 | 
					                    const k = opt.key
 | 
				
			||||||
                    const v = properties[k]
 | 
					                    const v = properties[k]
 | 
				
			||||||
| 
						 | 
					@ -264,7 +266,23 @@ export class And extends TagsFilter {
 | 
				
			||||||
                    if (v === undefined) {
 | 
					                    if (v === undefined) {
 | 
				
			||||||
                        continue
 | 
					                        continue
 | 
				
			||||||
                    }
 | 
					                    }
 | 
				
			||||||
                    if (v !== opt.value) {
 | 
					                    if(opt.invert){
 | 
				
			||||||
 | 
					                        // We should _not_ match this value
 | 
				
			||||||
 | 
					                        // If 'v' is given, we already know what value it should be
 | 
				
			||||||
 | 
					                        // If 'v' is the not-expected value, we have a conflict and return false
 | 
				
			||||||
 | 
					                        // Otherwise, we can safely drop this value
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                        const doesMatch = (typeof opt.value === "string" && v === opt.value) ||
 | 
				
			||||||
 | 
					                            (v.match(<RegExp> opt.value) !== null)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                        if(doesMatch){
 | 
				
			||||||
 | 
					                            // We have a conflict as 'opt' is inverted
 | 
				
			||||||
 | 
					                            return false
 | 
				
			||||||
 | 
					                        }else{
 | 
				
			||||||
 | 
					                            optimized.splice(i, 1)
 | 
				
			||||||
 | 
					                            i--
 | 
				
			||||||
 | 
					                        }
 | 
				
			||||||
 | 
					                    }else if (v !== opt.value) {
 | 
				
			||||||
                        // detected an internal conflict
 | 
					                        // detected an internal conflict
 | 
				
			||||||
                        return false
 | 
					                        return false
 | 
				
			||||||
                    }
 | 
					                    }
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,10 +1,10 @@
 | 
				
			||||||
import { TagsFilter } from "../../../Logic/Tags/TagsFilter"
 | 
					import {TagsFilter} from "../../../Logic/Tags/TagsFilter"
 | 
				
			||||||
import { And } from "../../../Logic/Tags/And"
 | 
					import {And} from "../../../Logic/Tags/And"
 | 
				
			||||||
import { Tag } from "../../../Logic/Tags/Tag"
 | 
					import {Tag} from "../../../Logic/Tags/Tag"
 | 
				
			||||||
import { TagUtils } from "../../../Logic/Tags/TagUtils"
 | 
					import {TagUtils} from "../../../Logic/Tags/TagUtils"
 | 
				
			||||||
import { Or } from "../../../Logic/Tags/Or"
 | 
					import {Or} from "../../../Logic/Tags/Or"
 | 
				
			||||||
import { RegexTag } from "../../../Logic/Tags/RegexTag"
 | 
					import {RegexTag} from "../../../Logic/Tags/RegexTag"
 | 
				
			||||||
import { describe, expect, it } from "vitest"
 | 
					import {describe, expect, it} from "vitest"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
describe("Tag optimalization", () => {
 | 
					describe("Tag optimalization", () => {
 | 
				
			||||||
    describe("And", () => {
 | 
					    describe("And", () => {
 | 
				
			||||||
| 
						 | 
					@ -71,6 +71,14 @@ describe("Tag optimalization", () => {
 | 
				
			||||||
            expect(TagUtils.toString(opt)).toBe("amenity=binoculars&bicycle=yes")
 | 
					            expect(TagUtils.toString(opt)).toBe("amenity=binoculars&bicycle=yes")
 | 
				
			||||||
        })
 | 
					        })
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        it("should correctly optimize key=A&key!=B into key=A", () => {
 | 
				
			||||||
 | 
					            const t = new And([new Tag("shop", "sports"), new RegexTag("shop", "mall", true)])
 | 
				
			||||||
 | 
					            const opt = t.optimize()
 | 
				
			||||||
 | 
					            expect(typeof opt !== "boolean").true
 | 
				
			||||||
 | 
					            expect(TagUtils.toString(<TagsFilter>opt)).toBe("shop=sports")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        })
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        it("should optimize nested ORs", () => {
 | 
					        it("should optimize nested ORs", () => {
 | 
				
			||||||
            const filter = TagUtils.Tag({
 | 
					            const filter = TagUtils.Tag({
 | 
				
			||||||
                or: [
 | 
					                or: [
 | 
				
			||||||
| 
						 | 
					@ -263,7 +271,7 @@ describe("Tag optimalization", () => {
 | 
				
			||||||
                or: [
 | 
					                or: [
 | 
				
			||||||
                    "club=climbing",
 | 
					                    "club=climbing",
 | 
				
			||||||
                    {
 | 
					                    {
 | 
				
			||||||
                        and: ["sport=climbing", { or: ["club~*", "office~*"] }],
 | 
					                        and: ["sport=climbing", {or: ["club~*", "office~*"]}],
 | 
				
			||||||
                    },
 | 
					                    },
 | 
				
			||||||
                    {
 | 
					                    {
 | 
				
			||||||
                        and: [
 | 
					                        and: [
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue