Compare commits

...

8 Commits

Author SHA1 Message Date
Martin Zimmermann f50b0b0ffb Back to development: 0.5.3
11 years ago
Martin Zimmermann 793f2dcb7f Preparing release 0.5.2
11 years ago
Martin Zimmermann 3fcf079ed1 use el.getAttribute instead of el.dataset to support IE10 m(
11 years ago
Martin Zimmermann 6c06b69dc5 another approach to fix #40 (return 403 on false Content-Type)
11 years ago
Martin Zimmermann 580f63606e Back to development: 0.5.2
11 years ago
Martin Zimmermann 9541f61900 Preparing release 0.5.1
11 years ago
Martin Zimmermann 30edf6ca28 add route for comment activation
11 years ago
Martin Zimmermann a43ac60552 translate deletion and activation links
11 years ago

@ -1,8 +1,14 @@
Changelog for Isso Changelog for Isso
================== ==================
0.5 (2013-11-17) 0.5.2 (2013-12-08)
---------------- ------------------
- Nothing changed yet.
0.5.1 (2013-11-21)
------------------
Major improvements: Major improvements:

@ -96,10 +96,10 @@ class SMTP(object):
key = self.isso.sign(comment["id"]) key = self.isso.sign(comment["id"])
rv.write("---\n") rv.write("---\n")
rv.write("Kommentar löschen: %s\n" % (uri + "/delete/" + key)) rv.write("Delete comment: %s\n" % (uri + "/delete/" + key))
if comment["mode"] == 2: if comment["mode"] == 2:
rv.write("Kommentar freischalten: %s\n" % (uri + "/activate/" + key)) rv.write("Activate comment: %s\n" % (uri + "/activate/" + key))
rv.seek(0) rv.seek(0)
return rv.read() return rv.read()

@ -31,25 +31,21 @@ define(["q"], function(Q) {
* .. code-block:: html * .. code-block:: html
* *
* <script data-isso="http://example.tld/path/" src="/.../embed.min.js"></script> * <script data-isso="http://example.tld/path/" src="/.../embed.min.js"></script>
*
* 2. use require.js (during development). When using require.js, we
* assume that the path to the scripts ends with `/js/`.
*/ */
var script, endpoint, var script, endpoint,
js = document.getElementsByTagName("script"); js = document.getElementsByTagName("script");
for (var i = 0; i < js.length; i++) { for (var i = 0; i < js.length; i++) {
if (js[i].dataset.isso !== undefined) { if (js[i].hasAttribute("data-isso")) {
endpoint = js[i].dataset.isso; endpoint = js[i].getAttribute("data-isso");
} else if (js[i].src.match("require\\.js$")) { break;
endpoint = js[i].dataset.main.replace(/\/js\/(embed|count)$/, "");
} }
} }
if (! endpoint) { if (! endpoint) {
for (i = 0; i < js.length; i++) { for (i = 0; i < js.length; i++) {
if (js[i].hasAttribute("async") || js[i].hasAttribute("defer")) { if (js[i].getAttribute("async") || js[i].getAttribute("defer")) {
throw "Isso's automatic configuration detection failed, please " + throw "Isso's automatic configuration detection failed, please " +
"refer to https://github.com/posativ/isso#client-configuration " + "refer to https://github.com/posativ/isso#client-configuration " +
"and add a custom `data-isso` attribute."; "and add a custom `data-isso` attribute.";
@ -57,14 +53,10 @@ define(["q"], function(Q) {
} }
script = js[js.length - 1]; script = js[js.length - 1];
endpoint = script.src.substring(0, script.src.length - "/js/embed.min.js".length);
if (script.dataset.prefix) {
endpoint = script.dataset.prefix;
} else {
endpoint = script.src.substring(0, script.src.length - "/js/embed.min.js".length);
}
} }
// strip trailing slash
if (endpoint[endpoint.length - 1] === "/") { if (endpoint[endpoint.length - 1] === "/") {
endpoint = endpoint.substring(0, endpoint.length - 1); endpoint = endpoint.substring(0, endpoint.length - 1);
} }
@ -93,6 +85,7 @@ define(["q"], function(Q) {
try { try {
xhr.open(method, url, true); xhr.open(method, url, true);
xhr.withCredentials = true; xhr.withCredentials = true;
xhr.setRequestHeader("Content-Type", "application/json");
if (method === "GET") { if (method === "GET") {
xhr.setRequestHeader("X-Origin", window.location.origin); xhr.setRequestHeader("X-Origin", window.location.origin);

@ -31,6 +31,30 @@ class JSON(Response):
return super(JSON, self).__init__(*args, content_type='application/json') return super(JSON, self).__init__(*args, content_type='application/json')
def xhr(func):
"""A decorator to check for CSRF on POST/PUT/DELETE using a <form>
element and JS to execute automatically (see #40 for a proof-of-concept).
When an attacker uses a <form> to downvote a comment, the browser *should*
add a `Content-Type: ...` header with three possible values:
* application/x-www-form-urlencoded
* multipart/form-data
* text/plain
If the header is not sent or requests `application/json`, the request is
not forged (XHR is restricted by CORS separately).
"""
def dec(self, env, req, *args, **kwargs):
if req.content_type and not req.content_type.startswith("application/json"):
raise Forbidden("CSRF")
return func(self, env, req, *args, **kwargs)
return dec
class API(object): class API(object):
FIELDS = set(['id', 'parent', 'text', 'author', 'website', 'email', FIELDS = set(['id', 'parent', 'text', 'author', 'website', 'email',
@ -47,6 +71,7 @@ class API(object):
('edit', ('PUT', '/id/<int:id>')), ('edit', ('PUT', '/id/<int:id>')),
('delete', ('DELETE', '/id/<int:id>')), ('delete', ('DELETE', '/id/<int:id>')),
('delete', ('GET', '/id/<int:id>/delete/<string:key>')), ('delete', ('GET', '/id/<int:id>/delete/<string:key>')),
('activate',('GET', '/id/<int:id>/activate/<string:key>')),
('like', ('POST', '/id/<int:id>/like')), ('like', ('POST', '/id/<int:id>/like')),
('dislike', ('POST', '/id/<int:id>/dislike')), ('dislike', ('POST', '/id/<int:id>/dislike')),
('checkip', ('GET', '/check-ip')) ('checkip', ('GET', '/check-ip'))
@ -90,6 +115,7 @@ class API(object):
return True, "" return True, ""
@xhr
@requires(str, 'uri') @requires(str, 'uri')
def new(self, environ, request, uri): def new(self, environ, request, uri):
@ -173,6 +199,7 @@ class API(object):
return Response(json.dumps(rv), 200, content_type='application/json') return Response(json.dumps(rv), 200, content_type='application/json')
@xhr
def edit(self, environ, request, id): def edit(self, environ, request, id):
try: try:
@ -216,6 +243,7 @@ class API(object):
resp.headers.add("X-Set-Cookie", cookie("isso-%i" % rv["id"])) resp.headers.add("X-Set-Cookie", cookie("isso-%i" % rv["id"]))
return resp return resp
@xhr
def delete(self, environ, request, id, key=None): def delete(self, environ, request, id, key=None):
try: try:
@ -253,7 +281,7 @@ class API(object):
resp.headers.add("X-Set-Cookie", cookie("isso-%i" % id)) resp.headers.add("X-Set-Cookie", cookie("isso-%i" % id))
return resp return resp
def activate(self, environ, request, _, key): def activate(self, environ, request, id, key):
try: try:
id = self.isso.unsign(key, max_age=2**32) id = self.isso.unsign(key, max_age=2**32)
@ -293,11 +321,13 @@ class API(object):
return JSON(json.dumps(rv), 200) return JSON(json.dumps(rv), 200)
@xhr
def like(self, environ, request, id): def like(self, environ, request, id):
nv = self.comments.vote(True, id, utils.anonymize(str(request.remote_addr))) nv = self.comments.vote(True, id, utils.anonymize(str(request.remote_addr)))
return Response(json.dumps(nv), 200) return Response(json.dumps(nv), 200)
@xhr
def dislike(self, environ, request, id): def dislike(self, environ, request, id):
nv = self.comments.vote(False, id, utils.anonymize(str(request.remote_addr))) nv = self.comments.vote(False, id, utils.anonymize(str(request.remote_addr)))

@ -17,7 +17,7 @@ else:
setup( setup(
name='isso', name='isso',
version='0.5', version='0.5.3.dev0',
author='Martin Zimmermann', author='Martin Zimmermann',
author_email='info@posativ.org', author_email='info@posativ.org',
packages=find_packages(), packages=find_packages(),

@ -2,6 +2,7 @@
import json import json
from werkzeug.test import Client
class FakeIP(object): class FakeIP(object):
@ -14,6 +15,13 @@ class FakeIP(object):
return self.app(environ, start_response) return self.app(environ, start_response)
class JSONClient(Client):
def open(self, *args, **kwargs):
kwargs.setdefault('content_type', 'application/json')
return super(JSONClient, self).open(*args, **kwargs)
class Dummy: class Dummy:
status = 200 status = 200

@ -12,14 +12,13 @@ try:
except ImportError: except ImportError:
from urllib import urlencode from urllib import urlencode
from werkzeug.test import Client
from werkzeug.wrappers import Response from werkzeug.wrappers import Response
from isso import Isso, core from isso import Isso, core
from isso.utils import http from isso.utils import http
from isso.views import comments from isso.views import comments
from fixtures import curl, loads, FakeIP from fixtures import curl, loads, FakeIP, JSONClient
http.curl = curl http.curl = curl
@ -37,7 +36,7 @@ class TestComments(unittest.TestCase):
self.app = App(conf) self.app = App(conf)
self.app.wsgi_app = FakeIP(self.app.wsgi_app, "192.168.1.1") self.app.wsgi_app = FakeIP(self.app.wsgi_app, "192.168.1.1")
self.client = Client(self.app, Response) self.client = JSONClient(self.app, Response)
self.get = self.client.get self.get = self.client.get
self.put = self.client.put self.put = self.client.put
self.post = self.client.post self.post = self.client.post
@ -133,7 +132,7 @@ class TestComments(unittest.TestCase):
def testDeleteWithReference(self): def testDeleteWithReference(self):
client = Client(self.app, Response) client = JSONClient(self.app, Response)
client.post('/new?uri=%2Fpath%2F', data=json.dumps({'text': 'First'})) client.post('/new?uri=%2Fpath%2F', data=json.dumps({'text': 'First'}))
client.post('/new?uri=%2Fpath%2F', data=json.dumps({'text': 'First', 'parent': 1})) client.post('/new?uri=%2Fpath%2F', data=json.dumps({'text': 'First', 'parent': 1}))
@ -160,7 +159,7 @@ class TestComments(unittest.TestCase):
--- [ comment 4, ref 2 ] --- [ comment 4, ref 2 ]
[ comment 5 ] [ comment 5 ]
""" """
client = Client(self.app, Response) client = JSONClient(self.app, Response)
client.post('/new?uri=%2Fpath%2F', data=json.dumps({'text': 'First'})) client.post('/new?uri=%2Fpath%2F', data=json.dumps({'text': 'First'}))
client.post('/new?uri=%2Fpath%2F', data=json.dumps({'text': 'Second', 'parent': 1})) client.post('/new?uri=%2Fpath%2F', data=json.dumps({'text': 'Second', 'parent': 1}))
@ -193,11 +192,11 @@ class TestComments(unittest.TestCase):
def testDeleteAndCreateByDifferentUsersButSamePostId(self): def testDeleteAndCreateByDifferentUsersButSamePostId(self):
mallory = Client(self.app, Response) mallory = JSONClient(self.app, Response)
mallory.post('/new?uri=%2Fpath%2F', data=json.dumps({'text': 'Foo'})) mallory.post('/new?uri=%2Fpath%2F', data=json.dumps({'text': 'Foo'}))
mallory.delete('/id/1') mallory.delete('/id/1')
bob = Client(self.app, Response) bob = JSONClient(self.app, Response)
bob.post('/new?uri=%2Fpath%2F', data=json.dumps({'text': 'Bar'})) bob.post('/new?uri=%2Fpath%2F', data=json.dumps({'text': 'Bar'}))
assert mallory.delete('/id/1').status_code == 403 assert mallory.delete('/id/1').status_code == 403
@ -263,10 +262,27 @@ class TestComments(unittest.TestCase):
def testDeleteCommentRemovesThread(self): def testDeleteCommentRemovesThread(self):
rv = self.client.post('/new?uri=%2F', data=json.dumps({"text": "..."})) rv = self.client.post('/new?uri=%2F', data=json.dumps({"text": "..."}))
assert '/' in self.app.db.threads assert '/' in self.app.db.threads
self.client.delete('/id/1') self.client.delete('/id/1')
assert '/' not in self.app.db.threads assert '/' not in self.app.db.threads
def testCSRF(self):
js = "application/json"
form = "application/x-www-form-urlencoded"
self.post('/new?uri=%2F', data=json.dumps({"text": "..."}))
# no header is fine (default for XHR)
assert self.post('/id/1/dislike', content_type="").status_code == 200
# x-www-form-urlencoded is definitely not RESTful
assert self.post('/id/1/dislike', content_type=form).status_code == 403
assert self.post('/new?uri=%2F', data=json.dumps({"text": "..."}),
content_type=form).status_code == 403
# just for the record
assert self.post('/id/1/dislike', content_type=js).status_code == 200
class TestModeratedComments(unittest.TestCase): class TestModeratedComments(unittest.TestCase):
@ -283,7 +299,7 @@ class TestModeratedComments(unittest.TestCase):
self.app = App(conf) self.app = App(conf)
self.app.wsgi_app = FakeIP(self.app.wsgi_app, "192.168.1.1") self.app.wsgi_app = FakeIP(self.app.wsgi_app, "192.168.1.1")
self.client = Client(self.app, Response) self.client = JSONClient(self.app, Response)
def tearDown(self): def tearDown(self):
os.unlink(self.path) os.unlink(self.path)
@ -314,7 +330,7 @@ class TestPurgeComments(unittest.TestCase):
self.app = App(conf) self.app = App(conf)
self.app.wsgi_app = FakeIP(self.app.wsgi_app, "192.168.1.1") self.app.wsgi_app = FakeIP(self.app.wsgi_app, "192.168.1.1")
self.client = Client(self.app, Response) self.client = JSONClient(self.app, Response)
def testPurgeDoesNoHarm(self): def testPurgeDoesNoHarm(self):
self.client.post('/new?uri=test', data=json.dumps({"text": "..."})) self.client.post('/new?uri=test', data=json.dumps({"text": "..."}))

@ -6,13 +6,12 @@ import json
import tempfile import tempfile
import unittest import unittest
from werkzeug.test import Client
from werkzeug.wrappers import Response from werkzeug.wrappers import Response
from isso import Isso, core from isso import Isso, core
from isso.utils import http from isso.utils import http
from fixtures import curl, loads, FakeIP from fixtures import curl, loads, FakeIP, JSONClient
http.curl = curl http.curl = curl
@ -33,7 +32,7 @@ class TestVote(unittest.TestCase):
app = App(conf) app = App(conf)
app.wsgi_app = FakeIP(app.wsgi_app, ip) app.wsgi_app = FakeIP(app.wsgi_app, ip)
return Client(app, Response) return JSONClient(app, Response)
def testZeroLikes(self): def testZeroLikes(self):

Loading…
Cancel
Save