一、使用.Net Core构建WebAPI并访问Docker中的Mysql数据库
这个的过程大概与我之前的文章《尝试.Net Core—使用.Net Core + Entity FrameWork Core构建WebAPI(一)》一致。
但是在我们这里,由于docker中无法部署sql server,所以我采用了Mysql数据库,顺便吐槽一下 SQL Server真的太贵了,阿里云的Mysql实例价格和SQL Server实例价格差10倍,真的好夸张。
Mysql官方的EF Core支持刚刚出来,由于文章是后续补充的,在我尝试的时候官方驱动还没有出,所以我使用的是第三方的驱动:Pomelo.EntityFrameworkCore.MySql,据说是国内一个90后的小伙儿写的,非常牛X.
引用之后,就和EF访问SQL Server没有什么区别了,但是:由于我们的Mysql实例是安装在Docker容器中的,那两个Docker容器之间怎么互相通信呢?
首先在启动API Image的时候,需要使用--link 命令,来建立API和Mysql 容器之间的访问通道,大概命令如下:
docker run -it -p 9100:9100 --name testapi --link msyqldb:msyqldb testapi
在Link的时候我们给需要连接的容器起了一个别名,这个别名就是一个主机名一样,可以直接用来访问,这个时候我们的连接字符串就类似于:
Data Source=mysqldb;port=3306;Initial Catalog=数据库名称;Integrated Security=False;Persist Security Info=False;User ID=用户名;Password=密码
这里,我们的DataSource,也就是实例的名称,就直接写主机名,也就是我们在Link的时候起的那个别名,就OK啦~只要你突破了这点,那访问基本上就没有什么问题了。
Coding 工作结束之后,发布你的API,并且最终打包成Image,启动容器,这一系列过程可以参考我的上一篇文章:《【Step By Step】将Dotnet Core部署到Docker上》
二、使用.Net Core构建一个Web程序并访问Docker中的WebAPI
在Docker中的Web程序访问另外一个Docker容器中的API, 需要解决的问题如下:
1.API的路径是啥?
2.跨域问题
其中第一个问题的解决思路和我们上面连接数据库的解决思路是完全一致的,只要在Link的时候,加上别名,就可以用http://别名:端口号 来访问你的api即可。
第二个问题不仅仅是在docker中会遇到的问题,在平时我们开发Web程序时,也会遇到这样子的问题,常用的解决方案有以下几种:
A.避免用js访问API,而采用后端代码去调用,就没有跨域这一说了
B.后端API HttpHeader增加Access-Control-Allow-Origin:* Access-Control-Allow-Headers:Content-type Access-Control-Allow-Methods:GET, POST, PUT, DELETE, OPTIONS三个选项
C.Jsonp,就是把数据当做一个js文件来加载~
这里我采用方法A来解决跨域问题(简单、暴力、最彻底):
第一步:通过Nuget引用Orion.ApiClientLight类库,这是一个封装了一系列调用api方法的类库,我们可以通过他很方便的调用API并序列化结果
第二步:在Controller中初始化APIClient对象,JsonApiClientLight apiClient = new JsonApiClientLight();
第三步:调用方法来访问API。
下面的代码是我调用一个CRUD的API来访问API的简单实现:
public class HomeController : Controller { JsonApiClientLight apiClient = new JsonApiClientLight(); public async Task<IActionResult> Index() { var list = await apiClient.GetAsync<List<Member>>("http://jb91api:9103/api/mermbers/mermber/list"); return View(list.Response); } [HttpGet] public IActionResult Create() { return View(); } [HttpPost,ActionName("Create")] public async Task<IActionResult> CreatePost(CreateMember model) { if(ModelState.IsValid) { var member = await apiClient.PostAsync("http://jb91api:9103/api/mermbers/mermber/", model); return RedirectToAction("Index"); } return View(model); } public async Task<IActionResult> Delete(int id) { if (id > 0) { await apiClient.DeleteAsync("http://jb91api:9103/api/mermbers/mermber?id=" + id.ToString()); } return RedirectToAction("Index"); } }