Asp.NET Core实现动态文件服务器

需求

我这边有一些3DTiles数据需要动态发布,3DTiles数据简单来说是把大规模的三维地理模型切成很多小片,在展示的时候按精度按范围调取需要的数据,以减轻网络和渲染压力,加快渲染速度的一个方案。因此他是有记录切片配置的json文件和b3dm格式的数据文件构成的,在数据的根目录下有一个根的配置文件,每个子目录下通常也会有子配置文件。

3DTiles文件结构

3DTiles数据大小得看数据规模和切片精度,通常城市级别的倾斜摄影模型切成3DTiles大小得按T计算,文件个数得按万计算,不太适合像普通文件一样上传然后通过接口访问。所以考虑将需要发布的数据先通过其他方式上传到服务器,然后通过文件服务器的方式展示出来。

当然其实也可以直接将某个文件夹通过IIS/Nginx发布出去,然后要求用户每次上传的数据都放在那个文件夹下也是可以的,但是这样灵活性和通用性就大打折扣。

因此设定的业务逻辑应该是客户通过FTP或者其他什么工具将数据上传到服务器,然后通过应用选择数据文件夹,设定虚拟目录(url子路径),发布,就可以通过url访问了。

技术栈

我这边习惯上后台使用Asp.Net Core Web API开发,现在到了.Net 6,是一个长期支持版本。当使用Visual Studio创建Asp.Net Core Web API后,入口文件Program.cs下会自动生成类似以下的代码,直接运行就会有一个天气预报的示例接口和Swagger接口文档页面(所以说.net core好用呀):

var apiAppBuilder = WebApplication.CreateBuilder(args);
apiAppBuilder.Services.AddControllers();
apiAppBuilder.Services.AddEndpointsApiExplorer();
apiAppBuilder.Services.AddSwaggerGen();

var apiApp = apiAppBuilder.Build();
if (apiApp.Environment.IsDevelopment())
{
    apiApp.UseSwagger();
    apiApp.UseSwaggerUI();
}
apiApp.UseAuthorization();
apiApp.MapControllers();

await apiApp.RunAsync();

文件服务

如果想直接让上面代码中的apiAPP支持文件服务,只需要给他绑定静态文件服务相关的内容就行:

var staticfile = new StaticFileOptions();
staticfile.ServeUnknownFileTypes = true;
staticfile.FileProvider = new PhysicalFileProvider(physicalPath);
staticfile.RequestPath = urlPath;
apiApp.UseStaticFiles(staticfile);

如果希望除了提供文件服务之外还可以在浏览器中浏览,就需要绑定文件夹浏览相关的内容:

文件夹浏览
var dirOp = new DirectoryBrowserOptions();
dirOp.FileProvider = new PhysicalFileProvider(physicalPath);
dirOp.RequestPath = urlPath;
apiApp.UseDirectoryBrowser(dirOp);

上面urlPath是在网址路径中的path,以'/'开头,比如我的服务是http://localhost:5000,这里的urlPath设置为/data,那通过http://localhost:5000/data访问到的就是physicalPath中的内容,UseStaticFiles和UseDirectoryBrowser都是可以反复添加的,通过这样的方式就可以添加多个文件夹。

寄生应用

很可惜,这些个是不能够动态设定的,也就是这些设置必须在apiApp.RunAsync()之前设定,启动之后再设置就没用了。但是如果按默认的设定,把apiApp停了整个服务就会挂掉,没法走设定后重启的路线,因此得在主应用之下加一个寄生应用作为文件服务的专有应用。

public class FileServerApp
{
    private static WebApplication AppInstance = null;
    private Dictionary<string,string> Directories = new Dictionary<string, string>();

    /// <summary>
    /// 添加文件夹
    /// </summary>
    /// <param name="key"></param>
    /// <param name="dir"></param>
    /// <returns></returns>
    public async Task AddDirectoryAsync(string urlPath, string physicalPath)
    {
        if (Directories.ContainsKey(urlPath))
        {
            Directories[urlPath] = physicalPath;
        }
        else
        {
            Directories.Add(urlPath, physicalPath);
        }
        await this.StopAsync();
        await this.StartAsync();
    }

    /// <summary>
    /// 停止
    /// </summary>
    /// <returns></returns>
    private async Task StopAsync()
    {
        if (AppInstance != null)
        {
            await AppInstance.StopAsync();
            await AppInstance.WaitForShutdownAsync();
            await AppInstance.DisposeAsync();
        }
    }

    /// <summary>
    /// 启动
    /// </summary>
    /// <returns></returns>
    private async Task StartAsync()
    {
        var fileAppBuilder = WebApplication.CreateBuilder();
        fileAppBuilder.Services.AddCors(options =>
        {
            options.AddPolicy("Any", builder =>
            {
                builder.AllowAnyOrigin()
                       .AllowAnyHeader()
                       .AllowAnyMethod();
            });
        });
        AppInstance = fileAppBuilder.Build();
        AppInstance.UseCors("Any");
        AppInstance.Urls.Add("http://*:6789");

        foreach (var urlPath in Directories.Keys)
        {
            var dirOp = new DirectoryBrowserOptions();
            dirOp.FileProvider = new PhysicalFileProvider(Directories[urlPath]);
            dirOp.RequestPath = urlPath;
            AppInstance.UseDirectoryBrowser(dirOp);
            var staticfile = new StaticFileOptions();
            staticfile.ServeUnknownFileTypes = true;
            staticfile.FileProvider = new PhysicalFileProvider(Directories[urlPath]);
            staticfile.RequestPath = urlPath;
            AppInstance.UseStaticFiles(staticfile);
        }

        await AppInstance.RunAsync();
    }
}

这里用Dictionary模拟持久化数据,只需要在相应的接口中执行以下代码就可以启动寄生应用了:

var fileServerApp = new FileServerApp();
fileServerApp.AddDirectoryAsync(urlPath, physicalPath);

注意这里的AddDirectoryAsync不能await,否则会阻塞住,直到该应用停止。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,417评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,921评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,850评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,945评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,069评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,188评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,239评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,994评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,409评论 1 304
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,735评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,898评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,578评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,205评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,916评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,156评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,722评论 2 363
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,781评论 2 351

推荐阅读更多精彩内容