用 Rust 生成 Ant-Design Table Columns

来源:https://www.cnblogs.com/jingdongkeji/archive/2023/07/27/17584618.html
-Advertisement-
Play Games

經常開發表格,是不是已經被手寫Ant-Design Table的Columns整煩了?尤其是ToB項目,表格經常動不動就幾十列。每次照著後端給的介面文檔一個個配置,太頭疼了,主要是有時還會粘錯就尷尬了。那有沒有辦法能自動生成columns配置呢? ...


經常開發表格,是不是已經被手寫Ant-Design Table的Columns整煩了?

尤其是ToB項目,表格經常動不動就幾十列。每次照著後端給的介面文檔一個個配置,太頭疼了,主要是有時還會粘錯就尷尬了。

那有沒有辦法能自動生成columns配置呢?

當然可以。

目前後端的介面文檔一般是使用Swagger來生成的,Swagger是基於OpenAPI規範的一種實現。(OpenAPI規範是一種描述RESTful API的語言無關的格式,它允許開發者定義API的操作、輸入和輸出參數、錯誤響應等信息,並提供了一種規範的方式來描述和交互API。)

那麼我們只需要解析Swagger的配置就可以反向生成前端代碼。

接下來我們就寫個CLI工具來生成Table Columns。

平常我們實現一個CLI工具一般都是用Node,今天我們搞點不一樣的,用Rust。

開始咯

swagger.json

打開後端用swagger生成的介面文檔中的一個介面,一般是下麵這樣的,可以看到其json配置文件,如下圖:
image

swagger: 2.0表明瞭這個文檔使用的swagger版本,不同版本json配置結構會不同。

paths這裡key是介面地址。

可以看到當前介面是“/api/operations/cate/rhythmTableList”。
順著往下看,“post.responses.200.schema.originalRef”,這就是我們要找的,這個介面對應的返回值定義。

definitions拿到上面的返回值定義,就可以在“definitions”里找到對應的值。
這裡是“definitions.ResponseResult«List«CateInsightRhythmListVO»».properties.data.items.originalRef”
通過他就可找到返回的實體類定義CateInsightRhythmListVO

CateInsightRhythmListVO這裡就是我們生成Table Columns需要的欄位定義了。

CLI

接下來製作命令行工具

起初我使用的是commander-rust,感覺用起來更符合直覺,全程採用macros定義即可。
但到發佈的時候才發現,Rust依賴必須有一個確定的版本,commander-rust目前使用的是分支解析。。。
最後還是換了clap

clap的定義就要繁瑣些,如下:

#[derive(Parser)]
#[command(author, version)]
#[command(about = "swagger_to - Generate code based on swagger.json")]
struct Cli {
    #[command(subcommand)]
    command: Option,
}

#[derive(Subcommand)]
enum Commands {
    /// Generate table columns for ant-design
    Columns(JSON),
}

#[derive(Args)]
struct JSON {
    /// path/to/swagger.json
    path: Option,
}


這裡使用#[command(subcommand)]#[derive(Subcommand)]來定義columns子命令
使用#[derive(Args)]定義了path參數,用來讓用戶輸入swagger.json的路徑

實現columns子命令

columns命令實現的工作主要是下麵幾步:

  1. 讀取用戶輸入的swagger.json

  2. 解析swager.json

  3. 生成ant-design table columns

  4. 生成對應Typescript類型定義

讀取用戶輸入的swagger.json

這裡用到了一個crate,serde_json, 他可以將swagger.json轉換為對象。

let file = File::open(json).expect("File should open");
let swagger_json: Value = serde_json::from_reader(file).expect("File should be proper JSON");


解析swager.json

有了swagger_json對象,我們就可以按照OpenAPI的結構來解析它。

/// openapi.rs

pub fn parse_openapi(swagger_json: Value) -> Vec {
    let paths = swagger_json["paths"].as_object().unwrap();
    let apis = paths
        .iter()
        .map(|(path, path_value)| {
            let post = path_value["post"].as_object().unwrap();
            let responses = post["responses"].as_object().unwrap();
            let response = responses["200"].as_object().unwrap();
            let schema = response["schema"].as_object().unwrap();
            let original_ref = schema["originalRef"].as_str().unwrap();
            let data = swagger_json["definitions"][original_ref]["properties"]["data"]
                .as_object()
                .unwrap();
            let items = data["items"].as_object().unwrap();
            let original_ref = items["originalRef"].as_str().unwrap();
            let properties = swagger_json["definitions"][original_ref]["properties"]
                .as_object()
                .unwrap();
            let response = properties
                .iter()
                .map(|(key, value)| {
                    let data_type = value["type"].as_str().unwrap();
                    let description = value["description"].as_str().unwrap();
                    ResponseDataItem {
                        key: key.to_string(),
                        data_type: data_type.to_string(),
                        description: description.to_string(),
                    }
                })
                .collect();
            Api {
                path: path.to_string(),
                model_name: original_ref.to_string(),
                response: response,
            }
        })
        .collect();
    return apis;
}


這裡我寫了一個parse_openapi()方法,用來將swagger.json解析成下麵這種形式:

[
  {
    path: 'xxx',
    model_name: 'xxx',
    response: [
      {
        key: '欄位key',
        data_type: 'number',
        description: '欄位名'
      }
    ]
  }
]


對應的Rust結構定義是這樣的:

pub struct ResponseDataItem {
    pub key: String,
    pub data_type: String,
    pub description: String,
}

pub struct Api {
    pub path: String,
    pub model_name: String,
    pub response: Vec<ResponseDataItem>,
}


生成ant-design table columns

有了OpenAPI對象就可以生成Table Column了,這裡寫了個generate_columns()方法:

/// generator.rs

pub fn generate_columns(apis: &mut Vec) -> String {
    let mut output_text = String::new();
    output_text.push_str("import type { ColumnsType } from 'antd'\n");
    output_text.push_str("import type * as Types from './types'\n");
    output_text.push_str("import * as utils from './utils'\n\n");

    for api in apis {
        let api_name = api.path.split('/').last().unwrap();
        output_text.push_str(
            &format!(
                "export const {}Columns: ColumnsType = [\n",
                api_name,
                api.model_name
            )
        );
        for data_item in api.response.clone() {
            output_text.push_str(
                &format!(
                    "  {{\n    title: '{}',\n    key: '{}',\n    dataIndex: '{}',\n    {}\n  }},\n",
                    data_item.description,
                    data_item.key,
                    data_item.key,
                    get_column_render(data_item.clone())
                )
            );
        }
        output_text.push_str("]\n");
    }

    return output_text;
}


這裡主要就是採用字元串模版的形式,將OpenAPI對象遍歷生成ts代碼。

生成對應Typescript類型定義

Table Columns的類型使用generate_types()來生成,原理和生成columns一樣,採用字元串模版:

/// generator.rs

pub fn generate_types(apis: &mut Vec) -> String {
    let mut output_text = String::new();

    for api in apis {
        let api_name = api.path.split('/').last().unwrap();
        output_text.push_str(
            &format!(
                "export type {} = {{\n",
                Some(api.model_name.clone()).unwrap_or(api_name.to_string())
            )
        );
        for data_item in api.response.clone() {
            output_text.push_str(&format!("  {}: {},\n", data_item.key, data_item.data_type));
        }
        output_text.push_str("}\n\n");
    }

    return output_text;
}


main.rs

然後我們在main.rs中分別調用上面這兩個方法即可

/// main.rs

let mut apis = parse_openapi(swagger_json);
    let columns = generator::generate_columns(&mut apis);
    let mut columns_ts = File::create("columns.ts").unwrap();
    write!(columns_ts, "{}", columns).expect("Failed to write to output file");
    let types = generator::generate_types(&mut apis);
    let mut types_ts = File::create("types.ts").unwrap();
    write!(types_ts, "{}", types).expect("Failed to write to output file");


對於columns和types分別生成兩個文件,columns.ts和types.ts。

!這裡有一點需要註意

當時開發的時候對Rust理解不是很深,起初拿到parse_openapi返回的apis我是直接分別傳給generate_columns(apis)和generate_types(apis)的。但編譯的時候報錯了:

image

這對於js很常見的操作竟然在Rust中報錯了。原來Rust所謂不依賴運行時垃圾回收而管理變數分配引用的特點就體現在這裡。
我就又回去讀了遍Rust教程里的“引用和借用”那篇,算是搞懂了。這裡實際上是Rust變數所有權、引用和借用的問題。讀完了自然你也懂了。

看看效果

安裝

cargo install swagger_to


使用

swagger_to columns path/to/swagger.json


會在swagger.json所在同級目錄生成三個文件:

columns.tsant-design table columns的定義

types.tscolumns對應的類型定義

utils.tscolumn中render對number類型的欄位添加了格式化工具

image

Enjoy

作者:京東零售 於弘達

來源:京東雲開發者社區


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1. 介紹 阿裡巴巴 Arthas 是一個診斷工具,可以用於監視、分析和解決 Java 應用程式的問題。使用 Arthas 的一個主要優點是,我們不需要修改代碼,甚至不需要重新啟動我們想要監視的 Java 服務。 在本教程中,我們將首先安裝 Arthas,在此之後,通過一個簡單的案例來演示 Arth ...
  • Mac/Win 最新 IntelliJ IDEA 2023.2 激活破解教程,附激活碼(持續更新~),適用於 JetBrains 全家桶的所有工具。 ...
  • 部署容器是使用Docker和容器化管理應用程式更高效、易於擴展和確保跨環境一致性性能的關鍵步驟。本主題將為您概述如何部署Docker容器以創建和運行應用程式。 ## 概述 Docker容器是輕量級、可移植且自我包含的環境,可以運行應用程式及其依賴項。部署容器涉及啟動、管理和擴展這些隔離的環境,以便順 ...
  • ## 開篇介紹 Java 8 中新增的特性旨在幫助程式員寫出更好的代碼,其中對核心類庫的改進是很關鍵的一部分,也是本章的主要內容。對核心類庫的改進主要包括集合類的 API 和新引入的流(Stream),流使程式員得以站在更高的抽象層次上對集合進行操作。下麵將介紹stream流的用法。 ## 1.初始 ...
  • # 同步電路與非同步電路 - ## 同步電路 - 電路中所有觸發器均連接同一個時鐘脈衝源,觸發器的狀態變化均與時鐘脈衝信號同步; - 電路中所有時鐘同源同相; - 同相位時鐘:始終頻率不同,但是時鐘邊沿對齊; - ![](https://img2023.cnblogs.com/blog/1964011 ...
  • 上節討論瞭如何保障數據中台的數據質量,讓數據“準”。除了“快”和“準”,數據中台還離不開“省”。隨數據規模越來越大,成本越來越高,如不合理控製成本,還沒等你挖掘出數據應用價值,企業利潤就被消耗完。 能否做到精細化成本管理,關乎數據中台項目成敗。 某電商業務數據建設資源增長趨勢(CU= 1vcpu + ...
  • # 事務 - **基本介紹** 1. JDBC 程式中當一個Connection對象創建時,預設情況下是自動提交事務:每次執行一個 SQL 語句時,如果執行成功,就會向資料庫自動提交,而不能回滾。 2. JDBC程式中為了多個SQL語句作為一個整體執行,需要==使用事務==。 3. 調用 Conne ...
  • ## 教程簡介 Excel Power View 是一種數據可視化技術,用於創建互動式圖表、圖形、地圖和其他視覺效果,以便直觀呈現數據。 Excel Power View中,可以快速創建各種可視化效果,從表格和矩陣到餅圖、條形圖和氣泡圖,以及多個圖表的集合。要創建各種可視化效果,請首先從表格開始著手 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...