Я использую xmbc для просмотра своих телешоу. До того, как я начал использовать xmbc, я загрузил несколько первых сезонов "Bleach". Что я могу сказать, я фанат аниме ;-). Они называются: "bleachxx.mp4", где xx - это номер эпизода по отношению квся серия, а не к сезону. поэтому "bleach21.mp4" - это 1-й эпизод второго сезона и 21-й в целом. Однако сами файлы разделены на собственные папки сезона.
Я знаю, что можно использовать команду "rename" для пакетного переименования файлов. После долгих манипуляций с ней с помощью этого руководства:http://tips.webdesign10.com/how-to-bulk-renamed-files-in-linux-in-the-terminalЯ получил такую команду:
rename -n 's/.*(\d{2}).*$/Bleach S0XE$1/' *
Проблема в том, что команда переименует все файлы в «Bleach S0XExx», но поскольку файлы имеют общие номера файлов, bleach52.mp4 --> «Bleach S03E52.mp4», хотя в третьем сезоне явно нет 52 серий.
Тогда я хотел бы узнать, есть ли способ применить математические операции к переименованию файлов. Это решило бы проблему, поскольку я мог бы вычесть количество эпизодов в предыдущих сезонах из общего числа и по сути получить номер сезона.
Пример: если в первом сезоне 20 серий, то 25-20=5, поэтому 25-я серия является пятой во втором сезоне, и переименование будет выполнено как обычно.
Итак, можно ли как-то изменить значения переименования, применив математические операции?
PS Извините за длинное объяснение, но я не был уверен, как это сделать и будет ли мой вопрос понятен без него.
решение1
Если вы достаточно преданы своему делу, чтобы возиться с регулярными выражениями (в приведенном примере), то я предлагаю вам сделать еще один шаг и написать небольшой скрипт - скажем, на Python. Таким образом, вы сможете выполнять абсолютно любые преобразования имен файлов.
Я бы оценил, что скрипт на Python будет не длиннее 15-20 строк, так что это определенно не огромная задача. Использование только регулярных выражений, как вы пытаетесь, гораздо более ограничено.
Вот мое мнение по поводу такого сценария:
#!/usr/bin/python
import os,re
files = os.listdir('.')
SEASONS = (
(1, 1, 3), # the format is - season number, first episode, last episode
(2, 50,52),
(3, 53,55),
(4, 56,99),
)
for f in files:
# skip all files which are not .mp4
if not f.endswith(".mp4"):
continue
# find the first number in the filename
matches = re.findall("\d+", f)
if not len(matches):
print "skipping", f
num = int(matches[0])
for season in SEASONS:
if num <= season[2]:
season_num = season[0]
ep_num = num - season[1] + 1
new_file_name = "BleachS%02dE%02d.mp4" % (season_num, ep_num)
# This is for testing
print "%s ==> %s" % (f, new_file_name)
# Uncomment the following when you're satisfied with the test runs
# os.rename(f, new_file_name)
break
print "Done"
Похоже, я недооценил размер скрипта (сейчас он составляет 36 строк), хотя я уверен, что если вы зайдете на StackOverflow с этим кодом, то получите множество предложений, которые будут гораздо более элегантными.
И только потому, что я сказал, что это можно сделать в 15 строк... далее следует 20 строк, 5 из которых - конфигурация :P
#!/usr/bin/python
import os, re, glob
SEASONS = (
{'num':1, 'first':1, 'last':3}, # the format is - season number, first episode, last episode
{'num':2, 'first':50, 'last':52},
{'num':3, 'first':53, 'last':55},
{'num':4, 'first':56, 'last':99},
)
files = glob.glob('bleach*.mp4')
for f in files:
num = int(re.findall("\d+", f)[0]) # find the first number in the filename
for season in SEASONS:
if num > season['last']: continue
new_file_name = "BleachS%02dE%02d.mp4" % (season['num'], num - season['first'] + 1)
print "%s ==> %s" % (f, new_file_name) # This is for testing
# os.rename(f, new_file_name) # Uncomment this when you're satisfied with the test runs
break
решение2
Я написал вам скрипт на Python.
import sys
import os
import glob
import re
def rename():
episode_counts = (0, 20, 41, 63)
episode_pairs = []
for index, num in enumerate(episode_counts):
if index < len(episode_counts) - 1:
episode_pairs.append((num, episode_counts[index+1]))
episodes = glob.glob(os.path.join(sys.argv[1], '*'))
for episode in episodes:
match = re.search('.*(\d{2}).*$', os.path.basename(episode))
episode_num = match.group(1)
for season, pair in enumerate(episode_pairs):
if int(episode_num) in range(pair[0]+1, pair[1]+1):
dirname = os.path.dirname(episode)
path = os.path.join(dirname, 'Bleach S{0}E{1}'.format(season+1, episode_num))
os.rename(episode, path)
if __name__ == "__main__":
rename()
Я совсем новичок в python (отчасти поэтому я это и написал, для практики), так что это, возможно, не лучший скрипт в мире. Но я попробовал его на нескольких тестовых файлах, и, кажется, он работает.
Просто отредактируйте строку около верха episode_counts = ...
до последних номеров серий сезона. Я вставил первые три изэта страница.
Сохраните код как-то так episode_renamer.py
и используйте его с python episode_renamer.py /path/to/episodes/
.
решение3
В репозитории доступен файловый менеджер Thunar, имеющий возможность массового переименования.
Более подробную информацию можно найти на этом сайте: http://thunar.xfce.org/pwiki/documentation/bulk_renamer
Установить Thunar можно в Центре программного обеспечения или в командной строке с помощью:
sudo apt-get install thunar
Есть и другие способы сделать это в bash или путем написания кода, но графический инструмент, вероятно, даст вам то, что нужно, быстрее.
решение4
Продолжать стегать эту лошадь... если это одноразовое переименование, я бы сделал что-то одноразовое, например:
cd /path/to/stuff
ls bleach*-lq.mp4 >x
paste x x >y
sed 's-^-mv -' <y >z
vim z # or use your favorite editor
# to rename the files manually
bash -x z
rm x y z
Если вы собираетесь использовать его более одного раза и для более чем одного типа эпизодов... то Python — это то, что вам нужно, и приведенные выше примеры вполне подойдут.