Ubuntu 开机启动图形界面应用程序

概述

目标:开机全屏运行浏览器,自动打开指定网址;

环境:Ubuntu 18.04

分析

整理一下大致思路,比较简单:

1. 取消桌面启动,减少资源浪费;
2. 用户自动登陆;
3. 自动打开图形界面应用程序;

准备

更换apt-get源

系统启动后,第一件事就是把apt-get的源修改为国内镜像,本文用的是阿里源,配置过程如下:

cd /etc/apt/
# 习惯性操作
$ sudo cp sources.list sources.list-bak
# 将下面一段代码替换掉sources.list原来内容
$ sudo vi sources.list
# ------------------start---------------------
deb http://mirrors.aliyun.com/ubuntu/ trusty main restricted universe multiverse
deb http://mirrors.aliyun.com/ubuntu/ trusty-security main restricted universe multiverse
deb http://mirrors.aliyun.com/ubuntu/ trusty-updates main restricted universe multiverse
deb http://mirrors.aliyun.com/ubuntu/ trusty-proposed main restricted universe multiverse
deb http://mirrors.aliyun.com/ubuntu/ trusty-backports main restricted universe multiverse
deb-src http://mirrors.aliyun.com/ubuntu/ trusty main restricted universe multiverse
deb-src http://mirrors.aliyun.com/ubuntu/ trusty-security main restricted universe multiverse
deb-src http://mirrors.aliyun.com/ubuntu/ trusty-updates main restricted universe multiverse
deb-src http://mirrors.aliyun.com/ubuntu/ trusty-proposed main restricted universe multiverse
deb-src http://mirrors.aliyun.com/ubuntu/ trusty-backports main restricted universe multiverse
# ------------------end---------------------

# 更新源地址
$ sudo apt-get update

软件包安装

chromium-browser是必须的,其他软件包根据自己的需求。

$ sudo apt-get install vim
$ sudo apt-get install nginx
$ sudo apt-get install chromium-browser

开始

设置系统运行级别

系统启动默认进入字符界面的配置有2种方式:

1. 修改grubquiet splash改为text

2. 使用systemd来设置runlevels
本文主要是第二种方式,Ubuntu使用systemd而不是init,因此runlevels的概念被术语target所取代。因此init-based runlevelssystemd-based targets之间确实存在关系,映射如下:
┌─────────┬───────────────────┐
│Runlevel │ Target │
├─────────┼───────────────────┤
│0 │ poweroff.target │
├─────────┼───────────────────┤
│1 │ rescue.target │
├─────────┼───────────────────┤
│2, 3, 4 │ multi-user.target │
├─────────┼───────────────────┤
│5 │ graphical.target │
├─────────┼───────────────────┤
│6 │ reboot.target │
└─────────┴───────────────────┘
如果只是修改当前runlevels,执行 sudo systemctl isolate multi-user.target即可
咱们要修改默认runlevels,执行以下命令:

$ sudo systemctl enable multi-user.target
$ sudo systemctl set-default multi-user.target
# 设置后查看runlevels,结果为:N 3
$ runlevels

重启后系统自动进入字符界面,如果想启动桌面,执行startx即可

设置用户自动登陆

$ sudo mkdir  /etc/systemd/system/getty@tty1.service.d/
$ sudo vim /etc/systemd/system/getty@tty1.service.d/override.conf
# 将以下内容写入文件保存
# ------------------start---------------------
[Service]
ExecStart=
ExecStart=-/sbin/agetty --noissue --autologin sql %I $TERM
Type=idle
# ------------------end---------------------

设置浏览器自启动

编写启动脚本

启动浏览器需要有X11的支持,借鉴startx的思路:先启动xserver,再启动client;其实把client部分换掉就行了。

$ cd /usr/bin/
$ sudo cp startx mystartx
$ sudo vim mystartx

mystartx中找到这段内容:

xinit "$client" $clientargs -- "$server" $display $serverargs

在它前面插入:

client=/usr/bin/chromium-browser
clientargs=" --kiosk --incognito --start-maximized --start-fullscreen --disable-session-crashed-bubble http://www.u396.com"

注意1:此时针对1080*1920分辨率,不能全屏,解决方法:

cd .config/chromium/Default/
cat Preferences |json_pp > tmp

修改tmp内容:

      "window_placement" : {
         "work_area_right" : 1920,
         "bottom" : 1080,
         "work_area_top" : 0,
         "top" : 10,
         "left" : 10,
         "right" : 1920,
         "maximized" : false,
         "work_area_bottom" : 1080,
         "work_area_left" : 0
      }

然后替换文件

mv tmp Preferences

 


注意2:外接HDMI没有声音输出问题
1.alsamixer检查是否默认设置了HDMI
2.aplay -l 列出音频设备卡和设备编号

**** List of PLAYBACK Hardware Devices ****

card 0: PCH [HDA Intel PCH], device 0: ALC892 Analog [ALC892 Analog]

  Subdevices: 1/1

  Subdevice #0: subdevice #0

card 0: PCH [HDA Intel PCH], device 1: ALC892 Digital [ALC892 Digital]

  Subdevices: 1/1

  Subdevice #0: subdevice #0

card 0: PCH [HDA Intel PCH], device 3: HDMI 0 [HDMI 0]

  Subdevices: 1/1

  Subdevice #0: subdevice #0

card 0: PCH [HDA Intel PCH], device 7: HDMI 1 [HDMI 1]

  Subdevices: 1/1

  Subdevice #0: subdevice #0

card 0: PCH [HDA Intel PCH], device 8: HDMI 2 [HDMI 2]

  Subdevices: 1/1

  Subdevice #0: subdevice #0

3.sudo vim /etc/asound.conf 设置默认设备,然后重启

defaults.pcm.card 0
defaults.pcm.device 3

设置自启动

打开~/.bashrc文件,在最后追加一下内容:

cc=`ps axuw|grep chromium-browser|wc -l`
xc=`ps axuw|grep xserver|wc -l`
echo "chromium-browser process:${cc}, x11 server process:${xc}"
if [ $cc -lt 2 ] && [ $xc -lt 2 ]; then
       	/usr/bin/mystartx	
fi

结束语

折腾了几个小时,重启服务器,面子上算是搞定,其中的问题欢迎大家指正。一直以来主要用centos,最后一次使用ubuntu大概十年前了,记得当时免费申请光盘,要了一箱,送给同学,然而没送出去几套,不禁有些感慨。当年在学校时,在大牛民工于某的影响下,几个发行版都玩了玩,其中印象比较深还有Magic,中文支持比较好,现在不知道怎么样了。

继续阅读

sqlplus Linux版本安装和使用

以下介绍SQLPLUS的Linux版本安装及使用,其他版本下载及文档见Oracle Instant Client Downloads

下载

我的服务器为x86-64,要找对应系统版本,打开Instant Client for Linux x86-64进入下载页面,优先考虑当前最新版本为12.2.0.1.0,选择接受许可协议“Accept License Agreement”,下载以下三个包:

  1. 必要的基础程序 instantclient-basic-linux.x64-12.2.0.1.0.zip
  2. 一些库和sqlplus可执行文件 instantclient-sqlplus-linux.x64-12.2.0.1.0.zip
  3. 包含应用开发需要的头文件以及Makefile示例 instantclient-sdk-linux.x64-12.2.0.1.0.zip

需要登录方可下载

安装

解压压缩包

解压后会自动创建并将文件解压至instantclient_12_2文件夹,将instantclient_12_2移至/usr/local并对动态库做软连接

mv instantclient_12_2 /usr/local/
cd /usr/local/instantclient_12_2
ln -s libclntsh.so.12.1 libclntsh.so
ln -s libocci.so.12.1 libocci.so

安装 libaio

sudo yum install libaio

设置环境变量

echo 'export LD_LIBRARY_PATH=/usr/local/instantclient_12_2:$LD_LIBRARY_PATH' >> ~/.bash_profile
echo 'export PATH=/usr/local/instantclient_12_2:$PATH' >> ~/.bash_profile
source ~/.bash_profile

优化命令行使用

其实到此已经可以使用了,为了用起来更方便,我们需要安装rlwraprlwrap是一个wrapper,它接收我们输入的命令并交给我们指定的程序去执行,同时使用GNU readline库提供命令行编辑功能并且记录命令历史,将会使你的命令行用起来更加方便(主要是sqlplus太难用了)。

yum install -y rlwrap
echo 'alias sqlplus="rlwrap sqlplus"' >> ~/.bashrc
source ~/.bashrc

OK,现在可以放心去用了。

连接使用数据库

  • 连接本机数据库 sqlplus username/password
  • 连接远程数据库 sqlplus username/password@//domain:port/dbname

继续阅读

Gulp插件制作 – 使用流处理

强烈建议写插件的时候支持流处理,本文是关于创建插件支持流的介绍。确保遵循规范处理错误,在内容处理的同时进行异常捕获。

使用流处理

我们来实现一个文件文本处理的插件,这个插件支持`file.contents`所有可能的形式。

var through = require('through2');
var gutil = require('gulp-util');
var PluginError = gutil.PluginError;

// consts
const PLUGIN_NAME = 'gulp-prefixer';

function prefixStream(prefixText) {
  var stream = through();
  stream.write(prefixText);
  return stream;
}

// plugin level function (dealing with files)
function gulpPrefixer(prefixText) {
  if (!prefixText) {
    throw new PluginError(PLUGIN_NAME, 'Missing prefix text!');
  }

  prefixText = new Buffer(prefixText); // allocate ahead of time

  // creating a stream through which each file will pass
  var stream = through.obj(function(file, enc, cb) {
    if (file.isBuffer()) {
      this.emit('error', new PluginError(PLUGIN_NAME, 'Buffers not supported!'));
      return cb();
    }

    if (file.isStream()) {
      // define the streamer that will transform the content
      var streamer = prefixStream(prefixText);
      // catch errors from the streamer and emit a gulp plugin error
      streamer.on('error', this.emit.bind(this, 'error'));
      // start the transformation
      file.contents = file.contents.pipe(streamer);
    }