最新消息:雨落星辰是一个专注网站SEO优化、网站SEO诊断、搜索引擎研究、网络营销推广、网站策划运营及站长类的自媒体原创博客

javascript - Easily detect when user alters DOM - Stack Overflow

programmeradmin0浏览0评论

Is it possible to easily detect DOM manipulation by the user?

When a user uses the console in any modern browser, he/she can manipulate the DOM in ways the developer did not intend.

I have a web app that is very much tied to the DOM being in certain states and should the user do anything to the DOM via a console, I'd like to be notified.

The answer:

  1. Doesn't need to be browser agnostic
  2. Doesn't need to be perfect. I fully understand that most, if not all, methods could be circumvented, but I'd like a good general solution.
  3. Can't be too convoluted. I'm not interested in registering an event handler with all DOM events that checks some flag set when my code performs an DOM manipulation

Edit:

There appears to be some confusion in the answers I've received thus far. As pointed out in #2 above, I understand that most, if not all, methods can be circumvented.

In addition, this is an internal tool and thus is protect by a VPN. Further more, there is server-side checking. However, there are reasons, which I cannot elaborate upon, for me wanting to know when a user (who are few in number) manipulated the DOM.

To be clear, this isn't for security reasons. I'm not trying to stop malicious users here. Think of this more as out of curiosity.

Is it possible to easily detect DOM manipulation by the user?

When a user uses the console in any modern browser, he/she can manipulate the DOM in ways the developer did not intend.

I have a web app that is very much tied to the DOM being in certain states and should the user do anything to the DOM via a console, I'd like to be notified.

The answer:

  1. Doesn't need to be browser agnostic
  2. Doesn't need to be perfect. I fully understand that most, if not all, methods could be circumvented, but I'd like a good general solution.
  3. Can't be too convoluted. I'm not interested in registering an event handler with all DOM events that checks some flag set when my code performs an DOM manipulation

Edit:

There appears to be some confusion in the answers I've received thus far. As pointed out in #2 above, I understand that most, if not all, methods can be circumvented.

In addition, this is an internal tool and thus is protect by a VPN. Further more, there is server-side checking. However, there are reasons, which I cannot elaborate upon, for me wanting to know when a user (who are few in number) manipulated the DOM.

To be clear, this isn't for security reasons. I'm not trying to stop malicious users here. Think of this more as out of curiosity.

Share Improve this question edited Nov 24, 2012 at 0:17 Avi asked Nov 23, 2012 at 23:57 AviAvi 1,35610 silver badges26 bronze badges 4
  • 1 What's wrong with the user manipulating the DOM? Your app shouldn't do anything sensitive (such as handling authentication) client-side. If the app is "tied to the DOM being in certain states", and the user changes the DOM, they can't plain when the app stops working. – Xenon Commented Nov 24, 2012 at 0:00
  • "DOM Mutation Events" might be the keyword: developer.mozilla/en-US/docs/DOM/Mutation_events – Samuli Hakoniemi Commented Nov 24, 2012 at 0:00
  • stackoverflow./questions/2844565/… – Sidharth Mudgal Commented Nov 24, 2012 at 0:01
  • 2 You can not trust the client, server needs to validate everything submitted to it. The user does not even need to use your webpage to alter it, they can submit requests with mand lines. :) – epascarello Commented Nov 24, 2012 at 0:01
Add a ment  | 

5 Answers 5

Reset to default 7

Don't do that. Code your web site to not trust user input and then don't care what the user does. If invalid input is submitted then reject it. Everyone is happy.

It's easy to think that you own the user's browser. You don't. It's serving you but only at the whim of the user.

If you really must know when the DOM is modified, you can use mutation events, and now, MutationObserver (more on this). For really old browsers that don't offer this—and this seems a really fragile design—then just do what amounts to calculating checksums. After each legitimate step of the site's approved function, traverse the DOM elements you care about and record their positions, values, or whatever you are concerned with. At intervals, validation time, or a next UI interaction, pare. This is the only prehensive way to detect DOM changes in old browsers.

Regarding mutation events, be sure to look for whatever the next technology is to be sure they haven't been replaced.

Ultimately, nothing you do can stop someone determined to defeat your scheme. If anything, the user can copy the browser's POST request using developer tools, tweak it, and write a tiny program to submit his own malicious POST request. It is more important to protect your server from malicious input than it is to make your web page supposedly bullet-proof (because it won't be).

DOM mutation events work in current versions of all major browsers and do what you want. The following will cover mon DOM modifications within the whole document:

function handleDomChange(evt) {
    console.log("DOM changed via event of type " + evt.type);
}

document.addEventListener("DOMNodeInserted", handleDomChange, false);
document.addEventListener("DOMNodeRemoved", handleDomChange, false);
document.addEventListener("DOMCharacterDataModified", handleDomChange, false);

DOM mutation events will eventually be replaced by mutation observers, which are implemented in recent Mozilla and WebKit browsers.

This is a pretty interesting question, and I think DOM mutation events may be a best solution. One thing I was initially thinking I might do is run a timed function that checks the DOM for specific modules, based on data- attributes or IDs. If I was building my page entirely client-side through JS, I would have a build configuration object for each module (DOM element like:

<div id='weather-widget' data-module-type='widget'>
    <h1 data-module-name='weather'>Weather</h1>
    <!-- etc etc -->
</div>

Anyhow, my config object would contain all of these things like module type, module name, etc, etc:

//Widget configuration object
var weatherWidgetConfig = {
    type: 'widget',
    name: 'weather'
}

and I would inspect the DOM element and all of its children to make sure the data- attributes still matched the configuration object, that they existed, and that they have not been changed. If they have, I would call a module.destroy() and module.build() again with the correct configuration.

I've received a lot of answers in which the respondent delivers advice about how to build a web app. While that may be useful to some readers, that isn't answering the question. Some, however, have attempted to answer. The closest I seen to a plete answer was given by @Keith. The only problem is that it fails the 'easy' test.

It appears that the correct answer, as some have said, is NO - it isn't possible to easily detect DOM manipulation by a user.

I recently discovered "Selector Listener", a technique that relies on css to detect DOM changes. It doesn't work in IE 9-. Applying it to the whole DOM doesn't sound like a good idea, the intent is rather to work with specific selectors.

More details can be found in this blog post.

发布评论

评论列表(0)

  1. 暂无评论