Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Sign in / Register
Toggle navigation
N
nfkit
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Locked Files
Issues
0
Issues
0
List
Boards
Labels
Service Desk
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Security & Compliance
Security & Compliance
Dependency List
License Compliance
Packages
Packages
List
Container Registry
Analytics
Analytics
CI / CD
Code Review
Insights
Issues
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
nanahira
nfkit
Commits
467ade8b
Commit
467ade8b
authored
Feb 07, 2026
by
nanahira
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
add ProtoMiddlewareDispatcher
parent
9e394bd4
Changes
9
Hide whitespace changes
Inline
Side-by-side
Showing
9 changed files
with
264 additions
and
2 deletions
+264
-2
src/i18n/i18n.ts
src/i18n/i18n.ts
+1
-1
src/middleware-dispatcher/dynamic-middleware-dispatcher.ts
src/middleware-dispatcher/dynamic-middleware-dispatcher.ts
+54
-0
src/middleware-dispatcher/index.ts
src/middleware-dispatcher/index.ts
+4
-0
src/middleware-dispatcher/middleware-dispatcher.ts
src/middleware-dispatcher/middleware-dispatcher.ts
+29
-0
src/middleware-dispatcher/proto-middleware-dispatcher.ts
src/middleware-dispatcher/proto-middleware-dispatcher.ts
+98
-0
src/middleware-dispatcher/types.ts
src/middleware-dispatcher/types.ts
+30
-0
src/types.ts
src/types.ts
+2
-0
tests/i18n-lookup.spec.ts
tests/i18n-lookup.spec.ts
+1
-1
tests/proto-middleware-dispatcher.spec.ts
tests/proto-middleware-dispatcher.spec.ts
+45
-0
No files found.
src/i18n/i18n.ts
View file @
467ade8b
import
{
MiddlewareDispatcher
}
from
'
../middleware-dispatcher
'
;
import
{
MiddlewareDispatcher
}
from
'
../middleware-dispatcher
/middleware-dispatcher
'
;
import
{
parseI18n
}
from
'
../utility/parse-i18n
'
;
import
{
parseI18n
}
from
'
../utility/parse-i18n
'
;
import
{
I18nMiddleware
,
I18nOptions
}
from
'
./types
'
;
import
{
I18nMiddleware
,
I18nOptions
}
from
'
./types
'
;
import
{
patchStringInObject
}
from
'
../patch-string-in-object
'
;
import
{
patchStringInObject
}
from
'
../patch-string-in-object
'
;
...
...
src/middleware-dispatcher.ts
→
src/middleware-dispatcher
/dynamic-middleware-dispatcher
.ts
View file @
467ade8b
import
{
Awaitable
}
from
'
./types
'
;
import
{
AnyFunc
,
type
AnyFunc
=
(...
args
:
any
[])
=>
any
;
MiddlewareDispatcherOptions
,
Middleware
,
type
MiddlewareValue
<
F
extends
AnyFunc
>
=
Awaited
<
ReturnType
<
F
>>
;
MiddlewareResult
,
type
MiddlewareResult
<
F
extends
AnyFunc
>
=
Promise
<
MiddlewareValue
<
F
>>
;
MiddlewareAcceptResult
,
type
MiddlewareNext
<
F
extends
AnyFunc
>
=
()
=>
MiddlewareResult
<
F
>
;
MiddlewareErrorHandler
,
type
MiddlewareArgs
<
F
extends
AnyFunc
>
=
[
MiddlewareReturn
,
...
args
:
Parameters
<
F
>
,
}
from
'
./types
'
;
next
:
MiddlewareNext
<
F
>
,
];
export
class
DynamicMiddlewareDispatcher
<
F
extends
AnyFunc
>
{
type
MiddlewareReturn
<
F
extends
AnyFunc
>
=
Awaitable
<
MiddlewareValue
<
F
>>
;
constructor
(
protected
options
:
MiddlewareDispatcherOptions
<
F
>
=
{})
{}
export
type
Middleware
<
F
extends
AnyFunc
>
=
(
async
buildMiddlewares
(...
args
:
Parameters
<
F
>
):
Promise
<
Middleware
<
F
>
[]
>
{
...
args
:
MiddlewareArgs
<
F
>
return
[];
)
=>
MiddlewareReturn
<
F
>
;
export
type
MiddlewareAcceptResult
<
F
extends
AnyFunc
>
=
(
s
:
MiddlewareValue
<
F
>
,
)
=>
Awaitable
<
boolean
>
;
export
type
MiddlewareErrorHandler
<
F
extends
AnyFunc
>
=
(
e
:
any
,
args
:
Parameters
<
F
>
,
next
:
MiddlewareNext
<
F
>
,
)
=>
Awaitable
<
MiddlewareValue
<
F
>>
;
export
interface
MiddlewareDispatcherOptions
<
F
extends
AnyFunc
>
{
acceptResult
?:
MiddlewareAcceptResult
<
F
>
;
errorHandler
?:
MiddlewareErrorHandler
<
F
>
;
}
export
class
MiddlewareDispatcher
<
F
extends
AnyFunc
>
{
constructor
(
private
options
:
MiddlewareDispatcherOptions
<
F
>
=
{})
{}
middlewares
:
Middleware
<
F
>
[]
=
[];
middleware
(
mw
:
Middleware
<
F
>
,
prior
=
false
)
{
if
(
prior
)
{
this
.
middlewares
.
unshift
(
mw
);
}
else
{
this
.
middlewares
.
push
(
mw
);
}
return
this
;
}
removeMiddleware
(
mw
:
Middleware
<
F
>
)
{
const
index
=
this
.
middlewares
.
indexOf
(
mw
);
if
(
index
>=
0
)
{
this
.
middlewares
.
splice
(
index
,
1
);
}
return
this
;
}
}
dispatch
(...
args
:
Parameters
<
F
>
):
MiddlewareResult
<
F
>
{
async
dispatch
(...
args
:
Parameters
<
F
>
):
MiddlewareResult
<
F
>
{
const
mws
=
this
.
middlewares
;
const
mws
=
await
this
.
buildMiddlewares
(...
args
)
;
const
acceptResult
:
MiddlewareAcceptResult
<
F
>
=
const
acceptResult
:
MiddlewareAcceptResult
<
F
>
=
this
.
options
.
acceptResult
||
((
res
)
=>
res
!=
null
);
this
.
options
.
acceptResult
||
((
res
)
=>
res
!=
null
);
const
errorHandler
:
MiddlewareErrorHandler
<
F
>
=
const
errorHandler
:
MiddlewareErrorHandler
<
F
>
=
...
...
src/middleware-dispatcher/index.ts
0 → 100644
View file @
467ade8b
export
*
from
'
./types
'
;
export
*
from
'
./middleware-dispatcher
'
;
export
*
from
'
./dynamic-middleware-dispatcher
'
;
export
*
from
'
./proto-middleware-dispatcher
'
;
src/middleware-dispatcher/middleware-dispatcher.ts
0 → 100644
View file @
467ade8b
import
{
AnyFunc
,
Middleware
}
from
'
./types
'
;
import
{
DynamicMiddlewareDispatcher
}
from
'
./dynamic-middleware-dispatcher
'
;
export
class
MiddlewareDispatcher
<
F
extends
AnyFunc
,
>
extends
DynamicMiddlewareDispatcher
<
F
>
{
middlewares
:
Middleware
<
F
>
[]
=
[];
middleware
(
mw
:
Middleware
<
F
>
,
prior
=
false
)
{
if
(
prior
)
{
this
.
middlewares
.
unshift
(
mw
);
}
else
{
this
.
middlewares
.
push
(
mw
);
}
return
this
;
}
removeMiddleware
(
mw
:
Middleware
<
F
>
)
{
const
index
=
this
.
middlewares
.
indexOf
(
mw
);
if
(
index
>=
0
)
{
this
.
middlewares
.
splice
(
index
,
1
);
}
return
this
;
}
async
buildMiddlewares
(...
args
:
Parameters
<
F
>
):
Promise
<
Middleware
<
F
>
[]
>
{
return
this
.
middlewares
;
}
}
src/middleware-dispatcher/proto-middleware-dispatcher.ts
0 → 100644
View file @
467ade8b
import
{
Middleware
,
MiddlewareResult
}
from
'
./types
'
;
import
{
AnyClass
,
ClassType
}
from
'
../types
'
;
import
{
DynamicMiddlewareDispatcher
}
from
'
./dynamic-middleware-dispatcher
'
;
export
type
ProtoMiddlewareArgs
<
A
extends
any
[],
T
=
any
>
=
[
...
args
:
A
,
inst
:
T
,
];
export
type
ProtoMiddlewareFunc
<
A
extends
any
[],
T
=
any
>
=
(
...
args
:
ProtoMiddlewareArgs
<
A
,
T
>
)
=>
T
;
export
class
ProtoMiddlewareDispatcher
<
A
extends
any
[],
>
extends
DynamicMiddlewareDispatcher
<
ProtoMiddlewareFunc
<
A
>>
{
private
middlewareProtoMap
=
new
Map
<
AnyClass
,
Middleware
<
ProtoMiddlewareFunc
<
A
>>
[]
>
();
private
middlewareProtoMapPrior
=
new
Map
<
AnyClass
,
Middleware
<
ProtoMiddlewareFunc
<
A
>>
[]
>
();
middleware
<
T
>
(
cls
:
ClassType
<
T
>
,
mw
:
Middleware
<
ProtoMiddlewareFunc
<
A
,
T
>>
,
prior
=
false
,
)
{
const
map
=
prior
?
this
.
middlewareProtoMapPrior
:
this
.
middlewareProtoMap
;
const
mws
=
map
.
get
(
cls
)
||
[];
mws
.
push
(
mw
);
map
.
set
(
cls
,
mws
);
return
this
;
}
removeMiddleware
<
T
>
(
cls
:
ClassType
<
T
>
,
mw
:
Middleware
<
ProtoMiddlewareFunc
<
A
,
T
>>
,
)
{
for
(
const
map
of
[
this
.
middlewareProtoMap
,
this
.
middlewareProtoMapPrior
])
{
const
mws
=
map
.
get
(
cls
);
if
(
mws
)
{
const
index
=
mws
.
indexOf
(
mw
);
if
(
index
>=
0
)
{
mws
.
splice
(
index
,
1
);
}
}
}
return
this
;
}
async
dispatch
<
T
>
(
...
args
:
ProtoMiddlewareArgs
<
A
,
T
>
):
MiddlewareResult
<
ProtoMiddlewareFunc
<
A
,
T
>>
{
return
super
.
dispatch
(...
args
);
}
async
buildMiddlewares
(...
args
:
ProtoMiddlewareArgs
<
A
>
)
{
// buildMiddlewares 只需要知道 inst
if
(
args
.
length
===
0
)
return
[];
const
inst
=
args
[
args
.
length
-
1
];
if
(
!
inst
||
typeof
inst
!==
'
object
'
)
return
[];
// 1. 收集原型链(Base → Sub)
const
chain
:
AnyClass
[]
=
[];
let
cur
:
any
=
inst
.
constructor
;
while
(
cur
&&
cur
!==
Object
)
{
chain
.
push
(
cur
);
cur
=
Object
.
getPrototypeOf
(
cur
.
prototype
)?.
constructor
;
}
chain
.
reverse
();
const
result
:
Middleware
<
ProtoMiddlewareFunc
<
A
>>
[]
=
[];
// 2. prior:Base → Sub
for
(
const
cls
of
chain
)
{
const
mws
=
this
.
middlewareProtoMapPrior
.
get
(
cls
);
if
(
mws
)
{
result
.
push
(...
mws
);
}
}
// 3. normal:Sub → Base
for
(
let
i
=
chain
.
length
-
1
;
i
>=
0
;
i
--
)
{
const
cls
=
chain
[
i
];
const
mws
=
this
.
middlewareProtoMap
.
get
(
cls
);
if
(
mws
)
{
result
.
push
(...
mws
);
}
}
return
result
;
}
}
src/middleware-dispatcher/types.ts
0 → 100644
View file @
467ade8b
import
{
Awaitable
}
from
'
../types
'
;
export
type
AnyFunc
=
(...
args
:
any
[])
=>
any
;
export
type
MiddlewareValue
<
F
extends
AnyFunc
>
=
Awaited
<
ReturnType
<
F
>>
;
export
type
MiddlewareResult
<
F
extends
AnyFunc
>
=
Promise
<
MiddlewareValue
<
F
>>
;
export
type
MiddlewareNext
<
F
extends
AnyFunc
>
=
()
=>
MiddlewareResult
<
F
>
;
export
type
MiddlewareArgs
<
F
extends
AnyFunc
>
=
[
...
args
:
Parameters
<
F
>
,
next
:
MiddlewareNext
<
F
>
,
];
export
type
MiddlewareReturn
<
F
extends
AnyFunc
>
=
Awaitable
<
MiddlewareValue
<
F
>>
;
export
type
Middleware
<
F
extends
AnyFunc
>
=
(
...
args
:
MiddlewareArgs
<
F
>
)
=>
MiddlewareReturn
<
F
>
;
export
type
MiddlewareAcceptResult
<
F
extends
AnyFunc
>
=
(
s
:
MiddlewareValue
<
F
>
,
)
=>
Awaitable
<
boolean
>
;
export
type
MiddlewareErrorHandler
<
F
extends
AnyFunc
>
=
(
e
:
any
,
args
:
Parameters
<
F
>
,
next
:
MiddlewareNext
<
F
>
,
)
=>
Awaitable
<
MiddlewareValue
<
F
>>
;
export
interface
MiddlewareDispatcherOptions
<
F
extends
AnyFunc
>
{
acceptResult
?:
MiddlewareAcceptResult
<
F
>
;
errorHandler
?:
MiddlewareErrorHandler
<
F
>
;
}
src/types.ts
View file @
467ade8b
export
type
Awaitable
<
T
>
=
T
|
Promise
<
T
>
;
export
type
Awaitable
<
T
>
=
T
|
Promise
<
T
>
;
export
type
AnyClass
=
new
(...
args
:
any
[])
=>
any
;
export
type
ClassType
<
T
=
any
>
=
new
(...
args
:
any
[])
=>
T
;
tests/i18n-lookup.spec.ts
View file @
467ade8b
// i18n-lookup.spec.ts
// i18n-lookup.spec.ts
import
{
I18nLookupMiddleware
}
from
'
../src/i18n
'
;
import
{
I18nLookupMiddleware
}
from
'
../src/i18n
'
;
import
{
MiddlewareDispatcher
}
from
'
../src/middleware-dispatcher
'
;
import
{
MiddlewareDispatcher
}
from
'
../src/middleware-dispatcher
/middleware-dispatcher
'
;
type
F
=
(
locale
:
string
,
key
:
string
)
=>
string
|
undefined
;
type
F
=
(
locale
:
string
,
key
:
string
)
=>
string
|
undefined
;
...
...
tests/proto-middleware-dispatcher.spec.ts
0 → 100644
View file @
467ade8b
import
{
ProtoMiddlewareDispatcher
}
from
'
../src/middleware-dispatcher
'
;
describe
(
'
ProtoMiddlewareDispatcher
'
,
()
=>
{
class
Base
{}
class
Sub
extends
Base
{}
it
(
'
builds middlewares by prototype chain (prior: Base->Sub, normal: Sub->Base)
'
,
async
()
=>
{
const
d
=
new
ProtoMiddlewareDispatcher
<
[
res
:
number
]
>
();
const
order
:
string
[]
=
[];
d
.
middleware
(
Base
,
async
(
x
,
inst
,
next
)
=>
{
order
.
push
(
'
base:prior
'
);
return
next
();
},
true
,
);
d
.
middleware
(
Sub
,
async
(
x
,
inst
,
next
)
=>
{
order
.
push
(
'
sub:prior
'
);
return
next
();
},
true
,
);
d
.
middleware
(
Sub
,
async
(
x
,
inst
,
next
)
=>
{
order
.
push
(
'
sub:normal
'
);
return
next
();
});
d
.
middleware
(
Base
,
async
(
x
,
inst
)
=>
{
order
.
push
(
'
base:normal
'
);
return
x
*
10
;
});
const
res
=
await
d
.
dispatch
(
3
,
new
Sub
());
expect
(
res
).
toBe
(
30
);
expect
(
order
).
toEqual
([
'
base:prior
'
,
'
sub:prior
'
,
'
sub:normal
'
,
'
base:normal
'
,
]);
});
});
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment