ASP.NET跨平台最佳实践

时间:2022-06-21 09:19:47

前言

八年的坚持敌不过领导的固执,最终还是不得不阔别已经成为我第二语言的C#,转战Java阵营。有过短暂的失落和迷茫,但技术转型真的没有想象中那么难。回头审视,其实单从语言本身来看,C#确实比Java更优秀(并非C#天生丽质,而是它站在了巨人的肩膀上)。

本文并非为.NET正名而来,而仅仅是分享作者近几年在ASP.NET跨平台方面的研究与实践经验,算是对八年的.NET之路作一个阶段性的总结。

.NET技术自诞生以来,便一直因其跨平台能力差而广受诟病。这里面有微软有意为之,也有别有用心之人在混淆视听。.NET在一开始便是按公开的语言规范进行设计,随着微软的逐渐妥协,.NET Framework已完全开源。.NET跨平台技术迎来了前所未有的机会,各种.NET跨平台技术必将如雨后春笋般涌现。本文将介绍作者在.NET跨平台方面的最佳实践经验,希望藉以降低.NET跨平台的技术难度,让.NET真正成为跨平台的生产技术。

环境

操作系统选择开源社区较有代表性的Linux服务器版本CentOS(本文所述的跨平台思路可以成功应用于绝大多数的Linux系统,也包括国产操作系统如中标麒麟);技术平台选择久负盛名的Mono;Web应用中间件选择中国制造的Jexus。涉及的环境与技术详细情况为:

  • 操作系统:CentOS_6.4_64bit
  • .NET框架:Mono_4.0.4.1
  • Web应用中间件:Jexus_5.6.5
  • 数据库:MySQL_5.1.73

目标

本文所述实践,将实现ASP.NET应用程序在Linux系统进行部署,并作产品化尝试,使ASP.NET应用在Linux平台易于部署和维护。

Web应用程序

本文的Web应用程序选用ASP.NET MVC 4技术开发,持久层选用ADO.NET Entity Framework 6技术。你可以在src/demo目录下找到它。

跨平台部署

一、安装系统

最小化安装CentOS系统。作为实验环境,可以考虑选用具有快照功能的虚拟机,作者选用的VMWare虚拟机进行试验。系统安装在此不再赘述。系统安装好后,做一个快照留待后续验证产品化安装包正确性所用。

如果你对跨平台部署ASP.NET应用的实现过程没有兴趣,只想看结果的话,可以直接跳过后续步骤,直接进入产品化章节的通过安装包部署ASP.NET应用部分。

二、搭建Linux编译环境

由于本文选择源码安装Mono,所以需要先搭建Linux编译环境。搭建Linux编译环境需要让系统连接互联网,并进行系统更新。

首先,需要让你的系统连接互联网。此处以VMWare虚拟机为例讲解如何连接互联网。VMWare虚拟机连接互联网的方式很多,作者选择NAT方式,首先需要将虚拟机网络连接方式设为NAT,如图 1所示。

ASP.NET跨平台最佳实践

1 VMWare虚拟机网络连接方式设置

设置VMnet8为自动获取IP,如图 2所示。

ASP.NET跨平台最佳实践

2 设置VMnet8为自动获取IP地址

然后将Linux虚拟机设为DHCP自动分配IP,编辑网卡配置文件:

[root@localhost ~]# vi /etc/sysconfig/network-scripts/ifcfg-eth0

修改该配置文件内容如下:

DEVICE=eth0
HWADDR=:0C::F5::
TYPE=Ethernet
UUID=74b949f0-57bb-4baa-a5f2-2c97fb533a8b
ONBOOT=yes
NM_CONTROLLED=yes
BOOTPROTO=dhcp

重启网络服务,让网卡设置生效:

[root@localhost ~]# service network restart

确认虚拟机已连接互联网:

[root@localhost ~]# ping yilin.cnblogs.com
PING yilin.cnblogs.com (42.121.252.58) () bytes of data.
bytes from 42.121.252.58: icmp_seq= ttl= time=77.7 ms
bytes from 42.121.252.58: icmp_seq= ttl= time=78.1 ms
bytes from 42.121.252.58: icmp_seq= ttl= time=77.5 ms

更新系统:

[root@localhost ~]# yum –y update

安装Mono源码安装需要的组件:

[root@localhost ~]# yum -y install wget glib2-devel libtiff libtiff-devel libjpeg libjpeg-devel giflib giflib-devel libpng libpng-devel libX11 libX11-devel freetype freetype-devel fontconfig fontconfig-devel libexif libexif-devel gcc-c++ gettext unzip zip

三、安装GDI+组件

安装Mono之前,需要先安装其依赖的GDI+组件。联网下载libgdiplus源码安装包(如果在Linux系统中直接下载源码包出现停滞的情况,请返回Windows系统下载libgdiplus源码安装包,并将其上传到Linux系统相应路径后进行安装。后续其他组件安装遇此情况与此雷同,不再赘述):

[root@localhost ~]# cd /usr
[root@localhost usr]# wget http://download.mono-project.com/sources/libgdiplus/libgdiplus-2.10.tar.bz2

解压libgdiplus源码安装包:

[root@localhost usr]# tar jxvf libgdiplus-2.10.tar.bz2

配置libgdiplus组件安装路径(这里指定安装路径,是为后文产品化制作安装包做准备,Mono、Jexus和MySQL安装也是如此,不再赘述):

[root@localhost usr]# cd libgdiplus-2.10
[root@localhost libgdiplus-2.10]# ./configure --prefix=/usr/apollo/hostd/mono/

编译libgdiplus源码:

[root@localhost libgdiplus-2.10]# make

安装libgdiplus组件:

[root@localhost libgdiplus-2.10]# make install

四、安装Mono

Mono是Linux平台的.NET Framework实现,是.NET程序移植到Linux平台的不二选择。首先,联网下载Mono源码安装包:

[root@localhost libgdiplus-2.10]# cd /usr
[root@localhost usr]# wget http://download.mono-project.com/sources/mono/mono-4.0.4.1.tar.bz2

解压Mono源码安装包:

[root@localhost usr]# tar jxvf mono-4.0.4.1.tar.bz2

配置Mono安装路径:

[root@localhost usr]# cd mono-4.0.
[root@localhost mono-4.0.]# ./configure --prefix=/usr/apollo/hostd/mono

编译Mono源码(此过程耗时一般为半小时到一小时,视系统软硬件配置而定):

[root@localhost mono-4.0.]# make

安装Mono:

[root@localhost mono-4.0.]# make install

通过查看Mono版本,确认Mono是否安装成功(出现如下的版本信息表示Mono安装成功):

[root@localhost mono-4.0.]# cd /usr/apollo/hostd/mono/bin/
[root@localhost bin]# ./mono -V
Mono JIT compiler version 4.0. (Stable 4.0.4.1/5ab4c0d Fri Oct :: CST )
Copyright (C) - Novell, Inc, Xamarin Inc and Contributors. www.mono-project.com
TLS: __thread
SIGSEGV: altstack
Notifications: epoll
Architecture: amd64
Disabled: none
Misc: softdebug
LLVM: supported, not enabled.
GC: sgen

五、安装Jexus

Jexus是国人开发的Linux平台上的ASP.NET Web应用中间件,类似于Windows平台的IIS。实践证明Jexus安装简单,运行稳定,是Linux平台架设ASP.NET应用的不错选择。首先,联网下载Jexus安装包:

[root@localhost ~]# cd /usr
[root@localhost usr]# wget http://www.linuxdot.net/down/jexus-5.6.5.tar.gz

解压Jexus安装包:

[root@localhost usr]# tar zxvf jexus-5.6..tar.gz

修改Jexus安装路径:

[root@localhost usr]# cd jexus-5.6.
[root@localhost jexus-5.6.]# vi install #!/bin/sh SRC_DIR=$(cd $(dirname $);pwd)
DAT_DIR=${SRC_DIR}/data
JWS_DIR='/usr/apollo/hostd/jexus'

安装Jexus:

[root@localhost jexus-5.6.]# sudo ./install

修改Jexus关于mono路径的配置:

[root@localhost jexus-5.6.]# cd /usr/apollo/hostd/jexus
[root@localhost jexus]# vi jws #!/bin/sh JWS_HOME=$(cd $(dirname $);pwd) export LANG="zh_CN.UTF-8"
export PATH=/usr/bin:${JWS_HOME}/../mono/bin:$PATH
export LD_LIBRARY_PATH=/usr/lib:${JWS_HOME}/../mono/lib:$LD_LIBRARY_PATH

启动Jexus服务,测试Jexus安装是否正确:

[root@localhost jexus]# ./jws start

设置Jexus服务开机自启动,增加如下高亮行::

[root@localhost jexus]# vi /etc/rc.d/rc.local

touch /var/lock/subsys/local
/usr/apollo/hostd/jexus/jws start

修改该文件权限并重启:

[root@localhost jexus]# chmod +x /etc/rc.d/rc.local

六、部署网站

创建网站目录:

[root@localhost jexus]# cd /usr/apollo/
[root@localhost apollo]# mkdir webapps
[root@localhost apollo]# cd webapps/
[root@localhost webapps]# mkdir default
[root@localhost webapps]# cd default/
[root@localhost default]# touch index.html
[root@localhost default]# vi index.html
<html>
<head>
<title>ASP.NET跨平台最佳实践</title>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/>
</head>
<body>
ASP.NET跨平台最佳实践,看到该页面,表示你的第一个网站已成功部署到Linux系统。
</body>
</html>

修改Jexus网站配置:

[root@localhost default]# cd /usr/apollo/hostd/jexus/siteconf
[root@localhost siteconf]# vi default ######################
# Web Site: Default
######################################## port=
root=/ /usr/apollo/webapps/default
hosts=* #OR your.com,*.your.com

开放防火墙80端口:

[root@localhost siteconf]# vi /etc/sysconfig/iptables

# Firewall configuration written by system-config-firewall
# Manual customization of this file is not recommended.
*filter
:INPUT ACCEPT [0:0]
:FORWARD ACCEPT [0:0]
:OUTPUT ACCEPT [0:0]
-A INPUT -m state --state ESTABLISHED,RELATED -j ACCEPT
-A INPUT -p icmp -j ACCEPT
-A INPUT -i lo -j ACCEPT
-A INPUT -m state --state NEW -m tcp -p tcp --dport 22 -j ACCEPT
-A INPUT -m state --state NEW -m tcp -p tcp --dport 80 -j ACCEPT
-A INPUT -j REJECT --reject-with icmp-host-prohibited
-A FORWARD -j REJECT --reject-with icmp-host-prohibited
COMMIT

重启iptables,使防火墙设置生效:

[root@localhost siteconf]# /etc/init.d/iptables restart

重启Jexus服务:

[root@localhost siteconf]# /usr/apollo/hostd/jexus/jws restart

终端浏览器输入网址http://ip:port/index.html访问网站:

ASP.NET跨平台最佳实践

3 ASP.NET网站成功部署到Linux平台

七、数据库

Linux平台免费关系数据库首选当然是MySQL,首先下载MySQL源码安装包:

[root@localhost siteconf]# cd /usr
[root@localhost usr]# wget http://downloads.mysql.com/archives/get/file/mysql-5.1.72-linux-x86_64-glibc23.tar.gz

解压MySQL源码安装包到指定目录,并修改目录名称:

[root@localhost usr]# tar zxvf mysql-5.1.-linux-x86_64-glibc23.tar.gz -C /usr/apollo
[root@localhost usr]# cd apollo
[root@localhost apollo]# mv mysql-5.1.-linux-x86_64-glibc23 data

将mysql配置文件拷贝至指定目录:

[root@localhost apollo]# cd data
[root@localhost data]# cp support-files/my-medium.cnf /etc/my.cnf

编辑mysql配置文件,在[client]节和[mysqld]节中加入以下高亮行::

[root@localhost data]# vi /etc/my.cnf

[client]
#password = your_password
port =
socket = /tmp/mysql.sock
default-character-set = utf8 # Here follows entries for some specific programs # The MySQL server
[mysqld]
port =
socket = /tmp/mysql.sock
skip-locking
key_buffer_size = 16M
max_allowed_packet = 1M
table_open_cache =
sort_buffer_size = 512K
net_buffer_length = 8K
read_buffer_size = 256K
read_rnd_buffer_size = 512K
myisam_sort_buffer_size = 8M
basedir = /usr/apollo/data
datadir = /usr/apollo/data/data
character-set-server = utf8
collation-server = utf8_general_ci

创建mysql组及用户,并设定目录访问权限:

[root@localhost data]# groupadd mysql
[root@localhost data]# useradd -g mysql mysql
[root@localhost data]# chown -R mysql .
[root@localhost data]# chgrp -R mysql .
[root@localhost data]# chown -R root .
[root@localhost data]# chown -R mysql data

初始化数据库:

[root@localhost data]# scripts/mysql_install_db --user=mysql

运行mysql服务:

[root@localhost data]# bin/mysqld_safe --user=mysql &

这种方式启动MySQL是阻塞式的,需要另开一个会话登录Linux系统,继续后续操作。

设置root用户密码:

[root@localhost data]# bin/mysqladmin -uroot password 

设置mysql服务开机自动启动:

[root@localhost data]# cp support-files/mysql.server /etc/rc.d/init.d/mysqld
[root@localhost data]# chmod /etc/init.d/mysqld
[root@localhost data]# chkconfig --add mysqld
[root@localhost data]# chkconfig --level mysqld on

现在,可以停止之前会话启动的MySQL服务(快捷键Ctrl + C),使用service命令后台启动MySQL服务。

[root@localhost data]# service mysqld start

将mysql命令加入系统环境变量中,在文件末尾加上以下两行代码:

[root@localhost data]# vi /etc/profile

...
PATH=$PATH:/usr/apollo/data/bin
export

执行配置,并重启系统,让环境变量生效:

[root@localhost data]# source /etc/profile

待Linux系统重启后,MySQL服务可以自动启动,MySQL命令也包含在了环境变量中。此时,MySQL服务已经可以在本地访问了,可以通过MySQL命令行创建数据库、执行SQL文件等操作。另外,可以通过MySQL命令行配置允许远程访问MySQL数据库(当网站服务和MySQL数据库服务在一起时,可以不进行该配置):

[root@localhost ~]# mysql -uroot -p11111111
mysql> GRANT ALL PRIVILEGES ON *.* TO 'root'@'%' IDENTIFIED BY '' WITH GRANT OPTION;
mysql> FLUSH PRIVILEGES;
mysql> exit;

还需要防火墙开放3306端口,编辑防火墙规则文件,在防火墙规则文件中添加如下高亮行:

[root@localhost ~]# vi /etc/sysconfig/iptables

# Firewall configuration written by system-config-firewall
# Manual customization of this file is not recommended.
*filter
:INPUT ACCEPT [:]
:FORWARD ACCEPT [:]
:OUTPUT ACCEPT [:]
-A INPUT -m state --state ESTABLISHED,RELATED -j ACCEPT
-A INPUT -p icmp -j ACCEPT
-A INPUT -i lo -j ACCEPT
-A INPUT -m state --state NEW -m tcp -p tcp --dport -j ACCEPT
-A INPUT -m state --state NEW -m tcp -p tcp --dport -j ACCEPT
-A INPUT -m state --state NEW -m tcp -p tcp --dport -j ACCEPT
-A INPUT -j REJECT --reject-with icmp-host-prohibited
-A FORWARD -j REJECT --reject-with icmp-host-prohibited
COMMIT

重启防火墙服务:

[root@localhost ~]# /etc/init.d/iptables restart

此时可以远程访问MySQL数据库了。

八、部署ASP.NET应用

本示例提供一个ASP.NET MVC应用,放置在后文制作的tar包webapps目录下,该示例通过Entity Framework实现数据库的自动创建及数据的增删改查。将该目录上传至Linux系统的/usr/apollo/webapps目录下,并配置Jexus网站配置:

[root@localhost ~]# cd /usr/apollo/hostd/jexus/siteconf/
[root@localhost siteconf]# cp default demo
[root@localhost siteconf]# vi demo …
port=
root=/ /usr/apollo/webapps/demo
hosts=* #OR your.com,*.your.com

Jexus支持承载多个Web站点,所以这里新增的demo站点和之前创建的default站点可以共存,仅需配置不同的端口号即可。同样的需要开放防火墙8080端口:

[root@localhost siteconf]# vi /etc/sysconfig/iptables

# Firewall configuration written by system-config-firewall
# Manual customization of this file is not recommended.
*filter
:INPUT ACCEPT [:]
:FORWARD ACCEPT [:]
:OUTPUT ACCEPT [:]
-A INPUT -m state --state ESTABLISHED,RELATED -j ACCEPT
-A INPUT -p icmp -j ACCEPT
-A INPUT -i lo -j ACCEPT
-A INPUT -m state --state NEW -m tcp -p tcp --dport -j ACCEPT
-A INPUT -m state --state NEW -m tcp -p tcp --dport -j ACCEPT
-A INPUT -m state --state NEW -m tcp -p tcp --dport -j ACCEPT
-A INPUT -j REJECT --reject-with icmp-host-prohibited
-A FORWARD -j REJECT --reject-with icmp-host-prohibited
COMMIT

重启iptables,使防火墙设置生效:

[root@localhost siteconf]# /etc/init.d/iptables restart

重启Jexus服务:

[root@localhost siteconf]# /usr/apollo/hostd/jexus/jws restart

现在可以在终端通过浏览器访问ASP.NET MVC站点了。

ASP.NET跨平台最佳实践

4 Linux环境部署ASP.NET MVC应用

产品化

通过上述一系列步骤,我们实现了ASP.NET MVC应用在Linux平台的部署,但这仅仅是技术预研,离产品化还有一定的距离。这一系列步骤技术性太强,需要联网,不易操作,这些对于生产和用户环境都是难于实现的。我们需要将.NET跨平台技术产品化,使得ASP.NET应用易于部署。

一、初始化脚本

如前文所述,ASP.NET应用在Linux平台部署需要设置防火墙策略、MySQL用户与分组、Jexus与MySQL服务开机自启动等,相当繁琐。我们可以将这些设置集中在一个初始化脚本里执行。

#!/bin/sh

#echo off
DIR="$( cd "$( dirname "${BASH_SOURCE[0]}" )" && pwd )" #创建软链接,以支持将应用部署到任意目录
#ln -s -T $DIR /usr/apollo #拷贝事先配置好的MySQL配置文件到指定目录
cp -f $DIR/conf/my.cnf /etc/my.cnf #创建MySQL需要的用户和用户组
groupadd mysql
useradd -g mysql mysql
chown -R root .
chown -R mysql data
chgrp -R mysql data #设置MySQL服务开机自启动
cp $DIR/data/support-files/mysql.server /etc/rc.d/init.d/mysqld
chmod /etc/init.d/mysqld
chkconfig --add mysqld
chkconfig --level mysqld on #将MySQL命令加入系统环境变量
sudo cat $DIR/conf/profile >> /etc/profile
source /etc/profile #设置防火墙策略
cp -f $DIR/conf/iptables /etc/sysconfig/iptables #设置Jexus服务开机自启动
sudo cat $DIR/conf/rc.local >> /etc/rc.d/rc.local
chmod +x /etc/rc.d/rc.local
chmod a+x $DIR/hostd/jexus/jws
chmod a+x -R $DIR/hostd/mono/bin #重启系统
reboot

二、制作安装包

将Linux平台部署ASP.NET应用需要的技术包括Mono、Jexus、MySQL等集成并进行打包分发。

[root@localhost ~]# tar jcvf crossplatform-1.0.0.0-centos6.-x86_32_64.tar.bz2 /usr/apollo

三、通过安装包部署ASP.NET应用

至此,Linux平台Web应用部署包已打好,将其下载到终端。将虚拟机恢复到初始安装快照状态,然后将终端上的部署包上传到Linux系统中并解压:

[root@localhost usr]# tar jxvf crossplatform-1.0.0.0-centos6.-x86_32_64.tar.bz2

执行初始化脚本完成安装:

[root@localhost usr]# cd apollo
[root@localhost apollo]# sh ./install

在终端通过浏览器访问ASP.NET MVC站点,确认部署成功。

ASP.NET跨平台最佳实践

5 Linux平台产品化部署ASP.NET MVC应用

总结

本文详细讲解了Linux平台部署ASP.NET应用的最佳实践过程,通过该实践过程了解了ASP.NET应用跨平台部署的方方面面,该过程同样适用于其他Linux分发版本和国产操作系统。另外,本文对Linux平台部署ASP.NET应用进行了产品化包装,制作的安装包可以直接应用于产品打包,你只需将应用放入指定目录即可。希望本文所阐述的技术对你有所帮助。

附件下载:示例源码+本文PDF版本

附件下载:产品化部署TAR包

其他最佳实践建议(持续更新)

  1. Mono对ASP.NET WebForm技术支持还不是很好,尽量不要选择ASP.NET WebForm技术,ASP.NET MVC技术是更好的选择;
  2. Mono似乎暂时不支持ValidateInputAttribute,所以如果你的Action需要禁用输入参数验证,需要在Web.config中增加“<httpRuntime requestValidationMode="2.0"/>”全局配置;
  3. Mono似乎暂时不支持BundleConfig,所以不要用这种方式来绑定前台资源;
  4. 如果你用了Sping.NET IoC技术,你一定会爱上她针对ASP.NET MVC Controller的自动注入功能,但是目前在Mono上运行有问题,原因暂未查明,只能舍弃而改用传统的IObjectFactory.GetObject方法。