2018年7月

之前写的一个python小程序,在windows服务器上跑的还可以,偶尔出点小意外,打算换到linux服务器上去跑,本来想着应该没啥问题,结果把环境配好以后,一运行,出问题了,报错 FileNotFoundError: [Errno 2] No such file or directory ,这是什么鬼,找不到文件???而且还是在windows环境下能跑,linux环境下跑就出问题了?看错误信息,好像是subprocess模块报的错,难道是环境没配好?子程序缺文件?不能运行?直接在shell里运行一下子命令看看,没有问题啊,可以正常执行的,这还真是奇怪了。

google了一下,发现是一个参数惹的祸,shell=False,因为这个程序用了subprocess模块来执行一些子程序

subprocess.run(args, *, stdin=None, input=None, stdout=None, stderr=None, shell=False, cwd=None, timeout=None, check=False, encoding=None, errors=None, text=None, env=None)

shell参数默认是False,因为在windows和linux环境下subprocess模块执行子程序的方式不一样,官方文档是这样写的

Execute a child program in a new process. On POSIX, the class uses os.execvp()-like behavior to execute the child program. On Windows, the class uses the Windows CreateProcess() function. The arguments to Popen are as follows.
args should be a sequence of program arguments or else a single string. By default, the program to execute is the first item in args if args is a sequence. If args is a string, the interpretation is platform-dependent and described below. See the shell and executable arguments for additional differences from the default behavior. Unless otherwise stated, it is recommended to pass args as a sequence.
On POSIX, if args is a string, the string is interpreted as the name or path of the program to execute. However, this can only be done if not passing arguments to the program.
On Windows, if args is a sequence, it will be converted to a string in a manner described in Converting an argument sequence to a string on Windows. This is because the underlying CreateProcess() operates on strings.
The shell argument (which defaults to False) specifies whether to use the shell as the program to execute. If shell is True, it is recommended to pass args as a string rather than as a sequence.
On POSIX with shell=True, the shell defaults to /bin/sh. If args is a string, the string specifies the command to execute through the shell. This means that the string must be formatted exactly as it would be when typed at the shell prompt. This includes, for example, quoting or backslash escaping filenames with spaces in them. If args is a sequence, the first item specifies the command string, and any additional items will be treated as additional arguments to the shell itself.
On Windows with shell=True, the COMSPEC environment variable specifies the default shell. The only time you need to specify shell=True on Windows is when the command you wish to execute is built into the shell (e.g. dir or copy). You do not need shell=True to run a batch file or console-based executable.
args is required for all calls and should be a string, or a sequence of program arguments. Providing a sequence of arguments is generally preferred, as it allows the module to take care of any required escaping and quoting of arguments (e.g. to permit spaces in file names). If passing a single string, either shell must be True or else the string must simply name the program to be executed without specifying any arguments.

所以在linux环境下,当shell=False(默认)时,subprocess.Popen使用os.execvp()来执行子程序,args参数需要是一个列表,如果args参数是个字符串的话,会被当做是命令本身或者可执行文件的路径,也就是说字符串只能是命令本身,而不能有额外的命令参数,如果字符串里带有命令参数的话,那参数也会被当作命令本身传递给os.execvp()来执行,比如说你要执行 cat test.txt, cat test.txt这整个字符串都会被当作可执行文件的路径来执行,但是又不存在cat test.txt这样的一个可执行的命令或者可执行文件的路径,所以就会报FileNotFoundError: [Errno 2] No such file or directory这样的错了。

shell=True时,会直接使用shell来执行子程序,如果args参数是字符串会直接传递给shell执行,如果args参数是个列表,args[0] 被视为命令交给shell执行,args[1:] 则会被忽略或者如果是可以改变shell行为的有效参数则对shell做相应调整,比如subprocess.run('cat test.txt', shell=True)正常执行查看test.txt文件,而subprocess.run(['cat', 'test.txt'], shell=True)则只会执行cat命令,而不会查看test.txt文件,而test.txt会被当作shell本身的参数,但是test.txt又不是shell的有效参数所以被忽略。

简单说就是当args参数是个字符串时,需要设置shell=True,当args参数是个列表的时候,shell保持默认的False。

打算搭一个短网址系统自用,找了找开源的程序,发现有YOURLS,Polr两个程序用貌似不错,比较了一下,打算使用YOURLS,虽然界面有点老旧,不过没关系,自用足够了,而且貌似插件扩展开发什么的也比较简单,就他了。

直接从 https://github.com/YOURLS/YOURLS 下载最新的源码,上传到服务器上
user/config-sample.php 重命名为 user/config.php,并编辑相应配置
访问 http://your-own-domain-here.com/admin/ ,点击 Install YOURLS 开始安装

不过出问题了,报了500的错误,emmm~什么鬼,因为一开始用的是php7,所以还以为是YOURLS不兼容php7,切换到php5.6,结果发现还是一样的问题,安装报500的错误,这就奇怪了,看来应该不是php版本不兼容的问题,在github的issues里搜了一下,看到有把数据库的字符集排序规则设置为 utf8_general_ci 解决的,难道是这个原因?打开phpmyadmin,发现数据库的排序规则是用的utf8mb4_general_ci,改成 utf8_general_ci 试试,嘿,可以,还真是可以正常安装了。

哈哈,又下了一个adsense,用之前新搭的那个站,本来打算那个号如果再被拒的话,就换这个新搭的站申请,结果那个号出乎意料的下号了,但是想着这个站既然搭好了也别浪费,再申请一个号试试,结果很美丽,一次申请就下号了。

简单说一下这个站的情况,是用去年趁黑色星期五打折时注册的一个域名,然后就一直闲置着,这个站是6月26号上线的,用的wordpress,搞了几百篇文章,发布时间从去年开始,然后还另外加了些文章做定时发布,设置的每天自动更新一篇文章,然后就啥也没管了,流量是零,emmm~,这次也没有刷流量,不过发现google倒是收录了不少,才上线不到10天,就有几百条收录了,前天用这个站申请的Google AdSense,今天就下号了,一次过,所以这么看来过不过审应该跟有没有流量关系不大,主要应该还是看网站,这个站用的是一个新闻类型的主题,整站文章也多一些,所以网站看起来内容应该算是比较丰富的,哦对,这个号也是申请的美号,英文站。

总结,域名时间久一点,半年以上?,内容丰富一些,emmm~就是网站的文章看着要多,不能太少,一百篇以上?,如果是原创的话,可能就不需要数量了,不过估计得要有质量了吧,哈哈,或者说,没质量数量凑?有质量不要数量?。

搞点啥站呢?搞点啥站能有流量呢?没有流量,不都是扯淡,还要看怎么过PIN,怎么收款。

在aws上开了一台windows服务器,用的操作系统是Windows Server 2016,结果在命令行操作时,发现中文全变成小方框问号了,无法正常显示,也无法输入中文,一开始以为是英文的操作系统没有安装中文语言包的原因,赶紧在控制面板里把中文语言包下载安装一下,结果发现命令行还是不能正确显示中文。

Windows Server 2016 cmd命令行窗口中文变成小方框问号无法正确显示

搜一下,说是命令行窗口的字符集不对,要改一下设置字符集为UTF-8
输入命令 chcp 65001
回车显示 Active code page: 65001
不过中文还是无法正确显示,再把字符集换成GBK的试试
输入命令 chcp 936
回车显示 Invalid code page
呵呵,看来应该不是这个原因。

接着搜,说是要把系统语言改成中文的,试试看,在控制面板里修改

Control Panel->Region->Administrative->Change system locale->Chinese

Windows Server 2016 cmd命令行窗口中文变成小方框问号无法正确显示

修改完成,重启系统

Windows Server 2016 cmd命令行窗口中文变成小方框问号无法正确显示

OK,问题解决,中文可以正常显示了。