Initial commit
This commit is contained in:
commit
c32c8cbee4
5
stickynotes.cfg
Normal file
5
stickynotes.cfg
Normal file
@ -0,0 +1,5 @@
|
||||
[stickynotes]
|
||||
database_url = sqlite:///stickynotes.sqlite
|
||||
secret_key = yoursecretkeyhere
|
||||
recaptcha_site_key =
|
||||
recaptcha_secret_key =
|
40
stickynotes/__init__.py
Normal file
40
stickynotes/__init__.py
Normal file
@ -0,0 +1,40 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
"""
|
||||
StickyNotes, yet another paste bin
|
||||
"""
|
||||
import os
|
||||
from ConfigParser import SafeConfigParser
|
||||
|
||||
from flask import Flask
|
||||
from flask.ext.mako import MakoTemplates
|
||||
|
||||
from models import init_db
|
||||
from views import views
|
||||
|
||||
|
||||
def read_config():
|
||||
"""
|
||||
Read the configuration file and return the values in a dictionary
|
||||
"""
|
||||
config_file = os.path.abspath(os.path.join(os.path.dirname(__file__), u'..', u'stickynotes.cfg'))
|
||||
config_parser = SafeConfigParser()
|
||||
config_parser.read(config_file)
|
||||
config = {}
|
||||
for option in config_parser.options(u'stickynotes'):
|
||||
config[option.upper()] = config_parser.get(u'stickynotes', option)
|
||||
print(config)
|
||||
return config
|
||||
|
||||
|
||||
def make_app():
|
||||
"""
|
||||
Create the application object
|
||||
"""
|
||||
app = Flask(__name__)
|
||||
# Load the config file
|
||||
config = read_config()
|
||||
app.config.update(config)
|
||||
MakoTemplates(app)
|
||||
init_db(config[u'DATABASE_URL'])
|
||||
app.register_blueprint(views)
|
||||
return app
|
47
stickynotes/models.py
Normal file
47
stickynotes/models.py
Normal file
@ -0,0 +1,47 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
"""
|
||||
The models in use
|
||||
"""
|
||||
from datetime import datetime
|
||||
|
||||
from sqlalchemy import Column, Integer, String, Text, DateTime, create_engine, Boolean
|
||||
from sqlalchemy.ext.declarative import declarative_base
|
||||
from sqlalchemy.orm import sessionmaker, scoped_session
|
||||
|
||||
BaseModel = declarative_base()
|
||||
db_session = None
|
||||
|
||||
|
||||
class StickyNote(BaseModel):
|
||||
"""
|
||||
The main (only?) table in the system
|
||||
"""
|
||||
__tablename__ = u'sticky_notes'
|
||||
|
||||
id = Column(Integer, autoincrement=True, primary_key=True)
|
||||
title = Column(String(255))
|
||||
source = Column(Text)
|
||||
lexer = Column(String(255), default=u'text')
|
||||
created = Column(DateTime, default=datetime.now, index=True)
|
||||
expiry = Column(DateTime, default=None, index=True)
|
||||
url = Column(String(255), index=True)
|
||||
private = Column(Boolean, default=False, index=True)
|
||||
|
||||
|
||||
def init_db(database_url):
|
||||
"""
|
||||
Initialise the database connection
|
||||
|
||||
:param database_url: The database connection URL
|
||||
"""
|
||||
global db_session
|
||||
engine = create_engine(database_url, pool_recycle=3600)
|
||||
db_session = scoped_session(sessionmaker(bind=engine))()
|
||||
BaseModel.metadata.create_all(engine, checkfirst=True)
|
||||
|
||||
|
||||
def get_session():
|
||||
"""
|
||||
Get the current database session
|
||||
"""
|
||||
return db_session
|
44
stickynotes/static/custom.css
Normal file
44
stickynotes/static/custom.css
Normal file
@ -0,0 +1,44 @@
|
||||
html {
|
||||
position: relative;
|
||||
min-height: 100%;
|
||||
}
|
||||
|
||||
body {
|
||||
margin-bottom: 50px;
|
||||
}
|
||||
|
||||
body > .container {
|
||||
padding: 70px 15px 0;
|
||||
}
|
||||
|
||||
.container .text-muted {
|
||||
margin: 20px 0;
|
||||
}
|
||||
|
||||
.footer {
|
||||
position: absolute;
|
||||
bottom: 0;
|
||||
width: 100%;
|
||||
height: 50px;
|
||||
}
|
||||
|
||||
.footer > .container {
|
||||
padding-right: 15px;
|
||||
padding-left: 15px;
|
||||
}
|
||||
|
||||
code {
|
||||
font-size: 80%;
|
||||
}
|
||||
|
||||
.sourcetable {
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
.linenos {
|
||||
text-align: right;
|
||||
}
|
||||
|
||||
.note-links {
|
||||
margin-bottom: 10px;
|
||||
}
|
7
stickynotes/templates/about.mako
Normal file
7
stickynotes/templates/about.mako
Normal file
@ -0,0 +1,7 @@
|
||||
<%inherit file="base.mako"/>
|
||||
<div class="row">
|
||||
<div class="col-xs-12 col-sm-12 col-md-12 col-lg-12">
|
||||
<h2>About StickyNotes</h2>
|
||||
<p>StickyNotes is a quick code paste application written in Python with Flask, SQLAlchemy, Mako, Pygments and a few other Python libraries.</p>
|
||||
</div>
|
||||
</div>
|
71
stickynotes/templates/base.mako
Normal file
71
stickynotes/templates/base.mako
Normal file
@ -0,0 +1,71 @@
|
||||
<!DOCTYPE html>
|
||||
<html>
|
||||
<head lang="en">
|
||||
<meta charset="utf-8">
|
||||
<meta http-equiv="X-UA-Compatible" content="IE=edge">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1">
|
||||
<title>StickyNotes</title>
|
||||
<link href="//cdnjs.cloudflare.com/ajax/libs/bootstrap-select/1.9.3/css/bootstrap-select.min.css" rel="stylesheet" type="text/css">
|
||||
<link href="//maxcdn.bootstrapcdn.com/bootswatch/3.3.6/lumen/bootstrap.min.css" rel="stylesheet" type="text/css">
|
||||
<link href="//maxcdn.bootstrapcdn.com/font-awesome/4.5.0/css/font-awesome.min.css" rel="stylesheet">
|
||||
<link href="/static/custom.css" rel="stylesheet" type="text/css">
|
||||
<link href="/pygments.css" rel="stylesheet" type="text/css">
|
||||
</head>
|
||||
<body>
|
||||
<nav class="navbar navbar-default navbar-fixed-top">
|
||||
<div class="container">
|
||||
<div class="navbar-header">
|
||||
<button type="button" class="navbar-toggle collapsed" data-toggle="collapse" data-target="#navbar" aria-expanded="false" aria-controls="navbar">
|
||||
<span class="sr-only">Toggle navigation</span>
|
||||
<span class="icon-bar"></span>
|
||||
<span class="icon-bar"></span>
|
||||
<span class="icon-bar"></span>
|
||||
</button>
|
||||
<a href="/" class="navbar-brand">StickyNotes</a>
|
||||
</div>
|
||||
<div id="navbar" class="collapse navbar-collapse">
|
||||
<ul class="nav navbar-nav">
|
||||
% if self.name == u'self:index.mako':
|
||||
<li class="active"><a href="/">New</a></li>
|
||||
%else:
|
||||
<li><a href="/">New</a></li>
|
||||
% endif
|
||||
<%doc>
|
||||
% if self.name == u'self:list.mako':
|
||||
<li class="active"><a href="/notes">Notes</a></li>
|
||||
%else:
|
||||
<li><a href="/notes">Notes</a></li>
|
||||
% endif
|
||||
</%doc>
|
||||
% if self.name == u'self:about.mako':
|
||||
<li class="active"><a href="/about">About</a></li>
|
||||
%else:
|
||||
<li><a href="/about">About</a></li>
|
||||
% endif
|
||||
</ul>
|
||||
</div>
|
||||
</div>
|
||||
</nav>
|
||||
<div class="container">
|
||||
<% messages = get_flashed_messages(True) %>
|
||||
% if messages:
|
||||
% for category, message in messages:
|
||||
<div class="alert alert-${category}" role="alert">
|
||||
${message}
|
||||
</div>
|
||||
% endfor
|
||||
% endif
|
||||
${self.body()}
|
||||
</div>
|
||||
<footer class="footer">
|
||||
<div class="container">
|
||||
<p class="text-muted">Copyright © 2015 Raoul Snyman.</p>
|
||||
</div>
|
||||
</footer>
|
||||
<script src="//code.jquery.com/jquery-1.11.3.min.js"></script>
|
||||
<script src="//code.jquery.com/jquery-migrate-1.2.1.min.js"></script>
|
||||
<script src="//maxcdn.bootstrapcdn.com/bootstrap/3.3.2/js/bootstrap.min.js" type="application/javascript"></script>
|
||||
<script src="//cdnjs.cloudflare.com/ajax/libs/bootstrap-select/1.9.3/js/bootstrap-select.min.js"></script>
|
||||
<script src="https://www.google.com/recaptcha/api.js" async defer></script>
|
||||
</body>
|
||||
</html>
|
48
stickynotes/templates/index.mako
Normal file
48
stickynotes/templates/index.mako
Normal file
@ -0,0 +1,48 @@
|
||||
<%inherit file="base.mako"/>
|
||||
<div class="row">
|
||||
<div class="col-xs-12 col-sm-12 col-md-12 col-lg-12">
|
||||
<form role="form" action="/save" method="post" form>
|
||||
<div class="form-group">
|
||||
<textarea name="source" id="source" class="form-control" rows="20"></textarea>
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<input type="text" name="title" id="title" class="form-control" placeholder="Title of your snippet">
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label for="language">Language</label>
|
||||
<select name="language" id="language" class="form-control selectpicker" data-live-search="true" data-size="15">
|
||||
% for language in lexers:
|
||||
% if language[0] == 'text':
|
||||
<option value="${language[0]}" selected>${language[1]}</option>
|
||||
% else:
|
||||
<option value="${language[0]}">${language[1]}</option>
|
||||
% endif
|
||||
% endfor
|
||||
</select>
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label for="expiry">Expiry</label>
|
||||
<select name="expiry" id="expiry" class="form-control selectpicker">
|
||||
<option value="never" selected>Never</option>
|
||||
<option value="10min">10 Minutes</option>
|
||||
<option value="1h">1 Hour</option>
|
||||
<option value="1d">1 Day</option>
|
||||
<option value="1w">1 Week</option>
|
||||
<option value="2w">2 Weeks</option>
|
||||
<option value="1m">1 Month</option>
|
||||
</select>
|
||||
</div>
|
||||
<div class="checkbox">
|
||||
<label>
|
||||
<input type="checkbox" name="private"> Unlisted (doesn't appear in the list on the notes page)
|
||||
</label>
|
||||
</div>
|
||||
% if recaptcha_site_key:
|
||||
<div class="form-group">
|
||||
<div class="g-recaptcha" data-sitekey="${recaptcha_site_key}"></div>
|
||||
</div>
|
||||
% endif
|
||||
<button type="submit" class="btn btn-primary">Submit</button>
|
||||
</form>
|
||||
</div>
|
||||
</div>
|
1
stickynotes/templates/raw.mako
Normal file
1
stickynotes/templates/raw.mako
Normal file
@ -0,0 +1 @@
|
||||
${source}
|
9
stickynotes/templates/view.mako
Normal file
9
stickynotes/templates/view.mako
Normal file
@ -0,0 +1,9 @@
|
||||
<%inherit file="base.mako"/>
|
||||
<div class="row">
|
||||
<div class="col-xs-12 col-sm-12 col-md-12 col-lg-12 text-right note-links">
|
||||
<a href="/raw/${note.url}" class="btn btn-default btn-sm"><i class="fa fa-fw fa-file-text"></i> Raw</a>
|
||||
</div>
|
||||
<div class="col-xs-12 col-sm-12 col-md-12 col-lg-12">
|
||||
${source}
|
||||
</div>
|
||||
</div>
|
166
stickynotes/views.py
Normal file
166
stickynotes/views.py
Normal file
@ -0,0 +1,166 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
"""
|
||||
The views
|
||||
"""
|
||||
from datetime import timedelta, datetime
|
||||
import logging
|
||||
|
||||
from flask import Blueprint, redirect, request, flash, make_response, current_app
|
||||
from flask.ext.mako import render_template
|
||||
from pygments import highlight
|
||||
from pygments.formatters.html import HtmlFormatter
|
||||
from pygments.lexers import get_lexer_by_name, get_all_lexers
|
||||
from sqlalchemy import or_
|
||||
import requests
|
||||
import short_url
|
||||
|
||||
from models import StickyNote, get_session
|
||||
|
||||
log = logging.getLogger(__name__)
|
||||
views = Blueprint(u'views', __name__)
|
||||
|
||||
EXPIRY_DELTAS = {
|
||||
u'10min': timedelta(minutes=10),
|
||||
u'1d': timedelta(days=1),
|
||||
u'1w': timedelta(days=7),
|
||||
u'2w': timedelta(days=14),
|
||||
u'1m': timedelta(days=30)
|
||||
}
|
||||
|
||||
|
||||
def _is_recaptcha_valid(secret, response, remote_ip=None):
|
||||
"""
|
||||
POST to the recaptcha service to check if the recaptcha is valid
|
||||
"""
|
||||
data = {u'secret': secret, u'response': response}
|
||||
if remote_ip:
|
||||
data[u'remoteip'] = remote_ip
|
||||
response = requests.post(u'https://www.google.com/recaptcha/api/siteverify', data=data)
|
||||
try:
|
||||
json_response = response.json()
|
||||
print(json_response)
|
||||
return json_response[u'success']
|
||||
except ValueError:
|
||||
print response
|
||||
return False
|
||||
|
||||
|
||||
@views.route('/', methods=[u'GET'])
|
||||
def index():
|
||||
"""
|
||||
Add a new sticky note
|
||||
"""
|
||||
all_lexers = [(lexer[1][0], lexer[0]) for lexer in get_all_lexers()]
|
||||
all_lexers.sort(key=lambda x: x[1].lower())
|
||||
now = datetime.utcnow()
|
||||
recaptcha_site_key = current_app.config.get(u'RECAPTCHA_SITE_KEY')
|
||||
return render_template(u'index.mako', lexers=all_lexers, recaptcha_site_key=recaptcha_site_key)
|
||||
|
||||
|
||||
@views.route('/notes', methods=[u'GET'])
|
||||
def notes():
|
||||
"""
|
||||
Show a list of recent notes
|
||||
"""
|
||||
recent_notes = get_session().query(StickyNote)\
|
||||
.filter(or_(StickyNote.expiry == None, StickyNote.expiry < now))\
|
||||
.filter(~StickyNote.private)\
|
||||
.order_by(StickyNote.created.desc())\
|
||||
.limit(10)
|
||||
return render_template(u'notes.mako', recent=recent_notes)
|
||||
|
||||
|
||||
@views.route('/about', methods=[u'GET'])
|
||||
def about():
|
||||
"""
|
||||
Show the about page
|
||||
"""
|
||||
return render_template(u'about.mako')
|
||||
|
||||
|
||||
@views.route('/save', methods=[u'POST'])
|
||||
def save():
|
||||
"""
|
||||
Save a sticky note
|
||||
"""
|
||||
# Check if the recaptcha is valid
|
||||
recaptcha_secret_key = current_app.config.get(u'RECAPTCHA_SECRET_KEY')
|
||||
is_recaptcha_valid = False
|
||||
try:
|
||||
is_recaptcha_valid = _is_recaptcha_valid(recaptcha_secret_key, request.form[u'g-recaptcha-response'])
|
||||
except KeyError:
|
||||
flash(u'Unable to verify you, don\'t forget to complete the captcha.', u'danger')
|
||||
print(u'No form variable')
|
||||
if not is_recaptcha_valid:
|
||||
return redirect(u'/')
|
||||
# Save the note
|
||||
db_session = get_session()
|
||||
try:
|
||||
created = datetime.utcnow()
|
||||
expiry = EXPIRY_DELTAS.get(request.form[u'expiry'])
|
||||
if expiry:
|
||||
expiry = created + expiry
|
||||
source = request.form[u'source']
|
||||
lexer = request.form[u'language']
|
||||
title = request.form.get(u'title', u'')
|
||||
private = True if request.form.get(u'private') else False
|
||||
string_id = u''.join([source, lexer, title, created.isoformat()])
|
||||
url = short_url.encode_url(sum([ord(char) for char in string_id]), min_length=8)
|
||||
note = StickyNote(
|
||||
title=title,
|
||||
source=source,
|
||||
lexer=lexer,
|
||||
created=created,
|
||||
expiry=expiry,
|
||||
private=private,
|
||||
url=url
|
||||
)
|
||||
db_session.add(note)
|
||||
db_session.commit()
|
||||
return redirect(u'/' + note.url)
|
||||
except Exception as e:
|
||||
flash(str(e), 'danger')
|
||||
db_session.rollback()
|
||||
return redirect(u'/')
|
||||
|
||||
|
||||
@views.route('/<string:note_url>', methods=[u'GET'])
|
||||
def view(note_url):
|
||||
"""
|
||||
Show a sticky note
|
||||
|
||||
:param note_url: The note to show
|
||||
"""
|
||||
note = get_session().query(StickyNote).filter(StickyNote.url == note_url).scalar()
|
||||
if not note:
|
||||
flash(u'That note does not exist', 'danger')
|
||||
return redirect(u'/')
|
||||
|
||||
lexer = get_lexer_by_name(note.lexer)
|
||||
formatter = HtmlFormatter(linenos=True, cssclass=u'source')
|
||||
result = highlight(note.source, lexer, formatter)
|
||||
return render_template(u'view.mako', note=note, source=result)
|
||||
|
||||
|
||||
@views.route('/raw/<string:note_url>', methods=[u'GET'])
|
||||
def raw(note_url):
|
||||
"""
|
||||
Show the raw version of a sticky note
|
||||
|
||||
:param note_url: The note to show
|
||||
"""
|
||||
note = get_session().query(StickyNote).filter(StickyNote.url == note_url).scalar()
|
||||
if not note:
|
||||
flash(u'That note does not exist', 'danger')
|
||||
return redirect(u'/')
|
||||
return render_template(u'raw.mako', source=note.source), 200, {'Content-Type': 'text/plain; charset=utf-8'}
|
||||
|
||||
|
||||
@views.route(u'/pygments.css', methods=[u'GET'])
|
||||
def pygments_css():
|
||||
"""
|
||||
Return the Pygments CSS to the browser
|
||||
"""
|
||||
response = make_response(HtmlFormatter().get_style_defs())
|
||||
response.headers['Content-Type'] = 'text/css'
|
||||
return response
|
10
wsgiapp.py
Normal file
10
wsgiapp.py
Normal file
@ -0,0 +1,10 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
"""
|
||||
This is the entry point for the WSGI server
|
||||
"""
|
||||
from stickynotes import make_app
|
||||
|
||||
application = make_app()
|
||||
|
||||
if __name__ == '__main__':
|
||||
application.run(debug=True)
|
Loading…
Reference in New Issue
Block a user