Security is a topic that comes up from time to time. It is not an issue from the beginning, but once something bad happens it is usually considered the one to blame. Software is complex, the human programming the machine is far from being perfect, and the user may not follow best practices either. So how can we create a secure system?
The web is one of the most insecure places possible. Computers with potential security risks connected to each other. Servers that can receive arbitrary data. Clients that execute code from unknown sources. While we cannot control the security of the servers, we have to do something to protect the clients. Even though JavaScript could be considered a secure scripting language, the code for any ActiveX, Flash or Silverlight plugins is definitely not. Furthermore, even though JavaScript itself is sandboxed, it might be used in such a way that the user will trigger insecure actions.
In this tutorial we will see the web security model in action. We will go into best practices and general guidelines for building secure web applications. We’ll learn what the cross-origin resource sharing policy is and how we can control it. Finally we will also talk about sandboxing (external) content.
One of the most important guidelines has nothing directly to do with HTML directly: Use HTTPS! The relation to HTML lies of course in the distribution of our hypertext documents. Nevertheless we have to realize that using HTTPS for the transport of our document and using HTTPS for our resources are two different things. It is definitely necessary to check if all contained resources really use the https://
scheme.
Another important guideline is related to user-defined content. Once we allow users to enter data on a form, we need to be careful. Not only do we need to make sure that the webserver is protected against common attacks, such as SQL injection, we also need to make sure that stored data is not used within executable code without caution. For instance we should escape strings to not contain any HTML. HTML alone is not malicious, but it can trigger script execution or resource fetching. One way to allow users to write HTML, which will be placed on the output page without modification, is to white-list certain tags and attributes. The other elements will be escaped.
Our JavaScripts should also minimize exposure and trust to third-party libraries. Of course we use the immediately-invoke-function expression (IIFE) to prevent the global context from being polluted; however, another reason is to not leak (probably) crucial internal states, which may then be changed by other scripts willingly or by chance.
(function () { // Standard code here. })();
For ourselves it is certainly good practice to rely on 'use strict';
and the conveyed benefits. Nevertheless, constraining the running script does not prevent us from using APIs with potentially corrupted data. Cookies and content in the localStorage
may be changed or viewed by the user or other programs depending on conditions that cannot be influenced by us. We should therefore always implement some sanity checks, which give us a hint to detect integrity flaws as soon as possible.
Finally we should make sure to only use trusted third-party resources. Using scripts from other servers within our site makes it possible to mutate the page or breach the privacy of our users.
The concept of cross-origin resource sharing (CORS) is simple. The browser does not allow the embedding of special resources from different origins unless the origins explicitly allow it. Special resources can be, e.g., web fonts or anything requested via an XMLHttpRequest
. Cross-origin AJAX requests are forbidden by default because of their ability to perform advanced requests that introduce many scripting security issues.
The origin is basically defined via the used protocol, host and port combination. Therefore http://a.com is different from https://a.com, which is different from https://a.com:8080. They are all different to http://b.com.
Clients can be allowed to use resources by including a certain header in the response. The browser then determines if the current website is allowed to use the resource or not. The origin is usually determined via the domain of the current website.
Let’s have a look at an illustrative example. In the following we assume that our page is located at foo.com
. We request JSON data from a page hosted at bar.com
. For the JSON request we use the XMLHttpRequest
as shown below.
var xhr = new XMLHttpRequest(); xhr.open('GET', 'https://bar.com/users'); xhr.addEventListener('load', function (ev) { if (xhr.status === 200) { var result = JSON.parse(xhr.responseText); // ... } }, false); xhr.send();
The browser already anticipates the possibility of a CORS-enabled response by adding the Origin
header to the request:
Origin: http://foo.com
Now the server has to deliver the right response. Not only do we want the correct JSON to be transported, but even more importantly we require CORS specific headers. It is possible to use wildcards. For instance, the following example will grant the right to use the requested resource to any request.
Access-Control-Allow-Origin: *
CORS can also be used as an alternative to the JSONP workaround. JSONP uses scripts to make cross-origin AJAX requests resulting in JSON responses. Before CORS, cross-domain calls were prohibited in general, but including scripts from different domains was always acceptable. In most APIs a JSONP response was triggered by providing a special query parameter, naming the callback function.
Suppose the call to http://bar.com/api
results in the following JSON response:
{ "name": "example", "factor": 5, "active": true }
The JSONP call to, e.g., http://bar.com/api?jsonp=setResult
would give us:
setResult({ "name": "example", "factor": 5, "active": true });
Since the result of JSONP is only seen by the <script>
element, the GET request method is implied. There is no possibility to use anything different. CORS gives us much more freedom in that area, such that we can also determine other parameters. This is all enabled by allowing us to freely use the standardized XMLHttpRequest
object.
An ideal solution would only fall back to JSONP for legacy browsers, while embracing CORS for more recent ones. This will prevent many cross-site scripting (XSS) issues originating from compromised external sites. Embedding scripts from external pages has always been a risky business. An even better fallback would be to redirect the JSON request from our server to the target machine. This way we talk with our (trusted) server, which gets the response from the target, evaluates its sanity and returns the (valid) result.
Every document
comes with its own window
. Accessing this window
is mostly proxied to the window
of the current browsing context, which drives the tab that we see. The browsing context is created with several options, such as the parent context, the creator, and the initial page. Along with these options, security flags are set. The flags set up the capabilities and restrictions of the context. In effect it is possible to prevent certain behavior, such as running scripts or opening new tabs.
How can we set the security flags of a new context? We set the context via attributes that are assigned to the elements, which need to create a new context. Currently only the iframe
element has such an attribute, even though standard frames would also fit to the former description. However, standard frames are considered deprecated and are therefore not really well supported. Even though the HTML5 standard mentions them, they should not be used anymore.
There are a bunch of flags available. The most important ones are:
allow-top-navigation
(allows changing the top context)allow-plugins
(enables embed
, object
, …)allow-same-origin
(content from the same origin may be accessed)allow-forms
(forms can be submitted)allow-popups
(popups / new contexts won’t be blocked)allow-pointer-lock
(enables the pointer-lock API)allow-scripts
(allows script execution)The <iframe>
uses the sandbox
attribute to enter the sandboxed mode. If this attribute is not specified, everything is allowed as we know it. With this attribute, everything is forbidden. The previously stated flags therefore enable certain features.
Let’s take a closer look at the allow-same-origin
flag. By default, the policy for an iframe
is really relaxed. If we do not specify the sandbox
attribute then only embedded pages from the same domain are allowed to read, e.g., cookies or the browser’s local storage for the given domain. Of course other risks exist as well—that’s why we usually want to supply the sandbox
attribute.
The following picture shows the default behavior in a simple diagram. While the second iframe
is allowed to access the previously stated content, the first one isn’t. The reason is the different domain, highlighted in yellow.
So how does this picture change with the sandbox
attribute? Content from a different domain is still prohibited. Therefore we only look at the additional possibilities for contents from the same origin. By default, even content from the same origin is treated as if it came from a different origin.
There are more features that can be controlled or set for contexts, but these are unfortunately transported via their own attributes. A great example is the allowfullscreen
attribute. Again it is only available on an iframe
. In principle it allows applications to go into full-screen mode.
Furthermore, we should note the seamless
attribute for an <iframe>
element. This attribute enables the styling of the parent document on the contained document. It is especially powerful in conjunction with the srcdoc
attribute, which allows us to provide the source of an iframe
directly, without requiring HTTP requests or using a data URI. This way we can very easily allow user content without worrying about script execution.
An example to display user content in a sandboxed environment is shown below.
<iframe sandbox seamless srcdoc=" <div class=comment> <h3>Example User</h3> <span class=comment-content>Custom, but <strong>safe</strong>!</span> </div> "></iframe>
Sandboxed iframes can be used for many tasks. For instance, if we want to evaluate scripts safely, we could pass the content to evaluate to a special handler sitting inside an iframe
. The handler would call the eval
function. The inline frame is sandboxed to only allow scripting, nothing else. No popups can be opened, navigation cannot be used, and our whole DOM is separated.
The exact implementation also requires HTML5 messaging between documents. We send the evaluation request with the postMessage
method and listen for the response via the message
event.
We can also sandbox parts of the current page using the content security policy (CSP). This property is usually transported via the page’s HTTP headers, but HTML gives us the ability to set it in a <meta>
tag.
An example looks as follows. We need to use the right http-equiv
value. The content is a series of definitions specific for the page.
<meta http-equiv="Content-Security-Policy" content="default-src https://bar.com; child-src 'none'; object-src 'none'">
The different definitions (called policies) also accept wildcards. The exact purpose and look are strongly dependent on the policy being used.
Web security is possible, even though it is hard to achieve and depends on a lot of (often impossible to control) external parameters. If we can minimize these external parameters, such as inserted content or scripts, then we are usually in good shape. Most attack vectors can only be used from scripts.
We have seen that (inline) frames and hyperlinks can be adjusted with sandboxing flags. Our own resources should only be deployed with CORS in mind.
The Best Small Business Web Designs by DesignRush
/Create Modern Vue Apps Using Create-Vue and Vite
/How to Fix the “There Has Been a Critical Error in Your Website” Error in WordPress
How To Fix The “There Has Been A Critical Error in Your Website” Error in WordPress
/How Long Does It Take to Learn JavaScript?
/The Best Way to Deep Copy an Object in JavaScript
/Adding and Removing Elements From Arrays in JavaScript
/Create a JavaScript AJAX Post Request: With and Without jQuery
/5 Real-Life Uses for the JavaScript reduce() Method
/How to Enable or Disable a Button With JavaScript: jQuery vs. Vanilla
/How to Enable or Disable a Button With JavaScript: jQuery vs Vanilla
/Confirm Yes or No With JavaScript
/How to Change the URL in JavaScript: Redirecting
/15+ Best WordPress Twitter Widgets
/27 Best Tab and Accordion Widget Plugins for WordPress (Free & Premium)
/21 Best Tab and Accordion Widget Plugins for WordPress (Free & Premium)
/30 HTML Best Practices for Beginners
/31 Best WordPress Calendar Plugins and Widgets (With 5 Free Plugins)
/25 Ridiculously Impressive HTML5 Canvas Experiments
/How to Implement Email Verification for New Members
/How to Create a Simple Web-Based Chat Application
/30 Popular WordPress User Interface Elements
/Top 18 Best Practices for Writing Super Readable Code
/Best Affiliate WooCommerce Plugins Compared
/18 Best WordPress Star Rating Plugins
/10+ Best WordPress Twitter Widgets
/20+ Best WordPress Booking and Reservation Plugins
/Working With Tables in React: Part Two
/Best CSS Animations and Effects on CodeCanyon
/30 CSS Best Practices for Beginners
/How to Create a Custom WordPress Plugin From Scratch
/10 Best Responsive HTML5 Sliders for Images and Text… and 3 Free Options
/16 Best Tab and Accordion Widget Plugins for WordPress
/18 Best WordPress Membership Plugins and 5 Free Plugins
/25 Best WooCommerce Plugins for Products, Pricing, Payments and More
/10 Best WordPress Twitter Widgets
1 /12 Best Contact Form PHP Scripts for 2020
/20 Popular WordPress User Interface Elements
/10 Best WordPress Star Rating Plugins
/12 Best CSS Animations on CodeCanyon
/12 Best WordPress Booking and Reservation Plugins
/12 Elegant CSS Pricing Tables for Your Latest Web Project
/24 Best WordPress Form Plugins for 2020
/14 Best PHP Event Calendar and Booking Scripts
/Create a Blog for Each Category or Department in Your WooCommerce Store
/8 Best WordPress Booking and Reservation Plugins
/Best Exit Popups for WordPress Compared
/Best Exit Popups for WordPress Compared
/11 Best Tab & Accordion WordPress Widgets & Plugins
/12 Best Tab & Accordion WordPress Widgets & Plugins
1New Course: Practical React Fundamentals
/Preview Our New Course on Angular Material
/Build Your Own CAPTCHA and Contact Form in PHP
/Object-Oriented PHP With Classes and Objects
/Best Practices for ARIA Implementation
/Accessible Apps: Barriers to Access and Getting Started With Accessibility
/Dramatically Speed Up Your React Front-End App Using Lazy Loading
/15 Best Modern JavaScript Admin Templates for React, Angular, and Vue.js
/15 Best Modern JavaScript Admin Templates for React, Angular and Vue.js
/19 Best JavaScript Admin Templates for React, Angular, and Vue.js
/New Course: Build an App With JavaScript and the MEAN Stack
/Hands-on With ARIA: Accessibility Recipes for Web Apps
/10 Best WordPress Facebook Widgets
13 /Hands-on With ARIA: Accessibility for eCommerce
/New eBooks Available for Subscribers
/Hands-on With ARIA: Homepage Elements and Standard Navigation
/Site Accessibility: Getting Started With ARIA
/How Secure Are Your JavaScript Open-Source Dependencies?
/New Course: Secure Your WordPress Site With SSL
/Testing Components in React Using Jest and Enzyme
/Testing Components in React Using Jest: The Basics
/15 Best PHP Event Calendar and Booking Scripts
/Create Interactive Gradient Animations Using Granim.js
/How to Build Complex, Large-Scale Vue.js Apps With Vuex
1 /Examples of Dependency Injection in PHP With Symfony Components
/Set Up Routing in PHP Applications Using the Symfony Routing Component
1 /A Beginner’s Guide to Regular Expressions in JavaScript
/Introduction to Popmotion: Custom Animation Scrubber
/Introduction to Popmotion: Pointers and Physics
/New Course: Connect to a Database With Laravel’s Eloquent ORM
/How to Create a Custom Settings Panel in WooCommerce
/Building the DOM faster: speculative parsing, async, defer and preload
1 /20 Useful PHP Scripts Available on CodeCanyon
3 /How to Find and Fix Poor Page Load Times With Raygun
/Introduction to the Stimulus Framework
/Single-Page React Applications With the React-Router and React-Transition-Group Modules
12 Best Contact Form PHP Scripts
1 /Getting Started With the Mojs Animation Library: The ShapeSwirl and Stagger Modules
/Getting Started With the Mojs Animation Library: The Shape Module
/Getting Started With the Mojs Animation Library: The HTML Module
/Project Management Considerations for Your WordPress Project
/8 Things That Make Jest the Best React Testing Framework
/Creating an Image Editor Using CamanJS: Layers, Blend Modes, and Events
/New Short Course: Code a Front-End App With GraphQL and React
/Creating an Image Editor Using CamanJS: Applying Basic Filters
/Creating an Image Editor Using CamanJS: Creating Custom Filters and Blend Modes
/Modern Web Scraping With BeautifulSoup and Selenium
/Challenge: Create a To-Do List in React
1Deploy PHP Web Applications Using Laravel Forge
/Getting Started With the Mojs Animation Library: The Burst Module
/10 Things Men Can Do to Support Women in Tech
/A Gentle Introduction to Higher-Order Components in React: Best Practices
/Challenge: Build a React Component
/A Gentle Introduction to HOC in React: Learn by Example
/A Gentle Introduction to Higher-Order Components in React
/Creating Pretty Popup Messages Using SweetAlert2
/Creating Stylish and Responsive Progress Bars Using ProgressBar.js
/18 Best Contact Form PHP Scripts for 2022
/How to Make a Real-Time Sports Application Using Node.js
/Creating a Blogging App Using Angular & MongoDB: Delete Post
/Set Up an OAuth2 Server Using Passport in Laravel
/Creating a Blogging App Using Angular & MongoDB: Edit Post
/Creating a Blogging App Using Angular & MongoDB: Add Post
/Introduction to Mocking in Python
/Creating a Blogging App Using Angular & MongoDB: Show Post
/Creating a Blogging App Using Angular & MongoDB: Home
/Creating a Blogging App Using Angular & MongoDB: Login
/Creating Your First Angular App: Implement Routing
/Persisted WordPress Admin Notices: Part 4
/Creating Your First Angular App: Components, Part 2
/Persisted WordPress Admin Notices: Part 3
/Creating Your First Angular App: Components, Part 1
/How Laravel Broadcasting Works
/Persisted WordPress Admin Notices: Part 2
/Create Your First Angular App: Storing and Accessing Data
/Persisted WordPress Admin Notices: Part 1
/Error and Performance Monitoring for Web & Mobile Apps Using Raygun
/Using Luxon for Date and Time in JavaScript
7 /How to Create an Audio Oscillator With the Web Audio API
/How to Cache Using Redis in Django Applications
/20 Essential WordPress Utilities to Manage Your Site
/Introduction to API Calls With React and Axios
/Beginner’s Guide to Angular 4: HTTP
/Rapid Web Deployment for Laravel With GitHub, Linode, and RunCloud.io
/Beginners Guide to Angular 4: Routing
/Beginner’s Guide to Angular 4: Services
/Beginner’s Guide to Angular 4: Components
/Creating a Drop-Down Menu for Mobile Pages
/Introduction to Forms in Angular 4: Writing Custom Form Validators
/10 Best WordPress Booking & Reservation Plugins
/Getting Started With Redux: Connecting Redux With React
/Getting Started With Redux: Learn by Example
/Getting Started With Redux: Why Redux?
/How to Auto Update WordPress Salts
/How to Download Files in Python
/Eloquent Mutators and Accessors in Laravel
1 /10 Best HTML5 Sliders for Images and Text
/Site Authentication in Node.js: User Signup
/Creating a Task Manager App Using Ionic: Part 2
/Creating a Task Manager App Using Ionic: Part 1
/Introduction to Forms in Angular 4: Reactive Forms
/Introduction to Forms in Angular 4: Template-Driven Forms
/24 Essential WordPress Utilities to Manage Your Site
/25 Essential WordPress Utilities to Manage Your Site
/Get Rid of Bugs Quickly Using BugReplay
1 /Manipulating HTML5 Canvas Using Konva: Part 1, Getting Started
/10 Must-See Easy Digital Downloads Extensions for Your WordPress Site
/22 Best WordPress Booking and Reservation Plugins
/Understanding ExpressJS Routing
/15 Best WordPress Star Rating Plugins
/Creating Your First Angular App: Basics
/Inheritance and Extending Objects With JavaScript
/Introduction to the CSS Grid Layout With Examples
1Performant Animations Using KUTE.js: Part 5, Easing Functions and Attributes
Performant Animations Using KUTE.js: Part 4, Animating Text
/Performant Animations Using KUTE.js: Part 3, Animating SVG
/New Course: Code a Quiz App With Vue.js
/Performant Animations Using KUTE.js: Part 2, Animating CSS Properties
Performant Animations Using KUTE.js: Part 1, Getting Started
/10 Best Responsive HTML5 Sliders for Images and Text (Plus 3 Free Options)
/Single-Page Applications With ngRoute and ngAnimate in AngularJS
/Deferring Tasks in Laravel Using Queues
/Site Authentication in Node.js: User Signup and Login
/Working With Tables in React, Part Two
/Working With Tables in React, Part One
/How to Set Up a Scalable, E-Commerce-Ready WordPress Site Using ClusterCS
/New Course on WordPress Conditional Tags
/TypeScript for Beginners, Part 5: Generics
/Building With Vue.js 2 and Firebase
6 /Best Unique Bootstrap JavaScript Plugins
/Essential JavaScript Libraries and Frameworks You Should Know About
/Vue.js Crash Course: Create a Simple Blog Using Vue.js
/Build a React App With a Laravel RESTful Back End: Part 1, Laravel 5.5 API
/API Authentication With Node.js
/Beginner’s Guide to Angular: HTTP
/Beginner’s Guide to Angular: Routing
/Beginners Guide to Angular: Routing
/Beginner’s Guide to Angular: Services
/Beginner’s Guide to Angular: Components
/How to Create a Custom Authentication Guard in Laravel
/Learn Computer Science With JavaScript: Part 3, Loops
/Build Web Applications Using Node.js
/Learn Computer Science With JavaScript: Part 4, Functions
/Learn Computer Science With JavaScript: Part 2, Conditionals
/Create Interactive Charts Using Plotly.js, Part 5: Pie and Gauge Charts
/Create Interactive Charts Using Plotly.js, Part 4: Bubble and Dot Charts
Create Interactive Charts Using Plotly.js, Part 3: Bar Charts
/Awesome JavaScript Libraries and Frameworks You Should Know About
/Create Interactive Charts Using Plotly.js, Part 2: Line Charts
/Bulk Import a CSV File Into MongoDB Using Mongoose With Node.js
/Build a To-Do API With Node, Express, and MongoDB
/Getting Started With End-to-End Testing in Angular Using Protractor
/TypeScript for Beginners, Part 4: Classes
/Object-Oriented Programming With JavaScript
/10 Best Affiliate WooCommerce Plugins Compared
/Stateful vs. Stateless Functional Components in React
/Make Your JavaScript Code Robust With Flow
/Build a To-Do API With Node and Restify
/Testing Components in Angular Using Jasmine: Part 2, Services
/Testing Components in Angular Using Jasmine: Part 1
/Creating a Blogging App Using React, Part 6: Tags
/React Crash Course for Beginners, Part 3
/React Crash Course for Beginners, Part 2
/React Crash Course for Beginners, Part 1
/Set Up a React Environment, Part 4
1 /Set Up a React Environment, Part 3
/New Course: Get Started With Phoenix
/Set Up a React Environment, Part 2
/Set Up a React Environment, Part 1
/Command Line Basics and Useful Tricks With the Terminal
/How to Create a Real-Time Feed Using Phoenix and React
/Build a React App With a Laravel Back End: Part 2, React
/Build a React App With a Laravel RESTful Back End: Part 1, Laravel 9 API
/Creating a Blogging App Using React, Part 5: Profile Page
/Pagination in CodeIgniter: The Complete Guide
/JavaScript-Based Animations Using Anime.js, Part 4: Callbacks, Easings, and SVG
/JavaScript-Based Animations Using Anime.js, Part 3: Values, Timeline, and Playback
/Learn to Code With JavaScript: Part 1, The Basics
/10 Elegant CSS Pricing Tables for Your Latest Web Project
/Getting Started With the Flux Architecture in React
/Getting Started With Matter.js: The Composites and Composite Modules
Getting Started With Matter.js: The Engine and World Modules
/10 More Popular HTML5 Projects for You to Use and Study
/Understand the Basics of Laravel Middleware
/Iterating Fast With Django & Heroku
/Creating a Blogging App Using React, Part 4: Update & Delete Posts
/Creating a jQuery Plugin for Long Shadow Design
/How to Register & Use Laravel Service Providers
2 /Unit Testing in React: Shallow vs. Static Testing
/Creating a Blogging App Using React, Part 3: Add & Display Post
/Creating a Blogging App Using React, Part 2: User Sign-Up
20 /Creating a Blogging App Using React, Part 1: User Sign-In
/Creating a Grocery List Manager Using Angular, Part 2: Managing Items
/9 Elegant CSS Pricing Tables for Your Latest Web Project
/Dynamic Page Templates in WordPress, Part 3
/Angular vs. React: 7 Key Features Compared
/Creating a Grocery List Manager Using Angular, Part 1: Add & Display Items
New eBooks Available for Subscribers in June 2017
/Create Interactive Charts Using Plotly.js, Part 1: Getting Started
/The 5 Best IDEs for WordPress Development (And Why)
/33 Popular WordPress User Interface Elements
/New Course: How to Hack Your Own App
/How to Install Yii on Windows or a Mac
/What Is a JavaScript Operator?
/How to Register and Use Laravel Service Providers
/
waly Good blog post. I absolutely love this…