implement content-security-policy, static assets cache bust, and fix
stripe back button ratchet issue because the only way to use stripe checkout is to run their proprietary JS, and we arent using a SPA, naturally what happens is, when you land on the stripe payment page if you hit the back button it goes back to the same page where you got re-directed to stripe. this commit fixes that.
This commit is contained in:
		@ -2,12 +2,15 @@ import logging
 | 
			
		||||
from logging.config import dictConfig as logging_dict_config
 | 
			
		||||
 | 
			
		||||
import os
 | 
			
		||||
import hashlib
 | 
			
		||||
 | 
			
		||||
import stripe
 | 
			
		||||
from dotenv import load_dotenv, find_dotenv
 | 
			
		||||
from flask import Flask
 | 
			
		||||
from flask_mail import Mail
 | 
			
		||||
from flask import render_template
 | 
			
		||||
from flask import url_for
 | 
			
		||||
from flask import current_app
 | 
			
		||||
 | 
			
		||||
from capsulflask import virt_model, cli
 | 
			
		||||
from capsulflask.btcpay import client as btcpay
 | 
			
		||||
@ -45,19 +48,19 @@ app.config.from_mapping(
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
logging_dict_config({
 | 
			
		||||
    'version': 1,
 | 
			
		||||
    'formatters': {'default': {
 | 
			
		||||
        'format': '[%(asctime)s] %(levelname)s in %(module)s: %(message)s',
 | 
			
		||||
    }},
 | 
			
		||||
    'handlers': {'wsgi': {
 | 
			
		||||
        'class': 'logging.StreamHandler',
 | 
			
		||||
        'stream': 'ext://flask.logging.wsgi_errors_stream',
 | 
			
		||||
        'formatter': 'default'
 | 
			
		||||
    }},
 | 
			
		||||
    'root': {
 | 
			
		||||
        'level': app.config['LOG_LEVEL'],
 | 
			
		||||
        'handlers': ['wsgi']
 | 
			
		||||
    }
 | 
			
		||||
  'version': 1,
 | 
			
		||||
  'formatters': {'default': {
 | 
			
		||||
    'format': '[%(asctime)s] %(levelname)s in %(module)s: %(message)s',
 | 
			
		||||
  }},
 | 
			
		||||
  'handlers': {'wsgi': {
 | 
			
		||||
    'class': 'logging.StreamHandler',
 | 
			
		||||
    'stream': 'ext://flask.logging.wsgi_errors_stream',
 | 
			
		||||
    'formatter': 'default'
 | 
			
		||||
  }},
 | 
			
		||||
  'root': {
 | 
			
		||||
    'level': app.config['LOG_LEVEL'],
 | 
			
		||||
    'handlers': ['wsgi']
 | 
			
		||||
  }
 | 
			
		||||
})
 | 
			
		||||
 | 
			
		||||
# app.logger.critical("critical")
 | 
			
		||||
@ -72,9 +75,9 @@ stripe.api_version = app.config['STRIPE_API_VERSION']
 | 
			
		||||
app.config['FLASK_MAIL_INSTANCE'] = Mail(app)
 | 
			
		||||
 | 
			
		||||
if app.config['VIRTUALIZATION_MODEL'] == "shell_scripts":
 | 
			
		||||
    app.config['VIRTUALIZATION_MODEL'] = virt_model.ShellScriptVirtualization()
 | 
			
		||||
  app.config['VIRTUALIZATION_MODEL'] = virt_model.ShellScriptVirtualization()
 | 
			
		||||
else:
 | 
			
		||||
    app.config['VIRTUALIZATION_MODEL'] = virt_model.MockVirtualization()
 | 
			
		||||
  app.config['VIRTUALIZATION_MODEL'] = virt_model.MockVirtualization()
 | 
			
		||||
 | 
			
		||||
app.config['BTCPAY_CLIENT'] = btcpay.Client(api_uri=app.config['BTCPAY_URL'], pem=app.config['BTCPAY_PRIVATE_KEY'])
 | 
			
		||||
 | 
			
		||||
@ -93,3 +96,49 @@ app.register_blueprint(cli.bp)
 | 
			
		||||
 | 
			
		||||
app.add_url_rule("/", endpoint="index")
 | 
			
		||||
 | 
			
		||||
@app.after_request
 | 
			
		||||
def security_headers(response):
 | 
			
		||||
  response.headers['X-Frame-Options'] = 'SAMEORIGIN'
 | 
			
		||||
  if 'Content-Security-Policy' not in response.headers:
 | 
			
		||||
    response.headers['Content-Security-Policy'] = "default-src 'self'"
 | 
			
		||||
  response.headers['X-Content-Type-Options'] = 'nosniff'
 | 
			
		||||
  return response
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@app.context_processor
 | 
			
		||||
def override_url_for():
 | 
			
		||||
  """
 | 
			
		||||
  override the url_for function built into flask 
 | 
			
		||||
  with our own custom implementation that busts the cache correctly when files change 
 | 
			
		||||
  """
 | 
			
		||||
  return dict(url_for=url_for_with_cache_bust)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def url_for_with_cache_bust(endpoint, **values):
 | 
			
		||||
  """
 | 
			
		||||
  Add a query parameter based on the hash of the file, this acts as a cache bust
 | 
			
		||||
  """
 | 
			
		||||
  
 | 
			
		||||
  if endpoint == 'static':
 | 
			
		||||
    filename = values.get('filename', None)
 | 
			
		||||
    if filename:
 | 
			
		||||
      if 'STATIC_FILE_HASH_CACHE' not in current_app.config:
 | 
			
		||||
        current_app.config['STATIC_FILE_HASH_CACHE'] = dict()
 | 
			
		||||
      
 | 
			
		||||
      if filename not in current_app.config['STATIC_FILE_HASH_CACHE']:
 | 
			
		||||
        filepath = os.path.join(current_app.root_path, endpoint, filename)
 | 
			
		||||
        #print(filepath)
 | 
			
		||||
        if os.path.isfile(filepath) and os.access(filepath, os.R_OK):
 | 
			
		||||
          
 | 
			
		||||
          with open(filepath, 'rb') as file:
 | 
			
		||||
            hasher = hashlib.md5()
 | 
			
		||||
            hasher.update(file.read())
 | 
			
		||||
            current_app.config['STATIC_FILE_HASH_CACHE'][filename] = hasher.hexdigest()[-6:]
 | 
			
		||||
            
 | 
			
		||||
      values['q'] = current_app.config['STATIC_FILE_HASH_CACHE'][filename]
 | 
			
		||||
 | 
			
		||||
  return url_for(endpoint, **values)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
  
 | 
			
		||||
 | 
			
		||||
@ -217,12 +217,13 @@ def ssh_public_keys():
 | 
			
		||||
      if method == "POST":
 | 
			
		||||
        parts = re.split(" +", request.form["content"])
 | 
			
		||||
        if len(parts) > 2 and len(parts[2].strip()) > 0:
 | 
			
		||||
          name = parts[2]
 | 
			
		||||
          name = parts[2].strip()
 | 
			
		||||
        else:
 | 
			
		||||
          name = parts[0]
 | 
			
		||||
          name = parts[0].strip()
 | 
			
		||||
      else:
 | 
			
		||||
        errors.append("Name is required")
 | 
			
		||||
    if not re.match(r"^[0-9A-Za-z_@\. -]+$", name):
 | 
			
		||||
    if not re.match(r"^[0-9A-Za-z_@. -]+$", name):
 | 
			
		||||
      print(name)
 | 
			
		||||
      errors.append("Name must match \"^[0-9A-Za-z_@. -]+$\"")
 | 
			
		||||
 | 
			
		||||
    if method == "POST":
 | 
			
		||||
@ -231,7 +232,7 @@ def ssh_public_keys():
 | 
			
		||||
        errors.append("Content is required")
 | 
			
		||||
      else:
 | 
			
		||||
        content = content.replace("\r", "").replace("\n", "")
 | 
			
		||||
        if not re.match(r"^(ssh|ecdsa)-[0-9A-Za-z+/_=@\. -]+$", content):
 | 
			
		||||
        if not re.match(r"^(ssh|ecdsa)-[0-9A-Za-z+/_=@. -]+$", content):
 | 
			
		||||
          errors.append("Content must match \"^(ssh|ecdsa)-[0-9A-Za-z+/_=@. -]+$\"")
 | 
			
		||||
 | 
			
		||||
      if get_model().ssh_public_key_name_exists(session["account"], name):
 | 
			
		||||
 | 
			
		||||
@ -40,7 +40,7 @@ def init_app(app):
 | 
			
		||||
  hasSchemaVersionTable = False
 | 
			
		||||
  actionWasTaken = False
 | 
			
		||||
  schemaVersion = 0
 | 
			
		||||
  desiredSchemaVersion = 4
 | 
			
		||||
  desiredSchemaVersion = 5
 | 
			
		||||
 | 
			
		||||
  cursor = connection.cursor()
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -175,6 +175,21 @@ class DBModel:
 | 
			
		||||
      self.cursor.fetchall()
 | 
			
		||||
    ))
 | 
			
		||||
 | 
			
		||||
  def payment_session_redirect(self, email, id):
 | 
			
		||||
    self.cursor.execute("SELECT redirected FROM payment_sessions WHERE email = %s AND id = %s", 
 | 
			
		||||
      (email, id)
 | 
			
		||||
    )
 | 
			
		||||
    row = self.cursor.fetchone()
 | 
			
		||||
    if row:
 | 
			
		||||
      self.cursor.execute("UPDATE payment_sessions SET redirected = TRUE WHERE email = %s AND id = %s", 
 | 
			
		||||
        (email, id)
 | 
			
		||||
      )
 | 
			
		||||
      self.connection.commit()
 | 
			
		||||
      return row[0]
 | 
			
		||||
    
 | 
			
		||||
    return None
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
  def consume_payment_session(self, payment_type, id, dollars):
 | 
			
		||||
    self.cursor.execute("SELECT email, dollars FROM payment_sessions WHERE id = %s AND type = %s", (id, payment_type))
 | 
			
		||||
    row = self.cursor.fetchone()
 | 
			
		||||
 | 
			
		||||
@ -2,9 +2,11 @@ import stripe
 | 
			
		||||
import json
 | 
			
		||||
import time
 | 
			
		||||
import decimal
 | 
			
		||||
from time import sleep
 | 
			
		||||
import re
 | 
			
		||||
 | 
			
		||||
from time import sleep
 | 
			
		||||
from flask import Blueprint
 | 
			
		||||
from flask import make_response
 | 
			
		||||
from flask import request
 | 
			
		||||
from flask import current_app
 | 
			
		||||
from flask import session
 | 
			
		||||
@ -165,14 +167,46 @@ def stripe_payment():
 | 
			
		||||
 | 
			
		||||
      #return redirect(f"https://checkout.stripe.com/pay/{stripe_checkout_session_id}")
 | 
			
		||||
 | 
			
		||||
      return redirect(f"/payment/stripe/{stripe_checkout_session_id}")
 | 
			
		||||
 | 
			
		||||
  for error in errors:
 | 
			
		||||
    flash(error)
 | 
			
		||||
 | 
			
		||||
  return render_template(
 | 
			
		||||
    "stripe.html", 
 | 
			
		||||
  return render_template("stripe.html")
 | 
			
		||||
 | 
			
		||||
@bp.route("/stripe/<string:stripe_checkout_session_id>")
 | 
			
		||||
@account_required
 | 
			
		||||
def redirect_to_stripe(stripe_checkout_session_id):
 | 
			
		||||
 | 
			
		||||
  if stripe_checkout_session_id and not re.match(r"^[a-zA-Z0-9_=-]+$", stripe_checkout_session_id):
 | 
			
		||||
    stripe_checkout_session_id = '___________'
 | 
			
		||||
 | 
			
		||||
  response = make_response(render_template(
 | 
			
		||||
    "stripe.html",
 | 
			
		||||
    stripe_checkout_session_id=stripe_checkout_session_id,
 | 
			
		||||
    stripe_public_key=current_app.config["STRIPE_PUBLISHABLE_KEY"]
 | 
			
		||||
  )
 | 
			
		||||
  ))
 | 
			
		||||
 | 
			
		||||
  if stripe_checkout_session_id is not None:
 | 
			
		||||
    response.headers['Content-Security-Policy'] = "default-src 'self' https://js.stripe.com"
 | 
			
		||||
  
 | 
			
		||||
  return response
 | 
			
		||||
 | 
			
		||||
@bp.route("/stripe/<string:stripe_checkout_session_id>/json")
 | 
			
		||||
@account_required
 | 
			
		||||
def stripe_checkout_session_json(stripe_checkout_session_id):
 | 
			
		||||
 | 
			
		||||
  if stripe_checkout_session_id and not re.match(r"^[a-zA-Z0-9_=-]+$", stripe_checkout_session_id):
 | 
			
		||||
    stripe_checkout_session_id = '___________'
 | 
			
		||||
 | 
			
		||||
  has_redirected_already = get_model().payment_session_redirect(session['account'], stripe_checkout_session_id)
 | 
			
		||||
 | 
			
		||||
  if has_redirected_already is None:
 | 
			
		||||
    abort(404, "Not Found")
 | 
			
		||||
 | 
			
		||||
  return jsonify(dict(hasRedirectedAlready=has_redirected_already))
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
def validate_stripe_checkout_session(stripe_checkout_session_id):
 | 
			
		||||
  checkout_session_completed_events = stripe.Event.list(
 | 
			
		||||
 | 
			
		||||
@ -0,0 +1,3 @@
 | 
			
		||||
ALTER TABLE payment_sessions DROP COLUMN redirected;
 | 
			
		||||
 | 
			
		||||
UPDATE schemaversion SET version = 4;
 | 
			
		||||
@ -0,0 +1,4 @@
 | 
			
		||||
ALTER TABLE payment_sessions 
 | 
			
		||||
ADD COLUMN redirected BOOLEAN NOT NULL DEFAULT FALSE;
 | 
			
		||||
 | 
			
		||||
UPDATE schemaversion SET version = 5;
 | 
			
		||||
							
								
								
									
										9
									
								
								capsulflask/static/create-capsul.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								capsulflask/static/create-capsul.js
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,9 @@
 | 
			
		||||
 | 
			
		||||
window.addEventListener('DOMContentLoaded', function(event) {
 | 
			
		||||
  var submitButton = document.getElementById('submit-button');
 | 
			
		||||
  var submitButtonClicked = document.getElementById('submit-button-clicked');
 | 
			
		||||
  document.getElementById('submit-button').onclick = function() {
 | 
			
		||||
    submitButton.className = "display-none";
 | 
			
		||||
    submitButtonClicked.className = "waiting-pulse";
 | 
			
		||||
  }
 | 
			
		||||
});
 | 
			
		||||
							
								
								
									
										39
									
								
								capsulflask/static/pay-with-stripe.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								capsulflask/static/pay-with-stripe.js
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,39 @@
 | 
			
		||||
 | 
			
		||||
window.addEventListener('DOMContentLoaded', function(event) {
 | 
			
		||||
 | 
			
		||||
  var httpRequest = new XMLHttpRequest();
 | 
			
		||||
  httpRequest.onloadend = () => {
 | 
			
		||||
    if (httpRequest.status < 300) {
 | 
			
		||||
      try {
 | 
			
		||||
        responseObject = JSON.parse(httpRequest.responseText);
 | 
			
		||||
 | 
			
		||||
        if(!responseObject.hasRedirectedAlready) {
 | 
			
		||||
          Stripe(document.getElementById("stripe_public_key").value)
 | 
			
		||||
          .redirectToCheckout({
 | 
			
		||||
            sessionId: document.getElementById("stripe_checkout_session_id").value,
 | 
			
		||||
          })
 | 
			
		||||
          .then(function(result) {
 | 
			
		||||
            if (result.error) {
 | 
			
		||||
              alert("Stripe.redirectToCheckout() failed with: " + result.error.message)
 | 
			
		||||
            }
 | 
			
		||||
          });
 | 
			
		||||
        } else {
 | 
			
		||||
          location.href = '/payment/stripe';
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
      } catch (err) {
 | 
			
		||||
        alert("could not redirect to stripe because capsul did not return valid json");
 | 
			
		||||
      }
 | 
			
		||||
    } else {
 | 
			
		||||
      alert("could not redirect to stripe because capsul returned HTTP" + httpRequest.status + ", expected HTTP 200");
 | 
			
		||||
    }
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  httpRequest.ontimeout = () => {
 | 
			
		||||
    alert("could not redirect to stripe because capsul timed out");
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  httpRequest.open("GET", "/payment/stripe/"+document.getElementById("stripe_checkout_session_id").value+"/json?q="+String(Math.random()).substring(2, 8));
 | 
			
		||||
  httpRequest.timeout = 10000;
 | 
			
		||||
  httpRequest.send();
 | 
			
		||||
});
 | 
			
		||||
@ -150,11 +150,7 @@ select {
 | 
			
		||||
  -webkit-appearance: none;
 | 
			
		||||
  -moz-appearance: none;
 | 
			
		||||
  appearance: none;
 | 
			
		||||
  /* 
 | 
			
		||||
    re-generate the following line from the source image with: 
 | 
			
		||||
    echo "background-image: url(data:image/png;base64,$(cat capsulflask/static/dropdown-handle.png | base64 -w 0));" 
 | 
			
		||||
  */
 | 
			
		||||
  background-image: url(); 
 | 
			
		||||
  background-image: url(/static/dropdown-handle.png); 
 | 
			
		||||
  background-repeat: no-repeat;
 | 
			
		||||
  background-position: bottom 0.65em right 0.8em;
 | 
			
		||||
  background-size: 0.5em;
 | 
			
		||||
 | 
			
		||||
@ -1,6 +1,5 @@
 | 
			
		||||
<html lang="en">
 | 
			
		||||
<head>
 | 
			
		||||
  <link href="" rel="icon">
 | 
			
		||||
  <title>{% block title %}{% endblock %}{% if self.title() %} - {% endif %}Capsul</title>
 | 
			
		||||
  <meta charset="utf-8">
 | 
			
		||||
  <meta name="viewport" content="width=device-width,initial-scale=1.0">
 | 
			
		||||
 | 
			
		||||
@ -67,16 +67,7 @@
 | 
			
		||||
        <input id="submit-button" type="submit" value="Create">
 | 
			
		||||
        <span  id="submit-button-clicked" class="display-none">..Creating...</span>
 | 
			
		||||
      </div>
 | 
			
		||||
      <script>
 | 
			
		||||
        window.addEventListener('DOMContentLoaded', function(event) {
 | 
			
		||||
            var submitButton = document.getElementById('submit-button');
 | 
			
		||||
            var submitButtonClicked = document.getElementById('submit-button-clicked');
 | 
			
		||||
            document.getElementById('submit-button').onclick = function() {
 | 
			
		||||
              submitButton.className = "display-none";
 | 
			
		||||
              submitButtonClicked.className = "waiting-pulse";
 | 
			
		||||
            }
 | 
			
		||||
        });
 | 
			
		||||
      </script>
 | 
			
		||||
      <script src="{{ url_for('static', filename='create-capsul.js') }}"></script>
 | 
			
		||||
    </form>
 | 
			
		||||
  {% endif %}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -10,33 +10,30 @@
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
{% block content %}
 | 
			
		||||
<div class="row third-margin">
 | 
			
		||||
  <h1>PAY WITH STRIPE</h1>
 | 
			
		||||
</div>
 | 
			
		||||
<div class="row half-margin">
 | 
			
		||||
  <form method="post">
 | 
			
		||||
    <div class="row justify-start">
 | 
			
		||||
      <label for="dollars">$</label>
 | 
			
		||||
      <input type="number" id="dollars" name="dollars"></input> 
 | 
			
		||||
    </div>
 | 
			
		||||
    <div class="row justify-end">
 | 
			
		||||
      <input type="submit" value="Pay With Stripe">
 | 
			
		||||
    </div>
 | 
			
		||||
  </form>
 | 
			
		||||
</div>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
{% if stripe_checkout_session_id %}
 | 
			
		||||
  <script>
 | 
			
		||||
    Stripe("{{ stripe_public_key }}")
 | 
			
		||||
    .redirectToCheckout({
 | 
			
		||||
      sessionId: "{{ stripe_checkout_session_id }}",
 | 
			
		||||
    })
 | 
			
		||||
    .then(function(result) {
 | 
			
		||||
      if (result.error) {
 | 
			
		||||
        alert("Stripe.redirectToCheckout() failed with: " + result.error.message)
 | 
			
		||||
      }
 | 
			
		||||
    });
 | 
			
		||||
  </script>
 | 
			
		||||
  <div class="row third-margin">
 | 
			
		||||
    <h1>REDIRECTING...</h1>
 | 
			
		||||
  </div>
 | 
			
		||||
  <input id="stripe_public_key" type="hidden" value="{{ stripe_public_key }}"/>
 | 
			
		||||
  <input id="stripe_checkout_session_id" type="hidden" value="{{ stripe_checkout_session_id }}"/>
 | 
			
		||||
  <script src="{{ url_for('static', filename='pay-with-stripe.js') }}"></script>
 | 
			
		||||
{% else %}
 | 
			
		||||
  <div class="row third-margin">
 | 
			
		||||
    <h1>PAY WITH STRIPE</h1>
 | 
			
		||||
  </div>
 | 
			
		||||
  <div class="row half-margin">
 | 
			
		||||
    <form method="post">
 | 
			
		||||
      <div class="row justify-start">
 | 
			
		||||
        <label for="dollars">$</label>
 | 
			
		||||
        <input type="number" id="dollars" name="dollars"></input> 
 | 
			
		||||
      </div>
 | 
			
		||||
      <div class="row justify-end">
 | 
			
		||||
        <input type="submit" value="Pay With Stripe">
 | 
			
		||||
      </div>
 | 
			
		||||
    </form>
 | 
			
		||||
  </div>
 | 
			
		||||
{% endif %}
 | 
			
		||||
 | 
			
		||||
{% endblock %}
 | 
			
		||||
 | 
			
		||||
		Reference in New Issue
	
	Block a user