admin管理员组

文章数量:1343904

If I know I have a value say 'JohnSmith' stored in some object in the global space of the browser and I hit some breakpoint, how do I know which object or variable has a value of 'JohnSmith'? I have hundreds of objects and I don't want to manually search for it in dev tools. I have used some scripts in the console or snippet which traverse the all the objects but they all have failed to search properly or give some error message.

I tried JS Runtime Inspector which looked promising but it doesn't work at all in my version 61 of Chrome. Its subtab doesn't show up like their video shows.

If I know I have a value say 'JohnSmith' stored in some object in the global space of the browser and I hit some breakpoint, how do I know which object or variable has a value of 'JohnSmith'? I have hundreds of objects and I don't want to manually search for it in dev tools. I have used some scripts in the console or snippet which traverse the all the objects but they all have failed to search properly or give some error message.

I tried JS Runtime Inspector which looked promising but it doesn't work at all in my version 61 of Chrome. Its subtab doesn't show up like their video shows.

Share Improve this question asked Oct 3, 2017 at 0:43 Tony_HenrichTony_Henrich 44.3k80 gold badges252 silver badges390 bronze badges 4
  • Do you have a list of objects? or Do you want to check all objects in global space? – zhuravlyov Commented Oct 3, 2017 at 0:52
  • All of them or starting from an object. So window or the name of the object can be an input parameter to the function doing the search – Tony_Henrich Commented Oct 3, 2017 at 1:22
  • Could the searched for object be in a variable declared using let? These don't exist as window properties and I am unaware you can search their name space. – traktor Commented Oct 3, 2017 at 1:56
  • Traktor53 App is patible with browsers which don't support let. So let is not used. – Tony_Henrich Commented Oct 3, 2017 at 3:45
Add a ment  | 

1 Answer 1

Reset to default 14

Here's a configurable digger that you can keep in devtools snippets panel and run via keyboard:

  • CtrlP (or P) then !name where name is the name of the snippet
  • CtrlEnter (or Enter) when editing the snippet's code.
((
  haystack = window, // $0 is the last inspected DOM element
  needle = 'foo', // 'string' or /RegExp/
  target = 'kv', // kv = keys and values, k = keys only, v = values only
  caseSensitive = true, // for a plain string needle
  ownKeysOnly = !true, // only dig into own keys in objects
  maxResults = 1000, // 0 = don't limit
  skipDOM = ['innerHTML', 'outerHTML', 'textContent', 'innerText', 'outerText'],
  skipValues = new Set([haystack, window, document.doctype]),
) => {
  let isRe, fnLow, fnIn, num = 0;
  const isSimpleId = /^[_$a-z][$\w]*$/iu;
  const path = [], pathKeys = [], inK = /k/i.test(target), inV = /v/i.test(target);
  if (!(isRe = needle instanceof RegExp)) {
    fnIn = dig.call.bind(''.includes);
    if (!caseSensitive) needle = (fnLow = dig.call.bind(''.toLowerCase))(needle);
  }
  dig(haystack);
  function check(v, name) {
    const t = typeof v;
    const n = typeof name === 'symbol' ? name.description : name;
    if (inK && (isRe ? needle.test(n) : fnIn(fnLow ? fnLow(n) : n, needle)) ||
        inV && (t === 'string' || t === 'number') &&
        (isRe ? needle.test(v) : fnIn(fnLow ? fnLow(v) : v, needle))) {
      let p = '';
      for (let k of pathKeys)
        p += !isSimpleId.test(k) ? `[${k}]` : p ? '.' + k : k;
      p = p.replace(/(^|[.\]])(children\[)(\d+)]((?:\.nextElementSibling)+)\b/g,
        (s, a, b, c, d) => a + b + (+c + d.split('.').length) + ']');
      console.log('#%s %s: %o in %o', ++num, name, v,
        {obj: path.at(-1), path: p, ancestry: path.slice(0)});
      if (!--maxResults) return 1;
    }
    if (v && t === 'object' && !skipValues.has(v)) {
      skipValues.add(v);
      pathKeys.push(name);
      return dig(v);
    }
  }
  function dig(o) {
    path.push(o);
    let res;
    if (Array.isArray(o)) {
      for (let len = o.length, i = 0; i < len; i++)
        if (check(o[i], i)) { res = 1; break; }
    } else if (o instanceof Map || o instanceof Set) {
      for (const e of o.entries())
        if (check(e[1], e[0])) { res = 1; break; }
    } else if (typeof o === 'object') {
      const isDom = skipDOM?.length && o instanceof Node;
      for (const k in o)
        if ((!ownKeysOnly || Object.hasOwn(o, k))
        && !(o === window && k.startsWith('webkit'))
        && !(isDom && skipDOM.includes(k)))
          try { if (check(o[k], k)) { res = 1; break; } } catch(e) {}
    }
    path.pop();
    pathKeys.pop();
    return res;
  }
})();

本文标签: