mirror of
https://github.com/arsenetar/dupeguru.git
synced 2026-01-28 17:31:38 +00:00
Compare commits
9 Commits
as/pyproje
...
334119ae6c
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
334119ae6c | ||
|
|
cd70c99c67 | ||
|
|
7eafb86c2b | ||
|
|
71672bc2c9 | ||
|
|
f4e6768caf | ||
|
|
8ce4080c4e | ||
|
|
f191b64cca | ||
|
|
a714a0860c | ||
|
|
5fb8caefa9 |
12
.github/workflows/default.yml
vendored
12
.github/workflows/default.yml
vendored
@@ -52,14 +52,4 @@ jobs:
|
||||
uses: actions/upload-artifact@v4
|
||||
with:
|
||||
name: modules ${{ matrix.python-version }}
|
||||
path: build/**/*.so
|
||||
merge-artifacts:
|
||||
needs: [test]
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- name: Merge Artifacts
|
||||
uses: actions/upload-artifact/merge@v4
|
||||
with:
|
||||
name: modules
|
||||
pattern: modules*
|
||||
delete-merged: true
|
||||
path: ${{ github.workspace }}/**/*.so
|
||||
|
||||
2
.vscode/launch.json
vendored
2
.vscode/launch.json
vendored
@@ -6,7 +6,7 @@
|
||||
"configurations": [
|
||||
{
|
||||
"name": "DupuGuru",
|
||||
"type": "debugpy",
|
||||
"type": "python",
|
||||
"request": "launch",
|
||||
"program": "run.py",
|
||||
"console": "integratedTerminal",
|
||||
|
||||
3
.vscode/settings.json
vendored
3
.vscode/settings.json
vendored
@@ -12,6 +12,5 @@
|
||||
"[python]": {
|
||||
"editor.formatOnSave": true,
|
||||
"editor.defaultFormatter": "ms-python.black-formatter"
|
||||
},
|
||||
"python.testing.pytestEnabled": true
|
||||
}
|
||||
}
|
||||
@@ -158,7 +158,7 @@ class SqliteCache:
|
||||
ids = ",".join(map(str, rowids))
|
||||
sql = (
|
||||
"select rowid, blocks, blocks2, blocks3, blocks4, blocks5, blocks6, blocks7, blocks8 "
|
||||
f"from pictures where rowid in ({ids})"
|
||||
f"from pictures where rowid in {ids}"
|
||||
)
|
||||
cur = self.con.execute(sql)
|
||||
return (
|
||||
|
||||
@@ -54,7 +54,7 @@ def get_cache(cache_path, readonly=False):
|
||||
return SqliteCache(cache_path, readonly=readonly)
|
||||
|
||||
|
||||
def prepare_pictures(pictures, cache_path, with_dimensions, match_rotated, j=job.nulljob):
|
||||
def prepare_pictures(pictures, cache_path, with_dimensions, j=job.nulljob):
|
||||
# The MemoryError handlers in there use logging without first caring about whether or not
|
||||
# there is enough memory left to carry on the operation because it is assumed that the
|
||||
# MemoryError happens when trying to read an image file, which is freed from memory by the
|
||||
@@ -76,14 +76,8 @@ def prepare_pictures(pictures, cache_path, with_dimensions, match_rotated, j=job
|
||||
if with_dimensions:
|
||||
picture.dimensions # pre-read dimensions
|
||||
try:
|
||||
if picture.unicode_path not in cache or (
|
||||
match_rotated and any(block == [] for block in cache[picture.unicode_path])
|
||||
):
|
||||
if match_rotated:
|
||||
if picture.unicode_path not in cache:
|
||||
blocks = [picture.get_blocks(BLOCK_COUNT_PER_SIDE, orientation) for orientation in range(1, 9)]
|
||||
else:
|
||||
blocks = [[]] * 8
|
||||
blocks[max(picture.get_orientation() - 1, 0)] = picture.get_blocks(BLOCK_COUNT_PER_SIDE)
|
||||
cache[picture.unicode_path] = blocks
|
||||
prepared.append(picture)
|
||||
except (OSError, ValueError) as e:
|
||||
@@ -193,7 +187,7 @@ def getmatches(pictures, cache_path, threshold, match_scaled=False, match_rotate
|
||||
j.set_progress(comparison_count, progress_msg)
|
||||
|
||||
j = j.start_subjob([3, 7])
|
||||
pictures = prepare_pictures(pictures, cache_path, not match_scaled, match_rotated, j=j)
|
||||
pictures = prepare_pictures(pictures, cache_path, with_dimensions=not match_scaled, j=j)
|
||||
j = j.start_subjob([9, 1], tr("Preparing for matching"))
|
||||
cache = get_cache(cache_path)
|
||||
id2picture = {}
|
||||
|
||||
@@ -37,7 +37,7 @@ class Photo(fs.File):
|
||||
def _plat_get_blocks(self, block_count_per_side, orientation):
|
||||
raise NotImplementedError()
|
||||
|
||||
def get_orientation(self):
|
||||
def _get_orientation(self):
|
||||
if not hasattr(self, "_cached_orientation"):
|
||||
try:
|
||||
with self.path.open("rb") as fp:
|
||||
@@ -95,13 +95,13 @@ class Photo(fs.File):
|
||||
fs.File._read_info(self, field)
|
||||
if field == "dimensions":
|
||||
self.dimensions = self._plat_get_dimensions()
|
||||
if self.get_orientation() in {5, 6, 7, 8}:
|
||||
if self._get_orientation() in {5, 6, 7, 8}:
|
||||
self.dimensions = (self.dimensions[1], self.dimensions[0])
|
||||
elif field == "exif_timestamp":
|
||||
self.exif_timestamp = self._get_exif_timestamp()
|
||||
|
||||
def get_blocks(self, block_count_per_side, orientation: int = None):
|
||||
if orientation is None:
|
||||
return self._plat_get_blocks(block_count_per_side, self.get_orientation())
|
||||
return self._plat_get_blocks(block_count_per_side, self._get_orientation())
|
||||
else:
|
||||
return self._plat_get_blocks(block_count_per_side, orientation)
|
||||
|
||||
@@ -96,8 +96,6 @@ class FilenameCategory(CriterionCategory):
|
||||
DOESNT_END_WITH_NUMBER = 1
|
||||
LONGEST = 2
|
||||
SHORTEST = 3
|
||||
LONGEST_PATH = 4
|
||||
SHORTEST_PATH = 5
|
||||
|
||||
def format_criterion_value(self, value):
|
||||
return {
|
||||
@@ -105,8 +103,6 @@ class FilenameCategory(CriterionCategory):
|
||||
self.DOESNT_END_WITH_NUMBER: tr("Doesn't end with number"),
|
||||
self.LONGEST: tr("Longest"),
|
||||
self.SHORTEST: tr("Shortest"),
|
||||
self.LONGEST_PATH: tr("Longest Path"),
|
||||
self.SHORTEST_PATH: tr("Shortest Path"),
|
||||
}[value]
|
||||
|
||||
def extract_value(self, dupe):
|
||||
@@ -120,10 +116,6 @@ class FilenameCategory(CriterionCategory):
|
||||
return 0 if ends_with_digit else 1
|
||||
else:
|
||||
return 1 if ends_with_digit else 0
|
||||
elif crit_value == self.LONGEST_PATH:
|
||||
return len(str(dupe.folder_path)) * -1
|
||||
elif crit_value == self.SHORTEST_PATH:
|
||||
return len(str(dupe.folder_path))
|
||||
else:
|
||||
value = len(value)
|
||||
if crit_value == self.LONGEST:
|
||||
@@ -138,8 +130,6 @@ class FilenameCategory(CriterionCategory):
|
||||
self.DOESNT_END_WITH_NUMBER,
|
||||
self.LONGEST,
|
||||
self.SHORTEST,
|
||||
self.LONGEST_PATH,
|
||||
self.SHORTEST_PATH,
|
||||
]
|
||||
]
|
||||
|
||||
|
||||
@@ -59,13 +59,13 @@ class BaseTestCaseCache:
|
||||
|
||||
def test_set_then_retrieve_blocks(self):
|
||||
c = self.get_cache()
|
||||
b = [[(0, 0, 0), (1, 2, 3)]] * 8
|
||||
b = [(0, 0, 0), (1, 2, 3)]
|
||||
c["foo"] = b
|
||||
eq_(b, c["foo"])
|
||||
|
||||
def test_delitem(self):
|
||||
c = self.get_cache()
|
||||
c["foo"] = [[]] * 8
|
||||
c["foo"] = ""
|
||||
del c["foo"]
|
||||
assert "foo" not in c
|
||||
with raises(KeyError):
|
||||
@@ -74,16 +74,16 @@ class BaseTestCaseCache:
|
||||
def test_persistance(self, tmpdir):
|
||||
DBNAME = tmpdir.join("hstest.db")
|
||||
c = self.get_cache(str(DBNAME))
|
||||
c["foo"] = [[(1, 2, 3)]] * 8
|
||||
c["foo"] = [(1, 2, 3)]
|
||||
del c
|
||||
c = self.get_cache(str(DBNAME))
|
||||
eq_([[(1, 2, 3)]] * 8, c["foo"])
|
||||
eq_([(1, 2, 3)], c["foo"])
|
||||
|
||||
def test_filter(self):
|
||||
c = self.get_cache()
|
||||
c["foo"] = [[]] * 8
|
||||
c["bar"] = [[]] * 8
|
||||
c["baz"] = [[]] * 8
|
||||
c["foo"] = ""
|
||||
c["bar"] = ""
|
||||
c["baz"] = ""
|
||||
c.filter(lambda p: p != "bar") # only 'bar' is removed
|
||||
eq_(2, len(c))
|
||||
assert "foo" in c
|
||||
@@ -92,9 +92,9 @@ class BaseTestCaseCache:
|
||||
|
||||
def test_clear(self):
|
||||
c = self.get_cache()
|
||||
c["foo"] = [[]] * 8
|
||||
c["bar"] = [[]] * 8
|
||||
c["baz"] = [[]] * 8
|
||||
c["foo"] = ""
|
||||
c["bar"] = ""
|
||||
c["baz"] = ""
|
||||
c.clear()
|
||||
eq_(0, len(c))
|
||||
assert "foo" not in c
|
||||
@@ -104,7 +104,7 @@ class BaseTestCaseCache:
|
||||
def test_by_id(self):
|
||||
# it's possible to use the cache by referring to the files by their row_id
|
||||
c = self.get_cache()
|
||||
b = [[(0, 0, 0), (1, 2, 3)]] * 8
|
||||
b = [(0, 0, 0), (1, 2, 3)]
|
||||
c["foo"] = b
|
||||
foo_id = c.get_id("foo")
|
||||
eq_(c[foo_id], b)
|
||||
@@ -127,10 +127,10 @@ class TestCaseSqliteCache(BaseTestCaseCache):
|
||||
fp.write("invalid sqlite content")
|
||||
fp.close()
|
||||
c = self.get_cache(dbname) # should not raise a DatabaseError
|
||||
c["foo"] = [[(1, 2, 3)]] * 8
|
||||
c["foo"] = [(1, 2, 3)]
|
||||
del c
|
||||
c = self.get_cache(dbname)
|
||||
eq_(c["foo"], [[(1, 2, 3)]] * 8)
|
||||
eq_(c["foo"], [(1, 2, 3)])
|
||||
|
||||
|
||||
class TestCaseCacheSQLEscape:
|
||||
@@ -152,7 +152,7 @@ class TestCaseCacheSQLEscape:
|
||||
|
||||
def test_delitem(self):
|
||||
c = self.get_cache()
|
||||
c["foo'bar"] = [[]] * 8
|
||||
c["foo'bar"] = []
|
||||
try:
|
||||
del c["foo'bar"]
|
||||
except KeyError:
|
||||
|
||||
@@ -112,7 +112,7 @@ msgstr "Размеры"
|
||||
|
||||
#: core\pe\result_table.py:21 core\se\result_table.py:21
|
||||
msgid "Size (KB)"
|
||||
msgstr "Размер (КБ)"
|
||||
msgstr "Размер (кБ)"
|
||||
|
||||
#: core\pe\result_table.py:24
|
||||
msgid "EXIF Timestamp"
|
||||
|
||||
@@ -100,15 +100,15 @@ msgstr ""
|
||||
|
||||
#: core\app.py:469
|
||||
msgid "Select a directory to copy marked files to"
|
||||
msgstr "Выберите каталог, в который вы хотите скопировать отмеченные файлы"
|
||||
msgstr "Выберите каталог, в который Вы хотите скопировать отмеченные файлы"
|
||||
|
||||
#: core\app.py:471
|
||||
msgid "Select a directory to move marked files to"
|
||||
msgstr "Выберите каталог, в который вы хотите переместить отмеченные файлы"
|
||||
msgstr "Выберите каталог для перемещения отмеченных файлов"
|
||||
|
||||
#: core\app.py:510
|
||||
msgid "Select a destination for your exported CSV"
|
||||
msgstr "Выберите назначение для экспортируемого "
|
||||
msgstr "Выберите назначение для Вашего экспортируемого CSV"
|
||||
|
||||
#: core\app.py:516 core\app.py:777 core\app.py:787
|
||||
msgid "Couldn't write to file: {}"
|
||||
@@ -124,7 +124,7 @@ msgstr "Вы собираетесь удалить %d файлов из резу
|
||||
|
||||
#: core\app.py:749
|
||||
msgid "{} duplicate groups were changed by the re-prioritization."
|
||||
msgstr "{} групп дубликатов было изменено при реприоритезации."
|
||||
msgstr "{} групп дубликатов было изменено при обновлении приоритета."
|
||||
|
||||
#: core\app.py:797
|
||||
msgid "The selected directories contain no scannable file."
|
||||
@@ -136,7 +136,7 @@ msgstr "Сбор файлов для сканирования"
|
||||
|
||||
#: core\app.py:863
|
||||
msgid "%s (%d discarded)"
|
||||
msgstr "%s. (%d отменено)"
|
||||
msgstr "%s (%d отменено)"
|
||||
|
||||
#: core\directories.py:191
|
||||
msgid "Collected {} files to scan"
|
||||
@@ -148,7 +148,7 @@ msgstr "Собрано {} каталогов для сканирования"
|
||||
|
||||
#: core\engine.py:27
|
||||
msgid "%d matches found from %d groups"
|
||||
msgstr "Найдено %d совпадений из %d групп"
|
||||
msgstr "%d совпадений найдено из %d групп"
|
||||
|
||||
#: core\gui\deletion_options.py:71
|
||||
msgid "You are sending {} file(s) to the Trash."
|
||||
@@ -262,3 +262,4 @@ msgstr "Почти готово! Вожусь с результатами..."
|
||||
#: core\se\scanner.py:18
|
||||
msgid "Folders"
|
||||
msgstr "Папки"
|
||||
|
||||
|
||||
@@ -93,7 +93,7 @@ msgid ""
|
||||
"Instead of sending files to trash, delete them directly. This option is "
|
||||
"usually used as a workaround when the normal deletion method doesn't work."
|
||||
msgstr ""
|
||||
"Удалить файлы с диска вместо отправки в Корзину. Используйте если нормальный"
|
||||
"Удалить файлы с диска вместо отправки в Корзину. Используйте, если нормальный "
|
||||
"метод удаления не работает."
|
||||
|
||||
#: qt/deletion_options.py:59 cocoa/en.lproj/Localizable.strings:0
|
||||
@@ -189,7 +189,7 @@ msgstr "Несохранённые результаты"
|
||||
|
||||
#: qt/directories_dialog.py:231 cocoa/en.lproj/Localizable.strings:0
|
||||
msgid "You have unsaved results, do you really want to quit?"
|
||||
msgstr "Имеются несохранённые результаты, вы действительно хотите выйти?"
|
||||
msgstr "Имеются несохранённые результаты, Вы действительно хотите выйти?"
|
||||
|
||||
#: qt/directories_dialog.py:239 cocoa/en.lproj/Localizable.strings:0
|
||||
msgid "Select a folder to add to the scanning list"
|
||||
@@ -213,7 +213,7 @@ msgstr "Начать новую проверку"
|
||||
|
||||
#: qt/directories_dialog.py:279 cocoa/en.lproj/Localizable.strings:0
|
||||
msgid "You have unsaved results, do you really want to continue?"
|
||||
msgstr "Имеются несохранённые результаты, вы действительно хотите продолжить?"
|
||||
msgstr "Имеются несохранённые результаты, Вы действительно хотите продолжить?"
|
||||
|
||||
#: qt/directories_model.py:23 cocoa/en.lproj/Localizable.strings:0
|
||||
msgid "Name"
|
||||
@@ -256,7 +256,7 @@ msgstr "Теги для проверки:"
|
||||
|
||||
#: qt/me/preferences_dialog.py:36 cocoa/en.lproj/Localizable.strings:0
|
||||
msgid "Track"
|
||||
msgstr "Трек"
|
||||
msgstr "Дорожка"
|
||||
|
||||
#: qt/me/preferences_dialog.py:38 cocoa/en.lproj/Localizable.strings:0
|
||||
msgid "Artist"
|
||||
@@ -323,7 +323,7 @@ msgstr "Уровень фильтрации:"
|
||||
|
||||
#: qt/preferences_dialog.py:69
|
||||
msgid "More Results"
|
||||
msgstr "Дополнительные результаты"
|
||||
msgstr "Больше результатов"
|
||||
|
||||
#: qt/preferences_dialog.py:74
|
||||
msgid "Fewer Results"
|
||||
@@ -379,7 +379,7 @@ msgstr ""
|
||||
|
||||
#: qt/problem_dialog.py:33 cocoa/en.lproj/Localizable.strings:0
|
||||
msgid "Problems!"
|
||||
msgstr "Проблемка!"
|
||||
msgstr "Проблема!"
|
||||
|
||||
#: qt/problem_dialog.py:37 cocoa/en.lproj/Localizable.strings:0
|
||||
msgid ""
|
||||
@@ -409,15 +409,15 @@ msgstr "Показать значения разницы"
|
||||
|
||||
#: qt/result_window.py:60
|
||||
msgid "Send Marked to Recycle Bin..."
|
||||
msgstr "Переместить отмеченные в Корзину…"
|
||||
msgstr "Переместить отмеченные в Корзину..."
|
||||
|
||||
#: qt/result_window.py:61 cocoa/en.lproj/Localizable.strings:0
|
||||
msgid "Move Marked to..."
|
||||
msgstr "Переместить отмеченные в…"
|
||||
msgstr "Переместить отмеченные в..."
|
||||
|
||||
#: qt/result_window.py:62 cocoa/en.lproj/Localizable.strings:0
|
||||
msgid "Copy Marked to..."
|
||||
msgstr "Скопировать отмеченные в…"
|
||||
msgstr "Скопировать отмеченные в..."
|
||||
|
||||
#: qt/result_window.py:63 cocoa/en.lproj/Localizable.strings:0
|
||||
msgid "Remove Marked from Results"
|
||||
@@ -477,7 +477,7 @@ msgstr "Экспорт в CSV"
|
||||
|
||||
#: qt/result_window.py:89 cocoa/en.lproj/Localizable.strings:0
|
||||
msgid "Save Results..."
|
||||
msgstr "Сохранить результаты…"
|
||||
msgstr "Сохранить результаты..."
|
||||
|
||||
#: qt/result_window.py:90 cocoa/en.lproj/Localizable.strings:0
|
||||
msgid "Invoke Custom Command"
|
||||
@@ -509,15 +509,15 @@ msgstr "Значения разницы"
|
||||
|
||||
#: qt/result_window.py:310 cocoa/en.lproj/Localizable.strings:0
|
||||
msgid "Select a file to save your results to"
|
||||
msgstr "Выберите файл, чтобы сохранить ваши результаты"
|
||||
msgstr "Выберите файл для сохранения Ваших результатов"
|
||||
|
||||
#: qt/se/preferences_dialog.py:41
|
||||
msgid "Ignore files smaller than"
|
||||
msgstr "Игнорировать файлы меньше чем"
|
||||
msgstr "Игнорировать файлы меньше, чем"
|
||||
|
||||
#: qt/se/preferences_dialog.py:52 cocoa/en.lproj/Localizable.strings:0
|
||||
msgid "KB"
|
||||
msgstr "КБ"
|
||||
msgstr "кБ"
|
||||
|
||||
#: cocoa/en.lproj/Localizable.strings:0
|
||||
msgid "%@ Results"
|
||||
@@ -549,7 +549,7 @@ msgstr "Все на передний план"
|
||||
|
||||
#: cocoa/en.lproj/Localizable.strings:0
|
||||
msgid "Check for update..."
|
||||
msgstr "Проверка обновлений…"
|
||||
msgstr "Проверка обновлений..."
|
||||
|
||||
#: cocoa/en.lproj/Localizable.strings:0
|
||||
msgid "Close Window"
|
||||
@@ -626,7 +626,7 @@ msgstr "Уровень фильтрации:"
|
||||
|
||||
#: cocoa/en.lproj/Localizable.strings:0
|
||||
msgid "Filter Results..."
|
||||
msgstr "Отфильтровать результаты…"
|
||||
msgstr "Отфильтровать результаты..."
|
||||
|
||||
#: cocoa/en.lproj/Localizable.strings:0
|
||||
msgid "Folder Selection Window"
|
||||
@@ -646,11 +646,11 @@ msgstr "Скрыть остальные"
|
||||
|
||||
#: cocoa/en.lproj/Localizable.strings:0
|
||||
msgid "Ignore files smaller than:"
|
||||
msgstr "Пропускать файлы меньше чем:"
|
||||
msgstr "Пропускать файлы меньше, чем:"
|
||||
|
||||
#: cocoa/en.lproj/Localizable.strings:0
|
||||
msgid "Load from file..."
|
||||
msgstr "Загрузить из файла…"
|
||||
msgstr "Загрузить из файла..."
|
||||
|
||||
#: cocoa/en.lproj/Localizable.strings:0
|
||||
msgid "Minimize"
|
||||
@@ -674,7 +674,7 @@ msgstr "Вставить"
|
||||
|
||||
#: cocoa/en.lproj/Localizable.strings:0
|
||||
msgid "Preferences..."
|
||||
msgstr "Настройки…"
|
||||
msgstr "Настройки..."
|
||||
|
||||
#: cocoa/en.lproj/Localizable.strings:0
|
||||
msgid "Quick Look"
|
||||
@@ -686,7 +686,7 @@ msgstr "Выйти из dupeGuru"
|
||||
|
||||
#: cocoa/en.lproj/Localizable.strings:0
|
||||
msgid "Reset to Default"
|
||||
msgstr "Восстановить значения по умолчанию"
|
||||
msgstr "Восстановить значение по умолчанию"
|
||||
|
||||
#: cocoa/en.lproj/Localizable.strings:0
|
||||
msgid "Reset To Defaults"
|
||||
@@ -706,7 +706,7 @@ msgstr "Выбрать все"
|
||||
|
||||
#: cocoa/en.lproj/Localizable.strings:0
|
||||
msgid "Send Marked to Trash..."
|
||||
msgstr "Переместить отмеченные в Корзину…"
|
||||
msgstr "Переместить отмеченные в Корзину..."
|
||||
|
||||
#: cocoa/en.lproj/Localizable.strings:0
|
||||
msgid "Services"
|
||||
@@ -754,11 +754,11 @@ msgstr "Выберите файл каталогов для загрузки"
|
||||
|
||||
#: qt\directories_dialog.py:338
|
||||
msgid "dupeGuru Results (*.dupegurudirs)"
|
||||
msgstr "Каталоги dupeGuru (*.dupegurudirs)"
|
||||
msgstr "Результаты dupeGuru (*.dupegurudirs)"
|
||||
|
||||
#: qt\directories_dialog.py:347
|
||||
msgid "Select a file to save your directories to"
|
||||
msgstr "Выберите файл для сохранения каталогов"
|
||||
msgstr "Выберите файл для сохранения Ваших каталогов"
|
||||
|
||||
#: qt\directories_dialog.py:348
|
||||
msgid "dupeGuru Directories (*.dupegurudirs)"
|
||||
@@ -835,7 +835,7 @@ msgstr "Переопределить значки темы на панели и
|
||||
#: qt\pe\preferences_dialog.py:58
|
||||
msgid ""
|
||||
"Use our own internal icons instead of those provided by the theme engine"
|
||||
msgstr "Используйте внутренние значки вместо значков, встроенных в тему"
|
||||
msgstr "Использовать наши внутренние значки вместо значков, встроенных в тему"
|
||||
|
||||
#: qt\pe\preferences_dialog.py:66
|
||||
msgid "Show scrollbars in image viewers"
|
||||
@@ -965,7 +965,7 @@ msgstr ""
|
||||
|
||||
#: qt\se\preferences_dialog.py:68
|
||||
msgid "Ignore files larger than"
|
||||
msgstr "Игнорировать файлы больше чем"
|
||||
msgstr "Игнорировать файлы больше, чем"
|
||||
|
||||
#: qt\app.py:135 qt\app.py:293
|
||||
msgid "Clear Cache"
|
||||
@@ -981,7 +981,7 @@ msgstr ""
|
||||
|
||||
#: qt\app.py:299
|
||||
msgid "Cache cleared."
|
||||
msgstr "Кэш очищен "
|
||||
msgstr "Кэш очищен."
|
||||
|
||||
#: qt\preferences_dialog.py:173
|
||||
msgid "Use dark style"
|
||||
@@ -994,8 +994,7 @@ msgstr "Сохранить профиль сканирования"
|
||||
#: qt\preferences_dialog.py:242
|
||||
msgid "Profile the scan operation and save logs for optimization."
|
||||
msgstr ""
|
||||
"В папке установленной или портативной программы, есть папка Data в которую "
|
||||
"сохраняется логи и файл с раширением *.profile для оптимизации."
|
||||
"Настройте операцию сканирования и сохраните журналы для оптимизации."
|
||||
|
||||
#: qt\preferences_dialog.py:246
|
||||
msgid "Logs located in: <a href=\"{}\">{}</a>"
|
||||
@@ -1023,11 +1022,11 @@ msgstr "Под лицензией GPLv3"
|
||||
|
||||
#: qt\about_box.py:68
|
||||
msgid "No update available."
|
||||
msgstr "У вас самая свежая версия"
|
||||
msgstr "Обновления недоступны."
|
||||
|
||||
#: qt\about_box.py:71
|
||||
msgid "New version {} available, download <a href=\"{}\">here</a>."
|
||||
msgstr "Обнаружена новая {} версия, загружать <a href=\"{}\">тут</a>."
|
||||
msgstr "Обнаружена новая {} версия, загрузить <a href=\"{}\">тут</a>."
|
||||
|
||||
#: qt\error_report_dialog.py:50
|
||||
msgid "Error Report"
|
||||
|
||||
@@ -1,86 +1,9 @@
|
||||
[build-system]
|
||||
requires = ["setuptools >= 75.3.1"]
|
||||
requires = ["setuptools"]
|
||||
build-backend = "setuptools.build_meta"
|
||||
|
||||
[project]
|
||||
name = "dupeGuru"
|
||||
description = "dupeGuru is a tool to find duplicate files on your computer."
|
||||
authors = [
|
||||
{name = "Andrew Senetar", email = "arsenetar@voltaicideas.net"}
|
||||
]
|
||||
readme = "README.md"
|
||||
license = "GPL-3.0-or-later"
|
||||
license-files = ["LICENSE"]
|
||||
keywords = ["deduplication"]
|
||||
classifiers = [
|
||||
"Development Status :: 5 - Production/Stable",
|
||||
"Intended Audience :: End Users/Desktop",
|
||||
"Operating System :: MacOS :: MacOS X",
|
||||
"Operating System :: Microsoft :: Windows",
|
||||
"Operating System :: POSIX",
|
||||
"Programming Language :: Python :: 3 :: Only",
|
||||
"Programming Language :: Python :: 3.7",
|
||||
"Programming Language :: Python :: 3.8",
|
||||
"Programming Language :: Python :: 3.9",
|
||||
"Programming Language :: Python :: 3.10",
|
||||
"Programming Language :: Python :: 3.11",
|
||||
"Programming Language :: Python :: 3.12",
|
||||
"Topic :: Desktop Environment :: File Managers",
|
||||
]
|
||||
requires-python = ">=3.7, <3.13"
|
||||
|
||||
dynamic = ["version"]
|
||||
|
||||
dependencies = [
|
||||
"distro>=1.8.0,<2.0.0",
|
||||
"mutagen>=1.46.0,<2.0.0",
|
||||
"polib>=1.1.0,<2.0.0",
|
||||
"PyQt5 >=5.15.0,<6.0; sys_platform != 'linux'",
|
||||
"pywin32>=304; sys_platform == 'win32'",
|
||||
"semantic-version>=2.0.0,<3.0.0",
|
||||
"Send2Trash>=1.8.2",
|
||||
"xxhash>=3.0.0,<4.0.0",
|
||||
]
|
||||
|
||||
[project.optional-dependencies]
|
||||
dev = [
|
||||
"pytest>=7,<8",
|
||||
"flake8",
|
||||
"black",
|
||||
]
|
||||
build = [
|
||||
"dupeGuru[dev]",
|
||||
"sphinx>=5.3.0,<8.0.0",
|
||||
"pyinstaller>=5.6,<6.0; sys_platform != 'linux'"
|
||||
]
|
||||
|
||||
[project.urls]
|
||||
Homepage = "https://dupeguru.voltaicideas.net/"
|
||||
Documentation = "https://dupeguru.voltaicideas.net/help/en/"
|
||||
Repository = "https://github.com/arsenetar/dupeguru.git"
|
||||
Issues = "https://github.com/arsenetar/dupeguru/issues"
|
||||
Releases = "https://github.com/arsenetar/dupeguru/releases"
|
||||
|
||||
[project.gui-scripts]
|
||||
dupeguru = "dupeguru.__main__:main"
|
||||
|
||||
[tool.black]
|
||||
line-length = 120
|
||||
|
||||
[tool.isort]
|
||||
# make it compatible with black
|
||||
profile = "black"
|
||||
skip_gitignore = true
|
||||
|
||||
[tool.setuptools.packages.find]
|
||||
include = ["core", "hscommon", "qt"]
|
||||
|
||||
[tool.setuptools.dynamic]
|
||||
version = {attr = "core.__version__"}
|
||||
|
||||
[tool.setuptools]
|
||||
ext-modules = [
|
||||
{name = "core.pe._block", sources = ["core/pe/modules/block.c", "core/pe/modules/common.c"], include-dirs = ["core/pe/modules"]},
|
||||
{name = "core.pe._cache", sources = ["core/pe/modules/cache.c", "core/pe/modules/common.c"], include-dirs = ["core/pe/modules"]},
|
||||
{name = "qt.pe._block_qt", sources = ["qt/pe/modules/block.c"]},
|
||||
]
|
||||
|
||||
4
requirements-extra.txt
Normal file
4
requirements-extra.txt
Normal file
@@ -0,0 +1,4 @@
|
||||
pytest>=7,<8
|
||||
flake8
|
||||
black
|
||||
pyinstaller>=5.6,<6.0; sys_platform != 'linux'
|
||||
9
requirements.txt
Normal file
9
requirements.txt
Normal file
@@ -0,0 +1,9 @@
|
||||
distro>=1.8.0,<2.0.0
|
||||
mutagen>=1.46.0,<2.0.0
|
||||
polib>=1.1.0,<2.0.0
|
||||
PyQt5 >=5.15.0,<6.0; sys_platform != 'linux'
|
||||
pywin32>=304; sys_platform == 'win32'
|
||||
semantic-version>=2.0.0,<3.0.0
|
||||
Send2Trash>=1.8.2,<2.0.0
|
||||
sphinx>=5.3.0,<8.0.0
|
||||
xxhash>=3.0.0,<4.0.0
|
||||
48
setup.cfg
Normal file
48
setup.cfg
Normal file
@@ -0,0 +1,48 @@
|
||||
[metadata]
|
||||
name = dupeGuru
|
||||
version = attr: core.__version__
|
||||
url = https://github.com/arsenetar/dupeguru
|
||||
project_urls =
|
||||
Bug Reports = https://github.com/arsenetar/dupeguru/issues
|
||||
author = Andrew Senetar
|
||||
author_email = arsenetar@voltaicideas.net
|
||||
license = GPLv3
|
||||
license_files = license
|
||||
description = dupeGuru is a tool to find duplicate files on your computer.
|
||||
long_description = file:README.md
|
||||
long_description_content_type = text/markdown
|
||||
classifiers =
|
||||
Development Status :: 5 - Production/Stable
|
||||
Intended Audience :: End Users/Desktop
|
||||
License :: OSI Approved :: GNU General Public License v3 (GPLv3)
|
||||
Operating System :: MacOS :: MacOS X
|
||||
Operating System :: Microsoft :: Windows
|
||||
Operating System :: POSIX
|
||||
Programming Language :: Python :: 3.7
|
||||
Programming Language :: Python :: 3.8
|
||||
Programming Language :: Python :: 3.9
|
||||
Programming Language :: Python :: 3.10
|
||||
Programming Language :: Python :: 3 :: Only
|
||||
Topic :: Desktop Environment :: File Managers
|
||||
|
||||
[options]
|
||||
packages = find:
|
||||
python_requires = >=3.7
|
||||
install_requires =
|
||||
Send2Trash>=1.8.2,<2.0.0
|
||||
mutagen>=1.46.0,<2.0.0
|
||||
distro>=1.8.0,<2.0.0
|
||||
PyQt5 >=5.15.0,<6.0; sys_platform != 'linux'
|
||||
pywin32>=228; sys_platform == 'win32'
|
||||
semantic-version>=2.0.0,<3.0.0
|
||||
xxhash>=3.0.0,<4.0.0
|
||||
setup_requires =
|
||||
sphinx>=3.0.0
|
||||
polib>=1.1.0
|
||||
tests_require =
|
||||
pytest >=6,<7
|
||||
include_package_data = true
|
||||
|
||||
[options.entry_points]
|
||||
console_scripts =
|
||||
dupeguru = run.py
|
||||
26
setup.py
Normal file
26
setup.py
Normal file
@@ -0,0 +1,26 @@
|
||||
from setuptools import setup, Extension
|
||||
from pathlib import Path
|
||||
|
||||
exts = [
|
||||
Extension(
|
||||
"core.pe._block",
|
||||
[
|
||||
str(Path("core", "pe", "modules", "block.c")),
|
||||
str(Path("core", "pe", "modules", "common.c")),
|
||||
],
|
||||
include_dirs=[str(Path("core", "pe", "modules"))],
|
||||
),
|
||||
Extension(
|
||||
"core.pe._cache",
|
||||
[
|
||||
str(Path("core", "pe", "modules", "cache.c")),
|
||||
str(Path("core", "pe", "modules", "common.c")),
|
||||
],
|
||||
include_dirs=[str(Path("core", "pe", "modules"))],
|
||||
),
|
||||
Extension("qt.pe._block_qt", [str(Path("qt", "pe", "modules", "block.c"))]),
|
||||
]
|
||||
|
||||
headers = [str(Path("core", "pe", "modules", "common.h"))]
|
||||
|
||||
setup(ext_modules=exts, headers=headers)
|
||||
Reference in New Issue
Block a user