book_result.go 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584
  1. package models
  2. import (
  3. "bytes"
  4. "io/ioutil"
  5. "os"
  6. "path/filepath"
  7. "strconv"
  8. "strings"
  9. "time"
  10. "encoding/base64"
  11. "github.com/PuerkitoBio/goquery"
  12. "github.com/astaxie/beego"
  13. "github.com/astaxie/beego/logs"
  14. "github.com/astaxie/beego/orm"
  15. "github.com/lifei6671/mindoc/conf"
  16. "github.com/lifei6671/mindoc/converter"
  17. "github.com/lifei6671/mindoc/utils/filetil"
  18. "github.com/lifei6671/mindoc/utils/ziptil"
  19. "gopkg.in/russross/blackfriday.v2"
  20. "regexp"
  21. "github.com/lifei6671/mindoc/utils/cryptil"
  22. "github.com/lifei6671/mindoc/utils/requests"
  23. )
  24. type BookResult struct {
  25. BookId int `json:"book_id"`
  26. BookName string `json:"book_name"`
  27. Identify string `json:"identify"`
  28. OrderIndex int `json:"order_index"`
  29. Description string `json:"description"`
  30. Publisher string `json:"publisher"`
  31. PrivatelyOwned int `json:"privately_owned"`
  32. PrivateToken string `json:"private_token"`
  33. DocCount int `json:"doc_count"`
  34. CommentStatus string `json:"comment_status"`
  35. CommentCount int `json:"comment_count"`
  36. CreateTime time.Time `json:"create_time"`
  37. CreateName string `json:"create_name"`
  38. RealName string `json:"real_name"`
  39. ModifyTime time.Time `json:"modify_time"`
  40. Cover string `json:"cover"`
  41. Theme string `json:"theme"`
  42. Label string `json:"label"`
  43. MemberId int `json:"member_id"`
  44. Editor string `json:"editor"`
  45. AutoRelease bool `json:"auto_release"`
  46. HistoryCount int `json:"history_count"`
  47. RelationshipId int `json:"relationship_id"`
  48. RoleId int `json:"role_id"`
  49. RoleName string `json:"role_name"`
  50. Status int `json:"status"`
  51. IsEnableShare bool `json:"is_enable_share"`
  52. IsUseFirstDocument bool `json:"is_use_first_document"`
  53. LastModifyText string `json:"last_modify_text"`
  54. IsDisplayComment bool `json:"is_display_comment"`
  55. IsDownload bool `json:"is_download"`
  56. }
  57. func NewBookResult() *BookResult {
  58. return &BookResult{}
  59. }
  60. // 根据项目标识查询项目以及指定用户权限的信息.
  61. func (m *BookResult) FindByIdentify(identify string, memberId int) (*BookResult, error) {
  62. if identify == "" || memberId <= 0 {
  63. return m, ErrInvalidParameter
  64. }
  65. o := orm.NewOrm()
  66. book := NewBook()
  67. err := o.QueryTable(book.TableNameWithPrefix()).Filter("identify", identify).One(book)
  68. if err != nil {
  69. return m, err
  70. }
  71. relationship := NewRelationship()
  72. err = o.QueryTable(relationship.TableNameWithPrefix()).Filter("book_id", book.BookId).Filter("member_id", memberId).One(relationship)
  73. if err != nil {
  74. return m, err
  75. }
  76. var relationship2 Relationship
  77. err = o.QueryTable(relationship.TableNameWithPrefix()).Filter("book_id", book.BookId).Filter("role_id", 0).One(&relationship2)
  78. if err != nil {
  79. logs.Error("根据项目标识查询项目以及指定用户权限的信息 => ", err)
  80. return m, ErrPermissionDenied
  81. }
  82. member, err := NewMember().Find(relationship2.MemberId)
  83. if err != nil {
  84. return m, err
  85. }
  86. m = NewBookResult().ToBookResult(*book)
  87. m.CreateName = member.Account
  88. if member.RealName != "" {
  89. m.RealName = member.RealName
  90. }
  91. m.MemberId = relationship.MemberId
  92. m.RoleId = relationship.RoleId
  93. m.RelationshipId = relationship.RelationshipId
  94. if m.RoleId == conf.BookFounder {
  95. m.RoleName = "创始人"
  96. } else if m.RoleId == conf.BookAdmin {
  97. m.RoleName = "管理员"
  98. } else if m.RoleId == conf.BookEditor {
  99. m.RoleName = "编辑者"
  100. } else if m.RoleId == conf.BookObserver {
  101. m.RoleName = "观察者"
  102. }
  103. doc := NewDocument()
  104. err = o.QueryTable(doc.TableNameWithPrefix()).Filter("book_id", book.BookId).OrderBy("modify_time").One(doc)
  105. if err == nil {
  106. member2 := NewMember()
  107. member2.Find(doc.ModifyAt)
  108. m.LastModifyText = member2.Account + " 于 " + doc.ModifyTime.Local().Format("2006-01-02 15:04:05")
  109. }
  110. return m, nil
  111. }
  112. func (m *BookResult) FindToPager(pageIndex, pageSize int) (books []*BookResult, totalCount int, err error) {
  113. o := orm.NewOrm()
  114. count, err := o.QueryTable(NewBook().TableNameWithPrefix()).Count()
  115. if err != nil {
  116. return
  117. }
  118. totalCount = int(count)
  119. sql := `SELECT
  120. book.*,rel.relationship_id,rel.role_id,m.account AS create_name,m.real_name
  121. FROM md_books AS book
  122. LEFT JOIN md_relationship AS rel ON rel.book_id = book.book_id AND rel.role_id = 0
  123. LEFT JOIN md_members AS m ON rel.member_id = m.member_id
  124. ORDER BY book.order_index DESC ,book.book_id DESC LIMIT ?,?`
  125. offset := (pageIndex - 1) * pageSize
  126. _, err = o.Raw(sql, offset, pageSize).QueryRows(&books)
  127. return
  128. }
  129. //实体转换
  130. func (m *BookResult) ToBookResult(book Book) *BookResult {
  131. m.BookId = book.BookId
  132. m.BookName = book.BookName
  133. m.Identify = book.Identify
  134. m.OrderIndex = book.OrderIndex
  135. m.Description = strings.Replace(book.Description, "\r\n", "<br/>", -1)
  136. m.PrivatelyOwned = book.PrivatelyOwned
  137. m.PrivateToken = book.PrivateToken
  138. m.DocCount = book.DocCount
  139. m.CommentStatus = book.CommentStatus
  140. m.CommentCount = book.CommentCount
  141. m.CreateTime = book.CreateTime
  142. m.ModifyTime = book.ModifyTime
  143. m.Cover = book.Cover
  144. m.Label = book.Label
  145. m.Status = book.Status
  146. m.Editor = book.Editor
  147. m.Theme = book.Theme
  148. m.AutoRelease = book.AutoRelease == 1
  149. m.IsEnableShare = book.IsEnableShare == 0
  150. m.IsUseFirstDocument = book.IsUseFirstDocument == 1
  151. m.Publisher = book.Publisher
  152. m.HistoryCount = book.HistoryCount
  153. m.IsDownload = book.IsDownload == 0
  154. if book.Theme == "" {
  155. m.Theme = "default"
  156. }
  157. if book.Editor == "" {
  158. m.Editor = "markdown"
  159. }
  160. doc := NewDocument()
  161. o := orm.NewOrm()
  162. err := o.QueryTable(doc.TableNameWithPrefix()).Filter("book_id", book.BookId).OrderBy("modify_time").One(doc)
  163. if err == nil {
  164. member2 := NewMember()
  165. member2.Find(doc.ModifyAt)
  166. m.LastModifyText = member2.Account + " 于 " + doc.ModifyTime.Local().Format("2006-01-02 15:04:05")
  167. }
  168. return m
  169. }
  170. //导出PDF、word等格式
  171. func (m *BookResult) Converter(sessionId string) (ConvertBookResult, error) {
  172. convertBookResult := ConvertBookResult{}
  173. outputPath := filepath.Join(conf.WorkingDirectory, "uploads", "books", strconv.Itoa(m.BookId))
  174. viewPath := beego.BConfig.WebConfig.ViewsPath
  175. pdfpath := filepath.Join(outputPath, "book.pdf")
  176. epubpath := filepath.Join(outputPath, "book.epub")
  177. mobipath := filepath.Join(outputPath, "book.mobi")
  178. docxpath := filepath.Join(outputPath, "book.docx")
  179. //先将转换的文件储存到临时目录
  180. tempOutputPath := filepath.Join(os.TempDir(), sessionId, m.Identify) //filepath.Abs(filepath.Join("cache", sessionId))
  181. os.MkdirAll(outputPath, 0766)
  182. os.MkdirAll(tempOutputPath, 0766)
  183. defer func(p string) {
  184. os.RemoveAll(p)
  185. }(tempOutputPath)
  186. if filetil.FileExists(pdfpath) && filetil.FileExists(epubpath) && filetil.FileExists(mobipath) && filetil.FileExists(docxpath) {
  187. convertBookResult.EpubPath = epubpath
  188. convertBookResult.MobiPath = mobipath
  189. convertBookResult.PDFPath = pdfpath
  190. convertBookResult.WordPath = docxpath
  191. return convertBookResult, nil
  192. }
  193. docs, err := NewDocument().FindListByBookId(m.BookId)
  194. if err != nil {
  195. return convertBookResult, err
  196. }
  197. tocList := make([]converter.Toc, 0)
  198. for _, item := range docs {
  199. if item.ParentId == 0 {
  200. toc := converter.Toc{
  201. Id: item.DocumentId,
  202. Link: strconv.Itoa(item.DocumentId) + ".html",
  203. Pid: item.ParentId,
  204. Title: item.DocumentName,
  205. }
  206. tocList = append(tocList, toc)
  207. }
  208. }
  209. for _, item := range docs {
  210. if item.ParentId != 0 {
  211. toc := converter.Toc{
  212. Id: item.DocumentId,
  213. Link: strconv.Itoa(item.DocumentId) + ".html",
  214. Pid: item.ParentId,
  215. Title: item.DocumentName,
  216. }
  217. tocList = append(tocList, toc)
  218. }
  219. }
  220. ebookConfig := converter.Config{
  221. Charset: "utf-8",
  222. Cover: m.Cover,
  223. Timestamp: time.Now().Format("2006-01-02 15:04:05"),
  224. Description: string(blackfriday.Run([]byte(m.Description))),
  225. Footer: "<p style='color:#8E8E8E;font-size:12px;'>本文档使用 <a href='https://www.iminho.me' style='text-decoration:none;color:#1abc9c;font-weight:bold;'>MinDoc</a> 构建 <span style='float:right'>- _PAGENUM_ -</span></p>",
  226. Header: "<p style='color:#8E8E8E;font-size:12px;'>_SECTION_</p>",
  227. Identifier: "",
  228. Language: "zh-CN",
  229. Creator: m.CreateName,
  230. Publisher: m.Publisher,
  231. Contributor: m.Publisher,
  232. Title: m.BookName,
  233. Format: []string{"epub", "mobi", "pdf", "docx"},
  234. FontSize: "14",
  235. PaperSize: "a4",
  236. MarginLeft: "72",
  237. MarginRight: "72",
  238. MarginTop: "72",
  239. MarginBottom: "72",
  240. Toc: tocList,
  241. More: []string{},
  242. }
  243. if m.Publisher != "" {
  244. ebookConfig.Footer = "<p style='color:#8E8E8E;font-size:12px;'>本文档由 <span style='text-decoration:none;color:#1abc9c;font-weight:bold;'>" + m.Publisher + "</span> 生成<span style='float:right'>- _PAGENUM_ -</span></p>"
  245. }
  246. if m.RealName != "" {
  247. ebookConfig.Creator = m.RealName
  248. }
  249. if tempOutputPath, err = filepath.Abs(tempOutputPath); err != nil {
  250. beego.Error("导出目录配置错误:" + err.Error())
  251. return convertBookResult, err
  252. }
  253. for _, item := range docs {
  254. name := strconv.Itoa(item.DocumentId)
  255. fpath := filepath.Join(tempOutputPath, name+".html")
  256. f, err := os.OpenFile(fpath, os.O_CREATE|os.O_RDWR, 0777)
  257. if err != nil {
  258. return convertBookResult, err
  259. }
  260. var buf bytes.Buffer
  261. if err := beego.ExecuteViewPathTemplate(&buf, "document/export.tpl", viewPath, map[string]interface{}{"Model": m, "Lists": item, "BaseUrl": conf.BaseUrl}); err != nil {
  262. return convertBookResult, err
  263. }
  264. html := buf.String()
  265. if err != nil {
  266. f.Close()
  267. return convertBookResult, err
  268. }
  269. bufio := bytes.NewReader(buf.Bytes())
  270. doc, err := goquery.NewDocumentFromReader(bufio)
  271. doc.Find("img").Each(func(i int, contentSelection *goquery.Selection) {
  272. if src, ok := contentSelection.Attr("src"); ok && strings.HasPrefix(src, "/") {
  273. //contentSelection.SetAttr("src", baseUrl + src)
  274. spath := filepath.Join(conf.WorkingDirectory, src)
  275. if ff, e := ioutil.ReadFile(spath); e == nil {
  276. encodeString := base64.StdEncoding.EncodeToString(ff)
  277. src = "data:image/" + filepath.Ext(src) + ";base64," + encodeString
  278. contentSelection.SetAttr("src", src)
  279. }
  280. }
  281. })
  282. html, err = doc.Html()
  283. if err != nil {
  284. f.Close()
  285. return convertBookResult, err
  286. }
  287. f.WriteString(html)
  288. f.Close()
  289. }
  290. filetil.CopyFile(filepath.Join(conf.WorkingDirectory, "static", "css", "kancloud.css"), filepath.Join(tempOutputPath, "styles", "css", "kancloud.css"))
  291. filetil.CopyFile(filepath.Join(conf.WorkingDirectory, "static", "css", "export.css"), filepath.Join(tempOutputPath, "styles", "css", "export.css"))
  292. filetil.CopyFile(filepath.Join(conf.WorkingDirectory, "static", "editor.md", "css", "editormd.preview.css"), filepath.Join(tempOutputPath, "styles", "editor.md", "css", "editormd.preview.css"))
  293. filetil.CopyFile(filepath.Join(conf.WorkingDirectory, "static", "prettify", "themes", "prettify.css"), filepath.Join(tempOutputPath, "styles", "prettify", "themes", "prettify.css"))
  294. filetil.CopyFile(filepath.Join(conf.WorkingDirectory, "static", "css,", "markdown.preview.css"), filepath.Join(tempOutputPath, "styles", "css", "markdown.preview.css"))
  295. filetil.CopyFile(filepath.Join(conf.WorkingDirectory, "static", "highlight", "styles", "vs.css"), filepath.Join(tempOutputPath, "styles", "highlight", "styles", "vs.css"))
  296. filetil.CopyFile(filepath.Join(conf.WorkingDirectory, "static", "katex", "katex.min.css"), filepath.Join(tempOutputPath, "styles", "katex", "katex.min.css"))
  297. eBookConverter := &converter.Converter{
  298. BasePath: tempOutputPath,
  299. OutputPath: strings.TrimSuffix(tempOutputPath, "sources"),
  300. Config: ebookConfig,
  301. Debug: true,
  302. }
  303. if err := eBookConverter.Convert(); err != nil {
  304. beego.Error("转换文件错误:" + m.BookName + " => " + err.Error())
  305. return convertBookResult, err
  306. }
  307. beego.Info("文档转换完成:" + m.BookName)
  308. filetil.CopyFile(mobipath, filepath.Join(tempOutputPath, "output", "book.mobi"))
  309. filetil.CopyFile(pdfpath, filepath.Join(tempOutputPath, "output", "book.pdf"))
  310. filetil.CopyFile(epubpath, filepath.Join(tempOutputPath, "output", "book.epub"))
  311. filetil.CopyFile(docxpath, filepath.Join(tempOutputPath, "output", "book.docx"))
  312. convertBookResult.MobiPath = mobipath
  313. convertBookResult.PDFPath = pdfpath
  314. convertBookResult.EpubPath = epubpath
  315. convertBookResult.WordPath = docxpath
  316. return convertBookResult, nil
  317. }
  318. //导出Markdown原始文件
  319. func (m *BookResult) ExportMarkdown(sessionId string) (string, error) {
  320. outputPath := filepath.Join(conf.WorkingDirectory, "uploads", "books", strconv.Itoa(m.BookId), "book.zip")
  321. os.MkdirAll(filepath.Dir(outputPath), 0644)
  322. tempOutputPath := filepath.Join(os.TempDir(), sessionId, "markdown")
  323. defer os.RemoveAll(tempOutputPath)
  324. bookUrl := conf.URLFor("DocumentController.Index",":key" , m.Identify) + "/"
  325. err := exportMarkdown(tempOutputPath, 0, m.BookId,tempOutputPath,bookUrl)
  326. if err != nil {
  327. return "", err
  328. }
  329. if err := ziptil.Compress(outputPath, tempOutputPath); err != nil {
  330. beego.Error("导出Markdown失败=>", err)
  331. return "", err
  332. }
  333. return outputPath, nil
  334. }
  335. //递归导出Markdown文档
  336. func exportMarkdown(p string, parentId int, bookId int,baseDir string,bookUrl string) error {
  337. o := orm.NewOrm()
  338. var docs []*Document
  339. _, err := o.QueryTable(NewDocument().TableNameWithPrefix()).Filter("book_id", bookId).Filter("parent_id", parentId).All(&docs)
  340. if err != nil {
  341. beego.Error("导出Markdown失败=>", err)
  342. return err
  343. }
  344. for _, doc := range docs {
  345. //获取当前文档的子文档数量,如果数量不为0,则将当前文档命名为READMD.md并设置成目录。
  346. subDocCount, err := o.QueryTable(NewDocument().TableNameWithPrefix()).Filter("parent_id", doc.DocumentId).Count()
  347. if err != nil {
  348. beego.Error("导出Markdown失败=>", err)
  349. return err
  350. }
  351. var docPath string
  352. if subDocCount > 0 {
  353. if doc.Identify != "" {
  354. docPath = filepath.Join(p, doc.Identify, "README.md")
  355. } else {
  356. docPath = filepath.Join(p, strconv.Itoa(doc.DocumentId), "README.md")
  357. }
  358. } else {
  359. if doc.Identify != "" {
  360. if strings.HasSuffix(doc.Identify,".md") || strings.HasSuffix(doc.Identify,".markdown") {
  361. docPath = filepath.Join(p, doc.Identify)
  362. }else {
  363. docPath = filepath.Join(p, doc.Identify+".md")
  364. }
  365. } else {
  366. docPath = filepath.Join(p, strings.TrimSpace(doc.DocumentName)+".md")
  367. }
  368. }
  369. dirPath := filepath.Dir(docPath)
  370. os.MkdirAll(dirPath, 0766)
  371. markdown := doc.Markdown
  372. //如果当前文档不为空
  373. if strings.TrimSpace(doc.Markdown) != "" {
  374. re := regexp.MustCompile(`!\[(.*?)\]\((.*?)\)`)
  375. //处理文档中图片
  376. markdown = re.ReplaceAllStringFunc(doc.Markdown, func(image string) string {
  377. images := re.FindAllSubmatch([]byte(image), -1)
  378. if len(images) <= 0 || len(images[0]) < 3 {
  379. return image
  380. }
  381. originalImageUrl := string(images[0][2])
  382. imageUrl := strings.Replace(string(originalImageUrl), "\\", "/", -1)
  383. //如果是本地路径,则需要将图片复制到项目目录
  384. if strings.HasPrefix(imageUrl, "http://") || strings.HasPrefix(imageUrl, "https://") {
  385. imageExt := cryptil.Md5Crypt(imageUrl) + filepath.Ext(imageUrl)
  386. dstFile := filepath.Join(baseDir, "uploads", time.Now().Format("200601"), imageExt)
  387. if err := requests.DownloadAndSaveFile(imageUrl, dstFile); err == nil {
  388. imageUrl = strings.TrimPrefix(strings.Replace(dstFile, "\\", "/", -1), strings.Replace(baseDir, "\\", "/", -1))
  389. if !strings.HasPrefix(imageUrl, "/") && !strings.HasPrefix(imageUrl, "\\") {
  390. imageUrl = "/" + imageUrl
  391. }
  392. }
  393. } else if strings.HasPrefix(imageUrl, "/") {
  394. filetil.CopyFile(filepath.Join(conf.WorkingDirectory, imageUrl), filepath.Join(baseDir, imageUrl))
  395. }
  396. imageUrl = strings.Replace(strings.TrimSuffix(image, originalImageUrl+")")+imageUrl+")", "\\", "/", -1)
  397. return imageUrl
  398. })
  399. linkRe := regexp.MustCompile(`\[(.*?)\]\((.*?)\)`)
  400. markdown = linkRe.ReplaceAllStringFunc(markdown, func(link string) string {
  401. links := linkRe.FindAllStringSubmatch(link, -1)
  402. if len(links) > 0 && len(links[0]) >= 3 {
  403. originalLink := links[0][2]
  404. //如果当前链接位于当前项目内
  405. if strings.HasPrefix(originalLink,bookUrl) {
  406. docIdentify := strings.TrimSpace(strings.TrimPrefix(originalLink, bookUrl))
  407. tempDoc := NewDocument()
  408. if id,err := strconv.Atoi(docIdentify);err == nil && id > 0 {
  409. err := o.QueryTable(NewDocument().TableNameWithPrefix()).Filter("document_id",id).One(tempDoc,"identify","parent_id","document_id")
  410. if err != nil {
  411. beego.Error(err)
  412. return link
  413. }
  414. }else{
  415. err := o.QueryTable(NewDocument().TableNameWithPrefix()).Filter("identify",docIdentify).One(tempDoc,"identify","parent_id","document_id")
  416. if err != nil {
  417. beego.Error(err)
  418. return link
  419. }
  420. }
  421. tempLink := recursiveJoinDocumentIdentify(tempDoc.ParentId,"") + strings.TrimPrefix(originalLink, bookUrl)
  422. if !strings.HasSuffix(tempLink,".md") && !strings.HasSuffix(doc.Identify,".markdown") {
  423. tempLink = tempLink + ".md"
  424. }
  425. link = strings.TrimSuffix(link, originalLink+")") + tempLink + ")"
  426. }
  427. }
  428. return link
  429. })
  430. }else{
  431. markdown = "# " + doc.DocumentName + "\n"
  432. }
  433. if err := ioutil.WriteFile(docPath, []byte(markdown), 0644); err != nil {
  434. beego.Error("导出Markdown失败=>", err)
  435. return err
  436. }
  437. if subDocCount > 0 {
  438. if err = exportMarkdown(dirPath, doc.DocumentId, bookId,baseDir,bookUrl); err != nil {
  439. return err
  440. }
  441. }
  442. }
  443. return nil
  444. }
  445. func recursiveJoinDocumentIdentify(parentDocId int,identify string) string {
  446. o := orm.NewOrm()
  447. doc := NewDocument()
  448. err := o.QueryTable(NewDocument().TableNameWithPrefix()).Filter("document_id",parentDocId).One(doc,"identify","parent_id","document_id")
  449. if err != nil {
  450. beego.Error(err)
  451. return identify
  452. }
  453. if doc.Identify == "" {
  454. identify = strconv.Itoa(doc.DocumentId) + "/" + identify
  455. }else{
  456. identify = doc.Identify + "/" + identify
  457. }
  458. if doc.ParentId > 0 {
  459. identify = recursiveJoinDocumentIdentify(doc.ParentId,identify)
  460. }
  461. return identify
  462. }
  463. //查询项目的第一篇文档
  464. func (m *BookResult) FindFirstDocumentByBookId(bookId int) (*Document, error) {
  465. o := orm.NewOrm()
  466. doc := NewDocument()
  467. err := o.QueryTable(doc.TableNameWithPrefix()).Filter("book_id", bookId).Filter("parent_id", 0).OrderBy("order_sort").One(doc)
  468. return doc, err
  469. }