Compare commits
32 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
cf0b394b05 | ||
|
|
a36b11f07d | ||
|
|
93c9ce0cad | ||
|
|
af88db42b2 | ||
|
|
0c6ccdc0a1 | ||
|
|
6d3f8171e5 | ||
|
|
3f1c570e84 | ||
|
|
cd88ae264e | ||
|
|
b2abcae319 | ||
|
|
41a5c675bf | ||
|
|
ac9f896a22 | ||
|
|
36d57914b2 | ||
|
|
800ba5dd0d | ||
|
|
a793f7b3b4 | ||
|
|
0d6dfb8319 | ||
|
|
c5d2e17615 | ||
|
|
93595f3877 | ||
|
|
c21329ac4e | ||
|
|
8500a3f7c6 | ||
|
|
4ffdd2cfb6 | ||
|
|
b4f2ba41be | ||
|
|
8ebc0cd965 | ||
|
|
405d10766a | ||
|
|
2a713c22f1 | ||
|
|
d6ccb7338c | ||
|
|
4d799182da | ||
|
|
f54f4d0afd | ||
|
|
988a9fb1a1 | ||
|
|
283a32e2bb | ||
|
|
330131f390 | ||
|
|
299649cea2 | ||
|
|
0a03a2c6fc |
@@ -4,13 +4,15 @@ stages:
|
|||||||
- go-test
|
- go-test
|
||||||
- build-src
|
- build-src
|
||||||
- build-docker
|
- build-docker
|
||||||
- docker-registry
|
- docker-registry-master
|
||||||
|
- docker-registry-tags
|
||||||
|
|
||||||
check-dco:
|
check-dco:
|
||||||
stage: check-dco
|
stage: check-dco
|
||||||
image: debian:buster
|
image: debian:buster
|
||||||
except:
|
except:
|
||||||
- master
|
- master
|
||||||
|
- tags
|
||||||
script:
|
script:
|
||||||
- apt update
|
- apt update
|
||||||
- apt install -y curl git jq
|
- apt install -y curl git jq
|
||||||
@@ -23,14 +25,21 @@ build-src:
|
|||||||
GOPATH: $CI_PROJECT_DIR
|
GOPATH: $CI_PROJECT_DIR
|
||||||
except:
|
except:
|
||||||
- master
|
- master
|
||||||
script:
|
- tags
|
||||||
|
before_script:
|
||||||
- apt update
|
- apt update
|
||||||
- apt install -y curl gnupg git make golang
|
- apt install -y curl gnupg git make golang
|
||||||
|
- mkdir -p /go/src /go/bin /go/pkg
|
||||||
|
- export GOPATH=/go
|
||||||
|
- export PATH=$PATH:/go/bin
|
||||||
|
- curl https://raw.githubusercontent.com/golang/dep/master/install.sh | sh
|
||||||
|
- ln -s $CI_PROJECT_DIR /go/src/$CI_PROJECT_NAME
|
||||||
- curl -sL https://deb.nodesource.com/setup_10.x | bash -
|
- curl -sL https://deb.nodesource.com/setup_10.x | bash -
|
||||||
- apt update
|
- apt update
|
||||||
- apt install -y nodejs
|
- apt install -y nodejs
|
||||||
- npm install -g html-minifier@3.5.7 uglify-js@3.4.1 sass@1.5.1
|
- npm install -g html-minifier@3.5.7 uglify-js@3.4.1 sass@1.5.1
|
||||||
- mkdir -p src/gitlab.com/commento && cd src/gitlab.com/commento && ln -s $CI_PROJECT_DIR && cd $CI_PROJECT_NAME
|
script:
|
||||||
|
- cd /go/src/$CI_PROJECT_NAME
|
||||||
- make devel
|
- make devel
|
||||||
- make prod
|
- make prod
|
||||||
|
|
||||||
@@ -41,6 +50,7 @@ build-docker:
|
|||||||
- docker:dind
|
- docker:dind
|
||||||
except:
|
except:
|
||||||
- master
|
- master
|
||||||
|
- tags
|
||||||
script:
|
script:
|
||||||
- docker build -t commento-ce .
|
- docker build -t commento-ce .
|
||||||
|
|
||||||
@@ -54,11 +64,17 @@ go-test:
|
|||||||
POSTGRES_PASSWORD: postgres
|
POSTGRES_PASSWORD: postgres
|
||||||
POSTGRES_DB: commento_test
|
POSTGRES_DB: commento_test
|
||||||
COMMENTO_POSTGRES: postgres://postgres:postgres@postgres/commento_test?sslmode=disable
|
COMMENTO_POSTGRES: postgres://postgres:postgres@postgres/commento_test?sslmode=disable
|
||||||
GOPATH: $CI_PROJECT_DIR
|
|
||||||
except:
|
except:
|
||||||
- master
|
- master
|
||||||
|
- tags
|
||||||
|
before_script:
|
||||||
|
- mkdir -p /go/src /go/bin /go/pkg
|
||||||
|
- export GOPATH=/go
|
||||||
|
- export PATH=$PATH:/go/bin
|
||||||
|
- curl https://raw.githubusercontent.com/golang/dep/master/install.sh | sh
|
||||||
|
- ln -s $CI_PROJECT_DIR /go/src/$CI_PROJECT_NAME
|
||||||
script:
|
script:
|
||||||
- mkdir -p src/gitlab.com/commento && cd src/gitlab.com/commento && ln -s $CI_PROJECT_DIR && cd $CI_PROJECT_NAME
|
- cd /go/src/$CI_PROJECT_NAME
|
||||||
- make test
|
- make test
|
||||||
|
|
||||||
go-fmt:
|
go-fmt:
|
||||||
@@ -66,12 +82,13 @@ go-fmt:
|
|||||||
image: golang:1.10.2
|
image: golang:1.10.2
|
||||||
except:
|
except:
|
||||||
- master
|
- master
|
||||||
|
- tags
|
||||||
script:
|
script:
|
||||||
- cd api
|
- cd api
|
||||||
- test -z $(go fmt)
|
- test -z $(go fmt)
|
||||||
|
|
||||||
docker-registry:
|
docker-registry-master:
|
||||||
stage: docker-registry
|
stage: docker-registry-master
|
||||||
image: docker:stable
|
image: docker:stable
|
||||||
services:
|
services:
|
||||||
- docker:dind
|
- docker:dind
|
||||||
@@ -83,3 +100,17 @@ docker-registry:
|
|||||||
- docker pull registry.gitlab.com/commento/commento-ce:latest || true
|
- docker pull registry.gitlab.com/commento/commento-ce:latest || true
|
||||||
- docker build --cache-from registry.gitlab.com/commento/commento-ce:latest --tag registry.gitlab.com/commento/commento-ce:latest .
|
- docker build --cache-from registry.gitlab.com/commento/commento-ce:latest --tag registry.gitlab.com/commento/commento-ce:latest .
|
||||||
- docker push registry.gitlab.com/commento/commento-ce:latest
|
- docker push registry.gitlab.com/commento/commento-ce:latest
|
||||||
|
|
||||||
|
docker-registry-tags:
|
||||||
|
stage: docker-registry-tags
|
||||||
|
image: docker:stable
|
||||||
|
services:
|
||||||
|
- docker:dind
|
||||||
|
only:
|
||||||
|
- tags
|
||||||
|
before_script:
|
||||||
|
- docker login -u gitlab-ci-token -p $CI_JOB_TOKEN registry.gitlab.com
|
||||||
|
script:
|
||||||
|
- apk add git
|
||||||
|
- docker build --tag registry.gitlab.com/commento/commento-ce:$(git describe --tags) .
|
||||||
|
- docker push registry.gitlab.com/commento/commento-ce:$(git describe --tags)
|
||||||
|
|||||||
@@ -4,7 +4,8 @@ FROM golang:1.10.2-alpine AS api-build
|
|||||||
COPY ./api /go/src/commento-ce/api
|
COPY ./api /go/src/commento-ce/api
|
||||||
WORKDIR /go/src/commento-ce/api
|
WORKDIR /go/src/commento-ce/api
|
||||||
|
|
||||||
RUN apk update && apk add bash make git
|
RUN apk update && apk add bash make git curl
|
||||||
|
RUN curl https://raw.githubusercontent.com/golang/dep/master/install.sh | sh
|
||||||
|
|
||||||
RUN make prod -j$(($(nproc) + 1))
|
RUN make prod -j$(($(nproc) + 1))
|
||||||
|
|
||||||
|
|||||||
10
README.md
10
README.md
@@ -38,6 +38,16 @@ Commento is possible only because of its community. If this is your first contri
|
|||||||
|
|
||||||
Help will always be given to those who ask for it. We use IRC for chat to collaborate with other developers. You're invited to [hang out with us](https://irc.commento.io) in the `#commento-dev` channel on freenode if you want to contribute to Commento!
|
Help will always be given to those who ask for it. We use IRC for chat to collaborate with other developers. You're invited to [hang out with us](https://irc.commento.io) in the `#commento-dev` channel on freenode if you want to contribute to Commento!
|
||||||
|
|
||||||
|
### Sponsors
|
||||||
|
|
||||||
|
Commento CE development is sponsored by [Mozilla](https://mozilla.org) and [DigitalOcean](https://www.digitalocean.com/) independently.
|
||||||
|
|
||||||
|
<p align="center">
|
||||||
|
<a href="https://www.mozilla.org/en-US/"><img src="https://user-images.githubusercontent.com/7521600/32265838-d05b2d08-bf0a-11e7-92e1-2cb183eae616.png" title="Mozilla" height="40"></a>
|
||||||
|
|
||||||
|
<a href="https://www.digitalocean.com"><img src="https://user-images.githubusercontent.com/7521600/32265839-d093c7da-bf0a-11e7-8d99-96a940041d06.png" title="DigitalOcean" height="40"></a>
|
||||||
|
</p>
|
||||||
|
|
||||||
#### License
|
#### License
|
||||||
|
|
||||||
```
|
```
|
||||||
|
|||||||
0
Gopkg.lock → api/Gopkg.lock
generated
0
Gopkg.lock → api/Gopkg.lock
generated
@@ -25,15 +25,15 @@ clean:
|
|||||||
# later down the line).
|
# later down the line).
|
||||||
|
|
||||||
devel-go:
|
devel-go:
|
||||||
go get .
|
dep ensure
|
||||||
go build -i -v -o $(GO_DEVEL_BUILD_BINARY)
|
go build -i -v -o $(GO_DEVEL_BUILD_BINARY)
|
||||||
|
|
||||||
prod-go:
|
prod-go:
|
||||||
go get .
|
dep ensure
|
||||||
go build -i -v -o $(GO_PROD_BUILD_BINARY)
|
go build -i -v -o $(GO_PROD_BUILD_BINARY)
|
||||||
|
|
||||||
test-go:
|
test-go:
|
||||||
go get .
|
dep ensure
|
||||||
go test -v .
|
go test -v .
|
||||||
|
|
||||||
$(shell mkdir -p $(GO_DEVEL_BUILD_DIR) $(GO_PROD_BUILD_DIR))
|
$(shell mkdir -p $(GO_DEVEL_BUILD_DIR) $(GO_PROD_BUILD_DIR))
|
||||||
|
|||||||
43
api/comment_count.go
Normal file
43
api/comment_count.go
Normal file
@@ -0,0 +1,43 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"net/http"
|
||||||
|
)
|
||||||
|
|
||||||
|
func commentCount(domain string, path string) (int, error) {
|
||||||
|
// path can be empty
|
||||||
|
if domain == "" {
|
||||||
|
return 0, errorMissingField
|
||||||
|
}
|
||||||
|
|
||||||
|
p, err := pageGet(domain, path)
|
||||||
|
if err != nil {
|
||||||
|
return 0, errorInternal
|
||||||
|
}
|
||||||
|
|
||||||
|
return p.CommentCount, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func commentCountHandler(w http.ResponseWriter, r *http.Request) {
|
||||||
|
type request struct {
|
||||||
|
Domain *string `json:"domain"`
|
||||||
|
Path *string `json:"path"`
|
||||||
|
}
|
||||||
|
|
||||||
|
var x request
|
||||||
|
if err := bodyUnmarshal(r, &x); err != nil {
|
||||||
|
bodyMarshal(w, response{"success": false, "message": err.Error()})
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
domain := domainStrip(*x.Domain)
|
||||||
|
path := *x.Path
|
||||||
|
|
||||||
|
count, err := commentCount(domain, path)
|
||||||
|
if err != nil {
|
||||||
|
bodyMarshal(w, response{"success": false, "message": err.Error()})
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
bodyMarshal(w, response{"success": true, "count": count})
|
||||||
|
}
|
||||||
54
api/comment_count_test.go
Normal file
54
api/comment_count_test.go
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestCommentCountBasics(t *testing.T) {
|
||||||
|
failTestOnError(t, setupTestEnv())
|
||||||
|
|
||||||
|
commenterHex, _ := commenterNew("test@example.com", "Test", "undefined", "http://example.com/photo.jpg", "google", "")
|
||||||
|
|
||||||
|
commentNew(commenterHex, "example.com", "/path.html", "root", "**foo**", "approved", time.Now().UTC())
|
||||||
|
commentNew(commenterHex, "example.com", "/path.html", "root", "**bar**", "approved", time.Now().UTC())
|
||||||
|
commentNew(commenterHex, "example.com", "/path.html", "root", "**baz**", "unapproved", time.Now().UTC())
|
||||||
|
|
||||||
|
count, err := commentCount("example.com", "/path.html")
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("unexpected error counting comments: %v", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if count != 2 {
|
||||||
|
t.Errorf("expected count=2 got count=%d", count)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestCommentCountNewPage(t *testing.T) {
|
||||||
|
failTestOnError(t, setupTestEnv())
|
||||||
|
|
||||||
|
count, err := commentCount("example.com", "/path.html")
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("unexpected error counting comments: %v", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if count != 0 {
|
||||||
|
t.Errorf("expected count=0 got count=%d", count)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestCommentCountEmpty(t *testing.T) {
|
||||||
|
if _, err := commentCount("example.com", ""); err != nil {
|
||||||
|
t.Errorf("unexpected error counting comments on empty path: %v", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, err := commentCount("", ""); err == nil {
|
||||||
|
t.Errorf("expected error not found counting comments with empty everything")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -6,7 +6,8 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
func commentList(commenterHex string, domain string, path string, includeUnapproved bool) ([]comment, map[string]commenter, error) {
|
func commentList(commenterHex string, domain string, path string, includeUnapproved bool) ([]comment, map[string]commenter, error) {
|
||||||
if commenterHex == "" || domain == "" || path == "" {
|
// path can be empty
|
||||||
|
if commenterHex == "" || domain == "" {
|
||||||
return nil, nil, errorMissingField
|
return nil, nil, errorMissingField
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -111,6 +112,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 +158,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,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -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
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
package main
|
package main
|
||||||
|
|
||||||
var edition = "ce"
|
var edition = "ce"
|
||||||
var version = "v1.0.0"
|
var version = "v1.3.1"
|
||||||
|
|||||||
@@ -60,6 +60,16 @@ func domainDelete(domain string) error {
|
|||||||
return errorInternal
|
return errorInternal
|
||||||
}
|
}
|
||||||
|
|
||||||
|
statement = `
|
||||||
|
DELETE FROM pages
|
||||||
|
WHERE pages.domain = $1;
|
||||||
|
`
|
||||||
|
_, err = db.Exec(statement, domain)
|
||||||
|
if err != nil {
|
||||||
|
logger.Errorf(statement, domain)
|
||||||
|
return errorInternal
|
||||||
|
}
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -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.")
|
||||||
|
|||||||
@@ -31,8 +31,8 @@ func ownerGetByOwnerToken(ownerToken string) (owner, error) {
|
|||||||
statement := `
|
statement := `
|
||||||
SELECT ownerHex, email, name, confirmedEmail, joinDate
|
SELECT ownerHex, email, name, confirmedEmail, joinDate
|
||||||
FROM owners
|
FROM owners
|
||||||
WHERE email IN (
|
WHERE ownerHex IN (
|
||||||
SELECT email FROM ownerSessions
|
SELECT ownerHex FROM ownerSessions
|
||||||
WHERE ownerToken = $1
|
WHERE ownerToken = $1
|
||||||
);
|
);
|
||||||
`
|
`
|
||||||
|
|||||||
10
api/page.go
Normal file
10
api/page.go
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import ()
|
||||||
|
|
||||||
|
type page struct {
|
||||||
|
Domain string `json:"domain"`
|
||||||
|
Path string `json:"path"`
|
||||||
|
IsLocked bool `json:"isLocked"`
|
||||||
|
CommentCount int `json:"commentCount"`
|
||||||
|
}
|
||||||
35
api/page_get.go
Normal file
35
api/page_get.go
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
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, commentCount
|
||||||
|
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, &p.CommentCount); 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
|
||||||
|
p.CommentCount = 0
|
||||||
|
} else {
|
||||||
|
logger.Errorf("error scanning page: %v", err)
|
||||||
|
return page{}, errorInternal
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return p, nil
|
||||||
|
}
|
||||||
43
api/page_get_test.go
Normal file
43
api/page_get_test.go
Normal 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
24
api/page_new.go
Normal 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
43
api/page_new_test.go
Normal 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
|
||||||
|
}
|
||||||
|
}
|
||||||
72
api/page_update.go
Normal file
72
api/page_update.go
Normal file
@@ -0,0 +1,72 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"net/http"
|
||||||
|
)
|
||||||
|
|
||||||
|
func pageUpdate(p page) error {
|
||||||
|
if p.Domain == "" {
|
||||||
|
return errorMissingField
|
||||||
|
}
|
||||||
|
|
||||||
|
// fields to not update:
|
||||||
|
// commentCount
|
||||||
|
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
43
api/page_update_test.go
Normal 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
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -31,9 +31,12 @@ func apiRouterInit(router *mux.Router) error {
|
|||||||
|
|
||||||
router.HandleFunc("/api/comment/new", commentNewHandler).Methods("POST")
|
router.HandleFunc("/api/comment/new", commentNewHandler).Methods("POST")
|
||||||
router.HandleFunc("/api/comment/list", commentListHandler).Methods("POST")
|
router.HandleFunc("/api/comment/list", commentListHandler).Methods("POST")
|
||||||
|
router.HandleFunc("/api/comment/count", commentCountHandler).Methods("POST")
|
||||||
router.HandleFunc("/api/comment/vote", commentVoteHandler).Methods("POST")
|
router.HandleFunc("/api/comment/vote", commentVoteHandler).Methods("POST")
|
||||||
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
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -27,6 +27,8 @@ type staticHtmlPlugs struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func staticRouterInit(router *mux.Router) error {
|
func staticRouterInit(router *mux.Router) error {
|
||||||
|
subdir := pathStrip(os.Getenv("ORIGIN"))
|
||||||
|
|
||||||
asset := make(map[string][]byte)
|
asset := make(map[string][]byte)
|
||||||
gzippedAsset := make(map[string][]byte)
|
gzippedAsset := make(map[string][]byte)
|
||||||
|
|
||||||
@@ -57,8 +59,6 @@ func staticRouterInit(router *mux.Router) error {
|
|||||||
|
|
||||||
gzip := (os.Getenv("GZIP_STATIC") == "true")
|
gzip := (os.Getenv("GZIP_STATIC") == "true")
|
||||||
|
|
||||||
subdir := pathStrip(os.Getenv("ORIGIN"))
|
|
||||||
|
|
||||||
asset[subdir+p] = []byte(prefix + string(contents))
|
asset[subdir+p] = []byte(prefix + string(contents))
|
||||||
if gzip {
|
if gzip {
|
||||||
gzippedAsset[subdir+p], err = gzipStatic(asset[subdir+p])
|
gzippedAsset[subdir+p], err = gzipStatic(asset[subdir+p])
|
||||||
@@ -101,6 +101,10 @@ func staticRouterInit(router *mux.Router) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
html := make(map[string]string)
|
html := make(map[string]string)
|
||||||
|
for _, page := range pages {
|
||||||
|
html[subdir+page] = ""
|
||||||
|
}
|
||||||
|
|
||||||
for _, page := range pages {
|
for _, page := range pages {
|
||||||
sl := string(os.PathSeparator)
|
sl := string(os.PathSeparator)
|
||||||
page = sl + page
|
page = sl + page
|
||||||
@@ -112,22 +116,30 @@ func staticRouterInit(router *mux.Router) error {
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
t, err := template.New(page).Delims("[[[", "]]]").Parse(string(contents))
|
result := string(contents)
|
||||||
if err != nil {
|
|
||||||
logger.Errorf("cannot parse %s%s template: %v", os.Getenv("STATIC"), file, err)
|
for {
|
||||||
return err
|
t, err := template.New(page).Delims("[[[", "]]]").Parse(result)
|
||||||
|
if err != nil {
|
||||||
|
logger.Errorf("cannot parse %s%s template: %v", os.Getenv("STATIC"), file, err)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
var buf bytes.Buffer
|
||||||
|
t.Execute(&buf, &staticHtmlPlugs{
|
||||||
|
Origin: os.Getenv("ORIGIN"),
|
||||||
|
CdnPrefix: os.Getenv("CDN_PREFIX"),
|
||||||
|
Footer: template.HTML(string(footer)),
|
||||||
|
})
|
||||||
|
|
||||||
|
result = buf.String()
|
||||||
|
if result == html[subdir+page] {
|
||||||
|
break
|
||||||
|
} else {
|
||||||
|
html[subdir+page] = result
|
||||||
|
continue
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
var buf bytes.Buffer
|
|
||||||
t.Execute(&buf, &staticHtmlPlugs{
|
|
||||||
Origin: os.Getenv("ORIGIN"),
|
|
||||||
CdnPrefix: os.Getenv("CDN_PREFIX"),
|
|
||||||
Footer: template.HTML(string(footer)),
|
|
||||||
})
|
|
||||||
|
|
||||||
subdir := pathStrip(os.Getenv("ORIGIN"))
|
|
||||||
|
|
||||||
html[subdir+page] = buf.String()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, page := range pages {
|
for _, page := range pages {
|
||||||
|
|||||||
9
db/20180922181651-page-attributes.sql
Normal file
9
db/20180922181651-page-attributes.sql
Normal 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);
|
||||||
15
db/20180923002745-comment-count.sql
Normal file
15
db/20180923002745-comment-count.sql
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
ALTER TABLE pages
|
||||||
|
ADD commentCount INTEGER NOT NULL DEFAULT 0;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION commentsInsertTriggerFunction() RETURNS TRIGGER AS $trigger$
|
||||||
|
BEGIN
|
||||||
|
UPDATE pages
|
||||||
|
SET commentCount = commentCount + 1
|
||||||
|
WHERE domain = new.domain AND path = new.path;
|
||||||
|
|
||||||
|
RETURN NEW;
|
||||||
|
END;
|
||||||
|
$trigger$ LANGUAGE plpgsql;
|
||||||
|
|
||||||
|
CREATE TRIGGER commentsInsertTrigger AFTER INSERT ON comments
|
||||||
|
FOR EACH ROW EXECUTE PROCEDURE commentsInsertTriggerFunction();
|
||||||
10
db/20180923004309-comment-count-build.sql
Normal file
10
db/20180923004309-comment-count-build.sql
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
-- Build the comments count column
|
||||||
|
|
||||||
|
UPDATE pages
|
||||||
|
SET commentCount = subquery.commentCount
|
||||||
|
FROM (
|
||||||
|
SELECT COUNT(commentHex) as commentCount
|
||||||
|
FROM comments
|
||||||
|
WHERE state = 'approved'
|
||||||
|
GROUP BY (domain, path)
|
||||||
|
) as subquery;
|
||||||
7
db/20181007230906-store-version.sql
Normal file
7
db/20181007230906-store-version.sql
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
CREATE TABLE IF NOT EXISTS config (
|
||||||
|
version TEXT NOT NULL
|
||||||
|
);
|
||||||
|
|
||||||
|
INSERT INTO
|
||||||
|
config (version)
|
||||||
|
VALUES ('v1.1.3');
|
||||||
2
db/20181007231407-v1.1.4.sql
Normal file
2
db/20181007231407-v1.1.4.sql
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
UPDATE config
|
||||||
|
SET version = 'v1.2.0';
|
||||||
@@ -175,12 +175,12 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- General Settings -->
|
<!-- Configure Domain -->
|
||||||
<div id="general-view" class="view hidden">
|
<div id="general-view" class="view hidden">
|
||||||
<div class="view-inside">
|
<div class="view-inside">
|
||||||
<div class="small-mid-view">
|
<div class="small-mid-view">
|
||||||
<div class="center center-title">
|
<div class="center center-title">
|
||||||
General Settings
|
Configure Domain
|
||||||
</div>
|
</div>
|
||||||
<div class="box">
|
<div class="box">
|
||||||
<div class="row">
|
<div class="row">
|
||||||
@@ -365,7 +365,7 @@
|
|||||||
</div>
|
</div>
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<div class="label">Website Name</div>
|
<div class="label">Website Name</div>
|
||||||
<input class="input gray-input" id="new-domain-name" type="text" placeholder="Billie Joe's Blog">
|
<input class="input gray-input" id="new-domain-name" type="text" placeholder="My Blog">
|
||||||
</div>
|
</div>
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<div class="label">Website Domain</div>
|
<div class="label">Website Domain</div>
|
||||||
|
|||||||
@@ -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);
|
||||||
|
|
||||||
@@ -462,7 +472,7 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
global.postComment = function(id) {
|
global.commentNew = function(id) {
|
||||||
var textarea = $(ID_TEXTAREA + id);
|
var textarea = $(ID_TEXTAREA + id);
|
||||||
|
|
||||||
var comment = textarea.value;
|
var comment = textarea.value;
|
||||||
@@ -577,6 +587,10 @@
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
cur.sort(function(a, b) {
|
||||||
|
return b.score - a.score;
|
||||||
|
});
|
||||||
|
|
||||||
var cards = create("div");
|
var cards = create("div");
|
||||||
cur.forEach(function(comment) {
|
cur.forEach(function(comment) {
|
||||||
var commenter = commenters[comment.commenterHex];
|
var commenter = commenters[comment.commenterHex];
|
||||||
@@ -590,8 +604,8 @@
|
|||||||
var edit = create("button");
|
var edit = create("button");
|
||||||
var reply = create("button");
|
var reply = create("button");
|
||||||
var collapse = create("button");
|
var collapse = create("button");
|
||||||
var upvote = create("div");
|
var upvote = create("button");
|
||||||
var downvote = create("div");
|
var downvote = create("button");
|
||||||
var approve = create("button");
|
var approve = create("button");
|
||||||
var remove = create("button");
|
var remove = create("button");
|
||||||
var children = commentsRecurse(parentMap, comment.commentHex);
|
var children = commentsRecurse(parentMap, comment.commentHex);
|
||||||
@@ -715,7 +729,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")
|
||||||
@@ -940,7 +957,7 @@
|
|||||||
classAdd(submit, "submit-button");
|
classAdd(submit, "submit-button");
|
||||||
classAdd(el, "button-margin");
|
classAdd(el, "button-margin");
|
||||||
|
|
||||||
attrSet(submit, "onclick", "postComment('" + id + "')");
|
attrSet(submit, "onclick", "commentNew('" + id + "')");
|
||||||
|
|
||||||
append(el, submit);
|
append(el, submit);
|
||||||
}
|
}
|
||||||
@@ -1037,6 +1054,7 @@
|
|||||||
classAdd(oauthButtonsContainer, "oauth-buttons-container");
|
classAdd(oauthButtonsContainer, "oauth-buttons-container");
|
||||||
classAdd(oauthButtons, "oauth-buttons");
|
classAdd(oauthButtons, "oauth-buttons");
|
||||||
classAdd(close, "login-box-close");
|
classAdd(close, "login-box-close");
|
||||||
|
classAdd(root, "root-min-height");
|
||||||
|
|
||||||
emailButton.innerText = "Continue";
|
emailButton.innerText = "Continue";
|
||||||
loginLink.innerText = "Already have an account? Log in.";
|
loginLink.innerText = "Already have an account? Log in.";
|
||||||
@@ -1257,6 +1275,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 +1327,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 +1360,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();
|
||||||
@@ -1297,6 +1383,7 @@
|
|||||||
var loginBoxContainer = $(ID_LOGIN_BOX_CONTAINER);
|
var loginBoxContainer = $(ID_LOGIN_BOX_CONTAINER);
|
||||||
|
|
||||||
classRemove(mainArea, "blurred");
|
classRemove(mainArea, "blurred");
|
||||||
|
classRemove(root, "root-min-height");
|
||||||
|
|
||||||
attrSet(loginBoxContainer, "style", "display: none");
|
attrSet(loginBoxContainer, "style", "display: none");
|
||||||
}
|
}
|
||||||
@@ -1346,6 +1433,7 @@
|
|||||||
|
|
||||||
selfGet(function() {
|
selfGet(function() {
|
||||||
commentsGet(function() {
|
commentsGet(function() {
|
||||||
|
modToolsCreate();
|
||||||
rootCreate(function() {
|
rootCreate(function() {
|
||||||
commentsRender();
|
commentsRender();
|
||||||
footerLoad();
|
footerLoad();
|
||||||
|
|||||||
@@ -27,7 +27,7 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"id": "general",
|
"id": "general",
|
||||||
"text": "General Settings",
|
"text": "Configure Domain",
|
||||||
"meaning": "Names, domains and the rest",
|
"meaning": "Names, domains and the rest",
|
||||||
"selected": false,
|
"selected": false,
|
||||||
"open": generalOpen,
|
"open": generalOpen,
|
||||||
@@ -35,13 +35,13 @@
|
|||||||
{
|
{
|
||||||
"id": "moderation",
|
"id": "moderation",
|
||||||
"text": "Moderation Settings",
|
"text": "Moderation Settings",
|
||||||
"meaning": "Approve and delete comments",
|
"meaning": "Manage list of moderators",
|
||||||
"selected": false,
|
"selected": false,
|
||||||
"open": moderationOpen,
|
"open": moderationOpen,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"id": "statistics",
|
"id": "statistics",
|
||||||
"text": "Statistics",
|
"text": "View Activity",
|
||||||
"meaning": "Usage and comment statistics",
|
"meaning": "Usage and comment statistics",
|
||||||
"selected": false,
|
"selected": false,
|
||||||
"open": statisticsOpen,
|
"open": statisticsOpen,
|
||||||
|
|||||||
@@ -71,7 +71,11 @@
|
|||||||
date.setTime(date.getTime() + (365*24*60*60*1000));
|
date.setTime(date.getTime() + (365*24*60*60*1000));
|
||||||
expires = "; expires=" + date.toUTCString();
|
expires = "; expires=" + date.toUTCString();
|
||||||
|
|
||||||
document.cookie = name + "=" + value + expires + "; path=/";
|
var cookieString = name + "=" + value + expires + "; path=/";
|
||||||
|
if (/^https:\/\//i.test(commentoOrigin))
|
||||||
|
cookieString += "; secure";
|
||||||
|
|
||||||
|
document.cookie = cookieString;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -152,7 +152,7 @@ textarea {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.commento-create-button {
|
.commento-create-button {
|
||||||
width: 120px;
|
width: 150px;
|
||||||
background: $pink-9;
|
background: $pink-9;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -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;
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,5 +1,9 @@
|
|||||||
@import url('https://fonts.googleapis.com/css?family=Source+Sans+Pro');
|
@import url('https://fonts.googleapis.com/css?family=Source+Sans+Pro');
|
||||||
|
|
||||||
|
.commento-root-min-height {
|
||||||
|
min-height: 350px;
|
||||||
|
}
|
||||||
|
|
||||||
.commento-root {
|
.commento-root {
|
||||||
font-family: "Source Sans Pro", "Segoe UI", Roboto, "Helvetica Neue", sans-serif;
|
font-family: "Source Sans Pro", "Segoe UI", Roboto, "Helvetica Neue", sans-serif;
|
||||||
font-size: 15px;
|
font-size: 15px;
|
||||||
@@ -8,7 +12,6 @@
|
|||||||
overflow-x: hidden;
|
overflow-x: hidden;
|
||||||
text-rendering: optimizeLegibility;
|
text-rendering: optimizeLegibility;
|
||||||
padding: 8px;
|
padding: 8px;
|
||||||
min-height: 350px;
|
|
||||||
|
|
||||||
@import "colors-main.scss";
|
@import "colors-main.scss";
|
||||||
@import "common-main.scss";
|
@import "common-main.scss";
|
||||||
@@ -41,6 +44,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 +102,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 {
|
||||||
|
|||||||
@@ -21,7 +21,7 @@
|
|||||||
outline: none;
|
outline: none;
|
||||||
padding: 5px;
|
padding: 5px;
|
||||||
padding-left: 10px;
|
padding-left: 10px;
|
||||||
width: calc(100% - 120px);
|
width: calc(100% - 150px);
|
||||||
}
|
}
|
||||||
|
|
||||||
.commento-input::placeholder {
|
.commento-input::placeholder {
|
||||||
|
|||||||
@@ -34,7 +34,7 @@
|
|||||||
|
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<div class="label">Full Name</div>
|
<div class="label">Full Name</div>
|
||||||
<input class="input" type="text" name="name" id="name" placeholder="Billie Joe Armstrong">
|
<input class="input" type="text" name="name" id="name" placeholder="Full Name">
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="row">
|
<div class="row">
|
||||||
|
|||||||
Reference in New Issue
Block a user