mirror of
https://github.com/Dushistov/sdcv.git
synced 2025-12-16 09:51:56 +00:00
Compare commits
19 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
49c8094b53 | ||
|
|
4346e65bd3 | ||
|
|
d144e0310c | ||
|
|
6e36e7730c | ||
|
|
abe5e9e72f | ||
|
|
488ec68854 | ||
|
|
b698445ead | ||
|
|
504e7807e6 | ||
|
|
6c80bf2d99 | ||
|
|
8742575c33 | ||
|
|
b294b76fb5 | ||
|
|
823ec3d840 | ||
|
|
6ab8b51e6c | ||
|
|
881657b336 | ||
|
|
911fc2f561 | ||
|
|
f488f5350b | ||
|
|
e72220e748 | ||
|
|
b77c0e793a | ||
|
|
ebaa6f2136 |
5
.github/workflows/main.yml
vendored
5
.github/workflows/main.yml
vendored
@@ -20,12 +20,13 @@ jobs:
|
|||||||
fail-fast: true
|
fail-fast: true
|
||||||
|
|
||||||
matrix:
|
matrix:
|
||||||
os: [ubuntu-latest]
|
os: [ubuntu-20.04, ubuntu-latest]
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v2
|
- uses: actions/checkout@v3
|
||||||
with:
|
with:
|
||||||
submodules: 'recursive'
|
submodules: 'recursive'
|
||||||
- uses: jwlawson/actions-setup-cmake@v1.4
|
- uses: jwlawson/actions-setup-cmake@v1.4
|
||||||
|
if: matrix.os != 'ubuntu-latest'
|
||||||
with:
|
with:
|
||||||
cmake-version: '3.5.1'
|
cmake-version: '3.5.1'
|
||||||
github-api-token: ${{ secrets.GITHUB_TOKEN }}
|
github-api-token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
|||||||
@@ -95,7 +95,7 @@ set(CPACK_PACKAGE_VENDOR "Evgeniy Dushistov <dushistov@mail.ru>")
|
|||||||
set(CPACK_PACKAGE_DESCRIPTION_FILE "${CMAKE_CURRENT_SOURCE_DIR}/README.org")
|
set(CPACK_PACKAGE_DESCRIPTION_FILE "${CMAKE_CURRENT_SOURCE_DIR}/README.org")
|
||||||
set(CPACK_PACKAGE_VERSION_MAJOR "0")
|
set(CPACK_PACKAGE_VERSION_MAJOR "0")
|
||||||
set(CPACK_PACKAGE_VERSION_MINOR "5")
|
set(CPACK_PACKAGE_VERSION_MINOR "5")
|
||||||
set(CPACK_PACKAGE_VERSION_PATCH "4")
|
set(CPACK_PACKAGE_VERSION_PATCH "5")
|
||||||
|
|
||||||
set(sdcv_VERSION
|
set(sdcv_VERSION
|
||||||
"${CPACK_PACKAGE_VERSION_MAJOR}.${CPACK_PACKAGE_VERSION_MINOR}.${CPACK_PACKAGE_VERSION_PATCH}")
|
"${CPACK_PACKAGE_VERSION_MAJOR}.${CPACK_PACKAGE_VERSION_MINOR}.${CPACK_PACKAGE_VERSION_PATCH}")
|
||||||
|
|||||||
10
NEWS
10
NEWS
@@ -1,3 +1,13 @@
|
|||||||
|
Version 0.5.5
|
||||||
|
- Avoid crashes when passing unknown dicts to the -u flag (by NiLuJe)
|
||||||
|
- Use off_t for stuff mainly assigned to a stat.st_size value
|
||||||
|
Version 0.5.4
|
||||||
|
- Use binary search for synonyms
|
||||||
|
- Various improvments in work with synonyms
|
||||||
|
- Added --json (same as --json-output) to match man
|
||||||
|
- Show all matched result
|
||||||
|
- More robust parsing of ifo file
|
||||||
|
- Prevent crash if file size of files not matched expecting one for .oft files
|
||||||
Version 0.5.3
|
Version 0.5.3
|
||||||
- Use single quotes around JSON data to reduce need for escaping
|
- Use single quotes around JSON data to reduce need for escaping
|
||||||
- Store integer magic in cache file
|
- Store integer magic in cache file
|
||||||
|
|||||||
@@ -1,6 +1,9 @@
|
|||||||
#+OPTIONS: ^:nil
|
#+OPTIONS: ^:nil
|
||||||
[[https://github.com/Dushistov/sdcv/actions?query=workflow%3ACI+branch%3Amaster][https://github.com/Dushistov/sdcv/workflows/CI/badge.svg]]
|
[[https://github.com/Dushistov/sdcv/actions?query=workflow%3ACI+branch%3Amaster][https://github.com/Dushistov/sdcv/workflows/CI/badge.svg]]
|
||||||
[[https://github.com/Dushistov/sdcv/blob/master/LICENSE][https://img.shields.io/badge/license-GPL%202-brightgreen.svg]]
|
[[https://github.com/Dushistov/sdcv/blob/master/LICENSE][https://img.shields.io/badge/license-GPL%202-brightgreen.svg]]
|
||||||
|
|
||||||
|
* sdcv
|
||||||
|
*sdcv* is a simple, cross-platform, text-based utility for working with dictionaries in [[http://stardict-4.sourceforge.net/][StarDict]] format.
|
||||||
* How to compile and install
|
* How to compile and install
|
||||||
#+BEGIN_SRC sh
|
#+BEGIN_SRC sh
|
||||||
mkdir /tmp/build-sdcv
|
mkdir /tmp/build-sdcv
|
||||||
|
|||||||
@@ -27,7 +27,7 @@ public:
|
|||||||
private:
|
private:
|
||||||
const char *start; /* start of mmap'd area */
|
const char *start; /* start of mmap'd area */
|
||||||
const char *end; /* end of mmap'd area */
|
const char *end; /* end of mmap'd area */
|
||||||
unsigned long size; /* size of mmap */
|
off_t size; /* size of mmap */
|
||||||
|
|
||||||
int type;
|
int type;
|
||||||
z_stream zStream;
|
z_stream zStream;
|
||||||
@@ -47,7 +47,7 @@ private:
|
|||||||
std::string origFilename;
|
std::string origFilename;
|
||||||
std::string comment;
|
std::string comment;
|
||||||
unsigned long crc;
|
unsigned long crc;
|
||||||
unsigned long length;
|
off_t length;
|
||||||
unsigned long compressedLength;
|
unsigned long compressedLength;
|
||||||
DictCache cache[DICT_CACHE_SIZE];
|
DictCache cache[DICT_CACHE_SIZE];
|
||||||
MapFile mapfile;
|
MapFile mapfile;
|
||||||
|
|||||||
@@ -22,13 +22,13 @@ public:
|
|||||||
~MapFile();
|
~MapFile();
|
||||||
MapFile(const MapFile &) = delete;
|
MapFile(const MapFile &) = delete;
|
||||||
MapFile &operator=(const MapFile &) = delete;
|
MapFile &operator=(const MapFile &) = delete;
|
||||||
bool open(const char *file_name, unsigned long file_size);
|
bool open(const char *file_name, off_t file_size);
|
||||||
gchar *begin() { return data; }
|
gchar *begin() { return data; }
|
||||||
|
|
||||||
private:
|
private:
|
||||||
char *data = nullptr;
|
char *data = nullptr;
|
||||||
unsigned long size = 0ul;
|
|
||||||
#ifdef HAVE_MMAP
|
#ifdef HAVE_MMAP
|
||||||
|
size_t size = 0u;
|
||||||
int mmap_fd = -1;
|
int mmap_fd = -1;
|
||||||
#elif defined(_WIN32)
|
#elif defined(_WIN32)
|
||||||
HANDLE hFile = 0;
|
HANDLE hFile = 0;
|
||||||
@@ -36,9 +36,8 @@ private:
|
|||||||
#endif
|
#endif
|
||||||
};
|
};
|
||||||
|
|
||||||
inline bool MapFile::open(const char *file_name, unsigned long file_size)
|
inline bool MapFile::open(const char *file_name, off_t file_size)
|
||||||
{
|
{
|
||||||
size = file_size;
|
|
||||||
#ifdef HAVE_MMAP
|
#ifdef HAVE_MMAP
|
||||||
if ((mmap_fd = ::open(file_name, O_RDONLY)) < 0) {
|
if ((mmap_fd = ::open(file_name, O_RDONLY)) < 0) {
|
||||||
// g_print("Open file %s failed!\n",fullfilename);
|
// g_print("Open file %s failed!\n",fullfilename);
|
||||||
@@ -46,14 +45,16 @@ inline bool MapFile::open(const char *file_name, unsigned long file_size)
|
|||||||
}
|
}
|
||||||
struct stat st;
|
struct stat st;
|
||||||
if (fstat(mmap_fd, &st) == -1 || st.st_size < 0 || (st.st_size == 0 && S_ISREG(st.st_mode))
|
if (fstat(mmap_fd, &st) == -1 || st.st_size < 0 || (st.st_size == 0 && S_ISREG(st.st_mode))
|
||||||
|| sizeof(st.st_size) > sizeof(file_size) || static_cast<unsigned long>(st.st_size) != file_size) {
|
|| st.st_size != file_size) {
|
||||||
close(mmap_fd);
|
close(mmap_fd);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
data = (gchar *)mmap(nullptr, file_size, PROT_READ, MAP_SHARED, mmap_fd, 0);
|
size = static_cast<size_t>(st.st_size);
|
||||||
|
data = (gchar *)mmap(nullptr, size, PROT_READ, MAP_SHARED, mmap_fd, 0);
|
||||||
if ((void *)data == (void *)(-1)) {
|
if ((void *)data == (void *)(-1)) {
|
||||||
// g_print("mmap file %s failed!\n",idxfilename);
|
// g_print("mmap file %s failed!\n",idxfilename);
|
||||||
|
size = 0u;
|
||||||
data = nullptr;
|
data = nullptr;
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|||||||
18
src/sdcv.cpp
18
src/sdcv.cpp
@@ -186,10 +186,13 @@ try {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// add bookname to list
|
// add bookname to list
|
||||||
gchar **p = get_impl(use_dict_list);
|
for (gchar **p = get_impl(use_dict_list); *p != nullptr; ++p) {
|
||||||
while (*p) {
|
auto it = bookname_to_ifo.find(*p);
|
||||||
order_list.push_back(bookname_to_ifo.at(*p));
|
if (it != bookname_to_ifo.end()) {
|
||||||
++p;
|
order_list.push_back(it->second);
|
||||||
|
} else {
|
||||||
|
fprintf(stderr, _("Unknown dictionary: %s\n"), *p);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
std::string ordering_cfg_file = std::string(g_get_user_config_dir()) + G_DIR_SEPARATOR_S "sdcv_ordering";
|
std::string ordering_cfg_file = std::string(g_get_user_config_dir()) + G_DIR_SEPARATOR_S "sdcv_ordering";
|
||||||
@@ -201,7 +204,12 @@ try {
|
|||||||
if (ordering_file != nullptr) {
|
if (ordering_file != nullptr) {
|
||||||
std::string line;
|
std::string line;
|
||||||
while (stdio_getline(ordering_file, line)) {
|
while (stdio_getline(ordering_file, line)) {
|
||||||
order_list.push_back(bookname_to_ifo.at(line));
|
auto it = bookname_to_ifo.find(line);
|
||||||
|
if (it != bookname_to_ifo.end()) {
|
||||||
|
order_list.push_back(it->second);
|
||||||
|
} else {
|
||||||
|
fprintf(stderr, _("Unknown dictionary: %s\n"), line.c_str());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
fclose(ordering_file);
|
fclose(ordering_file);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -429,7 +429,7 @@ public:
|
|||||||
if (idxfile)
|
if (idxfile)
|
||||||
fclose(idxfile);
|
fclose(idxfile);
|
||||||
}
|
}
|
||||||
bool load(const std::string &url, gulong wc, gulong fsize, bool verbose) override;
|
bool load(const std::string &url, gulong wc, off_t fsize, bool verbose) override;
|
||||||
const gchar *get_key(glong idx) override;
|
const gchar *get_key(glong idx) override;
|
||||||
void get_data(glong idx) override { get_key(idx); }
|
void get_data(glong idx) override { get_key(idx); }
|
||||||
const gchar *get_key_and_data(glong idx) override
|
const gchar *get_key_and_data(glong idx) override
|
||||||
@@ -489,7 +489,7 @@ public:
|
|||||||
{
|
{
|
||||||
}
|
}
|
||||||
~WordListIndex() { g_free(idxdatabuf); }
|
~WordListIndex() { g_free(idxdatabuf); }
|
||||||
bool load(const std::string &url, gulong wc, gulong fsize, bool verbose) override;
|
bool load(const std::string &url, gulong wc, off_t fsize, bool verbose) override;
|
||||||
const gchar *get_key(glong idx) override { return wordlist[idx]; }
|
const gchar *get_key(glong idx) override { return wordlist[idx]; }
|
||||||
void get_data(glong idx) override;
|
void get_data(glong idx) override;
|
||||||
const gchar *get_key_and_data(glong idx) override
|
const gchar *get_key_and_data(glong idx) override
|
||||||
@@ -615,7 +615,7 @@ bool OffsetIndex::save_cache(const std::string &url, bool verbose)
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool OffsetIndex::load(const std::string &url, gulong wc, gulong fsize, bool verbose)
|
bool OffsetIndex::load(const std::string &url, gulong wc, off_t fsize, bool verbose)
|
||||||
{
|
{
|
||||||
wordcount = wc;
|
wordcount = wc;
|
||||||
gulong npages = (wc - 1) / ENTR_PER_PAGE + 2;
|
gulong npages = (wc - 1) / ENTR_PER_PAGE + 2;
|
||||||
@@ -758,7 +758,7 @@ bool OffsetIndex::lookup(const char *str, std::set<glong> &idxs, glong &next_idx
|
|||||||
return bFound;
|
return bFound;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool WordListIndex::load(const std::string &url, gulong wc, gulong fsize, bool)
|
bool WordListIndex::load(const std::string &url, gulong wc, off_t fsize, bool)
|
||||||
{
|
{
|
||||||
gzFile in = gzopen(url.c_str(), "rb");
|
gzFile in = gzopen(url.c_str(), "rb");
|
||||||
if (in == nullptr)
|
if (in == nullptr)
|
||||||
@@ -771,7 +771,7 @@ bool WordListIndex::load(const std::string &url, gulong wc, gulong fsize, bool)
|
|||||||
if (len < 0)
|
if (len < 0)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
if (gulong(len) != fsize)
|
if (static_cast<off_t>(len) != fsize)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
wordlist.resize(wc + 1);
|
wordlist.resize(wc + 1);
|
||||||
@@ -920,7 +920,7 @@ bool Dict::Lookup(const char *str, std::set<glong> &idxs, glong &next_idx)
|
|||||||
|
|
||||||
bool Dict::load(const std::string &ifofilename, bool verbose)
|
bool Dict::load(const std::string &ifofilename, bool verbose)
|
||||||
{
|
{
|
||||||
gulong idxfilesize;
|
off_t idxfilesize;
|
||||||
if (!load_ifofile(ifofilename, idxfilesize))
|
if (!load_ifofile(ifofilename, idxfilesize))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
@@ -964,7 +964,7 @@ bool Dict::load(const std::string &ifofilename, bool verbose)
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Dict::load_ifofile(const std::string &ifofilename, gulong &idxfilesize)
|
bool Dict::load_ifofile(const std::string &ifofilename, off_t &idxfilesize)
|
||||||
{
|
{
|
||||||
DictInfo dict_info;
|
DictInfo dict_info;
|
||||||
if (!dict_info.load_from_ifo_file(ifofilename, false))
|
if (!dict_info.load_from_ifo_file(ifofilename, false))
|
||||||
|
|||||||
@@ -77,8 +77,8 @@ struct DictInfo {
|
|||||||
std::string website;
|
std::string website;
|
||||||
std::string date;
|
std::string date;
|
||||||
std::string description;
|
std::string description;
|
||||||
guint32 index_file_size;
|
off_t index_file_size;
|
||||||
guint32 syn_file_size;
|
off_t syn_file_size;
|
||||||
std::string sametypesequence;
|
std::string sametypesequence;
|
||||||
|
|
||||||
bool load_from_ifo_file(const std::string &ifofilename, bool istreedict);
|
bool load_from_ifo_file(const std::string &ifofilename, bool istreedict);
|
||||||
@@ -91,7 +91,7 @@ public:
|
|||||||
guint32 wordentry_size;
|
guint32 wordentry_size;
|
||||||
|
|
||||||
virtual ~IIndexFile() {}
|
virtual ~IIndexFile() {}
|
||||||
virtual bool load(const std::string &url, gulong wc, gulong fsize, bool verbose) = 0;
|
virtual bool load(const std::string &url, gulong wc, off_t fsize, bool verbose) = 0;
|
||||||
virtual const gchar *get_key(glong idx) = 0;
|
virtual const gchar *get_key(glong idx) = 0;
|
||||||
virtual void get_data(glong idx) = 0;
|
virtual void get_data(glong idx) = 0;
|
||||||
virtual const gchar *get_key_and_data(glong idx) = 0;
|
virtual const gchar *get_key_and_data(glong idx) = 0;
|
||||||
@@ -160,7 +160,7 @@ private:
|
|||||||
std::unique_ptr<IIndexFile> idx_file;
|
std::unique_ptr<IIndexFile> idx_file;
|
||||||
std::unique_ptr<SynFile> syn_file;
|
std::unique_ptr<SynFile> syn_file;
|
||||||
|
|
||||||
bool load_ifofile(const std::string &ifofilename, gulong &idxfilesize);
|
bool load_ifofile(const std::string &ifofilename, off_t &idxfilesize);
|
||||||
};
|
};
|
||||||
|
|
||||||
class Libs
|
class Libs
|
||||||
|
|||||||
Reference in New Issue
Block a user