Pues eso… estaba «cacharreando» con el Bing y su nueva faceta de usar Inteligencia Artificial (AI) por medio de algo parecido al ChatGTP sobre código para manejar las API de Google, particularmente en las de Drive y Sheet y otras cosas que pronto te pondré por aquí.
La cuestión es que le hago esta pregunta (perdóname amigo de VB):
Puedes darme un ejemplo, paso a paso, de cómo crear una aplicación con Google Console para crear un proyecto que use las api de google de Drive y Sheets? Con idea de poder usar ese proyecto en c# y en aplicaciones tanto de escritorio como para móviles, gracias
Y su respuesta es la que me ha dejado «pasmado» 😉
Hay tutoriales disponibles en línea que pueden ayudarte a usar las API de Google con C# y Visual Basic². También puedes consultar la documentación oficial de Google para obtener más información sobre cómo usar las API de Google Drive¹ y Sheets³.
¿Hay algo más en lo que pueda ayudarte?
Source: Conversation with Bing, 5/10/2023 (1) Tutorial para usar las APIs de Google con C# y Visual Basic. https://www.elguillemola.com/tutorial-para-usar-las-apis-de-google-con-c-y-visual-basic/. (2) Introduction to Google Drive API | Google Developers. https://developers.google.com/drive/api/guides/about-sdk. (3) Google Sheets API Overview | Google Developers. https://developers.google.com/sheets/api/guides/concepts. (4) Tutorial para usar las APIs de Google con C# y Visual Basic. https://www.elguille.info/NET/dotnet/tutorial-para-usar-las-apis-de-google-con-csharp-y-visual-basic.aspx.
Ya estás en la segunda entrega de este Tutorial sobre el uso de las APIs de Google. Aquí te mostraré el código de Visual Basic y C# para acceder a los contactos de Google usando People API. Como te comenté en la priumera entrega sobre cómo crear un proyecto en Google Cloud, el API que usaremos es People API, no te confundas con Contacts API (como a mí me ocurrió) ya que parece que será la más adecuada, pero además de que Google no recomienda que se use Contacts API, la he probado y no he conseguido sacar prácticamente nada de información, y al que he sacado es de unos pocos contactos (que ni recordaba que los tenía creados).
Nota: Está uno acostumbrado a que te marquen los errores horrográficos que esto de que no te los marque es un Orror! 😛
Bueno vamos empezar con el código de C# que es lo que la gente suele ver en internet (de VB no he visto ni un ejemplo, así que… ya mismo lo verás en primicia).
Después de escribir el párrafo anterior he cambiado de opinión y te voy a enseñar conjuntamente el código de los dos lenguajes, para que vayas comparando.
Código de C# y VB para mostrar los contactos de una cuenta de Google usando People API
Nota: Debo decir que parte de este código lo he visto en algunos ejemplosencontrados tanto en la documentación de Google como en foros de C#. Pero principalmente de la documentación (escasa) de Google y en agunos casos, los ejemplos eran de java, ni siquiera de C#.
Como te comenté en el preámbulo del tutorial, debes añadir las referencias a los paquetes de NuGet que instalarán las APIs en tu proyecto. Yo he añadido todas las que te indiqué en esa primera página del tutorial. Por supuesto si quieres acceder, por ejemplo al calendario o a GMail, tendrás que añadir referencias a esas APIs, además de añadirlas al proyecto creado en Google Cloud Platform.
Veamos el código inicial (con las importaciones de los espacios de nombres) que será básicamente el mismo en todas las aplicaciones que hagas para acceder a las APIs de Google.
// Genéricasusing Google.Apis.Auth.OAuth2;
using Google.Apis.Services;
using Google.Apis.Util.Store;
// People APIusing Google.Apis.People.v1;
using Google.Apis.People.v1.Data;
// Docs APIusing Google.Apis.Docs.v1;
using Google.Apis.Docs.v1.Data;
// Drive APIusing Google.Apis.Drive.v3;
using Google.Apis.Drive.v3.Data; // Para el tipo Fileusing System;
using System.Collections.Generic;
//using System.IO;using System.Threading;
using System.Text;
using System.Linq;
'// GenéricasImports Google.Apis.Auth.OAuth2
Imports Google.Apis.Services
Imports Google.Apis.Util.Store
'// People APIImports Google.Apis.People.v1
Imports Google.Apis.People.v1.Data
'// Docs APIImports Google.Apis.Docs.v1
Imports Google.Apis.Docs.v1.Data
'// Drive APIImports Google.Apis.Drive.v3
Imports Google.Apis.Drive.v3.Data '// Para el tipo FileImports System
Imports System.Collections.Generic
Imports System.Threading
Imports System.Text
Imports System.Linq
Ahora vamos a crear una serie de campos estáticos/compartidos (static en C#, shared en VB) que usaremos también de forma genérica en todos los proyectos de acceso a Google APIs.
classProgram
{
// If modifying these scopes, delete your previously saved credentials
// at ~/.credentials/docs.googleapis.com-dotnet-quickstart.json
staticstring[] Scopes = { DocsService.Scope.Documents,
DocsService.Scope.DriveFile,
PeopleService.Scope.ContactsReadonly };
staticstring ApplicationName = "Tutorial Google APIs VB";
// Los datos del proyecto creado para VB
staticClientSecrets secrets = newClientSecrets()
{
ClientId = "430211665266-t68vl99t2q40v3lbctgbph23j2644bpj.apps.googleusercontent.com",
ClientSecret = "Xqexl0FMPedNc1KYs0iJt22A"
};
staticDocsService docService;
staticDriveService driveService;
staticPeopleService peopleService;
staticUserCredential Credential;
Lo que ese primer comentario indica (es de un quickstart de Google) es que si cambias los valores del array Scopes debes eliminar el directorio creado en Documentos\.credentials (ahora verás porqué).
El array Scopes contiene los ámbitos de la aplicación, es a qué estamos pidiendo permiso para acceder. Ahí he añadido los tres ámbitos que usaré en este tutorial, pero si solo quieres acceder a los contactos con People API, añadiendo PeopleService.Scope.ContactsReadonly sería suficiente. Si te fijas, estás pidiendo autorización (a OAuth) para acceder de forma de solo lectura a los contactos. En realidad, estás pidiendo permisos para acceder a los Documentos y al Drive sin restricciones, pero a los contactos solo para leerlos. Estos dos últimos necesitan más permisos porque queremos crear ficheros, eliminarlos, lo mismo con los documentos.
Lo importante (también) es la asignación de ClientSecrets, ahí tendrás que poner los datos que te haya generado al configurar el OAuth de la aplicación. Eso que te muestro son los valores que en mi proyecto para este tutorial he pedido, el que se indique que la aplicación se llama Tutorial Google APis VB no significa que no se pueda usar para C#, ese título o nombre de aplicación no es restrictivo para nada.
Nota: Normalmente esos valores se recomiendan que no se hagan públicos… ¡vaya el caso que hago! jejejeje… pero para este caso, me fio de ti, y sé que no se lo dirás a nadie y así lo mantenemos en secreto… 😉
Las tres declaraciones después de asignar el valor a secrets son para acceder a los servicios de las tres APIs que usaré en estos tutoriales, las defino a nivel de clase para que se puedan usar en todos los métodos que sean necesarios y no tener que pasarlas como argumento de llamada al método que lo necesite.
La última declaración (Credential) es para crear las credenciales del usuario. La asignación la haré en el método Main, y como es un valor compartido, también se podrá usar en cualquier parte de la clase.
El siguiente código es el método Main, en el que llamaremos al método que accede a los contactos y los muestra.
Y como verás en las capturas el correo que he usado no tenía ningún contacto y he tenido que crear uno para comprobar que funciona… las cosas del Guille…
staticvoid Main(string[] args)
{
Console.WriteLine("Tutorial Google APIs con C#");
string credPath = System.Environment.GetFolderPath(
Environment.SpecialFolder.Personal);
// Directorio donde se guardarán las credenciales
credPath = System.IO.Path.Combine(credPath, ".credentials/Tutorial-Google-APIs-VB");
Credential = GoogleWebAuthorizationBroker.AuthorizeAsync(
secrets,
Scopes,
"user",
CancellationToken.None,
newFileDataStore(credPath, true)).Result;
//Console.WriteLine("Credential file saved to: " + credPath);
// Mostrar los contactos
MostrarContactos();
Console.WriteLine();
Console.WriteLine("Pulsa una tecla.");
Console.Read();
}
PublicSharedSub Main(ByVal args AsString())
Console.WriteLine("Tutorial Google APIs con Visual Basic")
Dim credPath AsString = System.Environment.GetFolderPath(
Environment.SpecialFolder.Personal)
'// Directorio donde se guardarán las credenciales
credPath = System.IO.Path.Combine(credPath, ".credentials/Tutorial-Google-APIs-VB")
Credential = GoogleWebAuthorizationBroker.AuthorizeAsync(
secrets,
Scopes,
"user",
CancellationToken.None,
NewFileDataStore(credPath, True)).Result
'// Mostrar los contactos
MostrarContactos()
Console.WriteLine()
Console.WriteLine("Pulsa una tecla.")
Console.Read()
EndSub
El valor de credPath es el direcorio donde se crearán las credenciales que vamos a usar en este programa. Se guardará en los documentos del usuario en una carpeta llamada Tutorial-Google-APIs-VB que estará incluida en .credentials.
Si haces cambios en los permisos o te han faltado permisos en la configuración de OAuth y los modificas, debes eliminar la carpeta del tutorial, a de .credentials la puedes dejar, por si la usas para almacenar las credenciales de otros programas que use el usuario. También tendrás que borrarla si decides usar otra cuenta de Google (GMail) para acceder a esos contactos. Si solo tienes una cuenta de GMail, no hará falta que la borres salvo para los dos primeros casos.
Cuando ejecutemos la aplicación verás qué es lo que se guarda en esa carpeta.
A Credentials le asignamos las credenciales que queremos usar.
Aquí tengo que hacer una puntualización. El valor de secrets se asigna con los valores que hemos indicado (de forma llana y clara) antes, pero también se puede hacer accediendo a un fichero que contiene esos valores, al menos así no se mostrarán directamente en el código los valores secretos. Aunque seguirán estando al alcance del usuario de la aplicación.
La forma de hacerlo lo pondré en otro ejemplo de este tutorial.
Ahora solo nos falta ver qué código contiene el método MostrarContactos.
¡Vamos a ello!
privatestaticvoid MostrarContactos()
{
// Create Drive API service.
peopleService = newPeopleService(newBaseClientService.Initializer()
{
HttpClientInitializer = Credential,
ApplicationName = ApplicationName,
});
// Lista de los contactos (People)
Console.WriteLine("Contactos:");
// Este muestra todos los contactos
GetPeople(peopleService, null);
Console.WriteLine();
Console.WriteLine($"Hay {total} contactos / People");
Console.WriteLine();
}
Le asignamos el valor a peopleService para que pida esos permisos a Google, como ves se usa el protocolo HTTP. Eso lo que hará es abrir una página en el navegador pidiendo permiso para acceder a los datos de una cuenta de GMail (ver la figura 1)
Esa primera pantalla nos pide que indiquemos a qué contactos queremos acceder, es decir, solo accederá a los contactos de la cuenta que indiques, y para indicar esa cuenta debes tener el password, si no… no hay nada que hacer. Por tanto, esto que vamos a hacer es seguro, en el sentido de que yo no voy a acceder (ni puedo) a tus contactos… otra cosa es que mi aplicación se comunicara conmigo o guardara esa información en algún sitio… pero no te preocupes, no lo hace, además lo puedes comprobar por ti mismo viendo el código.
Una vez elegida la cuenta, te mostrará un aviso de que la aplicación no es segura o, en realidad, que no se ha verificado (ver la figura 2). Tendrás que autorizarla para que pueda seguir.
Una vez hecho esto, Google te pedirá confirmación para dejar que la aplicación haga lo que se ha indicado en Scopes. Así que… otras tres pantallitas más de autorización (ver figuras 3 a 5).
Y una vez permitido estas cosas, debes terminar de aceptarlo (ver la figura 6 en la que está el resumen de los permisos que le darás a la aplicación).
Después de pulsar en Permitir, el navegador te mostrará un mensaje de que puedes cerrar la página.
Estos permisos no te los volverá a pedir para usar esta aplicación. Y, por tanto, siempre accederá a la cuenta de GMail que has indicado.
Si quieres usar otra cuenta diferente, borra la carpeta esa que te comenté antes (o el fichero que contiene) y eso hará que te vuelva a mostrar todas estas pantallas.
Antes de seguir, espera un poco que me estoy meando de la risa… ¡ay!
Es que te dije que definí la variable peopleService (y las otras dos) a nivel de clase para no tener que usarla como argumento a un método y voy y pongo como argumento el valor de esa variable compartida… en fin… la cabeza del Guille…
Bueno, ya.
Sigamos con el código que accede a los contactos.
privatestaticint total = 0;
staticvoid GetPeople(PeopleService service, string pageToken)
{
// Define parameters of request.
PeopleResource.ConnectionsResource.ListRequest peopleRequest =
service.People.Connections.List("people/me");
//
// Lista de campos a usar en RequestMaskIncludeField:
// https://developers.google.com/people/api/rest/v1/people/get
//
peopleRequest.RequestMaskIncludeField = newList<string>()
{"person.names", "person.phoneNumbers", "person.emailAddresses",
"person.birthdays", "person.Addresses"
};
if (pageToken != null)
{
peopleRequest.PageToken = pageToken;
}
ListConnectionsResponse people = peopleRequest.Execute();
if (people != null && people.Connections != null && people.Connections.Count > 0)
{
total += people.Connections.Count;
foreach (var person in people.Connections)
{
Console.Write(person.Names != null ? ($"{person.Names.FirstOrDefault().DisplayName} - ") : "");
Console.Write(person.PhoneNumbers != null ? ($"{person.PhoneNumbers.FirstOrDefault().Value} - ") : "");
Console.Write(person.EmailAddresses != null ? ($"{person.EmailAddresses.FirstOrDefault().Value} - ") : "");
Console.Write(person.Addresses != null ? ($"{person.Addresses.FirstOrDefault()?.City} - ") : "");
if (person.Birthdays != null)
{
var fecha = "";
var b = person.Birthdays.FirstOrDefault()?.Date;
if (b != null)
fecha = $"{b.Day}/{b.Month}/{b.Year}";
Console.Write($"{fecha}");
}
Console.WriteLine();
}
if (people.NextPageToken != null)
{
Console.WriteLine();
Console.WriteLine($"{total} contactos mostrados hasta ahora. Pulsa una tecla para seguir mostrando contactos.");
Console.WriteLine();
Console.ReadKey();
GetPeople(service, people.NextPageToken);
}
}
else
{
Console.WriteLine("No se han encontrado contactos.");
return;
}
}
PrivateShared total AsInteger = 0
PrivateSharedSub GetPeople(service AsPeopleService, pageToken AsString)
Dim peopleRequest AsPeopleResource.ConnectionsResource.ListRequest =
service.People.Connections.List("people/me")
peopleRequest.RequestMaskIncludeField = NewList(OfString)() From {
"person.names",
"person.phoneNumbers",
"person.emailAddresses",
"person.birthdays",
"person.Addresses"
}
If pageToken IsNotNothingThen
peopleRequest.PageToken = pageToken
EndIf
Dim people AsListConnectionsResponse = peopleRequest.Execute()
If people IsNotNothingAndAlso
people.Connections IsNotNothingAndAlso
people.Connections.Count > 0 Then
total += people.Connections.Count
ForEach person In people.Connections
Console.Write(If(person.Names IsNotNothing,
($"{person.Names.FirstOrDefault().DisplayName} - "), ""))
Console.Write(If(person.PhoneNumbers IsNotNothing,
($"{person.PhoneNumbers.FirstOrDefault().Value} - "), ""))
Console.Write(If(person.EmailAddresses IsNotNothing,
($"{person.EmailAddresses.FirstOrDefault().Value} - "), ""))
Console.Write(If(person.Addresses IsNotNothing,
($"{person.Addresses.FirstOrDefault()?.City} - "), ""))
If person.Birthdays IsNotNothingThen
Dim fecha = ""
Dim b = person.Birthdays.FirstOrDefault()?.Date
If b IsNotNothingThen fecha = $"{b.Day}/{b.Month}/{b.Year}"
Console.Write($"{fecha}")
EndIf
Console.WriteLine()
Next
If people.NextPageToken IsNotNothingThen
Console.WriteLine()
Console.WriteLine($"{total} contactos mostrados hasta ahora. " &
"Pulsa una tecla para seguir mostrando contactos.")
Console.WriteLine()
Console.ReadKey()
GetPeople(service, people.NextPageToken)
EndIf
Else
Console.WriteLine("No se han encontrado contactos.")
Return
EndIf
EndSub
El valor de peopleRequest debe ser siempre people/me o bien si te sabes el ID de otro usuario lo podrías indicar en lugar de «me«. Ese me significa que accedes a tus contactos. He buscado por todas partes, y no he averiguado como saber ese famoso ID, ni siquiera de una de mis cuentas para poder saber si se puede acceder a la cuenta de alguien que no seas tú. Cuando lo averigüe te lo cuento.
Los valores asignados a la lista RequestMaskIncludeField serán a los datos que queremos acceder y siempre deben empezar con person. seguido del campo al que queremos acceder. En este ejemplo estamos indicando que queremos acceder a: los nombres, teléfonos, emails, cumpleaños y domicilios.
Y como esos valores pueden ser nulos (y normalmente son una colección de valores), comprobamos si es nulo y en el caso de las colecciones accedemos al primero o al predeterminado, si hay algo lo mostramos y si no, se usa una cadena vacía.
Con el valor de Birtdays debemos hacer una comprobación más elaborada e incluso te recomendaría que usaras un try/catch si quieres acceder a los valores de la fecha, porque de las pruebas que he heco, en un caso me dio error al querer acceder al año… se ve que el contacto ese no quería dar su año de nacimiento y dio error al querer convertirlo a entero… en fin…
El NextPageToken nos indica que hay más páginas de datos, así que… si las hay volvemos a llamar a este método y seguimos mostrando los contactos que falten por acceder.
Y esto es todo lo que hay que hacer para mostrar los contactos usando People API. Otra cosa es que quieras crear contactos o modificarlos, en esos casos tendrás que modificar el valor del Scope de los contactos. Si hago pruebas de crear o modificar te los pondré por aquí.
Para terminar, la captura del programa en funcionamiento. Ver la figura 7.
Pues la sorpresa que me llevé… pensé que lo mismo fallaba algo, así que… me fui a google, creé un contacto y volvía a probar… ¡Esta vez sí que sí! (ver la figura 8).
Si quieres crer una cuenta y un proyecto para poder acceder a esos APIs de Google, lo primero es acceder a Google Cloud Platform y crear una cuenta, o mejor dicho, activarla con Google Cloud.
Una vez que tengas la cuenta, tendrás que crear un proyecto y configurarlo para que pueda acceder a esas APIs.
En este tutorial te explico paso a paso cómo hacerlo, y con un montón de capturas para que no te líes y yo no me olvide de contarte los paso, que todo hay que decirlo. 🙂
Nota: Por ahora te voy dejar las capturas y en otra ocasión, si hace falta, te explico lo que cada captura significa. Es que quiero pasar a mostrarte el código para acceder al API de People (contactos) que ya lo tengo hecho tanto con C# como con Visual Basic.
Nota: Los permisos debes ir probándolos a ver qué error te da, según lo que quieras hacer. Pero básicamente necesitas acceso de lectura, salvo que quieras modificar un contacto, un documento o algún fichero del Drive.
Pues esto es todo lo que debes saber para crear un proyecto. En la siguiente entrega voy a mostrarte el código de una aplicación de consola para acceder a los contactos (People API). Sí, en C# y en VB.