2021-01-01 03:52:25 +00:00
|
|
|
package zig
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"hash/crc32"
|
|
|
|
"unsafe"
|
|
|
|
|
|
|
|
"golang.org/x/text/encoding"
|
|
|
|
"golang.org/x/text/encoding/japanese"
|
|
|
|
)
|
|
|
|
|
|
|
|
/*
|
2021-01-10 19:51:14 +00:00
|
|
|
#cgo linux LDFLAGS: -lz
|
2022-04-23 22:34:14 +00:00
|
|
|
#cgo darwin LDFLAGS: -lz
|
2021-01-10 19:51:14 +00:00
|
|
|
#cgo windows LDFLAGS: -lz -lws2_32
|
2021-01-01 03:52:25 +00:00
|
|
|
#include "zig.h"
|
|
|
|
*/
|
|
|
|
import "C"
|
|
|
|
|
|
|
|
//export hookCallback
|
2021-01-01 04:35:21 +00:00
|
|
|
func hookCallback(book *C.EB_Book, appendix *C.EB_Appendix, container *C.void, hookCode C.EB_Hook_Code, argc C.int, argv *C.uint) C.EB_Error_Code {
|
|
|
|
var marker string
|
|
|
|
switch hookCode {
|
|
|
|
case C.EB_HOOK_NARROW_FONT:
|
|
|
|
marker = fmt.Sprintf("{{n_%d}}", *argv)
|
|
|
|
case C.EB_HOOK_WIDE_FONT:
|
|
|
|
marker = fmt.Sprintf("{{w_%d}}", *argv)
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(marker) > 0 {
|
|
|
|
markerC := C.CString(marker)
|
|
|
|
defer C.free(unsafe.Pointer(markerC))
|
|
|
|
C.eb_write_text_string(book, markerC)
|
|
|
|
}
|
|
|
|
|
|
|
|
return C.EB_SUCCESS
|
2021-01-01 03:52:25 +00:00
|
|
|
}
|
|
|
|
|
2021-01-10 04:49:12 +00:00
|
|
|
func formatError(code C.EB_Error_Code) string {
|
|
|
|
return C.GoString(C.eb_error_string(code))
|
|
|
|
}
|
|
|
|
|
2021-01-01 03:52:25 +00:00
|
|
|
type blockType int
|
|
|
|
|
|
|
|
const (
|
|
|
|
blockTypeHeading blockType = iota
|
|
|
|
blockTypeText
|
|
|
|
)
|
|
|
|
|
|
|
|
type BookEntry struct {
|
|
|
|
Heading string
|
|
|
|
Text string
|
|
|
|
}
|
|
|
|
|
|
|
|
type BookSubbook struct {
|
|
|
|
Title string
|
|
|
|
Copyright string
|
|
|
|
Entries []BookEntry
|
|
|
|
}
|
|
|
|
|
|
|
|
type Book struct {
|
|
|
|
DiscCode string
|
|
|
|
CharCode string
|
|
|
|
Subbooks []BookSubbook
|
|
|
|
}
|
|
|
|
|
|
|
|
type Context struct {
|
|
|
|
buffer []byte
|
|
|
|
decoder *encoding.Decoder
|
|
|
|
hookset *C.EB_Hookset
|
|
|
|
book *C.EB_Book
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Context) initialize() error {
|
|
|
|
if errEb := C.eb_initialize_library(); errEb != C.EB_SUCCESS {
|
2021-01-10 04:49:12 +00:00
|
|
|
return fmt.Errorf("eb_initialize_library failed with code: %s", formatError(errEb))
|
2021-01-01 03:52:25 +00:00
|
|
|
}
|
|
|
|
|
2021-01-10 19:51:14 +00:00
|
|
|
c.book = (*C.EB_Book)(C.calloc(1, C.size_t(unsafe.Sizeof(C.EB_Book{}))))
|
2021-01-01 03:52:25 +00:00
|
|
|
C.eb_initialize_book(c.book)
|
|
|
|
|
2021-01-02 03:12:55 +00:00
|
|
|
c.hookset = (*C.EB_Hookset)(C.calloc(1, C.size_t(unsafe.Sizeof(C.EB_Hookset{}))))
|
2021-01-01 03:52:25 +00:00
|
|
|
C.eb_initialize_hookset(c.hookset)
|
|
|
|
|
|
|
|
if err := c.installHooks(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
c.buffer = make([]byte, 22)
|
|
|
|
c.decoder = japanese.EUCJP.NewDecoder()
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Context) shutdown() {
|
|
|
|
C.eb_finalize_hookset(c.hookset)
|
|
|
|
C.free(unsafe.Pointer(c.hookset))
|
|
|
|
|
|
|
|
C.eb_finalize_book(c.book)
|
|
|
|
C.free(unsafe.Pointer(c.book))
|
|
|
|
|
|
|
|
C.eb_finalize_library()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Context) installHooks() error {
|
|
|
|
hookCodes := []C.EB_Hook_Code{
|
2021-01-01 04:35:21 +00:00
|
|
|
C.EB_HOOK_NARROW_FONT,
|
|
|
|
C.EB_HOOK_WIDE_FONT,
|
2021-01-01 03:52:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for _, hookCode := range hookCodes {
|
|
|
|
if errEb := C.installHook(c.hookset, hookCode); errEb != C.EB_SUCCESS {
|
2021-01-10 04:49:12 +00:00
|
|
|
return fmt.Errorf("eb_set_hook failed with code: %s", formatError(errEb))
|
2021-01-01 03:52:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Context) loadInternal(path string) (*Book, error) {
|
|
|
|
pathC := C.CString(path)
|
|
|
|
defer C.free(unsafe.Pointer(pathC))
|
|
|
|
if errEb := C.eb_bind(c.book, pathC); errEb != C.EB_SUCCESS {
|
2021-01-10 04:49:12 +00:00
|
|
|
return nil, fmt.Errorf("eb_bind failed with code: %s", formatError(errEb))
|
2021-01-01 03:52:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
var (
|
|
|
|
book Book
|
|
|
|
err error
|
|
|
|
)
|
|
|
|
|
|
|
|
if book.CharCode, err = c.loadCharCode(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if book.DiscCode, err = c.loadDiscCode(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if book.Subbooks, err = c.loadSubbooks(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &book, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Context) loadCharCode() (string, error) {
|
|
|
|
var charCode C.EB_Character_Code
|
|
|
|
if errEb := C.eb_character_code(c.book, &charCode); errEb != C.EB_SUCCESS {
|
2021-01-10 04:49:12 +00:00
|
|
|
return "", fmt.Errorf("eb_character_code failed with code: %s", formatError(errEb))
|
2021-01-01 03:52:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
switch charCode {
|
|
|
|
case C.EB_CHARCODE_ISO8859_1:
|
|
|
|
return "iso8859-1", nil
|
|
|
|
case C.EB_CHARCODE_JISX0208:
|
|
|
|
return "jisx0208", nil
|
|
|
|
case C.EB_CHARCODE_JISX0208_GB2312:
|
|
|
|
return "jisx0208/gb2312", nil
|
|
|
|
default:
|
|
|
|
return "invalid", nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Context) loadDiscCode() (string, error) {
|
|
|
|
var discCode C.EB_Disc_Code
|
|
|
|
if errEb := C.eb_disc_type(c.book, &discCode); errEb != C.EB_SUCCESS {
|
2021-01-10 04:49:12 +00:00
|
|
|
return "", fmt.Errorf("eb_disc_type failed with code: %s", formatError(errEb))
|
2021-01-01 03:52:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
switch discCode {
|
|
|
|
case C.EB_DISC_EB:
|
|
|
|
return "eb", nil
|
|
|
|
case C.EB_DISC_EPWING:
|
|
|
|
return "epwing", nil
|
|
|
|
default:
|
|
|
|
return "invalid", nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Context) loadSubbooks() ([]BookSubbook, error) {
|
|
|
|
var (
|
|
|
|
subbookCodes [C.EB_MAX_SUBBOOKS]C.EB_Subbook_Code
|
|
|
|
subbookCount C.int
|
|
|
|
)
|
|
|
|
|
|
|
|
if errEb := C.eb_subbook_list(c.book, &subbookCodes[0], &subbookCount); errEb != C.EB_SUCCESS {
|
2021-01-10 04:49:12 +00:00
|
|
|
return nil, fmt.Errorf("eb_subbook_list failed with code: %s", formatError(errEb))
|
2021-01-01 03:52:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
var subbooks []BookSubbook
|
|
|
|
for i := 0; i < int(subbookCount); i++ {
|
|
|
|
subbook, err := c.loadSubbook(subbookCodes[i])
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
subbooks = append(subbooks, *subbook)
|
|
|
|
}
|
|
|
|
|
|
|
|
return subbooks, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Context) loadSubbook(subbookCode C.EB_Subbook_Code) (*BookSubbook, error) {
|
|
|
|
if errEb := C.eb_set_subbook(c.book, subbookCode); errEb != C.EB_SUCCESS {
|
2021-01-10 04:49:12 +00:00
|
|
|
return nil, fmt.Errorf("eb_set_subbook failed with code: %s", formatError(errEb))
|
2021-01-01 03:52:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
var (
|
|
|
|
subbook BookSubbook
|
|
|
|
err error
|
|
|
|
)
|
|
|
|
|
|
|
|
if subbook.Title, err = c.loadTitle(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if subbook.Copyright, err = c.loadCopyright(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
blocksSeen := make(map[uint32]bool)
|
|
|
|
|
|
|
|
if errEb := C.eb_search_all_alphabet(c.book); errEb == C.EB_SUCCESS {
|
|
|
|
entries, err := c.loadEntries(blocksSeen)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
subbook.Entries = append(subbook.Entries, entries...)
|
|
|
|
}
|
|
|
|
|
|
|
|
if errEb := C.eb_search_all_kana(c.book); errEb == C.EB_SUCCESS {
|
|
|
|
entries, err := c.loadEntries(blocksSeen)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
subbook.Entries = append(subbook.Entries, entries...)
|
|
|
|
}
|
|
|
|
|
|
|
|
if errEb := C.eb_search_all_asis(c.book); errEb == C.EB_SUCCESS {
|
|
|
|
entries, err := c.loadEntries(blocksSeen)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
subbook.Entries = append(subbook.Entries, entries...)
|
|
|
|
}
|
|
|
|
|
|
|
|
return &subbook, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Context) loadEntries(blocksSeen map[uint32]bool) ([]BookEntry, error) {
|
|
|
|
var entries []BookEntry
|
|
|
|
|
|
|
|
for {
|
|
|
|
var (
|
|
|
|
hits [256]C.EB_Hit
|
|
|
|
hitCount C.int
|
|
|
|
)
|
|
|
|
|
|
|
|
if errEb := C.eb_hit_list(c.book, (C.int)(len(hits)), &hits[0], &hitCount); errEb != C.EB_SUCCESS {
|
2021-01-10 04:49:12 +00:00
|
|
|
return nil, fmt.Errorf("eb_hit_list failed with code: %s", formatError(errEb))
|
2021-01-01 03:52:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for _, hit := range hits[:hitCount] {
|
|
|
|
var (
|
|
|
|
entry BookEntry
|
|
|
|
err error
|
|
|
|
)
|
|
|
|
|
|
|
|
if entry.Heading, err = c.loadContent(hit.heading, blockTypeHeading); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2021-01-10 06:36:57 +00:00
|
|
|
if entry.Text, err = c.loadContent(hit.text, blockTypeText); err != nil {
|
2021-01-01 03:52:25 +00:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
hasher := crc32.NewIEEE()
|
|
|
|
hasher.Write([]byte(entry.Heading))
|
|
|
|
hasher.Write([]byte(entry.Text))
|
|
|
|
|
|
|
|
sum := hasher.Sum32()
|
|
|
|
if seen, _ := blocksSeen[sum]; !seen {
|
|
|
|
entries = append(entries, entry)
|
|
|
|
blocksSeen[sum] = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if hitCount == 0 {
|
|
|
|
return entries, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Context) loadTitle() (string, error) {
|
|
|
|
var data [C.EB_MAX_TITLE_LENGTH + 1]C.char
|
|
|
|
if errEb := C.eb_subbook_title(c.book, &data[0]); errEb != C.EB_SUCCESS {
|
2021-01-10 04:49:12 +00:00
|
|
|
return "", fmt.Errorf("eb_subbook_title failed with code: %s", formatError(errEb))
|
2021-01-01 03:52:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return c.decoder.String(C.GoString(&data[0]))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Context) loadCopyright() (string, error) {
|
|
|
|
if C.eb_have_copyright(c.book) == 0 {
|
|
|
|
return "", nil
|
|
|
|
}
|
|
|
|
|
|
|
|
var position C.EB_Position
|
|
|
|
if errEb := C.eb_copyright(c.book, &position); errEb != C.EB_SUCCESS {
|
2021-01-10 04:49:12 +00:00
|
|
|
return "", fmt.Errorf("eb_copyright failed with code: %s", formatError(errEb))
|
2021-01-01 03:52:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return c.loadContent(position, blockTypeText)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Context) loadContent(position C.EB_Position, blockType blockType) (string, error) {
|
|
|
|
for {
|
|
|
|
var (
|
|
|
|
data = (*C.char)(unsafe.Pointer(&c.buffer[0]))
|
2021-01-02 03:12:55 +00:00
|
|
|
dataSize = (C.size_t)(len(c.buffer))
|
|
|
|
dataUsed C.ssize_t
|
2021-01-01 03:52:25 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
if errEb := C.eb_seek_text(c.book, &position); errEb != C.EB_SUCCESS {
|
2021-01-10 04:49:12 +00:00
|
|
|
return "", fmt.Errorf("eb_seek_text failed with code: %s", formatError(errEb))
|
2021-01-01 03:52:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
switch blockType {
|
|
|
|
case blockTypeHeading:
|
|
|
|
if errEb := C.eb_read_heading(c.book, nil, c.hookset, nil, dataSize, data, &dataUsed); errEb != C.EB_SUCCESS {
|
2021-01-10 04:49:12 +00:00
|
|
|
return "", fmt.Errorf("eb_read_heading failed with code: %s", formatError(errEb))
|
2021-01-01 03:52:25 +00:00
|
|
|
}
|
|
|
|
case blockTypeText:
|
|
|
|
if errEb := C.eb_read_text(c.book, nil, c.hookset, nil, dataSize, data, &dataUsed); errEb != C.EB_SUCCESS {
|
2021-01-10 04:49:12 +00:00
|
|
|
return "", fmt.Errorf("eb_read_text failed with code: %s", formatError(errEb))
|
2021-01-01 03:52:25 +00:00
|
|
|
}
|
|
|
|
default:
|
|
|
|
panic("invalid block type")
|
|
|
|
}
|
|
|
|
|
2021-01-02 03:12:55 +00:00
|
|
|
if dataUsed+8 >= (C.ssize_t)(dataSize) {
|
2021-01-01 03:52:25 +00:00
|
|
|
c.buffer = make([]byte, dataSize*2)
|
|
|
|
} else {
|
|
|
|
return c.decoder.String(C.GoString(data))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func Load(path string) (*Book, error) {
|
|
|
|
var context Context
|
|
|
|
if err := context.initialize(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
defer context.shutdown()
|
|
|
|
return context.loadInternal(path)
|
|
|
|
}
|