Mac下关闭chrome浏览器的自动更新

方法一:

首先关闭Chrome浏览器,然后进入目录“/Library/Google/GoogleSoftwareUpdate”。(进入的是Mac系统目录下的“/Library/Google/GoogleSoftwareUpdate”这个目录)

cd /Library/Google/GoogleSoftwareUpdate

然后删除该目录下的GoogleSoftwareUpdate.bundle即可。可以通过命令行删除,或者Finder手动删除都可以。
表示我们禁用chrome的自动更新成功了。如果你点击了“设置自动更新”,则刚才的“GoogleSoftwareUpdate.bundle”文件就会再次出现了。点击“不再询问”大功告成。通过“帮助->关于Google Chrome”可以查看信息。

方法二:

可能在有些Mac上发现在“/Library”这个根目录下没有Google目录,那么其实在“~/Library”这个用户目录下也有一个Google目录( 进入的是 /Users/mac名/Library/Google目录 或者 ~/Library/Google目录 这个用户目录)。在该目录下执行操作同样可以禁用自动更新。请执行以下命令:

cd ~/Library/Google
sudo chown root:wheel GoogleSoftwareUpdate

相当于修改了GoogleSoftwareUpdate这个文件夹的拥有者,而不仅仅是修改了权限,使Google对该文件夹没有写入权限。事实证明这种方式是可行的。重启Chrome完成以后通过“帮助->关于Google Chrome”可以查看信息。


Ubuntu安装Plex Media Player

Plex官网上并没有针对Linux提供Plex Media Player安装包,但是好在有一些第三方基于官方APP制作的原生Linux GTK程序可以使用。
Ubuntu 64位桌面版可以去https://knapsu.eu/data/plex/ 下载最新的AppImage直接使用。目前也支持绝大多数Linux的桌面发行版。
下载后给下载好的AppImage文件(如Plex_Media_Player_2.58.1-ae73e074_x64.AppImage)加运行权限即可。
此处通过命令行的方式详细的讲一下过程:
先在Home目录下新建一个AppImage目录:

mkdir -p ~/AppImages
cd ~/AppImages

然后从https://knapsu.eu/data/plex/ 找到最新的AppImage的URL,截至到本文发布时间(2022年1月4日),最新版是2.58.1
将其下载到~/AppImages,并将下载的文件改名。

wget https://knapsu.eu/data/plex/Plex_Media_Player_2.58.1-ae73e074_x64.AppImage
mv Plex_Media_Player_2.58.1-ae73e074_x64.AppImage Plex_Media_Player.AppImage

给这个AppImage加运行权限:

sudo chmod +x Plex_Media_Player.AppImage

运行这个文件

./Plex_Media_Player.AppImage

首次运行会提示是否将Plex Media Player的快捷方式加入系统程序列表,建议选是。如果选否的话,以后依然可以通过双击AppImage或者通过./Plex_Media_Player.AppImage方式启动。
这样Plex Media Player就安装成功了。
首次运行需要通过plex.tv/link 进行账户关联,而不是直接通过输入用户名密码的形式登录账户。因此可以判断,该Appimage大概率是从移动端或者TV端打包来的。


Ubuntu安装Enpass

Enpass可以在 CentOS 7, Fedora 28, Mint 19 and Ubuntu 16.04+的64位Linux系统上安装。
在Ubuntu上的安装非常简单:
在/etc/apt/sources.list中添加源:

sudo -i
sudo echo "deb https://apt.enpass.io/ stable main" > /etc/apt/sources.list.d/enpass.list

导入key:

wget -O - https://apt.enpass.io/keys/enpass-linux.key | tee /etc/apt/trusted.gpg.d/enpass.asc

安装:

apt-get update
apt-get install enpass
exit

Ubuntu 18.04/20.04 生产力配置及优化美化

〇、首先,将软件源更换为国内源(可跳过),安装升级速度更快。

左下打开软件列表搜索并打开Softwares & Updates, 在Ubuntu Software -> Download from -> Other -> China中选择合适的镜像站就可以了,推荐阿里云(mirrors.aliyun.com)镜像。
然后更新一下,

sudo apt update && sudo apt upgrade -y

一、基础软件安装

WPS Office 2019

首先,卸载libreoffice

sudo apt remove libreoffice-common

官网下载WPS office 2019 for Linux

sudo dpkg -i wps-office_11.1.0.8372_amd64.deb

如果缺失字体,需要下载wps_symbol_fonts.7z,解压后:

sudo mv wps_symbol_fonts /usr/share/fonts/

中文输入法

安装搜狗输入法,从官网下载最新的安装包,

sudo dpkg -i sogoupinyin_2.4.0.3469_amd64.deb
sudo apt install -f

从设置找到语言->管理安装语言,待更新安装完成,设置为fcitx
重新登入,在右上角的托盘->配置,添加搜狗输入法

PDF阅读器

福昕阅读器下载.run格式的安装包,直接运行安装(如果没有执行权限,sudo chmod +x xxxxxxxxxxxxx.run)。

网易云音乐

目前官网1.1.0版本的安装包有点问题,安装后可以使用sudo运行起来。
或者第三方打包
从网上能找到的最完美的解决方案是使用重新打包过的安装包(传送门),安装完成后,删除~/.cache/netcase-cloud-music一次,应该是该文件夹访问权限的原因,之后就可以正常使用了。
官网更新了18.04的1.2.1版本,下载直接安装即可。

Chrome

官网下载安装即可,有可能要翻墙。

视频播放器VLC

sudo apt install vlc

安装zsh(也可以安装fish)

sudo apt install zsh
# 切换到zsh
chsh -s /bin/zsh 

安装Oh-my-zsh

sh -c "$(wget https://raw.github.com/ohmyzsh/ohmyzsh/master/tools/install.sh -O -)"

安装插件highlight,高亮语法

git clone https://github.com/zsh-users/zsh-syntax-highlighting.git ${ZSH_CUSTOM:-~/.oh-my-zsh/custom}/plugins/zsh-syntax-highlighting
git clone https://github.com/zsh-users/zsh-autosuggestions ${ZSH_CUSTOM:-~/.oh-my-zsh/custom}/plugins/zsh-autosuggestions

在Oh-my-zsh的配置文件中~/.zshrc中添加插件

plugins=(git z extract zsh-syntax-highlighting zsh-autosuggestions)

最后使配置生效

source ~/.zshrc

这里需要重新登入一下,否则zsh只会在当前终端中应用。

QQ/TIM

QQ官方发布了Linux版本,古董界面丑的很,也难用的很。推荐10.3的安装方法。

Wine QQ/TIM

其次,使用wine安装qq,推荐以下两个教程,但都有不同程度的问题,如果可以忍受可以使用:

  • https://phpcj.org/wineqq/
  • https://github.com/askme765cs/Wine-QQ-TIM

    使用deepin-wine 安装

    这个暂时应该是比较完美的安装方法了
    项目地址deepin-wine-for-ubuntu,这个项目最近没有维护,不过安装方法是一样的,有人发帖更新了安装脚本。

    启用“Minimize on Click”

    gsettings set org.gnome.shell.extensions.dash-to-dock click-action 'minimize'

二、gnome shell 扩展

Ubuntu Software => Add-ons => Shell extensions中安装(或直接在Ubuntu Software主界面直接搜索)。

  1. NetSpeed
    监控并显示实时网速。
  2. Coverflow Alt-Tab
    优化Alt-Tab,帅!
    三 美化
  3. Terminal
    在终端中右键->preferences中,禁止菜单,禁止滚动条,以及在配色中
  4. 主题
    首先安装 tweaks,使用该软件统一管理主题中的各个部分

    sudo apt install gnome-tweak-tool 

    而具体的主题等可以在gnome-look.org找到,非常多,下面选择几个我觉得好看的设置。

  5. Shell
    如果Shell显示不可修改,

    sudo apt install gnome-shell-extensions

    重新登入一下,然后在Tweaks->Extensions中将User themes打开,重启tweacks在Apperance -> Themes -> Shell选择,我直接选择使用Vimix主题。

  6. 合并Top bar & Dock
    将桌面打造成Windows风格,
    安装插件,并在TWeaks->extensions中打开

    sudo apt install gnome-shell-extension-dash-to-panel

    重新登入,打开该插件的设置,进行细微处的调节。

    三、针对双系统

  7. 双系统造成的windows时间错误
    先在ubuntu下更新一下时间,确保时间无误:

    sudo apt install ntpdate
    sudo ntpdate time.windows.com

    然后将时间更新到硬件上

    sudo hwclock --localtime --systohc

    方法2
    http://ubuntuhandbook.org/index.php/2016/05/time-differences-ubuntu-1604-windows-10/

  8. rEFInd 引导双系统
    如果你的电脑支持UEFI启动引导又嫌弃默认的启动界面丑,你可以使用rEFInd来管理你的启动项,推荐一个主题Minimal. 引导设置可参考rEFInd引导Win10+Ubuntu14双系统.
  9. 隐藏grub引导菜单
    如果使用了其他引导,可以隐藏ubuntu的grub引导菜单:

    sudo vim /etc/default/grub

    修改内容为

    GRUB_DEFAULT=0
    GRUB_HIDDEN_TIMEOUT=0
    GRUB_HIDDEN_TIMEOUT_QUIET=true
    GRUB_TIMEOUT=0
    GRUB_DISTRIBUTOR=`lsb_release -i -s 2> /dev/null || echo Debian`
    GRUB_CMDLINE_LINUX_DEFAULT="quiet splash"
    GRUB_CMDLINE_LINUX=""
    GRUB_DISABLE_OS_PROBER=true

    更新grub

    sudo update-grub

修改Ubuntu键位为MacOS键位及外观

〇、前言

本文假定读者正在使用标准Mac键盘(左Command键在左Option键的右边)操作Ubuntu
Ubuntu 18.04/20.04 Desktop 英文版

一、全局调换LCtrl和LCommand

1.编辑配置文件

sudo nano /usr/share/X11/xkb/symbols/pc

2.找到如下两条配置项

key <LCTL> { [ Control_L ] };

key <LWIN> { [ Super_L ] };

3.修改为:

key <LCTL> { [ Super_L ] };

key <LWIN> { [ Control_L ] };

4.重启

二、修改系统快捷键

此处修改的快捷键我们按照键盘表面的标注来表示,而不是已经对调过的键位来表示。
找到Activites –> Settings –> Keyboard Shortcuts
Launch Terminal : Control + Command + T
Search : Command + Space
Switch Applocations : Command + Tab
Switch Windows of an application : Command + `
Save a screenshot to Pictures : Command + Shift + 4
Record a short screencast : Command + Shift + 5
Eject : Command + E
Log out : Shift + Command + Q
Lock Screen : Ctrl + Command + Q
Close Window : Command + Q
Hide Window : Command + H
Toggle Full Screen Mode : Control + Command + F
Raise Window if coverd otherwise lower it : Command + M

三、修改文件管理器快捷键并取消删除确认

Ubuntu的文件管理器默认是nautilus,想修改nautilus的快捷键目前还没有很直观的办法,除非修改源代码自己重新编译。但这个工程确实是有点大了。
因此我们使用python-nautilus,使python可以调用gtk库,并自己用python编写一个插件。
安装python-nautilus:

sudo apt install python-nautilus

创建插件文件。

mkdir -p ~/.local/share/nautilus-python/extensions
nano ~/.local/share/nautilus-python/extensions/OSXKey.py

以下是插件代码:

#!/usr/bin/env python
# -*- coding: utf-8 -*-

import os, gi
gi.require_version('Nautilus', '3.0')
from gi.repository import GObject, Nautilus, Gtk, Gio, GLib

def ok():
    app = Gtk.Application.get_default()
    #Move to trash
    app.set_accels_for_action( "view.move-to-trash", ["<Primary>BackSpace", "Delete"] )
    #Delete permanently
    app.set_accels_for_action( "view.delete-permanently-shortcut", ["<Primary><alt>BackSpace", "<shift>Delete"] )
    #Rename
    app.set_accels_for_action( "view.rename", ["Return", "KP_Enter"] )
    #Open
    app.set_accels_for_action( "view.open-with-default-application", ["<Primary>o","<Primary>Down"] )
    #Go Home
    app.set_accels_for_action( "win.go-home", ["<shift><Primary>h"] )
    #Go to Path
    app.set_accels_for_action( "win.enter-location", ["<Primary>g","<Primary>k"] )
    #Go up
    app.set_accels_for_action( "win.up", ["<Primary>Up"] )
    #Go back
    app.set_accels_for_action( "win.back", ["<Primary>Left"] )
    #Go forward
    app.set_accels_for_action( "win.forward", ["<Primary>Right"] )
    #Show/hide Sidebar
    app.set_accels_for_action( "app.show-hide-sidebar", ["<Primary><alt>s"] )

class KeybindingRemap(GObject.GObject, Nautilus.LocationWidgetProvider):
    def __init__(self):
        pass

    def get_widget(self, uri, window):
        ok()
        return None

取消删除至回收站确认:
通过dconf-editor来修改Nautilus的设置,先安装:

sudo apt install dconf-editor
dconf-editor

打开dconf-editor,搜索nautilus,找到/org/gnome/nautilus/preferences
comfrim-Trash 改为关闭

四、修改Terminal快捷键

Terminal快捷键的差异主要在复制、粘贴、中断以及标签操作上。
打开我们刚刚安装的dconf-editor
查找:/org/gnome/terminal/legacy/keybindings/
修改如下项(修改前先取消"Use default value"):
close-Tab : <Ctrl>w
close-Window : <Ctrl>q
copy: <Ctrl>c
find: <Ctrl>f
new-tab: <Ctrl>t
new-window: <Ctrl>n
paste: <Ctrl>v
select-all: <Ctrl>a
zoom-in: <Ctrl>equal
save-contents: <Ctrl>s
经过以上修改,复制变为Command+C,但是同时,Ctrl+C也失去了中断程序的功能。因为Ctrl实际上被系统识别为Super,而Super的键值是无法被Terminal捕捉的。现在中断程序的快捷键在Terminal里被自动替换为Ctrl+Alt+C(在键盘上标注为Command+Alt+C)。当然,我们也可以直接关闭Terminal窗口来中断当前程序。也可以通过stty intr ^I 命令来修改中断快捷键为Ctrl+I(键盘标注为Command + I )或者将I替换成其他自定义单一字母,替换前建议先使用stty -a命令查看系统默认快捷键。

五、用Caps lock切换输入法

0.此处默认读者已经配置好fcitx并且正常启用。
1.安装 Gnome Tweak(已安装可跳过)

sudo apt-get install gnome-tweak-tool

2.用 Gnome Tweak 把 caps lock 改成 menu 键
打开 Tweaks,依次点击 Keyboard&Mouse -> Additional Layout Options
继续依次点击 Caps Lock behavior -> Make Caps Lock an additional Menu key,设置自动保存,关闭窗口即可。
3.安装你喜欢的输入法,如搜狗输入法、GooglePinyin(fcitx-googlepinyin)、Sunpinyin(fcitx-sunpinyin),并在fcitx中添加。fcitx中要保留一个En-US,一个中文输入法。
4.

fcitx-configtool

配置fcitx,点击 Global Config,找到Trigger Input Method,更改快捷键,这时按下键盘上的Caps lock键,设置后会显示"MENU"。这时应该可以通过Caps Lock来切换输入法了。
如果按Caps Lock时中文输入法名称在输入框下方一闪而过,但没有成功切换,然而双击Caps Lock却可以成功切换的话,那么继续下一步设置。
Activites –> Settings –> Universal Access(AccessX) –> Typing Assistant(AccessX) –> Bounce Keys调整为On
建议将Acceptence delay调整到非常低的一个值,较高的值会影响正常的打字。比如,用较快的速度输入"Good"时,系统会将"oo"削减成一个"o"。建议先调到最低值,然后逐渐调高,直至单击Caps Lock能正常切换输入法为止。
这样即可使用Caps Lock流畅的切换输入法了。

六、修改窗口的关闭、最小化、最大化按钮到左边

之前安装的Gnome Tweak可以实现。具体操作:
打开Gnome Tweak。
Window Titlebars –> Titlebar Buttons –> Placement –> Left

七、修改Ubuntu外观:BigSur主题

安装Application主题和Icon主题

sudo apt install git gtk2-engines-murrine gtk2-engines-pixbuf sassc optipng inkscape libglib2.0-dev-bin
git clone https://github.com/vinceliuice/WhiteSur-gtk-theme
./WhiteSur-gtk-theme/install.sh
git clone https://github.com/vinceliuice/WhiteSur-icon-theme
./WhiteSur-icon-theme/install.sh

打开Gnome Tweak。
Appearance –> Themes –> Applications –> WhiteSur
Appearance –> Themes –> Icons –> WhiteSur
效果应用是即时的。
安装Dash To Dock插件(可选),把Dash变成Dock的样式(20.04实测无效)

sudo apt install make gettext
git clone https://github.com/micheleg/dash-to-dock.git
cd dash-to-dock
make
make install

重启后打开Gnome Tweak。
Extensions –> Dash to Dock –> On


Python3 Ping功能的若干种实现方式

方式一:自己编写实现ping功能

#!/usr/bin/env python3
# -*- encoding = utf-8 -*-
# 优点:全部通过python3自带库实现,无需安装任何第三方依赖库
# 缺点:需要root权限
# 用法:sudo python3 ping.py <host>

import time
import struct
import socket
import select
import sys

def chesksum(data):
  """
  校验
  """
  n = len(data)
  m = n % 2
  sum = 0 
  for i in range(0, n - m ,2):
    sum += (data[i]) + ((data[i+1]) << 8)#传入data以每两个字节(十六进制)通过ord转十进制,第一字节在低位,第二个字节在高位
  if m:
    sum += (data[-1])
  #将高于16位与低16位相加
  sum = (sum >> 16) + (sum & 0xffff)
  sum += (sum >> 16) #如果还有高于16位,将继续与低16位相加
  answer = ~sum & 0xffff
  #主机字节序转网络字节序列(参考小端序转大端序)
  answer = answer >> 8 | (answer << 8 & 0xff00)
  return answer 

  '''
  连接套接字,并将数据发送到套接字
  '''
def raw_socket(dst_addr,imcp_packet):
  rawsocket = socket.socket(socket.AF_INET,socket.SOCK_RAW,socket.getprotobyname("icmp"))
  send_request_ping_time = time.time()
  #send data to the socket
  rawsocket.sendto(imcp_packet,(dst_addr,80))
  return send_request_ping_time,rawsocket,dst_addr

  '''
  request ping
  '''
def request_ping(data_type,data_code,data_checksum,data_ID,data_Sequence,payload_body):
  #把字节打包成二进制数据
  imcp_packet = struct.pack('>BBHHH32s',data_type,data_code,data_checksum,data_ID,data_Sequence,payload_body)
  icmp_chesksum = chesksum(imcp_packet)#获取校验和
  imcp_packet = struct.pack('>BBHHH32s',data_type,data_code,icmp_chesksum,data_ID,data_Sequence,payload_body)
  return imcp_packet
  '''
  reply ping
  '''
def reply_ping(send_request_ping_time,rawsocket,data_Sequence,timeout = 2):
  while True:
    started_select = time.time()
    what_ready = select.select([rawsocket], [], [], timeout)
    wait_for_time = (time.time() - started_select)
    if what_ready[0] == []: # Timeout
      return -1
    time_received = time.time()
    received_packet, addr = rawsocket.recvfrom(1024)
    icmpHeader = received_packet[20:28]
    type, code, checksum, packet_id, sequence = struct.unpack(
      ">BBHHH", icmpHeader
    )
    if type == 0 and sequence == data_Sequence:
      return time_received - send_request_ping_time
    timeout = timeout - wait_for_time
    if timeout <= 0:
      return -1

  '''
  实现 ping 主机/ip
  '''
def ping(host):
  data_type = 8 # ICMP Echo Request
  data_code = 0 # must be zero
  data_checksum = 0 # "...with value 0 substituted for this field..."
  data_ID = 0 #Identifier
  data_Sequence = 1 #Sequence number
  payload_body = b'abcdefghijklmnopqrstuvwabcdefghi' #data
  dst_addr = socket.gethostbyname(host)#将主机名转ipv4地址格式,返回以ipv4地址格式的字符串,如果主机名称是ipv4地址,则它将保持不变
  print("正在 Ping {0} [{1}] 具有 32 字节的数据:".format(host,dst_addr))
  for i in range(0,4):
    icmp_packet = request_ping(data_type,data_code,data_checksum,data_ID,data_Sequence + i,payload_body)
    send_request_ping_time,rawsocket,addr = raw_socket(dst_addr,icmp_packet)
    times = reply_ping(send_request_ping_time,rawsocket,data_Sequence + i)
    if times > 0:
      print("来自 {0} 的回复: 字节=32 时间={1}ms".format(addr,int(times*1000)))
      time.sleep(0.7)
    else:
      print("请求超时。")

if __name__ == "__main__":
  if len(sys.argv) < 2:
    sys.exit('Usage: ping.py <host>')
  ping(sys.argv[1])

方式二:通过更为强大的scapy库实现

#!/usr/bin/env python3
# -*- encoding = utf-8 -*-
# sudo apt install python3-scapy
# 调用scapy库实现ping功能
# 优点:几乎没有
# 缺点:需要root权限
from scapy.layers.inet import IP, ICMP
from scapy.packet import Raw
from scapy.sendrecv import sr1
import logging
logging.getLogger("scapy.runtime").setLevel(logging.ERROR)  # 清除报错信息

def ping_one(dst):  # 构建函数
    ping_pkt = IP(dst=dst)/ICMP() / b'welcome!'  # 构建数据包,目的地址为dst,数据部分为字节字符串welcome!
    ping_result = sr1(ping_pkt, timeout=2, verbose=False)  # 发送数据包并将返回结构赋值到ping_result中
    try:
        if ping_result.getlayer(IP).fields['src'] == dst and ping_result.getlayer(ICMP).fields['type'] == 0 \
                and ping_result.getlayer(Raw).fields['load'] == b'welcome!':
            # 判断返回数据包是否为reply,通过源地址,返回数据与返回类型进行判断
            return dst, 1  # 如果为返回数据包,函数返回输入的目的地址与1
        else:
            return dst, 2  # 如果不是返回数据包,则函数2
    except Exception:  # 如果无法ping通,则不会有返回结果,会出现报错,如果产生报错,直接返回dst与2
        return dst, 2

if __name__ == '__main__':
    try:
        while True:
            dst = input('请输入目的IP地址:')
            if dst == '':
                print('目的地址为空,请重新输入!')
            else:
                break
        result = ping_one(dst)
        if result[-1] == 1:
            print('目的', result[0], '可达!')
        else:
            print('目的', result[0], '不可达!')
    except KeyboardInterrupt:
        print('成功接收信号,退户程序!')

方式三:使用tcping实现

#!/usr/bin/env python3
# -*- encoding = utf-8 -*-
# 优点:无需root权限
# 缺点:需要安装依赖tcping,仅能通过tcp实现ping,必须指定端口
# python3 -m pip install tcping
# https://pypi.org/project/tcping/
from tcping import Ping
# ping_check(地址,端口,超时秒数,ping次数)
def ping_check(address,port,timeout,times):
    ping = Ping(address, port, timeout)
    ping.ping(times) #ping 4 次
    statistics = ping.result.rows
    # statistics = [Statistics(host='192.168.1.1', port='80', successed=4, failed=0, success_rate='100.00%', minimum='0.09ms', maximum='0.25ms', average='0.14ms')]
    print(statistics[0])
    ret = ping.result.raw
    print(ret)
    ret = ping.result.table
    print(ret)
    #从返回的statistics元组中获得成功次数和平均延迟
    successed = statistics[0][2]
    # 返回状态及平均延迟
    if successed > 0:
        return True, statistics[0][-1]
    else:
        return False, statistics[0][-1]
if __name__ == '__main__':
    i = input("请输入要Ping的地址和端口(例:192.168.1.1:80):")
    addr = i.split(":")
    ping_result = ping_check(addr[0],addr[1],5,4)
    print(ping_result)

Python脚本:基于ffmpeg的视频格式转换工具

现在NAS已经很普及了,NAS上经常会存储海量的高清电影,但有时候我们希望把电影格式转换成便携一些的格式,直接在服务器上转换会更加的高效,因此我便写了这一个工具,可以做到无人值守的批量转换整个目录下任意视频。
只需要把这个脚本放在需要转换视频的目录下,然后运行即可。如果需要压制字幕,字幕文件名需要跟视频文件名保持完全一致(大小写敏感)。字幕支持srt和ass,utf8编码。如果在压制带字幕的视频时出现红字错误,先检查字幕编码是否正确,通过命令行cat查看字幕文件,如果输出乱码,则需要先对字幕文件进行转码处理。
支持以下几种创建转换队列的方式:

  1. 通过输入文件编号,用逗号或者空格或者点隔开,如1,2 3.4,会选择1、2、3、4号文件加入转换队列
  2. 通过输入文件区间,用英文减号隔开,如1-4,会选择1、2、3、4文件加入转换队列

每次转换,整个队列的转换参数是一致的。如果需要不同的转换参数,请分多次运行本脚本。
如果需要转换的队列太大,建议直接实机运行或者通过VNC运行,如果通过ssh运行本脚本,ssh超时断开会导致任务中断。转换后的视频在原视频目录下的out目录中,格式为mp4。转换时在out目录下遇到同名文件会直接覆盖,不提示。
注意:本脚本可以在Linux和Mac上运行,python3.7以上版本,需要提前安装好ffmpeg,最好是编译安装,with-libx264, with-lamemp3,压制ass格式的外挂字幕的话,还需要with-libass
话不多说,直接上代码:(文末附原文件下载)

#!/usr/bin/env python3
# -*- coding: UTF-8 -*-
import subprocess
import os,sys
#指定当前.py脚本的绝对路径
current_path = sys.path[0]
#指定ffmpeg的路径:
ffmpeg_path = "/usr/local/ffmpeg/bin/ffmpeg"
print("\033[36m","="*40,"\033[0m")
print("\033[35m欢迎使用ffmpeg视频转码脚本 v1.2\033[0m")
print("\033[35mLast Update: 2021/12/21\033[0m")
print("\033[35mAuthor: 1Fan Tech\033[0m")
print("\033[35m将本脚本与需要转换的视频放在同一目录下\033[0m")
print("\033[35m转换后的mp4视频输出在out目录\033[0m")
print("\033[35m如果目标视频已存在会自动覆盖\033[0m")
print("\033[36m","="*40,"\033[0m")
print("\033[32m当前目录文件列表:\033[0m")

#输出当前目录文件列表
print("\033[32m",current_path,"\033[0m")
#判断文件是否为视频文件
def is_video(t_file):
    v_list = ["mp4","mkv","mpg","mpeg","webm","flv","avi","mov"]
    rst = False
    for v in v_list:
        if t_file.lower().endswith(v):
            rst = True
    return rst
# x - y 类型字符串转换为列表
def dash_into_list(t_str):
    out_list = []
    if "-" in t_str:
        t_list = t_str.replace(" ","").split("-")
        try:
            t_list.remove("")
        except:
            pass
        if len(t_list)==2:
            try:
                start = int(t_list[0])
                end = int(t_list[1])
                if start<end:
                    for x in range(start,end+1):
                        out_list.append(x)
            except:
                pass
    return out_list
file_index = 0
file_list = []
print("\033[34m",file_index,".\t\033[32m  退出\033[0m")
for tmp_file in os.listdir(current_path):
    if not os.path.isdir(tmp_file) and is_video(tmp_file):
        real_file = os.path.join(current_path, tmp_file)
        file_list.append(real_file)
        file_index+=1
        file_size = int(os.path.getsize(real_file)/10485.76)/100 #文件大小以两位小数显示Mb
        print("\033[34m",file_index,".\t\033[32m ",tmp_file,"\033[33m\t",file_size,"MB\033[0m")
if file_index>0:
    loop_indicator = 0
    new_error = 0
    while loop_indicator == 0:
        new_error = 0
        print("\033[34m请输入您要转换的文件序号(1 - ",file_index,")可选择多个文件以逗号或空格或点分隔,或者通过x-y批量选择:\033[0m", end="")
        nums = input("")
        selected_files_list = []
        if nums =="0":
            sys.exit(0)
        elif "-" in nums:
            selected_files_list = dash_into_list(nums)
            if selected_files_list == []:
                new_error = 1
        else:
            input_files_list = nums.replace(".",",").replace(" ",",").replace(",",",").replace("。",",").split(",")
            for num in input_files_list:
                if num !="":
                    try:
                        num = int(num)
                    except:
                        new_error = 1
                        num = file_index + 1
                    if num > file_index:
                        new_error = 1
                    else:
                        if num not in selected_files_list:selected_files_list.append(num)
        if new_error == 0:
            loop_indicator = 1
        else:
            print("\033[31m请输入正确的序号!\033[0m")
    print("已选择:")
    for num in selected_files_list:
        print("\033[34m",num,".\033[32m ",file_list[num-1],"\033[0m")
    #开始构建vf参数序列
    select_sub_ext = ""
    while select_sub_ext=="":
        print("\033[0;32;40m请选择输入的字幕文件后缀:\033[0m")
        print("0. 不附加字幕")
        print("1. ass")
        print("2. srt")
        a = input("输入数字:")
        if a == "1":
            select_sub_ext = ".ass"
        elif a=="2":
            select_sub_ext = ".srt"
        elif a=="0" or a =="":
            select_sub_ext = "no"
        else:
            print("\033[0;31;40m输入有误,请重新输入!\033[0m")
    scale = ""
    while scale=="":
        print("\033[32m请选择视频分辨率:\033[0m")
        print("0. 保留原始分辨率[直接回车]")
        print("1. 720p")
        print("2. 1080p")
        print("3. 4K")
        a = input("输入数字:")
        if a == "1":
            scale = "-1:720"
        elif a=="2":
            scale = "-1:1080"
        elif a=="3":
            scale = "-1:2160"
        elif a=="0" or a == "":
            scale = "no"
        else:
            print("\033[34m输入有误,请重新输入!\033[0m")
    #开始构建参数序列
    args_list = []
    #追加解码器
    args_list.append("-vcodec")
    args_list.append("libx264")
    args_list.append("-acodec")
    args_list.append("libfdk_aac")
    #追加视频音频码率
    bitrate = ""
    while bitrate=="":
        print("\033[32m请选择视频/音频质量:\033[0m")
        print("0. 保留原始质量[直接回车]")
        print("1. 高质量(10M/320k)")
        print("2. 中质量(5M/320k)")
        print("3. 低质量(2M/256k)")
        a = input("输入数字:")
        if a == "1":
            bitrate = "1"
            args_list.append("-b:v")
            args_list.append("10240k")
            args_list.append("-b:a")
            args_list.append("320k")
        elif a=="2":
            bitrate = "2"
            args_list.append("-b:v")
            args_list.append("5120k")
            args_list.append("-b:a")
            args_list.append("320k")
        elif a=="3":
            bitrate = "3"
            args_list.append("-b:v")
            args_list.append("2048k")
            args_list.append("-b:a")
            args_list.append("256k")
        elif a=="0" or a == "":
            bitrate = "no"
        else:
            print("\033[34m输入有误,请重新输入!\033[0m")
    print(args_list)
    start = ""
    for file_index in selected_files_list:
        file = file_list[file_index-1]
        #获取视频文件不带后缀的文件名
        videofilename = file[:file.rfind('.')]
        #获取不带路径不带后缀的文件名(输出时使用)
        pure_videofilename = videofilename[videofilename.rfind('/')+1:]
        #初始化执行列表
        exec_list = list([ffmpeg_path, "-y", "-i"])
        print("\033[32m发现视频文件: \033[35m",file,"\033[0m")
        #file_vid_list.append(file)
        if select_sub_ext != "no":
            sub_file = videofilename + select_sub_ext
            if os.path.isfile(sub_file):
                print("\033[32m发现字幕文件: \033[35m",sub_file,"\033[0m")
                #file_ass_list.append(sub_file)
                add_sub = select_sub_ext
            else:
                add_sub = "no"
                print("\033[34m未找到视频对应的字幕文件:\033[35m",sub_file,"\033[34m本视频不添加字幕!\033[0m")
        else:
            add_sub = "no"
        if start =="":
            #开始构建执行任务:
            #追加视频文件
            exec_list.append(file)
            #追加字幕及分辨率
            if add_sub!="no" or scale!="no":
                exec_list.append("-vf")
                if add_sub==".ass":
                    vf_args = "ass=" + sub_file
                    args_connector = "," #添加参数间隔符
                elif add_sub==".srt":
                    vf_args = "subtitles=" + sub_file
                    args_connector = "," #添加参数间隔符
                else:
                    vf_args = ""
                    args_connector = ""
                if scale != "no":
                    vf_args = vf_args + args_connector + "scale=" + scale 
                exec_list.append(vf_args)
            #追加之前用户收入的自定义参数
            exec_list.extend(args_list)
            # 输出文件名
            exec_list.append(current_path + "/out/" + pure_videofilename + ".mp4")
            #开始检查输出目录是否存在
            if not os.path.exists(os.path.join(current_path, "out")):
                os.makedirs(os.path.join(current_path, "out"))
            print("\033[32m开始执行转换命令:",exec_list,"\033[0m")
            #开始最终执行
            subprocess.call(exec_list)
else:
    sys.exit(0)

下载:videoconv.py


Ubuntu20.04使用ZeroTier搭建Moon服务器以及在OpenWRT上的应用

服务器安装Zerotier

公网服务器,需要有固定IP,一条命令安装Zerotier:

curl -s https://install.zerotier.com/ | sudo bash

出现如下提示则代表安装完成:

*** Enabling and starting ZeroTier service...
Synchronizing state of zerotier-one.service with SysV service script with /lib/systemd/systemd-sysv-install.
Executing: /lib/systemd/systemd-sysv-install enable zerotier-one

*** Waiting for identity generation...

*** Success! You are ZeroTier address [ 515xxxxxxx ].

生成Moon配置文件

cd /var/lib/zerotier-one
sudo zerotier-idtool initmoon identity.public > moon.json
# 然后编辑该配置文件
sudo nano moon.json

默认该配置文件是如下内容:

{
 "id": "515xxxxxxx",
 "objtype": "world",
 "roots": [
  {
   "identity": "515xxxxxxx:0:e4d8xxxxxx11d962bb14771fa1e9bf1e10f9519c82590c372abbad4c6b2b473f27c389f6f7c8c328d513ce449xxxxxxe4956a318af32e16df>
   "stableEndpoints": []
  }
 ],
 "signingKey": "54685533b0ef4fc66f92e6a010dc304c4531acdd68a4486d800436d42aabfc5a9eed745cb9b7a3fe0983f521adc672f5b153bed420782d431c35058e3110c3>
 "signingKey_SECRET": "3325fd28ec1ec8ae058c255825bc3259d1935bf4bb2b23e361eefec5729337e0f196caba311ba42245f2b77d707f04f5ed33c81b8bf74d31189524c>
 "updatesMustBeSignedBy": "54685533b0ef4fc66f92e6a010dc304c4531acdd68a4486d800436d42aabfc5a9eed745cb9b7a3fe0983f521adc672f5b153bed420782d431c3>
 "worldType": "moon"
}

其中"id"是本机器在zerotier中的id,此处默认无需修改
"stableEndpoints"需要填写服务器的公网IP(假设此处我们的服务器公网IP为114.114.114.114,默认9993端口, 且是UDP协议的, 此处在防火墙上需要开放UDP,否则其他设备连接不上Moon
"stableEndpoints": ["114.114.114.114/9993"]
仅需修改这一条后保存退出。

生成签名文件

sudo zerotier-idtool genmoon moon.json

运行后提示类似wrote 000000515xxxxxxx.moon (signed world with timestamp 1640010832xxxx)表示签名完成。文件名去掉开头的00000就是机器的ID

将 moon 节点加入网络

在/var/lib/zerotier-one中新建一个moons.d的目录,我们默认目前依旧在/var/lib/zerotier-one目录下,然后将刚才签名好的moon文件移动道moons.d中

sudo mkdir moons.d
sudo mv 000000515xxxxxxx.moon moons.d

重启zerotier服务以应用配置

sudo service zerotier-one restart 

此时,Moon服务器已经配置完成、正常运行,并且可以接受客户端的连接请求了
下一步准备配置客户端,首先通过SFTP或者其他方式,将刚刚生成的000000515xxxxxxx.moon下载到本地。

配置OpenWRT软路由

我们前提默认OpenWRT已经安装了Zerotier并且已经配置完毕
ssh登录OpenWRT后,在zerotier的目录下创建moons.d目录

mkdir /etc/config/zero/moons.d

加入MOON服务器方式一(推荐)

通过SFTP将moon服务器中的000000515xxxxxxx.moon文件上传到刚刚创建的moons.d目录中

加入MOON服务器方式二

如果不方便上传文件,那么可以简单通过命令加入:

zerotier-cli orbit 515xxxxxxx 515xxxxxxx

后面两串数字是MOON服务器的ID,两个的原因是orbit参数支持同时连接两台MOON服务器(同样,通过moon签名文件的方式也可以连接多个MOON服务器),本文中我们仅配置了一台MOON服务器,因此两个ID相同即可。
重启软路由,输入zerotier-cli listpeers以检查zerotier目前已经连接的节点是否包含我们刚刚配置好的moon服务器。
如果不出意外,反馈的应该是这样的:

200 listpeers <ztaddr> <path> <latency> <version> <role>
200 listpeers 296cxxxx17 192.168.2.191/29994;12457;5030 21 1.8.3 LEAF
200 listpeers 515xxxxxxx 114.114.114.114/9993;2347;2073 27 1.8.4 MOON
200 listpeers 61dxxxxx89 - -1 1.4.6 LEAF
200 listpeers 61dxxxxxcb 50.7.73.34/9993;2347;1448 905 - PLANET
200 listpeers 62fxxxxx71 50.7.252.138/9993;2347;1078 1266 - PLANET
200 listpeers 778xxxxx90 103.195.103.66/9993;2347;2050 285 - PLANET
200 listpeers 992xxxxxb7 195.181.173.159/9993;2347;2066 279 - PLANET

其中role列为MOON的那一条如果显示出IP与版本号,说明已经成功连接了。latency为MOON服务器的延迟。MOON为用户自建节点服务器,PLANET为官方节点服务器,LEAF为客户端。

疑问:MOON服务器本身,是否需要加入我们自己组建的Zerotier局域网呢?
答案:不需要且不能加入。因为加入之后,这台服务器就的角色就变成了LEAF。


MacOS编译安装ffmpeg

安装XCode

首先,必须安装XCode,XCode提供了Mac平台开发环境必须的gcc编译器。XCode安装后需要先打开一次,同意一下条款才能在命令行里使用。同时,如果你不确定你的Xcode版本是否提供了命令行工具,需要在终端里运行一下:

xcode-select --install 

如果不提示安装Command Line Tools,说明就已经默认安装好了命令行工具了。

下载并安装brew

homebrew是一个Mac平台的包管理工具,类似Debian/Ubuntu的apt,提供了许多Mac没有的Unix/Linux工具,例如wget。安装homebrew很简单:

ruby -e "$(curl -fsSkL raw.github.com/mxcl/homebrew/go)"

安装依赖库

brew install automake yasm git pkg-config fdk-aac libass libtool libvorbis libvpx opus sdl sdl2 shtool texi2html theora wget x264 x265 xvid speex lame

git克隆ffmpeg并安装

git clone https://git.ffmpeg.org/ffmpeg.git ffmpeg
cd ffmpeg
./configure --prefix=/usr/local/ffmpeg --enable-gpl --enable-nonfree --enable-libfdk-aac --enable-libmp3lame --enable-libx264 --enable-libx265 --enable-libass --enable-libspeex --enable-libopus --enable-libtheora --enable-libvorbis --enable-libvpx --enable-libfreetype --enable-videotoolbox --enable-filter=delogo --enable-debug --disable-optimizations --enable-shared --enable-pthreads --enable-version3 --enable-hardcoded-tables --cc=clang --host-cflags= --host-ldflags= --disable-x86asm
sudo make &&sudo make install

配置环境变量

sudo nano ~/.bash_profile

添加以下代码:

export PATH="$PATH:/usr/local/ffmpeg/bin"
export PKG_CONFIG_PATH="$PKG_CONFIG_PATH:/usr/local/lib/pkgconfig:/usr/lib/pkgconfig:/usr/local/Cellar/sdl2/2.0.8/lib/pkgconfig:/usr/local/ffmpeg/lib/pkgconfig"
PKG_CONFIG_PATH="$PKG_CONFIG_PATH:/usr/local/lib/pkgconfig:/usr/lib/pkgconfig:/usr/X11/lib/pkgconfig"
CFLAGS='freetype-config --cflags' LDFLAGS='freetype-config --libs'

Ubuntu20.04 下编译安装ffmpeg

安装编译必须的包

sudo apt install build-essential pkg-config checkinstall git cmake yasm nasm 

安装依赖库

sudo apt install libfaac-dev libjack-jackd2-dev libgmp3-dev libmp3lame-dev libopencore-amrnb-dev libopencore-amrwb-dev libsdl1.2-dev libtheora-dev libva-dev libvdpau-dev libvorbis-dev libx11-dev libxfixes-dev libxvidcore-dev texi2html zlib1g-dev  libx265-dev libx264-dev libspeex-dev libfdk-aac-dev libnuma-dev  gnutls-bin libaom-dev libass-dev libbluray-dev libopenmpt-dev libopus-dev libshine-dev libsnappy-dev libsoxr-dev libtwolame-dev libvo-amrwbenc-dev libvpx-dev libwavpack-dev libwebp-dev liblzma-dev

安装SDL2

SDL是编译 ffplayer 所必须的外部库,而 Ubuntu 环境并不会对其默认安装,故而需要用户手动安装。而 SDL1 有些过时,目前使用广泛的是 SDL2,因此这里推荐安装SDL2。
SDL 下载地址:

http://www.libsdl.org/download-2.0.php

下载后,同样解压,然后进入目录

./configure --prefix=/usr/local
sudo make
sudo make install

git编译安装ffmpeg

从gitee网站上clone ffmpeg 国内的网站速度会快很多

git clone https://gitee.com/mirrors/ffmpeg.git
cd ffmpeg
#切换到4.4 分支 可以切换到任何自己想要的分支上
git checkout release/4.4
cd ffmpeg

配置自己需要哪些库

./configure --prefix=/usr/local/ffmpeg --enable-shared --enable-gpl --enable-version3 --enable-sdl2 --enable-fontconfig --enable-iconv --enable-libass --enable-libbluray --enable-libmp3lame --enable-libopencore-amrnb --enable-libopencore-amrwb --enable-libopus --enable-libshine --enable-libsnappy --enable-libsoxr --enable-libtheora --enable-libtwolame --enable-libvpx --enable-libwebp --enable-libx264 --enable-libx265 --enable-libxml2 --enable-lzma --enable-zlib --enable-gmp --enable-libvorbis --enable-libvo-amrwbenc --enable-libspeex --enable-libxvid --enable-libaom --enable-libopenmpt --enable-libfdk-aac --enable-nonfree --enable-shared --enable-pthreads --enable-version3 --enable-hardcoded-tables --cc=gcc --host-cflags= --host-ldflags=

开始make

make
# 安装 安装位置为上面 --prefix= 后面的目录 可以随意更改 
sudo make install

手动指定ffmpeg自带的组件的库

cd /etc/ld.so.conf.d/
sudo touch ffmpeg.conf
sudo nano ffmpeg.conf

在文件中写入你的ffmpeg lib的位置 本文中的是/usr/local/ffmpeg/lib

/usr/local/ffmpeg/lib
sudo ldconfig

配置环境变量

将 FFmpeg 添加至环境变量:

sudo nano ~/.bashrc

将光标调至文件末尾
添加如下内容:

export PATH=$PATH:/usr/local/ffmpeg/bin

保存文件