-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathROPCFlow.pas
127 lines (117 loc) · 4.92 KB
/
ROPCFlow.pas
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
unit ROPCFlow;
interface
uses System.Classes, System.SysUtils, System.JSON, System.Threading, System.Net.URLClient, Winapi.ShellAPI, IdHTTP, IdSSLOpenSSL;
type
TOnErrorAccessToken = reference to procedure(Error, ErrorDescription: string);
TOnAfterAccessToken = reference to procedure(Access_Token, Token_Type: string; Expires_In: Integer; Scope: string);
TROPC_Flow = class
const
ROPCURL = 'https://login.microsoftonline.com/%s/oauth2/v2.0/token'; // ROPC Access Token URL
CLIENTIDSTRING = 'client_id=%s'; // ROPC Access Token post data -> client id
CLIENTSECRETSTRING = 'client_secret=%s'; // ROPC Access Token post data -> client secret
SCOPESTRING = 'scope=%s'; // ROPC Access Token -> scope
USERNAMESTRING = 'username=%s'; // Device Code Token post data -> username
PAWWORDSTRING = 'password=%s'; // Device Code Token post data -> password
GRANTTYPESTRING = 'grant_type=password'; // Device Code Token post data -> grant type
strict private
FTenantID: string;
FScope: string;
FClientID: string;
FClientSecret: string;
FPassword: string;
FUsername: string;
FVerification_URI: string;
FExpire_In: Integer;
FInterval: Integer;
IdHTTP_ROPC: TIdHTTP;
LHandler: TIdSSLIOHandlerSocketOpenSSL;
FOnAfterAccessToken: TOnAfterAccessToken;
FOnErrorAccessToken: TOnErrorAccessToken;
public
constructor Create;
destructor Destroy; override;
procedure Start;
property TenantID: string read FTenantID write FTenantID;
property ClientID: string read FClientID write FClientID;
property ClientSecret: string read FClientSecret write FClientSecret;
property Scope: string read FScope write FScope;
property Username: string read FUsername write FUsername;
property Password: string read FPassword write FPassword;
property OnAfterAccessToken: TOnAfterAccessToken read FOnAfterAccessToken write FOnAfterAccessToken;
property OnErrorAccessToken: TOnErrorAccessToken read FOnErrorAccessToken write FOnErrorAccessToken;
end;
implementation
{ TROPC_Flow }
constructor TROPC_Flow.Create;
begin
FClientID := '';
FClientSecret := '';
FTenantID := '';
FScope := '';
FUsername := '';
FPassword := '';
LHandler := TIdSSLIOHandlerSocketOpenSSL.Create(nil);
LHandler.SSLOptions.SSLVersions := [sslvTLSv1_2];
LHandler.SSLOptions.Mode := sslmClient;
LHandler.SSLOptions.VerifyMode := [];
LHandler.SSLOptions.VerifyDepth := 0;
IdHTTP_ROPC := TIdHTTP.Create(nil);
IdHTTP_ROPC.IOHandler := LHandler;
IdHTTP_ROPC.Request.ContentEncoding := 'UTF-8';
IdHTTP_ROPC.Request.ContentType := 'application/x-www-form-urlencoded';
end;
destructor TROPC_Flow.Destroy;
begin
if Assigned(LHandler) then FreeAndNil(LHandler);
if Assigned(IdHTTP_ROPC) then FreeAndNil(IdHTTP_ROPC);
inherited;
end;
procedure TROPC_Flow.Start;
var
postData: TStrings;
FResponseString: string;
FResponseJSON: TJSONObject;
FErrResponseJSON: TJSONObject;
begin
if (FClientID <> '') and
(FClientSecret <> '') and
(FTenantID <> '') and
(FScope <> '') and
(FUsername <> '') and
(FPassword <> '') then begin
try
try
// Post Data
postData := TStringList.Create;
postData.Add(Format(CLIENTIDSTRING, [FClientID]));
postData.Add(Format(CLIENTSECRETSTRING, [FClientSecret]));
postData.Add(Format(SCOPESTRING, [FScope]));
postData.Add(Format(USERNAMESTRING, [FUsername]));
postData.Add(Format(PAWWORDSTRING, [FPassword]));
postData.Add(GRANTTYPESTRING);
// Call Device Auth API
FResponseString := IdHTTP_ROPC.Post(Format(ROPCURL, [FTenantID]), postData);
// Response JSON
FResponseJSON := TJSONObject.ParseJSONValue(FResponseString) as TJSONObject;
// Callback Auth Code
if Assigned(FOnAfterAccessToken) then FOnAfterAccessToken(FResponseJSON.GetValue('access_token').AsType<string>,
FResponseJSON.GetValue('token_type').AsType<string>,
FResponseJSON.GetValue('expires_in').AsType<Integer>,
FResponseJSON.GetValue('scope').AsType<string>);
except
on E: EIdHTTPProtocolException do begin
// Http Error
FErrResponseJSON := TJSONObject.ParseJSONValue(E.ErrorMessage) as TJSONObject;
if Assigned(OnErrorAccessToken) then OnErrorAccessToken(FResponseJSON.GetValue('error').AsType<string>, FResponseJSON.GetValue('error_description').AsType<string>);
if Assigned(FErrResponseJSON) then FreeAndNil(FErrResponseJSON);
end;
end;
finally
if Assigned(postData) then FreeAndNil(postData);
if Assigned(FResponseJSON) then FreeAndNil(FResponseJSON);
end;
end else begin
raise Exception.Create('Not set Client ID or Client Secret or Tenant ID or Scope or Username or Password');
end;
end;
end.