2017-07-26 20:03:23 +01:00
|
|
|
// Copyright 2016 Documize Inc. <legal@documize.com>. All rights reserved.
|
|
|
|
//
|
|
|
|
// This software (Documize Community Edition) is licensed under
|
|
|
|
// GNU AGPL v3 http://www.gnu.org/licenses/agpl-3.0.en.html
|
|
|
|
//
|
|
|
|
// You can operate outside the AGPL restrictions by purchasing
|
|
|
|
// Documize Enterprise Edition and obtaining a commercial license
|
|
|
|
// by contacting <sales@documize.com>.
|
|
|
|
//
|
|
|
|
// https://documize.com
|
|
|
|
|
|
|
|
package link
|
|
|
|
|
|
|
|
import (
|
|
|
|
"database/sql"
|
2019-04-10 20:03:34 +01:00
|
|
|
"fmt"
|
2017-07-26 20:03:23 +01:00
|
|
|
"net/http"
|
|
|
|
"net/url"
|
|
|
|
|
2019-04-10 20:03:34 +01:00
|
|
|
"github.com/documize/community/core/stringutil"
|
|
|
|
|
2017-07-26 20:03:23 +01:00
|
|
|
"github.com/documize/community/core/env"
|
|
|
|
"github.com/documize/community/core/request"
|
|
|
|
"github.com/documize/community/core/response"
|
|
|
|
"github.com/documize/community/core/uniqueid"
|
|
|
|
"github.com/documize/community/domain"
|
2017-09-18 17:53:42 +01:00
|
|
|
"github.com/documize/community/domain/permission"
|
2018-09-27 15:14:48 +01:00
|
|
|
"github.com/documize/community/domain/store"
|
2017-07-26 20:03:23 +01:00
|
|
|
"github.com/documize/community/model/attachment"
|
|
|
|
"github.com/documize/community/model/link"
|
|
|
|
"github.com/documize/community/model/page"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Handler contains the runtime information such as logging and database.
|
|
|
|
type Handler struct {
|
|
|
|
Runtime *env.Runtime
|
2018-09-27 15:14:48 +01:00
|
|
|
Store *store.Store
|
2017-07-26 20:03:23 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// GetLinkCandidates returns references to documents/sections/attachments.
|
|
|
|
func (h *Handler) GetLinkCandidates(w http.ResponseWriter, r *http.Request) {
|
|
|
|
method := "link.Candidates"
|
|
|
|
ctx := domain.GetRequestContext(r)
|
|
|
|
|
2018-10-12 17:54:15 +01:00
|
|
|
spaceID := request.Param(r, "spaceID")
|
|
|
|
if len(spaceID) == 0 {
|
|
|
|
response.WriteMissingDataError(w, method, "spaceID")
|
2017-07-26 20:03:23 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
documentID := request.Param(r, "documentID")
|
|
|
|
if len(documentID) == 0 {
|
|
|
|
response.WriteMissingDataError(w, method, "documentID")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
pageID := request.Param(r, "pageID")
|
|
|
|
if len(pageID) == 0 {
|
|
|
|
response.WriteMissingDataError(w, method, "pageID")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// permission check
|
2017-09-18 17:53:42 +01:00
|
|
|
if !permission.CanViewDocument(ctx, *h.Store, documentID) {
|
2017-07-26 20:03:23 +01:00
|
|
|
response.WriteForbiddenError(w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// We can link to a section within the same document so
|
|
|
|
// let's get all pages for the document and remove "us".
|
|
|
|
pages, err := h.Store.Page.GetPagesWithoutContent(ctx, documentID)
|
2017-08-03 10:00:24 +01:00
|
|
|
if len(pages) == 0 {
|
|
|
|
pages = []page.Page{}
|
|
|
|
}
|
2017-07-26 20:03:23 +01:00
|
|
|
if err != nil && err != sql.ErrNoRows {
|
|
|
|
response.WriteServerError(w, method, err)
|
2017-08-03 10:00:24 +01:00
|
|
|
h.Runtime.Log.Error(method, err)
|
2017-07-26 20:03:23 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
pc := []link.Candidate{}
|
|
|
|
|
|
|
|
for _, p := range pages {
|
|
|
|
if p.RefID != pageID {
|
|
|
|
c := link.Candidate{
|
|
|
|
RefID: uniqueid.Generate(),
|
2018-10-12 17:54:15 +01:00
|
|
|
SpaceID: spaceID,
|
2017-07-26 20:03:23 +01:00
|
|
|
DocumentID: documentID,
|
|
|
|
TargetID: p.RefID,
|
2018-09-19 16:03:29 +01:00
|
|
|
LinkType: p.Type,
|
|
|
|
Title: p.Name,
|
2017-07-26 20:03:23 +01:00
|
|
|
}
|
|
|
|
pc = append(pc, c)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// We can link to attachment within the same document so
|
|
|
|
// let's get all attachments for the document.
|
|
|
|
files, err := h.Store.Attachment.GetAttachments(ctx, documentID)
|
2017-08-03 10:00:24 +01:00
|
|
|
if len(files) == 0 {
|
|
|
|
files = []attachment.Attachment{}
|
|
|
|
}
|
2017-07-26 20:03:23 +01:00
|
|
|
if err != nil && err != sql.ErrNoRows {
|
|
|
|
response.WriteServerError(w, method, err)
|
2017-08-03 10:00:24 +01:00
|
|
|
h.Runtime.Log.Error(method, err)
|
2017-07-26 20:03:23 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
fc := []link.Candidate{}
|
|
|
|
|
|
|
|
for _, f := range files {
|
|
|
|
c := link.Candidate{
|
|
|
|
RefID: uniqueid.Generate(),
|
2018-10-12 17:54:15 +01:00
|
|
|
SpaceID: spaceID,
|
2017-07-26 20:03:23 +01:00
|
|
|
DocumentID: documentID,
|
|
|
|
TargetID: f.RefID,
|
|
|
|
LinkType: "file",
|
|
|
|
Title: f.Filename,
|
|
|
|
Context: f.Extension,
|
|
|
|
}
|
|
|
|
|
|
|
|
fc = append(fc, c)
|
|
|
|
}
|
|
|
|
|
|
|
|
var payload struct {
|
|
|
|
Pages []link.Candidate `json:"pages"`
|
|
|
|
Attachments []link.Candidate `json:"attachments"`
|
|
|
|
}
|
|
|
|
|
|
|
|
payload.Pages = pc
|
|
|
|
payload.Attachments = fc
|
|
|
|
|
|
|
|
response.WriteJSON(w, payload)
|
|
|
|
}
|
|
|
|
|
|
|
|
// SearchLinkCandidates endpoint takes a list of keywords and returns a list of document references matching those keywords.
|
|
|
|
func (h *Handler) SearchLinkCandidates(w http.ResponseWriter, r *http.Request) {
|
|
|
|
method := "link.SearchLinkCandidates"
|
|
|
|
ctx := domain.GetRequestContext(r)
|
|
|
|
|
|
|
|
keywords := request.Query(r, "keywords")
|
|
|
|
decoded, err := url.QueryUnescape(keywords)
|
|
|
|
if err != nil {
|
2017-08-03 10:00:24 +01:00
|
|
|
h.Runtime.Log.Error(method, err)
|
2017-07-26 20:03:23 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
docs, pages, attachments, err := h.Store.Link.SearchCandidates(ctx, decoded)
|
|
|
|
if err != nil {
|
|
|
|
response.WriteServerError(w, method, err)
|
2017-08-03 10:00:24 +01:00
|
|
|
h.Runtime.Log.Error(method, err)
|
2017-07-26 20:03:23 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
var payload struct {
|
|
|
|
Documents []link.Candidate `json:"documents"`
|
|
|
|
Pages []link.Candidate `json:"pages"`
|
|
|
|
Attachments []link.Candidate `json:"attachments"`
|
|
|
|
}
|
|
|
|
|
|
|
|
payload.Documents = docs
|
|
|
|
payload.Pages = pages
|
|
|
|
payload.Attachments = attachments
|
|
|
|
|
|
|
|
response.WriteJSON(w, payload)
|
|
|
|
}
|
2019-04-10 20:03:34 +01:00
|
|
|
|
|
|
|
// GetLink returns link object for given ID.
|
|
|
|
func (h *Handler) GetLink(w http.ResponseWriter, r *http.Request) {
|
|
|
|
method := "link.GetLink"
|
|
|
|
ctx := domain.GetRequestContext(r)
|
|
|
|
|
|
|
|
// Param check.
|
|
|
|
linkID := request.Param(r, "linkID")
|
|
|
|
if len(linkID) == 0 {
|
|
|
|
response.WriteMissingDataError(w, method, "linkID")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Load link record.
|
|
|
|
link, err := h.Store.Link.GetLink(ctx, linkID)
|
|
|
|
if err != nil {
|
|
|
|
response.WriteServerError(w, method, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check document permissions.
|
|
|
|
if !permission.CanViewDocument(ctx, *h.Store, link.SourceDocumentID) {
|
|
|
|
response.WriteForbiddenError(w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Build URL for link
|
|
|
|
url := ""
|
|
|
|
|
|
|
|
// Jump-to-document link type.
|
|
|
|
if link.LinkType == "document" {
|
|
|
|
doc, err := h.Store.Document.Get(ctx, link.TargetDocumentID)
|
|
|
|
if err != nil {
|
|
|
|
response.WriteString(w, url)
|
|
|
|
}
|
|
|
|
url = ctx.GetAppURL(fmt.Sprintf("s/%s/%s/d/%s/%s",
|
|
|
|
doc.SpaceID, doc.SpaceID, doc.RefID, stringutil.MakeSlug(doc.Name)))
|
|
|
|
}
|
|
|
|
|
|
|
|
// Jump-to-section link type.
|
|
|
|
if link.LinkType == "section" || link.LinkType == "tab" {
|
|
|
|
doc, err := h.Store.Document.Get(ctx, link.TargetDocumentID)
|
|
|
|
if err != nil {
|
|
|
|
response.WriteString(w, url)
|
|
|
|
}
|
|
|
|
url = ctx.GetAppURL(fmt.Sprintf("s/%s/%s/d/%s/%s?currentPageId=%s",
|
|
|
|
doc.SpaceID, doc.SpaceID, doc.RefID,
|
|
|
|
stringutil.MakeSlug(doc.Name), link.TargetID))
|
|
|
|
}
|
|
|
|
|
|
|
|
response.WriteString(w, url)
|
|
|
|
}
|