Swoft 2.0.3 重大更新,發佈優雅的微服務治理

来源:https://www.cnblogs.com/stelin/archive/2019/07/09/11160679.html
-Advertisement-
Play Games

 什麼是 Swoft ? Swoft 是一款基於 Swoole 擴展實現的 PHP 微服務協程框架。Swoft 能像 Go 一樣,內置協程網路伺服器及常用的協程客戶端且常駐記憶體,不依賴傳統的 PHP-FPM。有類似 Go 語言的協程操作方式,有類似 Spring Cloud 框架靈活的註解、強大的 ...


什麼是 Swoft ?

Swoft 是一款基於 Swoole 擴展實現的 PHP 微服務協程框架。Swoft 能像 Go 一樣,內置協程網路伺服器及常用的協程客戶端且常駐記憶體,不依賴傳統的 PHP-FPM。有類似 Go 語言的協程操作方式,有類似 Spring Cloud 框架靈活的註解、強大的全局依賴註入容器、完善的服務治理、靈活強大的 AOP、標準的 PSR 規範實現等等。

Swoft 通過長達三年的積累和方向的探索,把 Swoft 打造成 PHP 界的 Spring Cloud, 它是 PHP 高性能框架和微服務治理的最佳選擇。

優雅的服務治理

Swoft 官方建議開發者使用 Service mesh 模式,比如 Istio/Envoy 框架,把業務和服務治理分開,但是 Swoft 也為中小型企業快速構建微服務提供了一套微服務組件。

服務註冊與發現

服務註冊與發現,需要用到 Swoft 官方提供的 swoft-consul 組件,如果其它第三方也類似。

註冊與取消服務

監聽 SwooleEvent::START 事件,註冊服務

/**
 * Class RegisterServiceListener
 *
 * @since 2.0
 *
 * @Listener(event=SwooleEvent::START)
 */
class RegisterServiceListener implements EventHandlerInterface
{
    /**
     * @Inject()
     *
     * @var Agent
     */
    private $agent;

    /**
     * @param EventInterface $event
     */
    public function handle(EventInterface $event): void
    {
        /* @var HttpServer $httpServer */
        $httpServer = $event->getTarget();

        $service = [
            // ....
        ];

        $scheduler = Swoole\Coroutine\Scheduler();
        $scheduler->add(function () use ($service) {
            // Register
            $this->agent->registerService($service);
            CLog::info('Swoft http register service success by consul!');
        });
        $scheduler->start();
    }
}

監聽 SwooleEvent::SHUTDOWN 事件,取消服務

/**
 * Class DeregisterServiceListener
 *
 * @since 2.0
 *
 * @Listener(SwooleEvent::SHUTDOWN)
 */
class DeregisterServiceListener implements EventHandlerInterface
{
    /**
     * @Inject()
     *
     * @var Agent
     */
    private $agent;

    /**
     * @param EventInterface $event
     */
    public function handle(EventInterface $event): void
    {
        /* @var HttpServer $httpServer */
        $httpServer = $event->getTarget();

        $scheduler = Swoole\Coroutine\Scheduler();
        $scheduler->add(function () use ($httpServer) {
            $this->agent->deregisterService('swoft');
        });
        $scheduler->start();
    }
}    

服務發現

定義服務提供者

/**
 * Class RpcProvider
 *
 * @since 2.0
 *        
 * @Bean()
 */
class RpcProvider implements ProviderInterface
{
    /**
     * @Inject()
     *
     * @var Agent
     */
    private $agent;

    /**
     * @param Client $client
     *
     * @return array
     * @example
     * [
     *     'host:port'
     * ]
     */
    public function getList(Client $client): array
    {
        // Get health service from consul
        $services = $this->agent->services();

        $services = [

        ];

        return $services;
    }
}

配置服務提供者

return [
    'user'           => [
      'class'   => ServiceClient::class,
      'provider' => bean(RpcProvider::class)
      // ...
    ]
];

服務熔斷

Swoft 使用 @Breaker 註解實現熔斷,可以在任何方法上面進行熔斷操作。 

/**
 * Class BreakerLogic
 *
 * @since 2.0
 *
 * @Bean()
 */
class BreakerLogic
{
    /**
     * @Breaker(fallback="funcFallback")
     *
     * @return string
     * @throws Exception
     */
    public function func(): string
    {
        // Do something

        throw new Exception('Breaker exception');
    }
    
    /**
     * @return string
     */
    public function funcFallback(): string
    {
        return 'funcFallback';
    }
}

服務限流

Swoft 中使用 @RateLimiter 註解實現服務限流,可以在任何方法上面限流,不僅僅是控制器,且 KEY 還支持 symfony/expression-language 表達式。

/**
 * Class LimiterController
 *
 * @since 2.0
 *
 * @Controller(prefix="limiter")
 */
class LimiterController
{
    /**
     * @RequestMapping()
     * @RateLimiter(key="request.getUriPath()", fallback="limiterFallback")
     *
     * @param Request $request
     *
     * @return array
     */
    public function requestLimiter(Request $request): array
    {
        $uri = $request->getUriPath();
        return ['requestLimiter', $uri];
    }

    /**
     * @param Request $request
     *
     * @return array
     */
    public function limiterFallback(Request $request): array
    {
        $uri = $request->getUriPath();
        return ['limiterFallback', $uri];
    }
}

配置中心

配置中心,需要用到 Swoft 官方提供的 Swoft-apollo 組件,如果其它第三方也類似。

聲明Agent

/**
 * Class AgentCommand
 *
 * @since 2.0
 *
 * @Command("agent")
 */
class AgentCommand
{
    /**
     * @Inject()
     *
     * @var Config
     */
    private $config;

    /**
     * @CommandMapping(name="index")
     */
    public function index(): void
    {
        $namespaces = [
            'application'
        ];

        while (true) {
            try {
                $this->config->listen($namespaces, [$this, 'updateConfigFile']);
            } catch (Throwable $e) {
                CLog::error('Config agent fail(%s %s %d)!', $e->getMessage(), $e->getFile(), $e->getLine());
            }
        }
    }

    /**
     * @param array $data
     *
     * @throws ContainerException
     * @throws ReflectionException
     */
    public function updateConfigFile(array $data): void
    {
        foreach ($data as $namespace => $namespaceData) {
            $configFile = sprintf('@config/%s.php', $namespace);

            $configKVs = $namespaceData['configurations'] ?? '';
            $content   = '<?php return ' . var_export($configKVs, true) . ';';
            Co::writeFile(alias($configFile), $content, FILE_NO_DEFAULT_CONTEXT);

            CLog::info('Apollo update success!');

            /** @var HttpServer $server */
            $server = bean('httpServer');
            $server->restart();
        }
    }
}

啟動Agent

Agent 只需要在服務(Http/RPC/Websocket)啟動前,運行即可。

php bin/swoft agent:index

更新內容

移除(Remove)

  • 移除 request->json() 方法(c9e8f04)

新增(Enhancement):

  • 新增介面依賴註入(6169f84)
  • 新增 getFile 方法獲取文件上傳保存之後的信息(fe7e3a6)
  • 新增 restart() 服務新增重啟方法(2ffec37)
  • 新增調用 1.x RPC 服務支持(30d73c3)
  • 新增 AOP 類名匹配支持正則表達式(bc5e479)
  • 新增 RPC Server /Http Server 中間件命名空間 use 錯誤提示(b1cec04)
  • 新增 驗證器排除屬性欄位 unfields(b1bf44f)
  • 新增 自動寫入時間戳(dc58011)
  • 新增 模型動作事件(dc58011)
  • 新增 資料庫遷移(26bb464)
  • 新增 實體自動與 json 和數組互轉(dc58011)
  • 新增 模型批量更新方法batchUpdateByIds(dc58011)

修複(Fixed):

  • 修複 cookies 設置時的一些問題,增加一些 withCookie 相關方法(b05afbb01)
  • 修複 在console使用協程方式運行命令時,沒有捕獲處理錯誤(8a5418bf)
  • 修複 websocket server 重啟命令沒有先停止舊server問題(db2d935)
  • 修複任務返回值為 null 問題(a69347c) 
  • 修複 RPC Server 只有類中間件無法使用問題()204bc7f
  • 修複 RPC Server 返回值為 null 問題(4d091be)
  • 修複 Logger 和 CLog 日誌等級無法覆蓋和無效問題(8eb8aba)
  • 修複 模型裡面的屬性不支持自定義表達式(dc58011)

更新(Update):

  • 驗證器優化,支持自定義驗證規則(d959a4f)
  • 重命名錯誤處理管理類 ErrorHanlders 為 ErrorManager (f3a8f04b)
  • console組件的異常處理改為由error組件提供的統一處理風格 (4f47204)
  • console組件允許設置禁用命令組(c5a0269)
  • 在預設的錯誤處理中,允許設置錯誤捕獲級別。預設級別是 E_ALL | E_STRICT (afff9029)
  • 優化 啟動ws server時同時啟用了http處理功能,信息面板添加提示(83a81170)
  • 優化 啟動ws server 並同時添加rpc server啟動,信息面板沒有顯示 rpc server信息(3d1d0d848)

擴展(Extra):

  • 文檔添加支持通過google進行搜索
  • 新增 apollo 組件
  • 新增 consul 組件
  • 新增 breaker 組件
  • 新增 limter 組件

資源


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

-Advertisement-
Play Games
更多相關文章
  • 上一篇文章 "MAT入門到精通(一)" 介紹了MAT的使用場景和基本概念,這篇文章開始介紹MAT的基本功能,後面還有兩篇,一篇是MAT的高級功能,另一篇是MAT實戰案例分析。 三、歡迎頁 使用MAT打開一個heap dump文件,解析完成後,預設會進入歡迎頁,歡迎頁里包含了一些常見的分析:最大記憶體占 ...
  • 如果這個: "leadcode的Hot100系列 62. 不同路徑 簡單的動態規劃" 看懂的話,那這題基本上是一樣的, 不同點在於: 1、這裡每條路徑相當於多了一個權值 2、結論不再固定,而是要比較不同走法哪個權值更小 針對第一點,需要把第一行和第一列的權值做一個累加: 假設這裡的權值都是1,則 | ...
  • 10-6 加法運算:提示用戶輸入提供數值輸入,常出現的一個問題是,用戶提供的是文本而不是數字。這種情況下,當你嘗試將輸入轉換為整數時,將 引發TypeError異常。編寫一個程式,提示用戶輸入兩個數字,再將它們相加並列印結果。在用戶輸入的任何一個值不是數字時都捕獲TypeError異常,並列印一條友 ...
  • 文件下載 1.寫action類 2.配置struts.xml 查看源碼發現要以流的方式返回給客戶端需要配置3個參數 上面那張圖流寫is根本調用不了getInputStream的方法要把is變數名更改為inputStream才可以。 下載完成 struts的OGNL OGNL是Object Graph ...
  • 題目比較清晰,簡單來說就是: | A | B | C | D | | | | | | | E | F | G | H | | I | J | K | L | 只能往右或者往下,從A到L,能有幾種走法。 這裡使用動態規劃的方法來做一下。 動態規劃最重要的就是動態方程,這裡簡單說下這個動態方程怎麼做出來 ...
  • 組合模式:將對象組合成樹形結構以表示“部分-整體”的層次結構。組合模式使得用戶對單個對象和組合對象的使用具有一致性。· 意思就是用樹狀結構來表示部分和整體,使他們具有一致性,一致性的意思就是都實現了相同的介面,舉個例子,一個公司可能有研發部門和銷售部門,然後這個公司又有子公司,子公司也有研發和銷售部 ...
  • 一、安裝和簡介 ElasticSearch是一個基於Lucene的搜索伺服器。它提供了一個分散式多用戶能力的全文搜索引擎,基於RESTful web介面。Elasticsearch是用Java開發的,並作為Apache許可條款下的開放源碼發佈,是當前流行的企業級搜索引擎。 搭建運行環境 "Linux ...
  • 一、Reids介紹 redis是一個key-value存儲系統。和Memcached類似,它支持存儲的value類型相對更多,包括string(字元串)、list(鏈表)、set(集合)、zset(sorted set --有序集合)和hash(哈希類型)。這些數據類型都支持push/pop、add ...
一周排行
    -Advertisement-
    Play Games
  • 前言 本文介紹一款使用 C# 與 WPF 開發的音頻播放器,其界面簡潔大方,操作體驗流暢。該播放器支持多種音頻格式(如 MP4、WMA、OGG、FLAC 等),並具備標記、實時歌詞顯示等功能。 另外,還支持換膚及多語言(中英文)切換。核心音頻處理採用 FFmpeg 組件,獲得了廣泛認可,目前 Git ...
  • OAuth2.0授權驗證-gitee授權碼模式 本文主要介紹如何筆者自己是如何使用gitee提供的OAuth2.0協議完成授權驗證並登錄到自己的系統,完整模式如圖 1、創建應用 打開gitee個人中心->第三方應用->創建應用 創建應用後在我的應用界面,查看已創建應用的Client ID和Clien ...
  • 解決了這個問題:《winForm下,fastReport.net 從.net framework 升級到.net5遇到的錯誤“Operation is not supported on this platform.”》 本文內容轉載自:https://www.fcnsoft.com/Home/Sho ...
  • 國內文章 WPF 從裸 Win 32 的 WM_Pointer 消息獲取觸摸點繪製筆跡 https://www.cnblogs.com/lindexi/p/18390983 本文將告訴大家如何在 WPF 裡面,接收裸 Win 32 的 WM_Pointer 消息,從消息裡面獲取觸摸點信息,使用觸摸點 ...
  • 前言 給大家推薦一個專為新零售快消行業打造了一套高效的進銷存管理系統。 系統不僅具備強大的庫存管理功能,還集成了高性能的輕量級 POS 解決方案,確保頁面載入速度極快,提供良好的用戶體驗。 項目介紹 Dorisoy.POS 是一款基於 .NET 7 和 Angular 4 開發的新零售快消進銷存管理 ...
  • ABP CLI常用的代碼分享 一、確保環境配置正確 安裝.NET CLI: ABP CLI是基於.NET Core或.NET 5/6/7等更高版本構建的,因此首先需要在你的開發環境中安裝.NET CLI。這可以通過訪問Microsoft官網下載並安裝相應版本的.NET SDK來實現。 安裝ABP ...
  • 問題 問題是這樣的:第三方的webapi,需要先調用登陸介面獲取Cookie,訪問其它介面時攜帶Cookie信息。 但使用HttpClient類調用登陸介面,返回的Headers中沒有找到Cookie信息。 分析 首先,使用Postman測試該登陸介面,正常返回Cookie信息,說明是HttpCli ...
  • 國內文章 關於.NET在中國為什麼工資低的分析 https://www.cnblogs.com/thinkingmore/p/18406244 .NET在中國開發者的薪資偏低,主要因市場需求、技術棧選擇和企業文化等因素所致。歷史上,.NET曾因微軟的閉源策略發展受限,儘管後來推出了跨平臺的.NET ...
  • 在WPF開發應用中,動畫不僅可以引起用戶的註意與興趣,而且還使軟體更加便於使用。前面幾篇文章講解了畫筆(Brush),形狀(Shape),幾何圖形(Geometry),變換(Transform)等相關內容,今天繼續講解動畫相關內容和知識點,僅供學習分享使用,如有不足之處,還請指正。 ...
  • 什麼是委托? 委托可以說是把一個方法代入另一個方法執行,相當於指向函數的指針;事件就相當於保存委托的數組; 1.實例化委托的方式: 方式1:通過new創建實例: public delegate void ShowDelegate(); 或者 public delegate string ShowDe ...