Slashdot Mirror


The Most Dangerous Programming Mistakes

snydeq writes "Fatal Exception's Neil McAllister discusses the most dangerous programming mistakes, and what can be done to avoid them. 'Even more than input validation errors, this year's list is rife with application security blunders of all kinds. Some of them sound fairly esoteric, such as "inclusion of functionality from untrusted control sphere." But of all such errors, the highest-ranking one on the list is "missing authentication for critical function" — in other words, the attacker was able to gain access because there was no lock on the door to begin with,' McAllister writes. 'With the pace of Internet attacks accelerating, now is not the time to cut QA staff or skimp on testing and code review.'"

18 of 213 comments (clear)

  1. Better link by Chris+Mattern · · Score: 5, Informative

    If you'd like to read what the mistakes *are*, instead of a fluff piece that amounts to "oh, they're so awful! And people make them all the time, too!", here's the actual original article: http://cwe.mitre.org/top25/index.html

    1. Re:Better link by frinkster · · Score: 5, Insightful

      If you'd like to read what the mistakes *are*, instead of a fluff piece that amounts to "oh, they're so awful! And people make them all the time, too!", here's the actual original article: http://cwe.mitre.org/top25/index.html

      Is one of the mistakes "Not being able to click on a link"? I would check myself, but I can't click on the link.

    2. Re:Better link by baKanale · · Score: 3, Insightful

      Switch back to the Classic Discussion System.

    3. Re:Better link by tepples · · Score: 3, Insightful
      Obviously, not all mitigations on the list apply to all situations. Here are some examples where they wouldn't apply so easily:

      Where possible, avoid implementing custom authentication routines and consider using authentication capabilities as provided by the surrounding framework, operating system, or environment.

      This can prove cost prohibitive when the authentication capabilities provided by the surrounding operating system are marketed for use only by privileged employees, not by the public. Consider the case of an operating system that charges per user account. (Microsoft calls this the "client access license" model.) One might be tempted to use or create an authentication and authorization library that runs independently of the operating system's own auth facility, so that one needs to buy a system user account for only the web server, not for each member of the public who creates a user account on the web site.

      For outbound authentication: store passwords, keys, and other credentials outside of the code in a strongly-protected, encrypted configuration file or database that is protected from access by all outsiders

      Say I encrypt the keys that a web server uses to communicate with other web services, such as the key used to communicate with a payment processor. Now how do I store the key to decrypt those keys?

      For inbound authentication: Rather than hard-code a default username and password, key, or other authentication credentials for first time logins, utilize a "first login" mode that requires the user to enter a unique strong password or key.

      So how do we prevent an attacker from attacking a system while it is still in "first login" mode?

      Clearly specify which data or resources are valuable enough that they should be protected by encryption.

      Firesheep shows that this includes users' passwords and cookies containing authenticated session tokens. But with StartSSL having suspended operations and Internet Explorer on Windows XP still not supporting Server Name Indication, how can hobbyist web developers get the certificate and dedicated IPv4 address needed to host an SSL site?

      If possible, create isolated accounts with limited privileges that are only used for a single task.

      Please see my comment above about the CAL pricing model.

      Generate a unique nonce for each form, place the nonce into the form, and verify the nonce upon receipt of the form.

      If you've ever seen errors about a "form key" on Slashdot, Slashdot is doing exactly this.

      Do not use the GET method for any request that triggers a state change.

      Is a hit counter a state change?

      Use a built-in path canonicalization function (such as realpath() in C)

      According to this page: "The realpath() function is not described in the C Standard." It's available only in UNIX, not in Windows.

      Avoid inconsistent messaging that might accidentally tip off an attacker about internal state, such as whether a username is valid or not.

      Does this mean don't bounce messages to nonexistent users but instead treat them as delivered and discard them? That would provide a bad user experience for people attempting to contact these users.

      Use code signing technologies such as Authenticode.

      How does a hobbyist afford the certificate for Authenticode?

      For all configuration files, executables, and libraries, make sure that they are only readable and writable by the software's administrator.

      Writable I agree with, but readable I'm not so sure. If configuration files are readable only by the adm

  2. Those aren't "programming" mistakes... by ThosLives · · Score: 4, Insightful

    ...Those are system design mistakes.

    A programming mistake is one where you meant to type x+1 and instead you write x-1. Missing something like authentication or checking is a requirements or design problem, not a programming problem.

    If software was a car, you wouldn't say it's a manufacturing problem if the car didn't have a place to install a lock - you'd say it's a design problem. It would only be a "programming" issue if it had a place for a lock but it was left uninstalled.

    (Yes, I don't consider "programming" to include the design aspects; I consider "programming" to mean "conversion of requirements into computer code." The errors about which this article talks are mostly requirements problems, not implementation problems.

    --
    "There are a dozen opinions on a matter until you know the truth. Then there is only one." - CS Lewis (paraprhase)
    1. Re:Those aren't "programming" mistakes... by Short+Circuit · · Score: 5, Interesting

      You seem to be advocating a distinction of responsibility of knowledge where programmers should not need knowledge of design. I would dispute that.

      First, all you've done is replace "programmer" with "compiler." If you posit that there is no need for programmers to do anything more than convert a design specification to code, then all you've done is define programmers as transcoders operating on a higher-level formal langauge than current compilers already do. That seems ridiculous; you'd be able to replace "programmers" with "compilers" for this higher-level language ("Technical writing in English") your design spec is written in. At that point, your designers are doing nothing more than programming in a higher-level language...making them programmers again. Look at the trends in new and redeveloped languages to include declarative behaviors for evidence of this already happening; dataflow-driven and declaration-driven language features are getting a lot of attention.

      Second, if your programmers aren't expected to have or build knowledge of good design and design practices, then they won't be able to identify mistakes--especially critical mistakes such as the ones discussed in TFA. People are people, people make mistakes. Without other people or tools (created by people) there to catch some the mistakes, more of the mistakes slip past. And while it's perhaps easy to build a unit test suite from a design document, that unit test suite is going to be better at detecting flaws in the code, not in the design.

  3. Headline reveals slashdot philosophy? by damn_registrars · · Score: 5, Insightful

    it's-probably-fine,-we'll-test-it-live

    Could describe every "upgrade" to slashdot that has happened since ... well probably ever.

    --
    Damn_registrars has no butt-hole. Damn_registrars has no use for a butt-hole.
  4. Did you try double right-click? by tepples · · Score: 3, Informative

    To work around Slashdot's brokenness, did you try double right-click, then open in new tab? It appears to work for me in Firefox 5.

    1. Re:Did you try double right-click? by EuclideanSilence · · Score: 3, Funny

      I got it.

      Double left click, right click, left click, triple right click, A, A, B, A, Up, Up, Up and I can see almost see slashdot as any other forum!

      It's pretty much the same command in Emacs.

  5. Re:why am I not surprised sql injection is first? by Joce640k · · Score: 3, Insightful

    Dear Web Developers,

    Stop using toy languages. A strongly typed language that only accepts type "SanitizedString" as an SQL function parameter will end this problem forever.

    --
    No sig today...
  6. Missing a Big One by Salamander · · Score: 3, Interesting

    The Mitre list does include "Use of a Broken or Risky Cryptographic Algorithm" but in my experience that's far less common than improper use of a perfectly good algorithm. Many algorithms and modes have known weaknesses that require specific generation/handling of keys and initialization vectors to maintain good security. Most algorithms and modes that are secure against unauthorized *reading* of data still require an extra MAC step to prevent unauthorized *modification* of that data (including targeted bit-flips). Developers often take shortcuts in these areas because doing all of "the right things" adds a lot of extra complexity and can absolutely kill performance. Look at recent events involving Dropbox and Jungledisk for examples. I don't think the Mitre list adequately conveys that cryptographic security requires not just good low-level algorithms like AES or Blowfish but also good higher-level (usually domain-specific) algorithms governing how the low-level algorithms and their inputs are used.

    --
    Slashdot - News for Herds. Stuff that Splatters.
  7. What kind of mistakes they are by DragonWriter · · Score: 4, Informative

    Those are system design mistakes.

    While TFS and TFA call them "programming" mistakes, the actual source refers to them as the "Top 25 Most Dangerous Software Errors".

    A programming mistake is one where you meant to type x+1 and instead you write x-1.

    No, that's a typographical error, not a programming mistake.

    A programming mistake is when you incorrectly analyze the requirements and think you need to type x-1 to correctly implement them when in fact you need to type x+1.

    But either one results in a "software error"; the list and the original source are fine, the fluff piece in between the original source and Slashdot (and, consequently, the Slashdot summary) is the only potential problem here.

    If software was a car, you wouldn't say it's a manufacturing problem if the car didn't have a place to install a lock - you'd say it's a design problem. It would only be a "programming" issue if it had a place for a lock but it was left uninstalled.

    While its fun to construct ways to point the finger somewhere else in an organization, or to pedantically categorize errors in to narrow boxes, what I'd say is that its a failure of each and every person who had sufficient contact with the product that they should have seen the relevant facts, and sufficient technical skill that they should have recognized the error, and who either did not recognize the error or who did recognize the error but did not take action to have it corrected [whether that was implementing a fix or providing notice up the line]. Plus all the people responsible for the process that produced the error.

    And most of the errors on the list are things that, whether or not they should be explicitly foreseen in requirements, programmers are positioned to recognize and ought to be taking steps to prevent. Programming isn't narrowly constrained assembly-line work, at least in any organization that expects to produce quality software.

  8. Re:QA - Microsoft is really to blame. by Joce640k · · Score: 4, Informative

    Microsoft's Visual C++ compiler will throw a huge number of warnings for things like strcpy, telling you to use strncpy_s or something like that.

    You shouldn't even be using strcpy(). std::string has been around for more than ten years now.

    Similarly arrays: Don't use them, use std::vector instead. Visual C++ vector even does range checking by default so this throws an exception instead of corrupting memory:

    std::vector foo(10);
    foo[11] = 123; // Will throw an exception in VC++...

    A few basic changes in programming style can make C++ as safe as Java (but with none of the drawbacks). If you're still writing C code with your C++ compiler you're Doing It Wrong.

    --
    No sig today...
  9. The unmentioned BIGGER mistake... by ka9dgx · · Score: 4, Insightful

    Using a system where the program has to be trusted to do its job correctly is the bigger mistake. When you hand your car keys to a valet, you don't also give him power of attorney to sell your house, liquidate your stocks, savings, etc... but every operating system out there does something like that when you tell it to run a program. The program you run can do anything you are authorized to do. The default assumption is that it should have permission to do anything, no matter how stupid, dangerous, or downright evil.

    This practice needs to end, about 10 years ago it should have ended... and we'll probably have to wait 10 more years because it's so freaking hard to get this idea across, nobody seems to be ready for it yet, by the way things seem to be going.

    A user should be able to decide exactly which and how much of the resources they are authorized to use will be allowed to be accessed by a program they choose to run. If you want to run a program with read/write access to /sandbox, and the ability to read from the internet using a filtered http driver (one that doesn't allow puts, for example), you should be able to do so, without having to do any fancy footwork.

    If put in to place, this type of system, which explicitly states what access things get, make it almost trivial to never get a virus or worm ever again. It's time to stop trusting programs, and only have to trust the hardware and OS to enforce our wishes.

    I impatiently await the arrival of capability based security.

    1. Re:The unmentioned BIGGER mistake... by ka9dgx · · Score: 4, Interesting

      It is unfortunate that it's necessary to try to overload an existing term with a new, slightly different definition, but that's what has happened here.

      Capabilities in a capability based system (cabsec for short) are fine grained access rights to things like single files, a directory, etc. They are explicitly granted at run time to a process, they are not persistent.

      This is in contrast to the Access Control List permissions of VAX/VMS, inherited by Windows, and the User/Group/World permissons of Unix/Linux, both of which are geared towards restricting the rights of a user, and are generally persistent.

      For example, if you give a process write access to a log file, and I/O access to the internet, and 1% of total cpu use....it could watch something on the net, and log the results. It could not access anything other than the log file, no matter what... this means a buffer overflow bug, or any other fault could not cause data to leak from the system.

      In a permissions/ACL based system, the administrator decides the rights of users, and the programs they run, without distinction

      With cabsec capabilities, the user decides which of their rights they wish to delegate to a process or program. This frees the user from the need to trust a given program with all of their rights. This also frees the system administrator from having to set up extra accounts with limited permission for each new service or daemon that needs to be run.

      I see cabsec as a very useful extension to the existing user permissions systems. Combined you get the best of both worlds.

  10. Re:why am I not surprised sql injection is first? by oursland · · Score: 3, Interesting

    Because you can doesn't mean that everyone should.

    You cited a few shining examples of people doing the right thing. Unfortunately, those languages make it so damn easy to do the wrong thing and that is precisely the problem.

  11. Re:why am I not surprised sql injection is first? by mvdwege · · Score: 3, Insightful

    Bull. 2 out of those three examples make it easy to do the right thing.

    Perl has DBI, which does parametrised queries very well, and in fact makes quoting a pain, because all the nice functions like binding variables to query results do not work well (if at all) with quoted queries. And if DBI is too low-level, there is always the DBIx::Class ORM.

    Python's DB-API 2.0 is the standard to do databases in Python, and it too makes parametrised queries a lot simpler than quoting. And on top of that you can use SQLAlchemy.

    I agree with PHP being bad. Sure it has parametrised query support in various libraries, but the default is still to use MySQL as backend and use the various mysql_yes_really_quote_it_correctly_this_time() functions. Given that until recently even the maintainers of the language didn't have clue as to how to do security right, PHP is by far the most unsuitable language to do secure web programming in.

    Mart

    --
    "I know I will be modded down for this": where's the option '-1, Asking for it'?
  12. Re:Table-valued parameters; query by example by mvdwege · · Score: 3, Insightful

    In Perl this is easy. Let's say that @args holds your argument list. Then it goes like this:

    my $query = "select * from table where column in (";

    # Use $#args to get the index value of the last member of the array @args. This gives us a loop that's one iteration shorter than the lenght of the argument list.

    for (1..$#args) { $query .= ' ?, '}

    $query .= '? )';

    my $sth = $dbh->prepare($query);

    $sth->execute(@args);

    # Bind result columns with $sth->bind_col

    while ($sth->fetch) {

    # Do something with the results.

    }

    Mart

    --
    "I know I will be modded down for this": where's the option '-1, Asking for it'?