Japanese/日本語

  • 雨崎しのぶ
  • 2024-07-12

Readme

Forgexは、すべてFortranで書かれた正規表現エンジンです。

このプロジェクトはFortranパッケージマネージャーで管理され、 正規表現の基本的な処理を提供し、MITライセンス のもとで利用可能なフリーソフトウェアです。 エンジンの核となるアルゴリズムには決定性有限オートマトン(Deterministic Finite Automaton, DFA)を使用しています。 この選択は実行時パフォーマンスを重視したものです。

機能

Forgexが処理を受け付ける正規表現の記法は以下の通りです。

メタキャラクター

  • | 選言(alternation)のバーティカルバー
  • * ゼロ回以上にマッチするアスタリスク
  • + 一回以上にマッチするプラス記号
  • ? ゼロ回または一回にマッチするクエスチョンマーク
  • \ メタキャラクターのエスケープ
  • . 任意の一文字にマッチするピリオド

文字クラス

  • 文字クラス(例: [a-z]
  • 否定クラス(例: [^a-z]
  • Unicode文字クラス(例: [α-ωぁ-ん]

否定クラスは制御文字にはマッチしないことに注意してください。

繰り返し回数の指定

  • {num},
  • {,max},
  • {min,},
  • {min, max}, ここで nummaxは0(ゼロ)以外の自然数を指定します。

アンカー

  • ^, 行頭にマッチ
  • $, 行末にマッチ

略記法

  • \t, タブ文字
  • \n, 改行文字 (LFまたはCRLF)
  • \r, 復帰文字 (CR)
  • \s, 空白文字 (半角スペース, タブ文字, CR, LF, FF, 全角スペース U+3000)
  • \S, 非空白文字
  • \w, ラテン文字アルファベット、半角数字及びアンダースコア([a-zA-Z0-9_])
  • \W, \wの否定クラス([^a-zA-Z0-9_])
  • \d, 半角数字 ([0-9])
  • \D, 非半角数字 ([^0-9])

ドキュメント

ドキュメントは英語と日本語で次のリンクから利用可能です。 https://shinobuamasaki.github.io/forgex.

使用方法

動作確認は以下のコンパイラーで行っています。

  • GNU Fortran (gfortran) v13.2.1
  • Intel Fortran Compiler (ifx) 2024.0.0 20231017

以下では、ビルドとAPIの使い方について解説しますが、Fortranパッケージマネージャー(fpm)を利用することを前提とします。

ビルド

まず初めに、あなたのプロジェクトのfpm.tomlに以下の記述を追加します。

[dependencies]
forgex = {git = "https://github.com/shinobuamasaki/forgex", tag="v2.0"}

代替の選択肢

macOSを使用している場合、このライブラリはMacPortsを使用して、以下のコマンドを実行することによりインストールすることができます。

sudo port install forgex

この場合には、Forgexのモジュールファイルforgex*.modは、/opt/local/include/forgexディレクトリに配置され、ライブラリのバイナリファイルは/opt/local/libに配置されるので、ソースファイルをコンパイルするためには以下のようなコマンドを実行する必要があります。

gfortran main.f90 -I/opt/local/include/forgex -L/opt/local/lib -lforgex

このインストールの方法を使用した場合に、fpmを使用してプロジェクトをビルドするには、次の変更をfpm.tomlに加えます。

[build]
external-modules = [ "forgex" ]
link = [ "forgex" ]

そしてそのプロジェクトを次のコマンドによってビルドします。

fpm build --flag "-I/opt/local/include/forgex" --link-flag "-L/opt/local/lib"

こちらも参照してください:https://ports.macports.org/port/forgex/details

APIの使い方

そのプロジェクトのプログラムのヘッダーにuse forgexと記述すると、.in..match.の演算子、 regexサブルーチンとregex_f関数が導入され、use文の有効なスコープでこれらの4つを使用することができます。

program main
   use :: forgex
   implicit none

.in.演算子は、文字列型を引数にとり、第一引数のパターンが、第二引数の文字列に含まれる場合に真を返します。

block
   character(:), allocatable :: pattern, str

   pattern = 'foo(bar|baz)'
   str = "foobarbaz"
   print *, pattern .in. str  ! T

   str = "foofoo"
   print *, pattern .in. str  ! F
end block

.match.演算子は、同様に指定されたパターンが、厳密に文字列と一致する場合に真を返します。

block
   character(:), allocatable :: pattern, str

   pattern = '\d{3}-\d{4}'
   str = '100-0001'
   print *, pattern .match. str  ! T

   str = '1234567'
   print *, pattern .match. str  ! F
end block

なお、.in..match.の演算子は無効なパターンの入力に対して偽を返します。

regex関数は、入力文字列の中でパターンに一致した部分文字列を返します。

block
   character(:), allocatable :: pattern, str, res
   integer :: length

   pattern = 'foo(bar|baz)'
   str = 'foobarbaz'

   call regex(pattern, str, res)
   print *, res                              ! foobar

   ! call regex(pattern, str, res, length)
        ! the value 6 stored in optional `length` variable.

end block

オプショナル引数のfrom/toを使用すると、与えた文字列から添字を指定して部分文字列を切り出すことができます。

block
   character(:), allocatable :: pattern, str, res
   integer :: from, to

   pattern = '[d-f]{3}'
   str = 'abcdefghi'

   call regex(pattern, str, res, from=from, to=to)
   print *, res                   ! def

   ! The `from` and `to` variables store the indices of the start and end points
   ! of the matched part of the string `str`, respectively.

   ! Cut out before the matched part.
   print *, str(1:from-1)        ! abc

   ! Cut out the matched part that equivalent to the result of the `regex` function.
   print *, str(from:to)         ! def

   ! Cut out after the matched part.
   print *, str(to+1:len(str))   ! ghi

end block

regexサブルーチンの宣言部(インタフェース)は次の通りです。

interface regex
   module procedure :: subroutine__regex
end interface

pure subroutine subroutine__regex(pattern, text, res, length, from, to, status, err_msg)
   implicit none
   character(*),              intent(in)    :: pattern, text
   character(:), allocatable, intent(inout) :: res
   integer, optional,         intent(inout) :: length, from, to, status
   character(*), optional,    intent(inout) :: err_msg

マッチした文字列を関数の戻り値として得たい場合には、regex_f関数を使用してください。

interface regex_f
   module procedure :: function__regex
end interface regex_f

pure function function__regex(pattern, text) result(res)
   implicit none
   character(*), intent(in)  :: pattern, text
   character(:), allocatable :: res

入力パターンの検証

バージョン4.0より、is_valid_regex関数が導入され、正規表現パターンを有効なものかどうかを確認できるようになりました。入力パターンを事前に検査する場合にはこの関数を使用してから上のAPIにパターンを渡してください。

この関数のインタフェースは次の通りです。

interface is_valid_regex
   module procedure :: is_valid_regex_pattern
end interfac

pure elemental function is_valid_regex_pattern (pattern) result(res)
   implicit none
   character(*), intent(in)  :: pattern
   logical                   :: res

UTF-8文字列のマッチング

UTF-8の文字列についても、ASCII文字と同様に正規表現のパターンで一致させることができます。 以下の例は、漢文の一節に対してマッチングを試みています。

block
   character(:), allocatable :: pattern, str
   integer :: length

   pattern = "夢.{1,7}胡蝶"
   str = "昔者莊周夢爲胡蝶 栩栩然胡蝶也"

   print *, pattern .in. str            ! T
   call regex(pattern, str, res, length)
   print *, res                         ! 夢爲胡蝶 栩栩然胡蝶
   print *, length                      ! 30 (is 3-byte * 10 characters)

end block

この例ではlength変数にバイト長が格納され、この場合は10個の3バイト文字に一致したので、その長さは30となります。

CLIツール

バージョン3.2以降では、Forgexエンジンを使用したコマンドラインツールforgex-cliが提供されてり、Forgexエンジン自体のデバッグ、正規表現マッチングのテストやベンチマークのために使用することができます。 以下のようにコマンドを実行することで、標準出力に結果を得ることができます。 使い方の詳細についてはドキュメンテーションを参照してください。

コマンド:

forgex-cli find match lazy-dfa '([a-z]*g+)n?' .match. 'assign'

fpm run経由で実行する場合:

fpm run forgex-cli --profile release -- find match lazy-dfa '([a-z]*g+)n?' .match. 'assign'

出力:

            pattern: ([a-z]*g+)n?
               text: 'assign'
         parse time:        46.5us
   compile nfa time:        74.9us
dfa initialize time:        78.4us
        search time:       661.7us
    matching result:         T
 memory (estimated):     10380

========== Thompson NFA ===========
state    1: (?, 5)
state    2: <Accepted>
state    3: (n, 2)(?, 2)
state    4: (g, 7)
state    5: (["a"-"f"], 6)(g, 6)(["h"-"m"], 6)(n, 6)(["o"-"z"], 6)(?, 4)
state    6: (?, 5)
state    7: (?, 8)
state    8: (g, 9)(?, 3)
state    9: (?, 8)
=============== DFA ===============
   1 : ["a"-"f"]=>2
   2 : ["o"-"z"]=>2 ["h"-"m"]=>2 g=>3
   3A: n=>4
   4A:
state    1  = ( 1 4 5 )
state    2  = ( 4 5 6 )
state    3A = ( 2 3 4 5 6 7 8 )
state    4A = ( 2 4 5 6 )
===================================

バージョン3.5以降、コマンドラインツールはForgexとは別のリポジトリで提供されるようになりました。このリンクを参照してください:ShinobuAmasaki/forgex-cli

注意

  • WindowおよびmacOS環境のgfortranでコンパイルされたプログラムでは、OpenMPの並列ブロックの中で割り付け可能文字列型変数を使用すると、セグメンテーション違反などでプログラムが停止する可能性があります。
  • コマンドラインツールforgex-cliをWindows上のPowerShellで利用する場合、Unicode文字を正しく入出力するには、システムのロケールをUTF-8に変更する必要があります。
  • is_valid_regex関数の追加に関わるAPIの内部的な変更として、.in..match.の演算子は無効なパターンの入力に対してFalseを返すようになりました(バージョン3.5以前ではerror stop文を実行して処理を終了します)。

To Do

  • Unicodeエスケープシーケンス\p{...}の追加
  • UTF-8において無効なバイトストリームへの対処
  • ✅ 無効なパターンの処理
  • ✅️ リテラル検索によるマッチングの最適化
  • ✅️ デバッグおよびベンチマーク用のCLIツールを追加 -> ShinobuAmasaki/forgex-cli
  • ✅️ すべてのAPI演算子にpure elemental属性を追加
  • ✅️ ドキュメントの公開
  • ✅️ UTF-8文字の基本的なサポート
  • ✅️ On-the-FlyのDFA構築
  • ✅️ CMakeによるビルドのサポート
  • ✅️ 簡単な時間計測ツールの追加 -> ShinobuAmasaki/forgex-cli
  • マッチングの並列化

コーディング規約

本プロジェクトに含まれるすべてのコードは、3スペースのインデントで記述されます。

謝辞

冪集合構成法のアルゴリズムと構文解析については、Russ Cox氏の論文と近藤嘉雪氏の本を参考にしました。 優先度付きキューの実装は、ue1221さんのコードに基づいています。 文字列に対して.in.演算子を適用するというアイデアは、soybeanさんのものにインスパイアされました。 forgex-cliのコマンドラインインターフェイスの設計については、Rust言語のregex-cliを参考にしました。

参考文献

  1. Russ Cox "Regular Expression Matching Can Be Simple And Fast", 2007年
  2. 近藤嘉雪, "定本 Cプログラマのためのアルゴリズムとデータ構造", 1998年, SB Creative.
  3. ue1221/fortran-utilities
  4. kazulagi, @soybean, Fortranでユーザー定義演算子.in.を作る - Qiita.com, 2022年
  5. rust-lang/regex/regex-cli

ライセンス

このプロジェクトはMITライセンスで提供されるフリーソフトウェアです (cf. LICENSE)。