api,db: add page attributes and thread locking

This commit is contained in:
Adhityaa 2018-07-05 10:36:52 +05:30 committed by Adhityaa Chandrasekar
parent 0a03a2c6fc
commit 299649cea2
16 changed files with 427 additions and 4 deletions

View File

@ -111,6 +111,12 @@ func commentListHandler(w http.ResponseWriter, r *http.Request) {
return return
} }
p, err := pageGet(domain, path)
if err != nil {
bodyMarshal(w, response{"success": false, "message": err.Error()})
return
}
commenterHex := "anonymous" commenterHex := "anonymous"
isModerator := false isModerator := false
if *x.CommenterToken != "anonymous" { if *x.CommenterToken != "anonymous" {
@ -151,6 +157,7 @@ func commentListHandler(w http.ResponseWriter, r *http.Request) {
"requireIdentification": d.RequireIdentification, "requireIdentification": d.RequireIdentification,
"isFrozen": d.State == "frozen", "isFrozen": d.State == "frozen",
"isModerator": isModerator, "isModerator": isModerator,
"attributes": p,
"configuredOauths": configuredOauths, "configuredOauths": configuredOauths,
}) })
} }

View File

@ -13,6 +13,16 @@ func commentNew(commenterHex string, domain string, path string, parentHex strin
return "", errorMissingField return "", errorMissingField
} }
p, err := pageGet(domain, path)
if err != nil {
logger.Errorf("cannot get page attributes: %v", err)
return "", errorInternal
}
if p.IsLocked {
return "", errorThreadLocked
}
commentHex, err := randomHex(32) commentHex, err := randomHex(32)
if err != nil { if err != nil {
return "", err return "", err
@ -31,6 +41,10 @@ func commentNew(commenterHex string, domain string, path string, parentHex strin
return "", errorInternal return "", errorInternal
} }
if err = pageNew(domain, path); err != nil {
return "", err
}
return commentHex, nil return commentHex, nil
} }

View File

@ -56,3 +56,18 @@ func TestCommentNewUpvoted(t *testing.T) {
return return
} }
} }
func TestCommentNewThreadLocked(t *testing.T) {
failTestOnError(t, setupTestEnv())
pageNew("example.com", "/path.html")
p, _ := pageGet("example.com", "/path.html")
p.IsLocked = true
pageUpdate(p)
_, err := commentNew("temp-commenter-hex", "example.com", "/path.html", "root", "**foo**", "approved", time.Now().UTC())
if err == nil {
t.Errorf("expected error not found creating a new comment on a locked thread")
return
}
}

View File

@ -41,3 +41,4 @@ var errorSelfVote = errors.New("You cannot vote on your own comment.")
var errorInvalidConfigFile = errors.New("Invalid config file.") var errorInvalidConfigFile = errors.New("Invalid config file.")
var errorInvalidConfigValue = errors.New("Invalid config value.") var errorInvalidConfigValue = errors.New("Invalid config value.")
var errorNewOwnerForbidden = errors.New("New user registrations are forbidden and closed.") var errorNewOwnerForbidden = errors.New("New user registrations are forbidden and closed.")
var errorThreadLocked = errors.New("This thread is locked. You cannot add new comments.")

9
api/page.go Normal file
View File

@ -0,0 +1,9 @@
package main
import ()
type page struct {
Domain string `json:"domain"`
Path string `json:"path"`
IsLocked bool `json:"isLocked"`
}

34
api/page_get.go Normal file
View File

@ -0,0 +1,34 @@
package main
import (
"database/sql"
)
func pageGet(domain string, path string) (page, error) {
// path can be empty
if domain == "" {
return page{}, errorMissingField
}
statement := `
SELECT isLocked
FROM pages
WHERE domain=$1 AND path=$2;
`
row := db.QueryRow(statement, domain, path)
p := page{Domain: domain, Path: path}
if err := row.Scan(&p.IsLocked); err != nil {
if err == sql.ErrNoRows {
// If there haven't been any comments, there won't be a record for this
// page. The sane thing to do is return defaults.
// TODO: the defaults are hard-coded in two places: here and the schema
p.IsLocked = false
} else {
logger.Errorf("error scanning page: %v", err)
return page{}, errorInternal
}
}
return p, nil
}

43
api/page_get_test.go Normal file
View File

@ -0,0 +1,43 @@
package main
import (
"testing"
)
func TestPageGetBasics(t *testing.T) {
failTestOnError(t, setupTestEnv())
pageNew("example.com", "/path.html")
p, err := pageGet("example.com", "/path.html")
if err != nil {
t.Errorf("unexpected error getting page: %v", err)
return
}
if p.IsLocked != false {
t.Errorf("expected p.IsLocked=false got %v", p.IsLocked)
return
}
if _, err := pageGet("example.com", "/path2.html"); err != nil {
t.Errorf("unexpected error getting page with non-existant record: %v", err)
return
}
}
func TestPageGetEmpty(t *testing.T) {
failTestOnError(t, setupTestEnv())
pageNew("example.com", "")
if _, err := pageGet("example.com", ""); err != nil {
t.Errorf("unexpected error getting page with empty path: %v", err)
return
}
if _, err := pageGet("", "/path.html"); err == nil {
t.Errorf("exepected error not found when getting page with empty domain")
return
}
}

24
api/page_new.go Normal file
View File

@ -0,0 +1,24 @@
package main
import ()
func pageNew(domain string, path string) error {
// path can be empty
if domain == "" {
return errorMissingField
}
statement := `
INSERT INTO
pages (domain, path)
VALUES ($1, $2 )
ON CONFLICT DO NOTHING;
`
_, err := db.Exec(statement, domain, path)
if err != nil {
logger.Errorf("error inserting new page: %v", err)
return errorInternal
}
return nil
}

43
api/page_new_test.go Normal file
View File

@ -0,0 +1,43 @@
package main
import (
"testing"
)
func TestPageNewBasics(t *testing.T) {
failTestOnError(t, setupTestEnv())
if err := pageNew("example.com", "/path.html"); err != nil {
t.Errorf("unexpected error creating page: %v", err)
return
}
}
func TestPageNewEmpty(t *testing.T) {
failTestOnError(t, setupTestEnv())
if err := pageNew("example.com", ""); err != nil {
t.Errorf("unexpected error creating page with empty path: %v", err)
return
}
if err := pageNew("", "/path.html"); err == nil {
t.Errorf("expected error not found creating page with empty domain")
return
}
}
func TestPageNewUnique(t *testing.T) {
failTestOnError(t, setupTestEnv())
if err := pageNew("example.com", "/path.html"); err != nil {
t.Errorf("unexpected error creating page: %v", err)
return
}
// no error should be returned when trying to duplicate insert
if err := pageNew("example.com", "/path.html"); err != nil {
t.Errorf("unexpected error creating same page twice: %v", err)
return
}
}

70
api/page_update.go Normal file
View File

@ -0,0 +1,70 @@
package main
import (
"net/http"
)
func pageUpdate(p page) error {
if p.Domain == "" {
return errorMissingField
}
statement := `
INSERT INTO
pages (domain, path, isLocked)
VALUES ($1, $2, $3 )
ON CONFLICT (domain, path) DO
UPDATE SET isLocked = $3;
`
_, err := db.Exec(statement, p.Domain, p.Path, p.IsLocked)
if err != nil {
logger.Errorf("error setting page attributes: %v", err)
return errorInternal
}
return nil
}
func pageUpdateHandler(w http.ResponseWriter, r *http.Request) {
type request struct {
CommenterToken *string `json:"commenterToken"`
Domain *string `json:"domain"`
Path *string `json:"path"`
Attributes *page `json:"attributes"`
}
var x request
if err := bodyUnmarshal(r, &x); err != nil {
bodyMarshal(w, response{"success": false, "message": err.Error()})
return
}
c, err := commenterGetByCommenterToken(*x.CommenterToken)
if err != nil {
bodyMarshal(w, response{"success": false, "message": err.Error()})
return
}
domain := domainStrip(*x.Domain)
isModerator, err := isDomainModerator(domain, c.Email)
if err != nil {
bodyMarshal(w, response{"success": false, "message": err.Error()})
return
}
if !isModerator {
bodyMarshal(w, response{"success": false, "message": errorNotModerator.Error()})
return
}
(*x.Attributes).Domain = *x.Domain
(*x.Attributes).Path = *x.Path
if err = pageUpdate(*x.Attributes); err != nil {
bodyMarshal(w, response{"success": false, "message": err.Error()})
return
}
bodyMarshal(w, response{"success": true})
}

43
api/page_update_test.go Normal file
View File

@ -0,0 +1,43 @@
package main
import (
"testing"
"time"
)
func TestPageUpdateBasics(t *testing.T) {
failTestOnError(t, setupTestEnv())
commenterHex, _ := commenterNew("test@example.com", "Test", "undefined", "https://example.com/photo.jpg", "google", "")
commentNew(commenterHex, "example.com", "/path.html", "root", "**foo**", "unapproved", time.Now().UTC())
p, _ := pageGet("example.com", "/path.html")
if p.IsLocked != false {
t.Errorf("expected IsLocked=false got %v", p.IsLocked)
return
}
p.IsLocked = true
if err := pageUpdate(p); err != nil {
t.Errorf("unexpected error updating page: %v", err)
return
}
p, _ = pageGet("example.com", "/path.html")
if p.IsLocked != true {
t.Errorf("expected IsLocked=true got %v", p.IsLocked)
return
}
}
func TestPageUpdateEmpty(t *testing.T) {
failTestOnError(t, setupTestEnv())
p := page{Domain: "", Path: "", IsLocked: false}
if err := pageUpdate(p); err == nil {
t.Errorf("expected error not found updating page with empty everything")
return
}
}

View File

@ -35,5 +35,7 @@ func apiRouterInit(router *mux.Router) error {
router.HandleFunc("/api/comment/approve", commentApproveHandler).Methods("POST") router.HandleFunc("/api/comment/approve", commentApproveHandler).Methods("POST")
router.HandleFunc("/api/comment/delete", commentDeleteHandler).Methods("POST") router.HandleFunc("/api/comment/delete", commentDeleteHandler).Methods("POST")
router.HandleFunc("/api/page/update", pageUpdateHandler).Methods("POST")
return nil return nil
} }

View File

@ -0,0 +1,9 @@
-- Introduces page attributes
CREATE TABLE IF NOT EXISTS pages (
domain TEXT NOT NULL ,
path TEXT NOT NULL ,
isLocked BOOLEAN NOT NULL DEFAULT false
);
CREATE UNIQUE INDEX pagesUniqueIndex ON pages(domain, path);

View File

@ -27,6 +27,8 @@
var ID_LOGIN_BOX_HR = "commento-login-box-hr"; var ID_LOGIN_BOX_HR = "commento-login-box-hr";
var ID_LOGIN_BOX_OAUTH_PRETEXT = "commento-login-box-oauth-pretext"; var ID_LOGIN_BOX_OAUTH_PRETEXT = "commento-login-box-oauth-pretext";
var ID_LOGIN_BOX_OAUTH_BUTTONS_CONTAINER = "commento-login-box-oauth-buttons-container"; var ID_LOGIN_BOX_OAUTH_BUTTONS_CONTAINER = "commento-login-box-oauth-buttons-container";
var ID_MOD_TOOLS = "commento-mod-tools";
var ID_MOD_TOOLS_LOCK_BUTTON = "commento-mod-tools-lock-button";
var ID_ERROR = "commento-error"; var ID_ERROR = "commento-error";
var ID_LOGGED_CONTAINER = "commento-logged-container"; var ID_LOGGED_CONTAINER = "commento-logged-container";
var ID_COMMENTS_AREA = "commento-comments-area"; var ID_COMMENTS_AREA = "commento-comments-area";
@ -62,8 +64,9 @@
var requireModeration = true; var requireModeration = true;
var isModerator = false; var isModerator = false;
var isFrozen = false; var isFrozen = false;
var chosenAnonymous = false;
var shownSubmitButton = {"root": false}; var shownSubmitButton = {"root": false};
var chosenAnonymous = false;
var isLocked = false;
var shownReply = {}; var shownReply = {};
var configuredOauths = []; var configuredOauths = [];
var loginBoxType = "signup"; var loginBoxType = "signup";
@ -336,6 +339,9 @@
requireIdentification = resp.requireIdentification; requireIdentification = resp.requireIdentification;
isModerator = resp.isModerator; isModerator = resp.isModerator;
isFrozen = resp.isFrozen; isFrozen = resp.isFrozen;
isLocked = resp.attributes.isLocked;
comments = resp.comments; comments = resp.comments;
commenters = resp.commenters; commenters = resp.commenters;
configuredOauths = resp.configuredOauths; configuredOauths = resp.configuredOauths;
@ -443,7 +449,11 @@
commentsArea.innerHTML = ""; commentsArea.innerHTML = "";
append(mainArea, textareaCreate("root")); if (!isLocked)
append(mainArea, textareaCreate("root"));
else
append(mainArea, messageCreate("This thread is locked. You cannot create new comments."));
append(mainArea, commentsArea); append(mainArea, commentsArea);
append(root, mainArea); append(root, mainArea);
@ -715,7 +725,10 @@
// append(options, edit); // uncomment when implemented // append(options, edit); // uncomment when implemented
append(options, downvote); append(options, downvote);
append(options, upvote); append(options, upvote);
append(options, reply);
if (!isLocked)
append(options, reply);
if (isModerator) { if (isModerator) {
append(options, remove); append(options, remove);
if (comment.state == "unapproved") if (comment.state == "unapproved")
@ -1257,6 +1270,45 @@
} }
function pageUpdate(callback) {
var attributes = {
"isLocked": isLocked,
};
var json = {
"commenterToken": commenterTokenGet(),
"domain": location.host,
"path": location.pathname,
"attributes": attributes,
};
post(origin + "/api/page/update", json, function(resp) {
if (!resp.success) {
errorShow(resp.message);
return
}
call(callback);
});
}
global.threadLockToggle = function() {
var lock = $(ID_MOD_TOOLS_LOCK_BUTTON);
isLocked = !isLocked;
lock.disabled = true;
pageUpdate(function(success) {
lock.disabled = false;
if (isLocked)
lock.innerHTML = "Unlock Thread";
else
lock.innerHTML = "Lock Thread";
});
}
function mainAreaCreate() { function mainAreaCreate() {
var mainArea = create("div"); var mainArea = create("div");
@ -1270,6 +1322,29 @@
} }
function modToolsCreate() {
var modTools = create("div");
var lock = create("button");
modTools.id = ID_MOD_TOOLS;
lock.id = ID_MOD_TOOLS_LOCK_BUTTON;
classAdd(modTools, "mod-tools");
classAdd(lock, "mod-tools-lock-button");
if (isLocked)
lock.innerHTML = "Unlock Thread";
else
lock.innerHTML = "Lock Thread";
attrSet(modTools, "style", "display: none");
attrSet(lock, "onclick", "threadLockToggle()");
append(modTools, lock);
append(root, modTools);
}
global.loadCssOverride = function() { global.loadCssOverride = function() {
if (cssOverride === undefined) if (cssOverride === undefined)
global.allShow(); global.allShow();
@ -1280,12 +1355,18 @@
global.allShow = function() { global.allShow = function() {
var mainArea = $(ID_MAIN_AREA); var mainArea = $(ID_MAIN_AREA);
var modTools = $(ID_MOD_TOOLS);
var loggedContainer = $(ID_LOGGED_CONTAINER); var loggedContainer = $(ID_LOGGED_CONTAINER);
var footer = $(ID_FOOTER); var footer = $(ID_FOOTER);
attrSet(mainArea, "style", ""); attrSet(mainArea, "style", "");
if (isModerator)
attrSet(modTools, "style", "");
if (loggedContainer) if (loggedContainer)
attrSet(loggedContainer, "style", ""); attrSet(loggedContainer, "style", "");
attrSet(footer, "style", ""); attrSet(footer, "style", "");
nameWidthFix(); nameWidthFix();
@ -1346,6 +1427,7 @@
selfGet(function() { selfGet(function() {
commentsGet(function() { commentsGet(function() {
modToolsCreate();
rootCreate(function() { rootCreate(function() {
commentsRender(); commentsRender();
footerLoad(); footerLoad();

View File

@ -177,3 +177,18 @@ textarea {
.commento-button-margin { .commento-button-margin {
padding-bottom: 60px; padding-bottom: 60px;
} }
.commento-mod-tools-lock-button {
color: $gray-6;
background: none;
border: none;
font-weight: bold;
font-size: 12px;
text-transform: uppercase;
margin-left: 12px;
padding: 2px;
}
.commento-mod-tools-lock-button:hover {
cursor: pointer;
}

View File

@ -41,6 +41,18 @@
font-weight: bold; font-weight: bold;
} }
.commento-mod-tools {
margin-bottom: 16px;
}
.commento-mod-tools::before {
content: "Moderators";
text-transform: uppercase;
color: $indigo-8;
font-size: 12px;
font-weight: bold;
}
.commento-moderation-notice { .commento-moderation-notice {
width: 100%; width: 100%;
border-radius: 4px; border-radius: 4px;
@ -87,7 +99,7 @@
border-top: 1px solid #f0f0f0; border-top: 1px solid #f0f0f0;
.commento-header { .commento-header {
padding-bottom: 12px; padding-bottom: 4px;
} }
.commento-avatar::after { .commento-avatar::after {