# dvpwa **Repository Path**: mirrors_Azure/dvpwa ## Basic Information - **Project Name**: dvpwa - **Description**: No description available - **Primary Language**: Unknown - **License**: MIT - **Default Branch**: master - **Homepage**: None - **GVP Project**: No ## Statistics - **Stars**: 0 - **Forks**: 0 - **Created**: 2024-10-29 - **Last Updated**: 2026-03-14 ## Categories & Tags **Categories**: Uncategorized **Tags**: None ## README =============================================== DVPWA -- Damn Vulnerable Python Web Application =============================================== .. image:: https://raw.githubusercontent.com/vshymanskyy/StandWithUkraine/main/banner2-direct.svg :alt: StandWithUkraine :align: center :target: https://stand-with-ukraine.pp.ua Description =========== DVPWA was inspired by famous `dvwa`_ project and `bobby-tables xkcd comics`_. The purpose of this project is to implement real-world like application in Python with as many vulnerabilities as possible while having a good design and intentions. This project was used as demonstration of vulnerabilities during my `Web vulnerabilities`_ presentation at EVO Summer Python Lab'17. Running ======= Docker-compose -------------- DVPWA is packaged into docker container. All the dependencies described in :code:`docker-compose.yml`. You can easiliy run it and its dependencies using a simple command: .. code-block :: bash docker-compose up Then visit http://localhost:8080 in your favorite browser. To rebuild the container, please use ``./recreate.sh`` script, which will delete old container and create new from scratch. This script is primarly used in order to rebuild application image. If you have screwed up the database (i.e. with ``DROP TABLE students;``, please issue the following commands to recreate database container: .. code-block :: bash docker-compose stop postgres docker-compose rm # make sure, you remove only images you want to recreate docker-compose up postgres # recreate container and run Natively -------- If for some reasons you cannot use docker or docker-compose you can run the application on your host system. Requirements ~~~~~~~~~~~~ - Python3.6.2 - PostgreSQL database for data storage - Redis for session storage Installing and running ~~~~~~~~~~~~~~~~~~~~~~ .. code-block :: bash # Install application dependencies. pip install -r requirements.txt # Set up postgresql database Further I assume your db user # is named postgres and database name is sqli # Create database schema by applying migration 000 psql -U postgres --d sqli --host localhot --port 5432 \ -f migrations/000-init-schema.sql # Load fixtures into database psql -U postgres --d sqli --host localhot --port 5432 \ -f migrations/001-fixtures.sql # Modify config/dev.yaml cat config/dev.yaml <Is this bold? Yes! 3. Submit the review by clicking "Save" button. 4. Observe the newly created review. Note that text "Is it bold?" is bold, which means review content is probably neither sanitized on input nor escaped on output. 5. Open http://localhost:8080/courses/1/review. 6. Fill in review content with the following payload: .. code-block:: html 7. Submit the review by clicking "Save" button. 8. Observe the result. Result ~~~~~~ Now whenever you load http://localhost:8080/courses/1, you will receive an alert, which displays your cookie. You can play with different ways to inject your custom javascript to the page now: event handlers (i.e. ````, links with javascript targets, etc. Mitigation ~~~~~~~~~~ Escape all untrusted content, when you output it. In this example, to mitigate this kind of attack you can set ``autoescape=True`` when setting up templating engine (Jinja2) in ``sqli/app.py``. You can also sanitize text, when users input it and prohibit different kinds of code injection. Bad choice for storing passwords -------------------------------- Description ~~~~~~~~~~~ As per `check_paswword function `_ and `database initialization script `_, passwords are not stored in the database themselves, but their md5 hashes. Here are the problems with such approach: - As hash function produces same output for same input, same passwords will produce the same hash. Passwords are vulnerable to statistical analysis: it is possible to determine how many people use the same password, how popular the password is, etc: .. code-block :: sql sqli=# select pwd_hash, array_agg(username), count(*) sqli-# from "users" sqli-# group by pwd_hash sqli-# order by count(*) desc; pwd_hash | array_agg | count ----------------------------------+----------------+------- 5f4dcc3b5aa765d61d8327deb882cf99 | {j.doe,s.king} | 2 1da0bac388e8e0409a83e121e1af6ef4 | {p.parker} | 1 17c4520f6cfd1ab53d8745e84681eb49 | {superadmin} | 1 (3 rows) - Md5 is considered quite a weak hash, thus collisions can be easily found. Moreover, this hash is easy to bruteforce, as well as a lot of rainbow tables exists for md5. For example, `CrackStation website `_ can be used for such purposes. Mitigation ~~~~~~~~~~ Password themselves should never be stored in database. Special hash functions for passwords exist, such as argon2, bcrypt, pbkdf2. These functions should be used instead of plain text passwords or weak hashes like md5, or fast hash functions like sha1, sha2. For examples, see `password hashing `_ section on PyNaCL documentation. Cross-site request forgery -------------------------- TBA .. _`dvwa`: http://dvwa.co.uk .. _`bobby-tables xkcd comics`: https://xkcd.com/327/ .. _`Web vulnerabilities`: https://www.slideshare.net/OlexandrKovalchuk/web-vulnerabilities-78366279