уважаемые посетители блога, если Вам понравилась, то, пожалуйста, помогите автору с лечением. Подробности тут.
На сегодняшний день интерфейс ClientLogin не рекомендуется использовать в API Google. Эта статья относится к архивным и не отражает современного положения дел. Для авторизации в API Google Вы можете использовать модуль для OAuth, или же, если у Вас Delphi XE5 и выше — библиотеку REST Client Library.

Сегодня наконец-то умудрился загрузить code.google.com, что не может не радовать.  Оказывается у моего провайдера периодически происходят какие-то косяки в оборудовании и, что самое интересное, только Google глючит — остальные адреса работают…Странный глюк, но сегодня не об этом.

Сегодня я расскажу про модуль, реализующий интерфейс ClientLogin. Этот интерфейс используется в Google API для аутентификации пользователей, использующих установленные приложения. Т.е. по сути этот интерфейс — основа для работы с различными API от Google из своих приложений Delphi, например доступ к Календарю Google, Analytics и т.д.

В начале о том, что дает нам использование ClientLogin. Самое главное — это получение маркера Auth, который действует в течение некоторого промежутка времени (в зависимости от используемой службы) и на который будет ссылаться наше будущее приложение, чтобы получить доступ к управлению данными.

Интерфейс использует стандартную систему защиты — Captcha, поэтому надо будет предусмотреть её загрузку и отправку данных  ответа на сервер.

Интерфейс НЕ используется для добавления и удаления служб и сервисов в аккаунт пользователя, т.е. вы не сможете использовать Auth, например, для добавления сервиса Google Analytics в свой аккаунт, но управлять уже подключенным сервисом — сможете.

Теперь о том, как получить этот самый маркер Auth. Для этого необходимо отправить запрос методом POST на адрес https://www.google.com/accounts/ClientLogin и проанализировать ответ сервера. Как видите, нам придётся иметь дело опять с HTTPS.

Параметры запроса следующие:

accountType - тип аккаунта, для которого выполняется аутентификация. Может принимать значения:

GOOGLE (выполнить аутентификацию аккаунта Google)
HOSTED (выполнить аутентификацию размещенного аккаунта)
HOSTED_OR_GOOGLE (выполнить аутентификацию размещенного аккаунта; в случае неудачи выполнить аутентификацию аккаунта Google)

Если мы изначально не знаем с каким типом аккаунта будем иметь дело — то лучше использовать HOSTED_OR_GOOGLE.

Email - адрес электронной почты пользователя.

Passwd - пароль пользователя.

service - имя службы Google, к которой вы запрашиваете доступ. Каждой из служб, использующих службу аутентификации, присвоено имя. Например, с Календарем Google связано имя «cl«, с Google Analytics — «analytics» и т.д. Если изначально неизвестно имя службы, то можно использовать универсальное имя «xapi«.

source - Короткая строка, идентифицирующая ваше приложение и применяемая для ведения журнала. Эта строка должна иметь следующий формат:
«companyName-applicationName-versionID».

logintoken - Маркер, представляющий конкретное значение CAPTCHA. Google присылает этот маркер и URL изображения CAPTCHA в ответе о неудачном входе с кодом ошибки «CaptchaRequired».

logincaptcha - Строка, указанная пользователем в качестве ответа на тест CAPTCHA.

Все перечисленные параметры должны содержаться в запросе. После отправки запроса нам придёт ответ, который может содержать:

  1. В случае успешной аутентификации — код 200 и 3 строки: SID, LSID и Auth. Первые два маркера в настоящее время зарезервированы и не используются.
  2. В случае ошибки нам возвращается код 403 и короткое сообщение (код) об ошибке
  3. В случае необходимости ввода каптчи — код ошибки «CaptchaRequired», URL Captcha и параметр logintoken.

Таким образом, от нас требуется не так уж и много — отправить запрос, проанализировать ответ и, если необходимо, получить каптчу и отправить ответ серверу.

Теперь, что касается модуля Delphi. Модуль GoogleLogin не использует в работе ни Synapse, ни Indy — только WinInet. Отправка данных на сервер осуществляется в следующей функции:

function TGoogleLogin.SendRequest(const ParamStr: string): AnsiString;
  function DataAvailable(hRequest: pointer; out Size : cardinal): boolean;
  begin
    result := wininet.InternetQueryDataAvailable(hRequest, Size, 0, 0);
  end;
var hInternet,hConnect,hRequest : Pointer;
    dwBytesRead,I,L : Cardinal;
begin
try
  hInternet := InternetOpen(PChar('GoogleLogin'),INTERNET_OPEN_TYPE_PRECONFIG,Nil,Nil,0);
  if Assigned(hInternet) then
    begin
     //Открываем сессию
     hConnect := InternetConnect(hInternet,PChar('www.google.com'),Flags_connection,nil,nil,INTERNET_SERVICE_HTTP,0,1);
     if Assigned(hConnect) then
       begin
        //Формируем запрос
        hRequest := HttpOpenRequest(hConnect,PChar(uppercase('post')),PChar('accounts/ClientLogin?'+ParamStr),HTTP_VERSION,nil,Nil,Flags_Request,1);
        if Assigned(hRequest) then
          begin
           //Отправляем запрос
           I := 1;
           if HttpSendRequest(hRequest,nil,0,nil,0) then
             begin
               repeat
                 DataAvailable(hRequest, L);//Получаем кол-во принимаемых данных
                 if L = 0 then break;
                 SetLength(Result,L + I);
                 if InternetReadFile(hRequest,@Result[I],sizeof(L),dwBytesRead) then//Получаем данные с сервера
                 else break;
                 inc(I,dwBytesRead);
               until dwBytesRead = 0;
               Result[I] := #0;
              end;
         end;
      InternetCloseHandle(hRequest);
     end;
   InternetCloseHandle(hConnect);
  end;
 InternetCloseHandle(hInternet);
except
  InternetCloseHandle(hRequest);
  InternetCloseHandle(hConnect);
  InternetCloseHandle(hInternet);
end;
end;

На выходе имеем строку, содержащую ответ серверу. Привожу код именно этой функции, т.к. она Вам может пригодиться не только для работы с Google, но и, например, для авторизации ВКонтакте — главное передать верные параметры на вход.

В модуле определен следующий класс:

type
TGoogleLogin = class
private
//регистрационные данные
  FAccountType  : TAccountType;
  FLastResult   : TLoginResult;
  FEmail        : string;
  FPassword     : string;
//данные ответа/запроса
  FSID          : string;//в настоящее время не используется
  FLSID         : string;//в настоящее время не используется
  FAuth         : string;
  FService      : string;//сервис к которому необходимо получить доступ
  FSource       : string;//имя вызывающего приложения
  FLogintoken   : string;
  FLogincaptcha : string;
//параметры Captcha
  FCaptchaURL    : string;
  function SendRequest(const ParamStr: string):AnsiString;
  function ExpertLoginResult(const LoginResult:string):TLoginResult;
  function GetLoginError(const str: string):TLoginResult;
  function GetCaptchaToken(const cList:TStringList):String;
  function GetCaptchaURL(const cList:TStringList):string;
  function GetResultText:string;
  procedure SetEmail(cEmail:string);
  procedure SetPassword(cPassword:string);
  procedure SetService(cService:string);
  procedure SetSource(cSource: string);
  procedure SetCaptcha(cCaptcha:string);
public
  constructor Create(const aEmail, aPassword: string);
  function Login(aLoginToken:string='';aLoginCaptcha:string=''):TLoginResult;
  procedure CloseConect;//удаляет все данные по авторизации
  property AccountType: TAccountType read FAccountType write FAccountType;
  property LastResult: TLoginResult read FLastResult;
  property LastResultText:string read GetResultText;
  property Email: string read FEmail write SetEmail;
  property Password:string read FPassword write SetPassword;
  property Service: string read FService write SetService;
  property Source: string read FSource write FSource;
  property Auth: string read FAuth;
  property SID: string read FSID;
  property LSID: string read FLSID;
  property CaptchaURL: string read FCaptchaURL;
  property LoginToken: string read FLogintoken;
  property LoginCaptcha: string read FLogincaptcha write FLogincaptcha;
end;

При работе с классом следует учитывать, что изменение в процессе работы свойств Email, Password и Source ведет к обнулению всех данных об аутентификации. То есть, если вы меняете одно из этих свойств или все сразу, то после этого надо обязательно выполнить метод Login для получения нового значения Auth.

Работа с классом элементарна, поэтому не вижу причин расписывать каждое поле, свойство и метод в отдельности, а покажу простой пример его использования. Например, можно авторизоваться так:

[...]
var Account: TGoogleLogin;
begin
  Account:=TGoogleLogin.Create(Edit1.Text,Edit2.Text);
  Account.Login();
  ShowMessage(Account.LastResultText);
[...]

Метод LastResultText вернет вам сообщение об успешной аутентификации, либо текст, содержащий описание ошибки. А параметры запроса будут следующими:

service=xapi

source=Noname-MyCompany-1.0

accountType=HOSTED_OR_GOOGLE

Можно изменить любые из свойств и перелогиниться. А если Google попросит ввести каптчу, то узнать об этом можно, например так:

if Account.LastResult=lrCaptchaRequired then
  ShowMessage('URL для загрузки Captcha '+Account.CaptchaURL);

и после ввода пользователем ответа, отправить новые запрос на аутентификацию таким образом:

Account.LoginCaptcha:=тут ответ пользователя;
Account.Login(Account.LoginToken,Account.LoginCaptcha);

В этом модуле я не стал реализовывать загрузку каптчи т.к. по сути это действие должно выполняться в готовом приложении — надо будет не только загрузить, но и где-то показать картинку с кодом. Хотя, если потребуется в будущем, можно будет реализовать загрузку в простой TBitmap и потом из него уже выводить куда угодно. Но это пока в планах.

0 0 голоса
Рейтинг статьи
уважаемые посетители блога, если Вам понравилась, то, пожалуйста, помогите автору с лечением. Подробности тут.
Подписаться
Уведомить о
21 Комментарий
Межтекстовые Отзывы
Посмотреть все комментарии
Алексей Тимохин

Ай, молодец! =)))
Отличное начало!

Andy
04/02/2010 10:34

Автор, ждем описания работы с Google Data API SDK. Для Delphi это пока никто не реализовал.

Виталий
Виталий
19/03/2010 13:59

Ув. автор. Ссылка не рабочая. Пере залейте пожалуйста куда нибудь. Или отправте мне на E- Mail. Буду очень признателен.

Filmix
Filmix
29/08/2010 20:57

Как раз то что надо, для работы с gmail!

Вася
Вася
03/01/2011 21:36

Подключил пример в C++ Builder 6 при попытке логинения выдаётся: ‘Неизвестная ошибка’, что-то видимо не подцепилось в Buildere. Кто-нить знает что?

Семёныч
Семёныч
25/08/2011 14:16

В примере, после создания TGoogleLogin.Create зачем-то сбрасываются следующие значения:
  Account.Email := ’email’;
  Account.Password := ‘password’;
  Account.Service := ‘service’;
Если эти строки закомментировать, то всё ОК. Спасибо!

Жека
Жека
05/04/2012 19:13

Погодите плз, пытаюсь залогиниться через idHttp выдает
Project Project2.exe raised exception class EIdHTTPProtocolException with message ‘HTTP/1.0 403 Forbidden’.

не подскажите чего делать? 
 
 

Жека
Жека
06/04/2012 11:19

ну если делать по примерэтому исходнику через winhttp то с темеже параметрами проходит..я делаю так
 url:= ‘https://www.google.com/youtube/accounts/ClientLogin’;  // так же пробовал линк без youtube
 str:=tstringlist.Create;
 str.add(‘Email=’+YOUTUBE_EMAIL+’&’);
 str.add(‘Passwd=’+YOUTUBE_PASS+’&’);
 str.add(‘service=youtube&’);
 str.add(‘accountType=HOSTED_OR_GOOGLE’);
 str.Add(‘source=test’);
 req:=IdHTTP1.Put(url,str); 

Жека
Жека
06/04/2012 11:21

забыл добавить .. в исходнике нет передачи ключа.(или я не заметил?) но он работает, правда почему-то выдает просто что не подходит логин или пароль (хотя странно) потому через кучу раз говорит что нада капчу вводить

Жека
Жека
06/04/2012 11:26

да чтож такое,нельзя редактировать свой пост, там у меня в конце не put а post

Жека
Жека
06/04/2012 12:14

кароче вопрос решен.. я просто тупо описался в емыле

Жека
Жека
06/04/2012 14:11

может кто нибудь делал загрузку видео на youtube ? немного не врублюсь.. при всяких разных вариантах пишут код 401 Unauthorized

Жека
Жека
06/04/2012 14:13

   str.add(‘Authorization: AuthSub token=»‘+fauth+'»&’); // тот что мне в ответе прислали
   str.add(‘GData-Version: 2&’);
   str.add(‘X-GData-Client: 26r8LaDLsfwQUqE2iN5D_-&’);
   str.add(‘X-GData-Key: key=’+developer_key+’&’);
   str.add(‘GSlug: c:\vidos.avi&’);

Жека
Жека
06/04/2012 16:00

вообще я так и не понял как отправлять файл, всмысле в мануле пишут что нада еще передавать xml с описание видеофайла.. как передавать этот хмл… 
в общем получилось получить список файлов  и это уже радует

Олег
Олег
21/09/2012 04:03

А ссылка на файл не рабочая (ни одна)… :(