2013-10-30 00:48:23 +01:00
|
|
|
|
unit MVCFramework.Router;
|
|
|
|
|
|
|
|
|
|
interface
|
|
|
|
|
|
|
|
|
|
uses
|
|
|
|
|
Web.HTTPApp,
|
|
|
|
|
RTTIUtilsU,
|
|
|
|
|
MVCFramework.Commons,
|
|
|
|
|
System.RTTI,
|
|
|
|
|
MVCFramework,
|
|
|
|
|
System.Generics.Collections;
|
|
|
|
|
|
|
|
|
|
type
|
|
|
|
|
TMVCRouter = class
|
|
|
|
|
private
|
2014-04-01 00:02:31 +02:00
|
|
|
|
FCTX: TRttiContext;
|
2013-10-30 00:48:23 +01:00
|
|
|
|
FMethodToCall: TRTTIMethod;
|
|
|
|
|
FMVCControllerClass: TMVCControllerClass;
|
|
|
|
|
FMVCConfig: TMVCConfig;
|
2014-04-01 00:02:31 +02:00
|
|
|
|
function IsHTTPContentTypeCompatible(AWebRequestMethodType: TMVCHTTPMethodType; AContentType: AnsiString;
|
|
|
|
|
AAttributes: TArray<TCustomAttribute>): Boolean;
|
|
|
|
|
function IsHTTPAcceptCompatible(AWebRequestMethodType: TMVCHTTPMethodType; AAccept: AnsiString;
|
2013-11-09 14:22:11 +01:00
|
|
|
|
AAttributes: TArray<TCustomAttribute>): Boolean;
|
2014-06-27 15:30:39 +02:00
|
|
|
|
function GetFirstMimeType(const AContentType: string): string;
|
2013-10-30 00:48:23 +01:00
|
|
|
|
protected
|
|
|
|
|
function IsHTTPMethodCompatible(AMethodType: TMVCHTTPMethodType;
|
|
|
|
|
AAttributes: TArray<TCustomAttribute>): Boolean; virtual;
|
|
|
|
|
function IsCompatiblePath(AMVCPath: string; APath: string;
|
|
|
|
|
var AParams: TMVCRequestParamsTable): Boolean; virtual;
|
2013-11-09 14:22:11 +01:00
|
|
|
|
function GetAttribute<T: TCustomAttribute>(AAttributes
|
|
|
|
|
: TArray<TCustomAttribute>): T;
|
2013-10-30 00:48:23 +01:00
|
|
|
|
|
|
|
|
|
public
|
|
|
|
|
class function StringMethodToHTTPMetod(const Value: AnsiString)
|
|
|
|
|
: TMVCHTTPMethodType;
|
|
|
|
|
constructor Create(AMVCConfig: TMVCConfig);
|
|
|
|
|
function ExecuteRouting(AWebRequestPathInfo: AnsiString;
|
2014-04-01 00:02:31 +02:00
|
|
|
|
AWebRequestMethodType: TMVCHTTPMethodType;
|
|
|
|
|
AWebRequestContentType: AnsiString;
|
|
|
|
|
AWebRequestAccept: AnsiString;
|
2013-10-30 00:48:23 +01:00
|
|
|
|
AMVCControllers: TList<TMVCControllerClass>;
|
2014-06-27 15:30:39 +02:00
|
|
|
|
ADefaultContentType: string;
|
|
|
|
|
ADefaultContentCharset: string;
|
2013-10-30 00:48:23 +01:00
|
|
|
|
var AMVCRequestParams: TMVCRequestParamsTable;
|
2014-06-27 15:30:39 +02:00
|
|
|
|
out AResponseContentType: string;
|
|
|
|
|
out AResponseContentEncoding: string)
|
2013-11-09 14:22:11 +01:00
|
|
|
|
: Boolean; overload;
|
2013-10-30 00:48:23 +01:00
|
|
|
|
property MethodToCall: TRTTIMethod read FMethodToCall;
|
|
|
|
|
property MVCControllerClass: TMVCControllerClass read FMVCControllerClass;
|
|
|
|
|
end;
|
|
|
|
|
|
|
|
|
|
implementation
|
|
|
|
|
|
|
|
|
|
uses
|
|
|
|
|
System.StrUtils,
|
|
|
|
|
System.RegularExpressions,
|
|
|
|
|
System.SysUtils,
|
|
|
|
|
idURI;
|
|
|
|
|
|
|
|
|
|
{ TMVCRouter }
|
|
|
|
|
|
|
|
|
|
constructor TMVCRouter.Create(AMVCConfig: TMVCConfig);
|
|
|
|
|
begin
|
|
|
|
|
inherited Create;
|
|
|
|
|
FMVCConfig := AMVCConfig;
|
|
|
|
|
end;
|
|
|
|
|
|
|
|
|
|
function TMVCRouter.ExecuteRouting(AWebRequestPathInfo: AnsiString;
|
2014-04-01 00:02:31 +02:00
|
|
|
|
AWebRequestMethodType: TMVCHTTPMethodType; AWebRequestContentType: AnsiString; AWebRequestAccept: AnsiString;
|
2013-10-30 00:48:23 +01:00
|
|
|
|
AMVCControllers: TList<TMVCControllerClass>;
|
2014-06-27 15:30:39 +02:00
|
|
|
|
ADefaultContentType, ADefaultContentCharset: string;
|
2013-10-30 00:48:23 +01:00
|
|
|
|
var AMVCRequestParams: TMVCRequestParamsTable;
|
2014-06-27 15:30:39 +02:00
|
|
|
|
out AResponseContentType: string;
|
|
|
|
|
out AResponseContentEncoding: string): Boolean;
|
2013-10-30 00:48:23 +01:00
|
|
|
|
var
|
|
|
|
|
controllerClass: TMVCControllerClass;
|
|
|
|
|
_type: TRttiType;
|
|
|
|
|
_methods: TArray<TRTTIMethod>;
|
|
|
|
|
_method: TRTTIMethod;
|
|
|
|
|
_attribute: TCustomAttribute;
|
2014-04-01 00:02:31 +02:00
|
|
|
|
_attributes: TArray<TCustomAttribute>;
|
2013-10-30 00:48:23 +01:00
|
|
|
|
i: Integer;
|
|
|
|
|
ControllerMappedPath: string;
|
|
|
|
|
MethodPathAttribute: string;
|
2013-11-09 14:22:11 +01:00
|
|
|
|
MVCProduceAttr: MVCProducesAttribute;
|
2013-10-30 00:48:23 +01:00
|
|
|
|
begin
|
|
|
|
|
FMethodToCall := nil;
|
|
|
|
|
FMVCControllerClass := nil;
|
|
|
|
|
|
|
|
|
|
if trim(AWebRequestPathInfo) = EmptyStr then
|
|
|
|
|
AWebRequestPathInfo := '/'
|
|
|
|
|
else
|
|
|
|
|
begin
|
|
|
|
|
if AWebRequestPathInfo[1] <> '/' then
|
|
|
|
|
AWebRequestPathInfo := '/' + AWebRequestPathInfo;
|
|
|
|
|
end;
|
|
|
|
|
|
|
|
|
|
// daniele
|
|
|
|
|
AWebRequestPathInfo := TIdURI.URLDecode(AWebRequestPathInfo);
|
|
|
|
|
|
2014-02-24 10:20:34 +01:00
|
|
|
|
{ ISAPI CHANGE THE REQUEST PATH INFO START }
|
|
|
|
|
if IsLibrary then
|
|
|
|
|
begin
|
2014-06-27 15:30:39 +02:00
|
|
|
|
AWebRequestPathInfo := string(AWebRequestPathInfo).Remove(0, FMVCConfig.Value[TMVCConfigKey.ISAPIPath].Length);
|
2014-02-24 10:20:34 +01:00
|
|
|
|
if Length(AWebRequestPathInfo) = 0 then
|
|
|
|
|
AWebRequestPathInfo := '/';
|
|
|
|
|
end;
|
|
|
|
|
{ ISAPI CHANGE THE REQUEST PATH INFO END }
|
|
|
|
|
|
2014-04-01 00:02:31 +02:00
|
|
|
|
TMonitor.Enter(Lock); // start of lock
|
|
|
|
|
try
|
2013-10-30 00:48:23 +01:00
|
|
|
|
|
2014-04-01 00:02:31 +02:00
|
|
|
|
Result := False;
|
|
|
|
|
ControllerMappedPath := '';
|
|
|
|
|
for controllerClass in AMVCControllers do
|
|
|
|
|
begin
|
|
|
|
|
SetLength(_attributes, 0);
|
|
|
|
|
_type := FCTX.GetType(controllerClass.ClassInfo);
|
|
|
|
|
_attributes := _type.GetAttributes;
|
|
|
|
|
if not Assigned(_attributes) then
|
|
|
|
|
Continue;
|
2013-10-30 00:48:23 +01:00
|
|
|
|
|
2014-04-01 00:02:31 +02:00
|
|
|
|
for _attribute in _attributes do
|
|
|
|
|
if _attribute is MVCPathAttribute then
|
|
|
|
|
begin
|
|
|
|
|
ControllerMappedPath := MVCPathAttribute(_attribute).Path;
|
|
|
|
|
Break;
|
|
|
|
|
end;
|
2013-10-30 00:48:23 +01:00
|
|
|
|
|
2014-04-01 00:02:31 +02:00
|
|
|
|
if ControllerMappedPath.IsEmpty then
|
|
|
|
|
raise EMVCException.Create('Controller ' + _type.Name +
|
|
|
|
|
' doesn''t have MVCPath attribute');
|
2013-10-30 00:48:23 +01:00
|
|
|
|
|
2014-04-01 00:02:31 +02:00
|
|
|
|
if ControllerMappedPath = '/' then // WE WANT TO AVOID '//' AS MVCPATH
|
|
|
|
|
ControllerMappedPath := '';
|
2013-10-30 00:48:23 +01:00
|
|
|
|
|
2014-04-01 00:02:31 +02:00
|
|
|
|
if (not ControllerMappedPath.IsEmpty) and
|
|
|
|
|
(Pos(ControllerMappedPath, AWebRequestPathInfo) <> 1) then
|
|
|
|
|
Continue;
|
|
|
|
|
|
|
|
|
|
_methods := _type.GetMethods;
|
|
|
|
|
for _method in _methods do
|
2013-10-30 00:48:23 +01:00
|
|
|
|
begin
|
2014-04-01 00:02:31 +02:00
|
|
|
|
_attributes := _method.GetAttributes;
|
|
|
|
|
for i := 0 to Length(_attributes) - 1 do
|
2013-10-30 00:48:23 +01:00
|
|
|
|
begin
|
2014-04-01 00:02:31 +02:00
|
|
|
|
_attribute := _attributes[i];
|
|
|
|
|
if _attribute is MVCPathAttribute then
|
2013-10-30 00:48:23 +01:00
|
|
|
|
begin
|
2014-04-01 00:02:31 +02:00
|
|
|
|
if IsHTTPMethodCompatible(AWebRequestMethodType, _attributes) and
|
|
|
|
|
IsHTTPContentTypeCompatible(AWebRequestMethodType, AWebRequestContentType, _attributes) and
|
|
|
|
|
IsHTTPAcceptCompatible(AWebRequestMethodType, AWebRequestAccept, _attributes) then
|
2013-10-30 00:48:23 +01:00
|
|
|
|
begin
|
2014-04-01 00:02:31 +02:00
|
|
|
|
MethodPathAttribute := MVCPathAttribute(_attribute).Path;
|
|
|
|
|
if IsCompatiblePath(ControllerMappedPath + MethodPathAttribute,
|
|
|
|
|
AWebRequestPathInfo, AMVCRequestParams) then
|
2013-11-08 23:10:25 +01:00
|
|
|
|
begin
|
2014-04-01 00:02:31 +02:00
|
|
|
|
FMethodToCall := _method;
|
|
|
|
|
FMVCControllerClass := controllerClass;
|
|
|
|
|
// getting the default contenttype using MVCProduceAttribute
|
|
|
|
|
MVCProduceAttr := GetAttribute<MVCProducesAttribute>(_attributes);
|
|
|
|
|
if Assigned(MVCProduceAttr) then
|
|
|
|
|
begin
|
|
|
|
|
AResponseContentType := MVCProduceAttr.Value;
|
|
|
|
|
AResponseContentEncoding := MVCProduceAttr.ProduceEncoding;
|
|
|
|
|
end
|
|
|
|
|
else
|
|
|
|
|
begin
|
|
|
|
|
AResponseContentType := ADefaultContentType;
|
2014-04-10 13:56:23 +02:00
|
|
|
|
AResponseContentEncoding := ADefaultContentCharset;
|
2014-04-01 00:02:31 +02:00
|
|
|
|
end;
|
|
|
|
|
Exit(true);
|
|
|
|
|
end; // if is compatible path
|
|
|
|
|
end; // if is compatible method, contenttype and accept
|
|
|
|
|
end; // if attribute is mvcpath
|
|
|
|
|
end; // for each attributes on method
|
|
|
|
|
end; // for each methods
|
|
|
|
|
end; // for each controllers
|
|
|
|
|
finally
|
|
|
|
|
TMonitor.Exit(Lock);
|
2013-10-30 00:48:23 +01:00
|
|
|
|
end;
|
|
|
|
|
end;
|
|
|
|
|
|
|
|
|
|
function TMVCRouter.GetAttribute<T>(AAttributes: TArray<TCustomAttribute>): T;
|
|
|
|
|
var
|
|
|
|
|
a: TCustomAttribute;
|
|
|
|
|
begin
|
|
|
|
|
Result := nil;
|
|
|
|
|
for a in AAttributes do
|
|
|
|
|
if a is T then
|
|
|
|
|
Exit(T(a));
|
|
|
|
|
end;
|
|
|
|
|
|
2014-06-27 15:30:39 +02:00
|
|
|
|
function TMVCRouter.GetFirstMimeType(
|
|
|
|
|
const AContentType: string): string;
|
|
|
|
|
begin
|
|
|
|
|
Result := AContentType;
|
|
|
|
|
while Pos(',', Result) > 0 do
|
|
|
|
|
Result := Copy(Result, 1, Pos(',', Result) - 1);
|
|
|
|
|
while Pos(';', Result) > 0 do
|
|
|
|
|
Result := Copy(Result, 1, Pos(';', Result) - 1); // application/json;charset=UTF-8 {daniele}
|
|
|
|
|
end;
|
|
|
|
|
|
2013-10-30 00:48:23 +01:00
|
|
|
|
function TMVCRouter.IsCompatiblePath(AMVCPath: string; APath: string;
|
|
|
|
|
var AParams: TMVCRequestParamsTable): Boolean;
|
|
|
|
|
function ToPattern(const V: string; Names: TList<string>): string;
|
|
|
|
|
var
|
|
|
|
|
s: string;
|
|
|
|
|
begin
|
|
|
|
|
Result := V;
|
|
|
|
|
for s in Names do
|
|
|
|
|
Result := StringReplace(Result, '($' + s + ')',
|
|
|
|
|
'([ <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD>\.\_\,%\w\d\x2D\x3A]*)', [rfReplaceAll]);
|
|
|
|
|
end;
|
|
|
|
|
|
|
|
|
|
function GetParametersNames(const V: string): TList<string>;
|
|
|
|
|
var
|
|
|
|
|
s: string;
|
|
|
|
|
matches: TMatchCollection;
|
|
|
|
|
match: TMatch;
|
|
|
|
|
i: Integer;
|
|
|
|
|
begin
|
|
|
|
|
Result := TList<string>.Create;
|
|
|
|
|
s := '\(\$([A-Za-z0-9]+)\)';
|
|
|
|
|
matches := TRegEx.matches(V, s, [roIgnoreCase, roCompiled, roSingleLine]);
|
|
|
|
|
for match in matches do
|
|
|
|
|
for i := 0 to match.Groups.Count - 1 do
|
|
|
|
|
begin
|
|
|
|
|
s := match.Groups[i].Value;
|
|
|
|
|
if (Length(s) > 0) and (s[1] <> '(') then
|
|
|
|
|
begin
|
|
|
|
|
Result.Add(s);
|
|
|
|
|
Break;
|
|
|
|
|
end;
|
|
|
|
|
end;
|
|
|
|
|
end;
|
|
|
|
|
|
|
|
|
|
var
|
|
|
|
|
re: TRegEx;
|
|
|
|
|
m: TMatch;
|
|
|
|
|
pattern: string;
|
|
|
|
|
i: Integer;
|
|
|
|
|
Names: TList<string>;
|
|
|
|
|
begin
|
|
|
|
|
Names := GetParametersNames(AMVCPath);
|
|
|
|
|
try
|
|
|
|
|
pattern := ToPattern(AMVCPath, Names);
|
|
|
|
|
if APath = AMVCPath then
|
|
|
|
|
Exit(true)
|
|
|
|
|
else
|
|
|
|
|
begin
|
|
|
|
|
re := TRegEx.Create('^' + pattern + '$', [roIgnoreCase, roCompiled,
|
|
|
|
|
roSingleLine]);
|
|
|
|
|
m := re.match(APath);
|
|
|
|
|
Result := m.Success;
|
|
|
|
|
if Result then
|
|
|
|
|
for i := 1 to pred(m.Groups.Count) do
|
|
|
|
|
AParams.Add(Names[i - 1], TIdURI.URLDecode(m.Groups[i].Value));
|
|
|
|
|
end;
|
|
|
|
|
finally
|
|
|
|
|
Names.Free;
|
|
|
|
|
end;
|
|
|
|
|
end;
|
|
|
|
|
|
2014-04-01 00:02:31 +02:00
|
|
|
|
function TMVCRouter.IsHTTPAcceptCompatible(AWebRequestMethodType: TMVCHTTPMethodType; AAccept: AnsiString;
|
2013-11-09 14:22:11 +01:00
|
|
|
|
AAttributes: TArray<TCustomAttribute>): Boolean;
|
|
|
|
|
var
|
|
|
|
|
i: Integer;
|
2014-04-01 00:02:31 +02:00
|
|
|
|
MethodAccept: string;
|
|
|
|
|
FoundOneAttribProduces: Boolean;
|
|
|
|
|
begin
|
|
|
|
|
Result := False;
|
|
|
|
|
FoundOneAttribProduces := False;
|
|
|
|
|
for i := 0 to high(AAttributes) do
|
|
|
|
|
begin
|
|
|
|
|
if AAttributes[i] is MVCProducesAttribute then
|
|
|
|
|
begin
|
|
|
|
|
FoundOneAttribProduces := true;
|
|
|
|
|
MethodAccept := MVCProducesAttribute(AAttributes[i]).Value;
|
2014-06-27 15:30:39 +02:00
|
|
|
|
AAccept := GetFirstMimeType(AAccept);
|
|
|
|
|
// while Pos(',', AAccept) > 0 do
|
|
|
|
|
// AAccept := Copy(AAccept, 1, Pos(',', AAccept) - 1);
|
|
|
|
|
|
2014-04-01 00:02:31 +02:00
|
|
|
|
Result := SameText(AAccept, MethodAccept, loInvariantLocale);
|
|
|
|
|
if Result then
|
|
|
|
|
Break;
|
|
|
|
|
end;
|
|
|
|
|
end;
|
|
|
|
|
Result := (not FoundOneAttribProduces) or (FoundOneAttribProduces and Result);
|
|
|
|
|
end;
|
|
|
|
|
|
|
|
|
|
function TMVCRouter.IsHTTPContentTypeCompatible(AWebRequestMethodType: TMVCHTTPMethodType;
|
|
|
|
|
AContentType: AnsiString; AAttributes: TArray<TCustomAttribute>): Boolean;
|
|
|
|
|
var
|
|
|
|
|
i: Integer;
|
|
|
|
|
MethodContentType: string;
|
2013-11-09 14:22:11 +01:00
|
|
|
|
FoundOneAttribConsumes: Boolean;
|
|
|
|
|
begin
|
2014-03-24 17:37:08 +01:00
|
|
|
|
// content type is applicable only for PUT, POST and PATCH
|
|
|
|
|
if AWebRequestMethodType in [httpGET, httpDELETE, httpHEAD, httpOPTIONS] then
|
|
|
|
|
Exit(true);
|
|
|
|
|
|
2013-11-09 14:22:11 +01:00
|
|
|
|
Result := False;
|
|
|
|
|
FoundOneAttribConsumes := False;
|
|
|
|
|
for i := 0 to high(AAttributes) do
|
|
|
|
|
begin
|
|
|
|
|
if AAttributes[i] is MVCConsumesAttribute then
|
|
|
|
|
begin
|
|
|
|
|
FoundOneAttribConsumes := true;
|
2014-04-01 00:02:31 +02:00
|
|
|
|
MethodContentType := MVCConsumesAttribute(AAttributes[i]).Value;
|
2014-06-27 15:30:39 +02:00
|
|
|
|
AContentType := GetFirstMimeType(AContentType);
|
2014-04-01 00:02:31 +02:00
|
|
|
|
Result := SameText(AContentType, MethodContentType, loInvariantLocale);
|
|
|
|
|
if Result then
|
|
|
|
|
Break;
|
2013-11-09 14:22:11 +01:00
|
|
|
|
end;
|
|
|
|
|
end;
|
|
|
|
|
Result := (not FoundOneAttribConsumes) or (FoundOneAttribConsumes and Result);
|
|
|
|
|
end;
|
|
|
|
|
|
2013-10-30 00:48:23 +01:00
|
|
|
|
function TMVCRouter.IsHTTPMethodCompatible(AMethodType: TMVCHTTPMethodType;
|
|
|
|
|
AAttributes: TArray<TCustomAttribute>): Boolean;
|
|
|
|
|
var
|
|
|
|
|
i: Integer;
|
|
|
|
|
MustBeCompatible: Boolean;
|
|
|
|
|
CompatibleMethods: TMVCHTTPMethods;
|
|
|
|
|
begin
|
|
|
|
|
Result := False;
|
2013-11-09 14:22:11 +01:00
|
|
|
|
// if there aren't MVCHTTPMethod attributes defined, the action is compatibile with all methods
|
2013-10-30 00:48:23 +01:00
|
|
|
|
MustBeCompatible := False;
|
|
|
|
|
for i := 0 to high(AAttributes) do
|
|
|
|
|
begin
|
|
|
|
|
if AAttributes[i] is MVCHTTPMethodAttribute then
|
|
|
|
|
begin
|
|
|
|
|
MustBeCompatible := true;
|
|
|
|
|
CompatibleMethods := MVCHTTPMethodAttribute(AAttributes[i])
|
|
|
|
|
.MVCHTTPMethods;
|
|
|
|
|
Result := (AMethodType in CompatibleMethods);
|
|
|
|
|
end;
|
|
|
|
|
end;
|
2013-11-09 14:22:11 +01:00
|
|
|
|
Result := (not MustBeCompatible) or (MustBeCompatible and Result);
|
2013-10-30 00:48:23 +01:00
|
|
|
|
end;
|
|
|
|
|
|
2013-11-11 19:32:20 +01:00
|
|
|
|
class
|
|
|
|
|
function TMVCRouter.StringMethodToHTTPMetod(const Value: AnsiString)
|
2013-10-30 00:48:23 +01:00
|
|
|
|
: TMVCHTTPMethodType;
|
|
|
|
|
begin
|
|
|
|
|
if Value = 'GET' then
|
|
|
|
|
Exit(httpGET);
|
|
|
|
|
if Value = 'POST' then
|
|
|
|
|
Exit(httpPOST);
|
|
|
|
|
if Value = 'DELETE' then
|
|
|
|
|
Exit(httpDELETE);
|
|
|
|
|
if Value = 'PUT' then
|
|
|
|
|
Exit(httpPUT);
|
|
|
|
|
if Value = 'HEAD' then
|
|
|
|
|
Exit(httpHEAD);
|
|
|
|
|
if Value = 'OPTIONS' then
|
|
|
|
|
Exit(httpOPTIONS);
|
2014-03-24 17:37:08 +01:00
|
|
|
|
if Value = 'PATCH' then
|
|
|
|
|
Exit(httpPATCH);
|
|
|
|
|
if Value = 'TRACE' then
|
|
|
|
|
Exit(httpTRACE);
|
2013-10-30 00:48:23 +01:00
|
|
|
|
raise EMVCException.CreateFmt('Unknown HTTP method [%s]', [Value]);
|
|
|
|
|
end;
|
|
|
|
|
|
|
|
|
|
end.
|