How to Check Authenticity of an AJAX Request

后端 未结 15 871
悲&欢浪女
悲&欢浪女 2021-01-31 08:47

I am designing a web site in which users solve puzzles as quickly as they can. JavaScript is used to time each puzzle, and the number of milliseconds is sent to the server via A

相关标签:
15条回答
  • 2021-01-31 09:17

    there is a very fast implementation of cryptography in js here

    http://crypto.stanford.edu/sjcl/

    it allows public / private key encryption all on the client and I think you can adapt it to encrypt the Ajax communication between your server and the client browser

    here is a detailed explanation, which you can adapt to your needs http://crypto.stanford.edu/sjcl/#usage

    0 讨论(0)
  • 2021-01-31 09:19

    The way I would do this is that when the server sends the puzzle to the client, the current time is stored in a session. This ensures that the timing starts immediately after the puzzle has been sent. After the puzzle has been completed, and is sent to the server to check if the puzzle was done right, the server again checks the time and does a comparison.

    Obviously slow Internet connections can make this time bigger, but there's nothing you can do about it.

    0 讨论(0)
  • 2021-01-31 09:23

    -- Edit:

    This solution is somewhat flawed, as pointed out by ZoFrex below.

    -- Old:

    Here is a way (but you'll need to do some profiling).

    Send down a series of "problems" for the JavaScript to solve, while they are playing the puzzle. Previously, I've sufficiently-sized number N such that it is the result of: prime1 * prime2. This forces the client to factor the number (you can get code to do this in JavaScript) and this will take time (this is where profiling clients comes in, and sending down appropriately-sized primes [obviously, this opens you to degradation-attacks, but nevertheless]).

    Then, you just send down say, 500, of these prime-problems (or another type), and let the JavaScript solve them in the background. It will generate a list of solutions, and when you send the completed value, you also send this list. From the total count of answers supplied, you can determine how long they spent on the puzzle.

    Cons:

    • Requires profiling to determine capabilities of various clients (and hence difficulty of problems)
    • Can be downgrade-attacked
    • Slightly complicated
    • JavaScript computation may interrupt general puzzle-solving
    • Possible to write a bot to get solve problems faster than JS

    Pros:

    • Calculations must be done in order to submit the form
    • If implemented correctly, will prevent all but non-trivial attacks

    Clearly, it's attackable, (all proposed answers are), but I think it's reasonable. At least, it would be fun to work on :)

    In the end, though, you need to actually install a client-side system with a bit more security. And do note that Flash certainly is not this; it's trivial to decompile. Infact, there was an IQ test here in Australia once, and it was controlled via a Flash app that was done LIVE on television. Of course, the winner was a computer programmer, I wonder why :P

    -- Edit:

    OP, Also, I linked it in a comment to this post, but just incase you miss it, you are kind of interested in the Hashcash, which is the aim to show that a client has completed some amount of 'Work'. Even if my implementation isn't suitable, you may find a review of that field fruitful.

    0 讨论(0)
  • 2021-01-31 09:26

    You can't guarantee the security of the timings cryptographically, because the client's browser can't do secure computation. Any means for encrypting to/from the server could be bypassed by adjusting the actual timings.

    And timing on the server doesn't work, either - if you don't take account of latency in the round-trip-time, users with lower latency connections will have an advantage; if you do, users could thwart the compensation phase by adding extra latency there and then removing it later.

    You can, of course make it difficult for the users to modify this, but security by obscurity is an unsustainable policy anyway.

    So it comes down to either trusting your users somewhat (a reasonable assumption, most of the time) and designing the game so it's not trivial to circumvent the timings.

    0 讨论(0)
  • 2021-01-31 09:26

    Even a compiled application could be forged. If the user changes their system clock halfway through timing, your application will report an incorrect time to the server. The only way to get an accurate upper-bound on the time it takes them is to start timing on the server when the puzzle is given to them, and to stop timing when they supply the answer.

    As others have pointed out you can minimise the effect that slow connections have by making the load of the puzzle as small as possible. Load the entire page and "game engine" first, and then use an asynchronous request to load the puzzle itself (which should be a small amount of data) to level the playing field as much as possible.

    Unfortunately you can't do latency compensation as this would be open to tampering. However, on a connection that's not being used for anything else, the latency for a request like this would be greatly overshadowed by the time it takes a human to solve a puzzle, I don't think it will be a big deal.

    (Reasoning: 200ms is considered very bad lag, and that's the average human reaction time. The shortest possible "puzzle" for a human to complete would be a visual reaction speed test, in which case bad lag would have a 100% markup on their results. So as a timing solution this is 2-OPT. Any puzzle more complex will be impacted less by lag.)

    I would also put a banner on the page saying to not use the internet connection for anything else while playing for the best possible speeds, possibly linking to a speed / latency tester.

    0 讨论(0)
  • 2021-01-31 09:27

    You have to use server-side time here. Here is how I would do it:

    Make an AJAX request on document ready to ping the server. When server-side code receives the ping, store the server-side time as a session variable (making sure the variable does not already exist). When they finish the quiz, take the server-side time again and compare it with the session variable to determine their duration. Remove the session variable.

    Why this works:

    • You do not start the timer before they see the quiz
    • The network delay is factored in, because the timer does not start until the AJAX request comes in (if they have a slow connection, the AJAX request will be slow)
    • Ping is not spoofable because you make sure the session variable does not exist before storing

    EDIT: I wanted to add that you could continue to keep client-side time, and include it in the final post. Then you can compare it with your server-side calculated time. If they are reasonably close, then you can trust the client time.

    0 讨论(0)
提交回复
热议问题