Consumindo APIs de descanso com (nuvem) ABAP

Chamada de método de postagem

Ao trabalhar com APIs REST, o SAP usa API REST para comunicação entre aplicativos. Neste artigo, exploraremos como consumir APIs REST usando ABAP, especificamente com a funcionalidade da Cloud ABAP. Aqui estão os principais pontos que você precisa saber:

  1. API significa interface de programação de aplicativos e permite que dois aplicativos se comuniquem entre si.
  2. As APIs REST são um padrão de APIs de construção usando o protocolo HTTP e envia/recebimento de dados JSON ou XML através de URIs.
  3. Odata, popular no mundo SAP, é um tipo de API REST.
  4. Há informações limitadas disponíveis sobre o consumo de APIs externas no ABAP, especialmente as APIs de Whitelisted para Cloud ABAP.
  5. Neste tutorial, usaremos a API de espaço reservado JSON para fins de demonstração.
  6. O provedor da API oferece recursos como postagens, comentários, álbuns, fotos, Todos e usuários.
  7. Vamos nos concentrar no recurso de postagens para a simplicidade.
  8. Uma postagem tem um ID de ID, título, corpo e usuário.
  9. Para consumir APIs REST em ABAP, usaremos a API ABAP da lista de permissões chamado se_web_http_client.
  10. Também usaremos a biblioteca XCO para trabalhar com JSON.

Questões:

  1. O que significa API?
    API significa interface de programação de aplicativos.
  2. O que são APIs de descanso?
    As APIs REST são um padrão de APIs de construção usando o protocolo HTTP e envia/recebimento de dados JSON ou XML através de URIs.
  3. O que é Odata?
    Odata é um tipo de API REST que é popular no mundo da seiva.
  4. Há muitas informações disponíveis sobre o consumo de APIs externas em ABAP?
    Não, há informações limitadas disponíveis, especialmente para APIs de Whitelisted para Cloud ABAP.
  5. Qual provedor de API estaremos usando neste tutorial?
    Usaremos a API de espaço reservado JSON para fins de demonstração.
  6. Que recursos a API de espaço reservado JSON oferece?
    A API de espaço reservado JSON oferece recursos como postagens, comentários, álbuns, fotos, Todos e usuários.
  7. Em qual recurso estaremos focando neste tutorial?
    Estaremos focados no recurso de postagens para simplicidade.
  8. Que atributos um post tem?
    Uma postagem tem um ID de ID, título, corpo e usuário.
  9. Que API de ABAP de Whitelisted estaremos usando para consumir APIs REST?
    Usaremos a API IF_WEB_HTTP_CLIENT.
  10. Qual biblioteca estaremos usando para trabalhar com JSON?
    Estaremos usando a biblioteca XCO.

Respostas detalhadas:

  1. O que significa API?
    API significa interface de programação de aplicativos. É um conjunto de padrões que permite que dois aplicativos se comuniquem entre si.
  2. O que são APIs de descanso?
    As APIs REST são um padrão de APIs de construção usando o protocolo HTTP. Eles permitem que os aplicativos enviem e recebam dados JSON ou XML através do URIS. As APIs de REST baseadas em JSON são amplamente utilizadas.
  3. O que é Odata?
    Odata é um tipo de API REST que é muito popular no mundo da seiva. Permite fácil acesso e manipulação de dados armazenados em sistemas SAP.
  4. Há muitas informações disponíveis sobre o consumo de APIs externas em ABAP?
    Não, há informações limitadas disponíveis sobre o consumo de APIs externas em ABAP, especialmente para APIs em permissões que podem ser usadas com a nuvem ABAP. Este tutorial tem como objetivo fornecer orientações sobre o consumo de APIs de descanso usando a nuvem ABAP.
  5. Qual provedor de API estaremos usando neste tutorial?
    Usaremos a API de espaço reservado JSON, que é uma API de repouso online falsa gratuita para testes e prototipagem. Ele nos permite realizar ações CRUD (Criar, ler, atualizar, excluir).
  6. Que recursos a API de espaço reservado JSON oferece?
    A API de espaço reservado JSON oferece recursos como postagens, comentários, álbuns, fotos, Todos e usuários. Neste tutorial, vamos nos concentrar no recurso de postagens.
  7. Em qual recurso estaremos focando neste tutorial?
    Estaremos focados no recurso de postagens da API de espaço reservado JSON. Isso nos permitirá demonstrar como executar ações CRUD em uma API REST usando APIs ABAP de Whitelisted na plataforma SAP Cloud.
  8. Que atributos um post tem?
    Uma postagem tem um ID de ID, título, corpo e usuário. O ID representa o identificador exclusivo da postagem, e o ID do usuário representa o ID do usuário que criou a postagem.
  9. Que API de ABAP de Whitelisted estaremos usando para consumir APIs REST?
    Para consumir APIs REST em ABAP, usaremos a API IF_WEB_HTTP_CLIENT. Esta é uma API ABAP de permissões em WHITELEST que pode ser usada na plataforma SAP Cloud.
  10. Qual biblioteca estaremos usando para trabalhar com JSON?
    Para trabalhar com o JSON, usaremos a edição da plataforma em nuvem da biblioteca XCO (componentes de extensão). Esta biblioteca fornece funcionalidade útil para transformar dados JSON entre diferentes convenções de nomenclatura, como CamelCase para SUD_SCORE e vice -versa.

Seguindo este tutorial, você poderá consumir APIs REST usando ABAP, especificamente com a funcionalidade da Cloud ABAP.

Consumindo APIs de descanso com (nuvem) ABAP

Publicar Chamada ->

SAP usa API REST?

О эээ сйранibus

Ы з ззарегиgléria. С помощью этой страницы мы сможем определить, что запросы отправляете именно вы, а не робот. Почpels эээ моогitu произойth?

Эта страница отображается в тех случаях, когда автоматическими системами Google регистрируются исходящие из вашей сети запросы, которые нарушают Условия использования. Ponto. Ээth момо номттаая и оозз илэз и ээ и эз и эз и з и ззз и зз и ээз и ээз иth ээ эth ээзз эth эзз иthлз ио и зз и иth эз иээ эээо иth эз эээ ээо ээоо иth иэзз эth эзт эth эз ио эээ иth эз иэз иthлзз иоз ил иээ иээо иэээ иээо иth ио иээ эth иэ иээ эth иэ иээ эth ио иэ ээог seguir.

Ит и и и и и и и и и чззжfia м ирржжжfia м иржжжжfia м мжжжжжж<ь м м иржжжfia. não. Если вы используете общий доступ в Интернет, проблема может быть с компьютером с таким же IP-адресом, как у вас. Орратитеitivamente к с о и и с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с с а с с а с а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а а &rdquo;. ПодробнÉ.

Проверка по слову может также появляться, если вы вводите сложные запросы, обычно распространяемые автоматизированными системами, или же вводите запросы очень часто.

Consumindo APIs de descanso com (nuvem) ABAP

API significa interface de programação de aplicativos e compreende um conjunto de padrões que permitem que dois aplicativos conversem entre si. As APIs de repouso são um certo padrão de APIs de construção. Eles são baseados no protocolo HTTP, enviando e recebendo dados JSON ou XML através de URIs (Uniform Resource Identifier). As APIs de REST baseadas em JSON são predominantes. Também estaremos usando esse neste tutorial.

Odata, que é muito popular no mundo da seiva, é uma API REST. Há muitas informações por aí sobre como fornecer uma API REST da ABAP (i.e., Para publicar um serviço ODATA). Mas não há&rsquo;T muito sobre como consumir uma API externa em ABAP. E pelo pouco que existe, inclui ABAP ABAP não brançada, eu.e., Eles não podem ser usados ​​com a nuvem ABAP. Então, decidi escrever este tutorial sobre consumir APIs de descanso usando a nuvem ABAP.

Cenário

Provedor de API

Estaremos trabalhando com o espaço reservado JSON – um &ldquo;Livre para usar a API de repouso online falsa para teste e prototipagem&rdquo;. Isso nos permitirá executar todas as ações CRUD (Criar, ler, atualizar, excluir). Para ser justo, criar, atualizar e excluir não funcionará, mas o servidor vai fingir como se fizesse. O que é completamente o suficiente para o nosso caso de uso!

Recursos

Nosso provedor de API expõe Postagens, Comentários, álbuns, fotos, Todos, e Usuários. Pela simplicidade&rsquo;São, nós estaremos usando apenas o Postagens recurso, e finja o resto não&rsquo;t lá. A idéia principal do meu tutorial é fornecer um guia extremamente simples sobre como executar as ações CRUD em uma API REST. E faça isso usando APIs ABAP de Whitelisted na plataforma SAP Cloud (CP). Isso significa que você pode executar este código em uma conta de teste SAP CP.

Postagens recurso

Uma postagem tem um ID, título, corpo e ID do usuário, o que significa o ID do usuário que criou a postagem. Nós o representamos em ABAP da seguinte maneira:

TIPOS: Começando de post_s, user_id Tipo I, ID Tipo I, Tipo de título String, String do tipo de corpo, extremidade de post_s, post_tt Tipo Tabela de post_s com tecla vazia, começo de post_without_id_s, user_id Tipo I, string de tipo de título, string do tipo de corpo, extremidade de post_without_id_s. 

Precisamos da estrutura sem identificação porque o ID do post é atribuído automaticamente pela API REST. O que significa que não o fornecemos ao criar um novo post.

APIs de nuvem ABAP usadas

Enviando solicitações HTTP

Como mencionei anteriormente, o pequeno número de tutoriais existentes para consumir APIs de repouso em ABAP usam principalmente ABAP ABAP não branco. Por exemplo, o if_http_client um, cujo uso não é permitido na nuvem ABAP. A maneira de verificar as APIs ABAP da lista de permissões da plataforma SAP Cloud é navegar Objetos liberados listas. É acessível em Eclipse ABAP Development Tools (ADT) -> Project Explorer -> Objetos liberados. Então, a API ABAP pronta para a nuvem para enviar solicitação http é o if_web_http_client. Definimos o seguinte método para obter um cliente:

Métodos: Create_client Importação URL Tipo String 

Método create_client. Dados (dest) = cl_http_destination_provider => create_by_url (url). resultado = cl_web_http_client_manager => create_by_http_destination (dest). EndMethod. 

Observe que o URL é um parâmetro de entrada. O resultado retornado é o cliente HTTP da web criado.

Trabalhando com JSON

Para trabalhar com o JSON, usaremos a edição da plataforma em nuvem do XCO (Componentes de extensão) biblioteca. Leia mais sobre isso aqui e aqui. A classe específica, relevante para o nosso caso de uso é xco_cp_json. Algo extremamente valioso que oferece é a capacidade de transformar diferentes convenções de nomenclatura. Por exemplo, CamelCase to Sound_Score e o contrário.

Consumindo a API REST

Antes de chegar à parte divertida, apenas temos que definir algumas constantes. Obviamente, isso não é estritamente necessário, mas trabalhar com constantes em oposição aos literais de cordas é uma prática melhor e permite a reutilização.

Constantes: Base_url Tipo String Valor 'https: // jsonplaceholder.Typicode.com/postagens ', content_type Tipo String Value' content-type ', JSON_COntent Type String Value' Application/JSON; charset = utf-8 '. 

O URL base é simplesmente o endereço do Postagens recurso. As duas últimas constantes que precisamos para os casos em que enviaremos dados (i.e., criar e atualizar) para o servidor usando a API REST. Temos que informar o servidor que estamos enviando JSON.

Leia todas as postagens

O URL para ler todas as postagens é apenas o URL base. Então, criamos um cliente para ele, usamos o cliente para executar uma solicitação GET, fechar o cliente e converter o JSON recebido em uma tabela de postagens. A tabela de postagens é definida no Recurso de postagens seção acima. Você também pode se referir ao código completo no final.

Read_Posts Retorno Valor (resultado) Tipo post_tt Raising cx_static_check 

Método read_posts. "Obtenha JSON de todos os dados de postagens (URL) = |< base_url >|. Dados (cliente) = create_client (url). Dados (resposta) = client-> execute (if_web_http_client => get)-> get_text (). Cliente-> Close (). "Converta JSON para postar Tabela XCO_CP_JSON => Data-> From_string (Response)-> Aplicar (valor #((xco_cp_json => transformação-> camel_case_to_underscore))-> write_to (ref #(resultado)). EndMethod. 

Leia uma única postagem

O método para ler uma única postagem é semelhante, com as diferenças que tomamos como entrada de um ID do post e devolve uma estrutura (i.e., um único post,) em vez de uma mesa (eu.e., uma lista de postagens). O resto API&rsquo;S URL de ler um único post é o seguinte:

read_single_post Importar ID Tipo I Valor de retorno (resultado) Tipo post_s Raising cx_static_check 

Método read_single_post. "Obtenha JSON para dados de ID de entrada de entrada (URL) = |< base_url >/< id >|. Dados (cliente) = create_client (url). Dados (resposta) = client-> execute (if_web_http_client => get)-> get_text (). Cliente-> Close (). "Converta JSON para a estrutura de postagem XCO_CP_JSON => DATA-> From_string (Response)-> Aplicar (valor #((xco_cp_json => transformação-> camel_case_to_underscore))-> write_to (ref #(resultado)). EndMethod. 

Criar postagem

Como explicado anteriormente, postagens&rsquo; Os IDs são atribuídos automaticamente pela API REST. Então, para criar um post, usaremos o post_without_id_s tipo. Este será o nosso parâmetro de entrada. Vamos converter desta estrutura ABAP para JSON, mais uma vez usando a Biblioteca XCO. A partir daí, criamos um cliente. Em seguida, definimos o corpo da solicitação HTTP que enviaremos para ser o JSON que acabamos de criar e informamos ao servidor que enviaremos o tipo de conteúdo JSON. Por fim, executamos uma solicitação de postagem e retornamos o servidor&rsquo;S resposta. Se tudo fosse bom, o servidor&rsquo;A resposta S retornaria nossa postagem, juntamente com seu ID recém -gerado (101, porque atualmente existem 100 postagens).

create_post Importando post_without_id Tipo post_without_id_s Valor de retorno (resultado) Tipo String Raising cx_static_check 

Método create_post. "Converta a postagem de entrada em dados JSON (json_post) = xco_cp_json => data-> de_abap (post_without_id)-> APLIC. "Envie o JSON da postagem para o servidor e retorne os dados de resposta (URL) = |< base_url >|. Dados (cliente) = create_client (url). Dados (req) = client-> get_http_request (). req-> set_text (json_post). req-> set_header_field (i_name = content_type i_value = json_content). resultado = client-> execute (if_web_http_client => post)-> get_text (). Cliente-> Close (). EndMethod. 

Postagem de atualização

Estaremos atualizando com um pedido de put. Isso significa que vamos fornecer a postagem completa. Patch, por outro lado, nos permite fornecer apenas o campo atualizado (e.g., único título). Se você achar isso interessante, pode tentar fazer o patch solicitar – ele não deve&rsquo;Não serei muito difícil com os recursos fornecidos aqui!

Seguimos uma lógica semelhante ao do criar Ação. Também fornecemos uma postagem como um parâmetro de entrada, mas desta vez usamos a estrutura completa (com pós -identificação). O URL para atualizar uma postagem é o mesmo que acessar esta postagem (única):

Então, as únicas diferenças de criar Inclua o tipo alterado do parâmetro de entrada de postagem, o URL e o método de solicitação HTTP (put).

Update_post Importando Post Type Post_s Valor de retorno (resultado) Tipo String Raising cx_static_check

Método update_post. "Converta a postagem de entrada em dados JSON (json_post) = xco_cp_json => data-> de_abap (post)-> APLIC. "Envie o JSON da postagem para o servidor e retorne os dados de resposta (URL) = |< base_url >/< post-id >|. Dados (cliente) = create_client (url). Dados (req) = client-> get_http_request (). req-> set_text (json_post). req-> set_header_field (i_name = content_type i_value = json_content). resultado = client-> execute (if_web_http_client => put)-> get_text (). Cliente-> Close (). EndMethod. 

Apague a postagem

Excluir uma postagem é o pedido mais simples. Simplesmente pegamos o ID e enviamos uma solicitação HTTP de exclusão para o URL da postagem específica. Para deixar o usuário se algo der errado, verificamos o servidor&rsquo;s Código de resposta (deve ser 200 – significando ok).

Delete_Post Importação ID TIPO I RAISANDO CX_STATIC_CHECK 

Método delete_post. Dados (url) = |< base_url >/< id >|. Dados (cliente) = create_client (url). Dados (resposta) = cliente-> execute (if_web_http_client => delete). Se resposta-> get_status () -code ne 200. Aumentar o tipo de exceção cx_web_http_client_error. FIM SE. EndMethod.

Testando nosso código

Agora que fornecemos todas as funcionalidades do CRUD, vamos&rsquo;está confira! Para fazer isso, estaremos implementando o if_oo_adt_classrun interface, que permite executar uma classe como um aplicativo de console. Tem um método principal que é executado – semelhante ao Java.

Método if_oo_adt_classrun ~ main. TENTAR. "Leia os dados (all_Posts) = read_posts (). Dados (primeiro_post) = read_single_post (1). "Criar dados (create_restons) = create_post (valor #(user_id = 7 title = 'hello, mundo!'corpo =' :) '))). "Atualize First_Post-user_id = 777. Data (Update_Response) = UPDATE_POST (FIRST_POST). "Delete Delete_Post (9). "Resultados de impressão Out-> Write (All_Posts). Out-> Write (First_Post). fora. fora. Pegue cx_root em dados (exc). fora-> write (exc-> get_text ()). Final. EndMethod.

Executando com F9 imprime a seguinte saída:

Início%20of%20The%20Output%20in%20The%20ABAP%20Console

Início da saída no console ABAP

Final%20of%20The%20Output%20in%20The%20ABAP%20Console

Fim da saída no console ABAP

Conclusão

Isso termina o tutorial de como consumir APIs REST na nuvem ABAP. Espero que tenha sido útil para você. Se você se sentir aí&rsquo;é qualquer ponto de melhoria ou você tem alguma dúvida ou feedback para mim, deixe -me saber nos comentários!

Código completo

Classe ZSS_TESTER_2 Definição Public final Create Public. Seção pública. Interfaces: if_oo_adt_classrun. TIPOS: Começando de post_s, user_id Tipo I, ID Tipo I, Tipo de título String, String do tipo de corpo, extremidade de post_s, post_tt Tipo Tabela de post_s com tecla vazia, começo de post_without_id_s, user_id Tipo I, string de tipo de título, string do tipo de corpo, extremidade de post_without_id_s. METHODS: create_client IMPORTING url TYPE string RETURNING VALUE(result) TYPE REF TO if_web_http_client RAISING cx_static_check, read_posts RETURNING VALUE(result) TYPE post_tt RAISING cx_static_check, read_single_post IMPORTING id TYPE i RETURNING VALUE(result) TYPE post_s RAISING cx_static_check, create_post IMPORTING post_without_id TYPE post_without_id_s RETURNING VALUE(result) TYPE string RAISING cx_static_check, update_post IMPORTING post TYPE post_s RETURNING VALUE(result) TYPE string RAISING cx_static_check, delete_post IMPORTING id TYPE i RAISING cx_static_check. Seção particular. Constantes: Base_url Tipo String Valor 'https: // jsonplaceholder.Typicode.com/postagens ', content_type Tipo String Value' content-type ', JSON_COntent Type String Value' Application/JSON; charset = utf-8 '. Endclass. Classe ZSS_TESTER_2 Implementação. Método if_oo_adt_classrun ~ main. TENTAR. "Leia os dados (all_Posts) = read_posts (). Dados (primeiro_post) = read_single_post (1). "Criar dados (create_restons) = create_post (valor #(user_id = 7 title = 'hello, mundo!'corpo =' :) '))). "Atualize First_Post-user_id = 777. Data (Update_Response) = UPDATE_POST (FIRST_POST). "Delete Delete_Post (9). "Resultados de impressão Out-> Write (All_Posts). Out-> Write (First_Post). fora. fora. Pegue cx_root em dados (exc). fora-> write (exc-> get_text ()). Final. EndMethod. Método create_client. Dados (dest) = cl_http_destination_provider => create_by_url (url). resultado = cl_web_http_client_manager => create_by_http_destination (dest). EndMethod. Método read_posts. "Obtenha JSON de todos os dados de postagens (URL) = |< base_url >|. Dados (cliente) = create_client (url). Dados (resposta) = client-> execute (if_web_http_client => get)-> get_text (). Cliente-> Close (). "Converta JSON para postar Tabela XCO_CP_JSON => Data-> From_string (Response)-> Aplicar (valor #((xco_cp_json => transformação-> camel_case_to_underscore))-> write_to (ref #(resultado)). EndMethod. Método read_single_post. "Obtenha JSON para dados de ID de entrada de entrada (URL) = |< base_url >/< id >|. Dados (cliente) = create_client (url). Dados (resposta) = client-> execute (if_web_http_client => get)-> get_text (). Cliente-> Close (). "Converta JSON para a estrutura de postagem XCO_CP_JSON => DATA-> From_string (Response)-> Aplicar (valor #((xco_cp_json => transformação-> camel_case_to_underscore))-> write_to (ref #(resultado)). EndMethod. Método create_post. "Converta a postagem de entrada em dados JSON (json_post) = xco_cp_json => data-> de_abap (post_without_id)-> APLIC. "Envie o JSON da postagem para o servidor e retorne os dados de resposta (URL) = |< base_url >|. Dados (cliente) = create_client (url). Dados (req) = client-> get_http_request (). req-> set_text (json_post). req-> set_header_field (i_name = content_type i_value = json_content). resultado = client-> execute (if_web_http_client => post)-> get_text (). Cliente-> Close (). EndMethod. Método update_post. "Converta a postagem de entrada em dados JSON (json_post) = xco_cp_json => data-> de_abap (post)-> APLIC. "Envie o JSON da postagem para o servidor e retorne os dados de resposta (URL) = |< base_url >/< post-id >|. Dados (cliente) = create_client (url). Dados (req) = client-> get_http_request (). req-> set_text (json_post). req-> set_header_field (i_name = content_type i_value = json_content). resultado = client-> execute (if_web_http_client => put)-> get_text (). Cliente-> Close (). EndMethod. Método delete_post. Dados (url) = |< base_url >/< id >|. Dados (cliente) = create_client (url). Dados (resposta) = cliente-> execute (if_web_http_client => delete). Se resposta-> get_status () -code ne 200. Aumentar o tipo de exceção cx_web_http_client_error. FIM SE. EndMethod. Endclass.

Criação da API REST (Chamada de método Get & Post)

Uma API REST é uma interface de programação de aplicativos de transferência de estado representacional que está em conformidade com as restrições do estilo arquitetônico de repouso e permite interação com serviços RESTful.

Os métodos mais comuns são: obtenha, poste, coloque e exclua,

Esses métodos seriam usados, uma solicitação de get para recuperar um registro, uma solicitação de postagem para criar um, uma solicitação de put para atualizar um registro e uma solicitação de exclusão para excluir um

Cenário -> Precisa fornecer detalhes do driver com base no ID do driver.

Etapa 1 ->

Tabela de base de dados do driver.

Etapa 2 ->

Criar aula de manipulador de solicitação &lsquo;Zcl_driver_req_handler&rsquo; e herdar com classe padrão &lsquo;Cl_resthttp_handler&rsquo;

Nota -> é obrigatório implementar o método get_root_handler, caso contrário, ele dará um erro de sintaxe.

Etapa 3 ->

Criar aula de provedor de solicitação &lsquo;Zcl_driver_req_provider&rsquo; e herdar com classe padrão &lsquo;CL_REST_RESOURCE&rsquo;

Etapa 4 -> Agora implemente se_rest_resource ~ obtenha o método para ler os dados.

Depois de ler a chamada de dados /ui2 /cl_json => serialize () método para converter a estrutura ABAP em formato json.

Etapa 5 -> Implementar get_root_handler Método da classe de manipulador de solicitação.

Aqui precisamos vincular a classe de manipulador de solicitação e a classe de provedor de solicitação com a ajuda do roteador.

Etapa 6 -> Crie elemento de serviço, tcode sicf

Etapa 7 -> Lista de manipuladores de link, aqui precisamos definir nossa classe de manipulador de solicitação &lsquo;Zcl_driver_req_handler&rsquo;.

Etapa 8 -> Ative Service.

Etapa 9 -> Serviço de teste.

Resultado -> Aqui estamos passando o motorista e com base nos dados de identificação é exibido no formato JSON.

Publicar Chamada ->

Cenário -> Precisamos criar novos detalhes do driver na base de dados. Como você pode ver Id Driver &lsquo;002&rsquo; existe apenas no sistema e agora novo id &lsquo;001&rsquo; deve ser criado.

Na variável lv_data, os dados estão chegando na string json

Call /ui2 /cl_json => Deserialize () Método para converter a string json em estrutura ABAP.

Aqui está a nova entrada com ID &lsquo;001&rsquo;

Conclusão ->

Depois de ler este blog, você poderá criar uma API REST simples e também terá uma idéia de parâmetros de leitura na classe de provedor de solicitação.

Sinta -se à vontade para sugerir se alguma correção for necessária 🙂

Desenvolvendo uma API REST em ABAP

Em dois blogs recentes, demonstrei como escrever clientes da Web de APIs REST – com XML (aplicativo de demonstração aqui) ou JSON (aplicativo de demonstração aqui) como formato de transferência de dados. Neste blog, vou me concentrar no lado do servidor: como implementar uma API REST como manipulador de solicitação ABAP. Você pode inspecionar todo o código que estou discutindo aqui no site Migros BSP: It&rsquo;está tudo na classe zcl_job_data .

A árvore da ICF

Os manipuladores de solicitação são classes que implementam a interface if_http_extension, que consiste em um único método handle_request. Uma classe de manipulador de solicitação pode ser anexada a um caminho na transação SICF. Uma solicitação HTTP recebida será analisada pela estrutura de comunicação da Internet, tentando corresponder ao caminho da solicitação com o caminho SICF. O processo de correspondência é interrompido assim que um nó com um manipulador de solicitação anexado for encontrado. Se for esse o caso, uma instância da classe Handler será criada e o método handle_request será chamado.

Nosso serviço de exemplo está anexado ao caminho /Job/Atributos. A classe zcl_job_data é declarada responsável por todas as solicitações de entrada em que o caminho de solicitação começa com /Job/Atributos :

/wp-content/uploads/2013/01/sicf_176942.png

Primeira estratégia: método de solicitação HTTP

A implementação do método da interface if_http_extension ~ handle_request () forma o nível superior de processamento. Portanto, a implementação fornece apenas o esqueleto de processamento áspero: uma instância para operações de banco de dados, bem como uma instância para o processamento da operação restante é criada, o manuseio de solicitações é delegado a essa instância e há um bloco de captura para processamento de erros, caso nenhuma instância possa ser determinada para processamento da solicitação. Essa situação deve resultar em uma resposta HTTP com código de status &lsquo;400 – Pedido ruim&rsquo;.

Neste local, estamos usando o padrão de design da estratégia: dependendo do método HTTP (Get, Put, Post, Excluir, Opções), uma instância específica é criada. Cada instância possível corresponde a uma estratégia específica.

 Método if_http_extension ~ handle_request . Dados: LO_DB TIPO REF para LIF_DB, LO_REST TIPO REF para LIF_REST, LO_INVALID_METHOD TIPO REF para ZCX_ERROR, LV_ROASEL. tentar. * Objeto para operações de banco de dados LO_DB ?= get_db (io_server = servidor). * Obtenha a instância correta do manipulador de repouso, dependendo do verbo (Get, Cut, Post, Opções, Excluir) lo_rest ?= get_rest (io_server = servidor io_db = lo_db). * Faça a operação lo_rest-> handle_request (). Pegue zcx_not_found em lo_invalid_method. lv_Reason = lo_invalid_method-> get_text (). Servidor-> Resposta-> set_status (código = 400 "Razão de solicitação ruim = lv_Reason). final. EndMethod.

Estamos usando uma convenção de nomenclatura para a determinação da instância: a classe lcl_rest_get estará associada ao verbo http get, lcl_rest_put com put e assim por diante. Todas essas classes implementam a interface LIF_REST. Dessa forma, podemos usar a criação de instâncias dinâmicas. Como alternativa, poderíamos ter escrito um grande caso … declaração com muitos quando&rsquo;s. A vantagem do caso seria que o criar objeto A declaração pode ser verificada estaticamente quanto à correção sintática. Eu escolhi a variante dinâmica desde que acho mais clara e mais legível do que um monte de quando galhos.

Observe que o método de solicitação HTTP (Get, Put, Post,…) está disponível como pseudo -cabeçalho com o nome &lsquo;~ request_method&rsquo;:

método get_rest. Dados: lv_className Type SeoclsName, lv_method Type String, lv_message Tipo Text255. lv_method = io_server-> request-> get_header_field ('~ request_method'). Concatenate 'lcl_rest_' lv_method em lv_classname. tentar. Criar objeto EO_rest Type (lv_classname) Exportando io_request = io_server-> solicitação io_ressonse = io_server-> resposta io_db = io_db. Pegue cx_sy_create_object_error. lv_message = 'Método' 'e' 'não suportado' (001). Substitua '&' em lv_message com lv_method. _raise_with_text zcx_not_found lv_message. final. EndMethod.

Segunda estratégia: formato de transferência de dados

Agora temos diferentes classes de manipulador para os diferentes métodos de solicitação HTTP. Mas para todos esses manipuladores, existem algumas tarefas comuns. Uma dessas tarefas comuns é: determinar o formato atual de transferência de dados e converter a entrada – se disponível – em dados ABAP e vice -versa: para converter os dados do resultado ABAP na saída com o formato de transferência de dados desejado (XML ou JSON).

Agora, alguns métodos de solicitação como GET não exigem nenhum conteúdo de solicitação. Portanto, a conversão de dados recebidos é realizada pelos manipuladores de métodos que sabem que exigem dados de conteúdo. Por outro lado, sempre haverá o resultado do seguinte tipo de dados:

TIPOS: Começar em ty_result, msgtype tipo symsgty, comprimento do tipo C Mensagem 255, Jobs Tipo zjobs_tab, final de ty_result.

Nem sempre pode haver entradas na mesa de trabalho. Mas nem todo componente dessa estrutura será inicial. Se não houver tabela de trabalho, geralmente haverá uma mensagem. Portanto, a conversão do resultado sempre pode ser realizada.

Faz sentido trabalhar com uma classe de conversor abstrato, as subclasses específicas que contêm os algoritmos de conversão por tipo de conteúdo. Esta é a segunda aplicação do padrão de estratégia.

 Classe lcl_converter definição resumo. seção pública. Class-Methods get_instance importar iv_accept tipo string Valor de retorno (eo_instance) Tipo ref para lcl_converter. Métodos Content_type Resumo Valor de retorno (ev_content_type) Tipo String. Métodos get_ented_data Resumo Importação IV_CDATA TIPO STRIL. Métodos Result_to_cdata Resumo Importação IS_RESULT TYPE TY_RESULT EXPORTAMENTO EV_CDATA TIPO STRING. Endclass. "Definição lcl_converter

O método estático lcl_converter => get_instance () faz a distinção, dependendo do Aceitar Campo de cabeçalho da solicitação HTTP:

Classe LCL_Converter Implementation. método get_instance. Se iv_accept cs 'aplicação/json'. Criar objeto EO_Instance Type lcl_json_converter. outro. Criar objeto EO_Instance Type lcl_xml_converter. fim se. EndMethod. "get_instance endclass. "LCL_Converter Implementation

O enredo comum para todos os pedidos

Podemos extrair tarefas comuns para uma superclasse lcl_rest de todos os manipuladores de métodos específicos, implementando a interface Lif_rest ~ handle_request () uma vez para todas as subclasses.

O código comum na superclasse precisa ser misturado com código específico, implementado na subclasse e definindo o comportamento específico dessa subclasse. Para conseguir isso, chamamos no momento desejado em Lif_rest ~ handle_request (), um método abstrato fazer( ), que deve ser redefinido nas subclasses. Este método do () conterá a ação específica.

Agora, a implementação comum LIF_REST ~ Handle () na superclasse define apenas o fluxo do processamento, deixando as ações concretas para as subclasses ou delegados como go_converter:

  1. Executar a ação específica ligando fazer(),
  2. Manipulação de erros, com o código de erro HTTP 400 &ldquo;Pedido ruim&rdquo; Em caso de erro de conversão (dados de entrada errados) ou definir dados de resposta para uma mensagem de erro no caso de um erro de aplicativo,
  3. A estrutura de resultado é mapeada para a estrutura de dados de resposta (XML ou JSON), usando a instância do conversor correspondente,
  4. Finalmente, os dados de resposta são colocados no corpo da resposta HTTP, e também o tipo de resposta apropriado é definido: Aplicativo/JSON, ou texto/XML.

Este é o esboço geral – o processamento de resposta válido para todos Métodos de solicitação HTTP e para todos Tipos de conteúdo (XML e JSON). Os detalhes estão contidos nos métodos chamados.

Método Lif_rest ~ handle_request. Dados: LO_EX TIPO REF para CX_ROOT, LV_CDATA TIPO STRING, LS_RESULT TYPE TY_RESULT. tentar. * Execute a operação específica do (importar es_result = ls_result). Pegue zcx_parse_error em lo_ex. Go_Response-> set_status (código = 400 "Razão de solicitação ruim = lo_ex-> get_text ()). SET_RESPONSEIRO_PARAMETERS (). retornar. Pegue ZCX_Error em lo_ex. ls_result-message = lo_ex-> get_text (). ls_result-msgtype = 'e'. final. * Converta a estrutura de resultados em json ou xml, respectivamente, chame o método go_converter-> result_to_cdata exportando is_result = ls_result importando ev_cdata = lv_cdata. * Coloque o resultado no método de chamada do corpo da resposta Set_Response exportando iv_content_type = go_converter-> content_type () iv_cdata = lv_cdata. EndMethod. "handle_request

Uma tarefa específica – a solicitação de put

Deixar&rsquo;s Veja uma tarefa específica para ilustração: a solicitação de put – que sempre é uma tarefa para atualizar ou inserir atributos de trabalho para um determinado ID no banco de dados. Como segue do design, existe uma classe local lcl_rest_put manusear solicitações. Na verdade, para este manipulador de solicitação, houve apenas o fazer O próprio método a ser implementado (que é o mínimo absoluto para uma classe de tarefa específica implementar: fazer() é abstrato na classe dos pais. Sem uma implementação, nenhuma instância poderia ser construída.):

classe lcl_rest_put Definição herdando de lcl_rest. seção protegida. Métodos fazem redefinição. Endclass. "LCL_REST_PUPT Definição

A implementação é a seguinte:

  • O trabalho com o ID especificado é lido no banco de dados (se Um ID foi especificado – para novos empregos, esse não é o caso),
  • Os dados inseridos serão analisados ​​em uma estrutura LS_JOB, usando o apropriado go_converter instância,
  • E finalmente, o salvar() O método é chamado. É implementado na superclasse, pois outros métodos de solicitação também o usam.

 Classe lcl_rest_put implementação. método do. Dados: LS_JOB Type ZJobs, LV_ID TIPO ZJOBS-ID. tentar. get_job_by_id (importando es_job = ls_job). lv_id = ls_job-id. Pegue ZCX_NOT_FOUND. final. LS_JOB claro. Método de chamada go_converter-> get_entered_data exportando iv_cdata = go_request-> get_cdata () importando es_job = ls_job. Se ls_job não for inicial. Se lv_id não for inicial. ls_job-id = lv_id. fim se. Salvar (alteração de cs_job = ls_job). es_result-message = 'Job & foi salvo' (002). Substitua '&' em ES_RESULT-MESSAGEM por LS_JOB-ID. es_result msgtype = 's'. "Mensagem de sucesso Inserir ls_job na tabela es_result-jobs. fim se. EndMethod. "Faça a classe final. "LCL_REST_PUPT IMPLEMENTAÇÃO

Observe que a implementação desta tarefa não faz&rsquo;T Afaste -se com a estrutura de dados HTTP, o formato realmente em uso, nem com os detalhes do formato de dados de transferência. Simplesmente funciona com estruturas de dados ABAP ls_job para a entrada e ES_RESULT para a saída.

Sessão, identidade e bloqueio

Nos aplicativos de teste (nem no aplicativo JSON nem no aplicativo XML), não há login nem injeção dos dados. Como os aplicativos estão abertos para todos, isso funciona apenas porque eu não&rsquo;t realmente operar em uma tabela de banco de dados ZJOBS. Na verdade, cada cliente que chama o aplicativo está trabalhando com seus próprios dados de sessão, então ele não&rsquo;t conflito com outros usuários&rsquo; operações e ele próprio não é perturbado por outros usuários. Os dados da sessão são preservados para ele como cookies do lado do servidor, sobreviver à etapa de diálogo único (por exemplo, recarregar a página reproduziria o estado atual dos dados).

Quando um client da web é escrito como BSP, há um ID de sessão disponível no atributo Runtime-> server_id. Este ID da sessão identifica a instância do navegador específico que fez o pedido. No lado do cliente, este ID da sessão está sempre contido em um cookie chamado SAP-AppContext. Se um aplicativo tiver um estado que deve ser preservado com um ID da sessão, o ID deve ser extraído do cookie SAP-AppContext e deve ser passado como um parâmetro de consulta com todas as solicitações de Ajax. Aqui está a função que extrai o SAP-AppContext do cookie:

 função get_appcontext () < var lAppcontextCookie = document.cookie.match(/sap-appcontext=(.*?)(?:;|$)/); return lAppcontextCookie && ( lAppcontextCookie.length >= 2) && UNESCAPE (LappContextCookie [1]) || ""; >

O AppContext retornou desta função, pode ser passado como parâmetro de consulta com cada solicitação de Ajax. No lado do servidor, o ID da sessão pode ser extraído desse parâmetro:

Método get_session_id. Dados: lv_app_context tipo string, lv_app_context64 string de tipo. * Leia o campo do formulário, fornecido pelo Ajax Request lv_app_context64 = io_server-> request-> get_form_field ('SAP_APPCONTEXT'). Se lv_app_context64 não for inicial. * Base64 decodificar lv_app_context = cl_http_utility => decode_base64 (lv_app_context64). * Extraia o ID da sessão, encontre regex 'SAP-SessionId = ([^;]+) (?:; | $) 'em lv_app_context envios ev_session_id. fim se. Se ev_session_id for inicial. ev_session_id = io_server-> session_id. fim se. EndMethod.

Como fallback, na linha 22, o servidor-> session_id é usado. No entanto, haverá um novo servidor-> session_id para cada solicitação, o que resulta em novos dados da sessão com cada etapa de diálogo. Se você realmente precisa de gerenciamento de sessão, é essencial que o ID da sessão seja passado para o servidor.

É uma boa ideia combinar o ID da sessão com o procedimento de login: se o usuário autenticar, seu navegador recebe um ID da sessão com uma validade limitada. Esse ID da sessão deve ser aprovado a cada operação sucessiva de descanso. No ABAP, ele pode ser usado para armazenar e recuperar dados específicos da sessão na tabela de banco de dados SSCOOKIE, por meio de sua classe de acesso ao banco de dados CL_BSP_SERVER_SED_COOKIE.

Esse acoplamento de um ID da sessão com o login é – aproximadamente – a maneira como a API REST para o centro de qualidade HP funciona.

Usando ABAP&rsquo;S conversor JSON embutido

Embora a instância do conversor XML seja bastante direta para implementar -chamando uma transformação XSLT para XML -> ABAP, e outro para o caminho de volta -pode ser uma surpresa que a conversão JSON possa ser tratada exatamente da mesma maneira: com transformações. Isso é possível, pois o transformação de chamada A declaração suporta o formato JSON (pelo menos conforme SAP_BASIS 702). JSON é detectado automaticamente e analisado em um formato intermediário JSON-XML. Isso pode ser processado com uma transformação arbitrária XSLT e convertida em outros documentos XML ou para ABAP Data.

Por exemplo, uma solicitação de put de nosso aplicativo de teste pode enviar os seguintes dados JSON para o servidor:

Se uma string com este conteúdo for passada como &ldquo;Fonte XML&rdquo; para ABAP&rsquo;S Declaração de transformação de chamadas, o JSON será analisado em uma representação XML como esta (o formato é fácil de entender – acho que uma explicação detestável não é necessária aqui):

   0001 Rsnast00 Uxpd_kube_kv 2 X Saída de todas as confirmações de pedidos de vendas Rainer Zufl 

Ao processar uma transformação arbitrária do XSLT, com a declaração de transformação de chamadas e passando uma string json como fonte, o XSLT operará nessa representação interna do JSON-XML. É fácil transformar um documento JSON-XML em dados ABAP-para ser mais preciso: transformá-lo em uma representação ASXML de dados ABAP. Por exemplo, considere a seguinte transformação XSLT:

Quando aplicado à string json, ele produzirá o seguinte resultado:

     0001 Rsnast00 Uxpd_kube_kv 2 X Rainer Zufl Saída de todas as confirmações de pedidos de vendas   

Esta é uma descrição válida de dados ABAP. Se a transformação for nomeada ZJSON2JOB, os dados podem ser simplesmente importados para uma estrutura de dados ABAP com o ID dos componentes, repid e assim por diante – assim como a estrutura ES_JOB na seguinte implementação do conversor JSON.

Classe lcl_json_converter implementação. Método get_entered_data. Dados: LO_EX TIPO REF para CX_TRANSFORMATIONS_ERROR. claro ES_JOB. Verifique iv_cdata cn espaço. tentar. Transformação de chamada ZJSON2JOB Fonte XML IV_CDATA RESULTO Job = ES_JOB. Pegue cx_transformation_error em lo_ex. Raise_parse_error (lo_ex). final. EndMethod. "get_entered_data

Muitas coisas podem ser feitas com o ID da transformação da identidade, sem a necessidade de definir uma própria transformação XSLT. Se você pode impor a estrutura de dados JSON a ser usada no aplicativo da web, é vantagem usar tal &ldquo;canônico&rdquo; estrutura. Por exemplo, considere envolver o hash json com os atributos do trabalho em outro hash, tornando -o o valor para algum nome de chave simbólico como &ldquo;TRABALHO&rdquo;:

Em seguida, os dados podem ser analisados ​​em uma estrutura sem a necessidade de desenvolver uma transformação XSLT personalizada, simples usando a identidade:

 Fonte de ID da transformação de chamada XML IV_CDATA Job = ES_JOB.

Neste exemplo, desde que escrevi o client e o processamento do lado do servidor, eu poderia ter escolhido isso mais &ldquo;canônico&rdquo; formatar. Mas, ao não escolher, aprendi a trabalhar com formatos de dados JSON mais flexíveis.

Existem várias razões para trabalhar com &ldquo;não canônico&rdquo; JSON Representações de dados ABAP:

  • Um formato JSON pode ser projetado em favor do aplicativo da web – para otimizar a legibilidade do código JavaScript do cliente trabalhando nos dados.
  • Pode haver componentes do cliente que exigem formatos JSON específicos. Por exemplo, o JQuery Datatable exige que os dados da tabela sejam passados ​​como uma matriz de matrizes: http: // www.Datatables.net/release-dados/exemplos/data_sources/Ajax.html
  • Os serviços de terceiros baseados em JSON podem ser chamados do lado ABAP (com um objeto cliente HTTP)
  • Os dados ABAP podem ser projetado para os dados essenciais, reduzindo o tamanho da mensagem para os dados que são realmente necessários.

Só para ilustrar, vamos&rsquo;s Dê uma olhada na outra conversão – a saída do servidor para o cliente. Novamente, o formato difere um pouco do &ldquo;canônico&rdquo; Formato json, que simplificaria o manuseio do lado da ABAP consideravelmente. Como mencionado, a estrutura de dados do resultado contém

  • uma mensagem,
  • um tipo de mensagem,
  • e uma tabela de atributos de trabalho:

TIPOS: Começar em ty_result, msgtype tipo symsgty, comprimento do tipo C Mensagem 255, Jobs Tipo zjobs_tab, final de ty_result.

O formato a seguir seria um pingente JSON perfeito para esta estrutura. Poderia ser simplesmente produzido com a transformação da identidade, passando como &ldquo;resultado da fonte = ls_result&rdquo; (onde ls_result é uma estrutura do tipo ty_result):

  • Todos os nomes dos componentes combinam perfeitamente com os nomes de chave do JSON Hash,
  • Uma tabela interna é mapeada como uma variedade de hashs json, cada um hash representando uma entrada da tabela,
  • E há um hash de nível superior com um nome simbólico &ldquo;RESULTADO&rdquo; Para a coisa completa:

Mas o formato JSON que a API REST suporta, na verdade difere em alguns detalhes:

  • Os trabalhos são projetados não como uma matriz, mas como um hash, com o ID como hash -chave.
  • Não há hash redundante, envolvendo tudo como o valor para alguma chave.
  • O componente para msgtype é diferente. É simplesmente chamado de tipo.

Aqui está uma instância de exemplo:

 < "JOBS": < "0001": < "REPID": "RSNAST00", "VARID": "UXPD_KUBE_KV", "PRIO": "2", "RESTART": "X", "CONTACT": "Rainer Zufall", "DESCR": "Output all sales order confirmations" >, "0002": < "REPID": "RBDAPP01", "VARID": "UXPD_EDI_GUT02", "PRIO": "3", "RESTART": "X", "CONTACT": "Herbert Hurtig", "DESCR": "Credit Memos" >>, "Mensagem": "", "tipo": "">

Procedemos de maneira semelhante à acima, apenas na outra direção: com base no tipo de dados ABAP ty_result, Escrevemos uma transformação XSLT para obter o formato interno JSON-XML correspondente a esta string de dados JSON.

O formato de dados JSON-XML da sequência de dados JSON desejada se parece com o seguinte:

     Rsnast00 Uxpd_kube_kv 2 X Rainer Zufl Saída de todas as confirmações de pedidos de vendas  Rbdapp01 Uxpd_edi_gut02 3 X Herbert Hurtig Notas de crédito   Teste EU 

Portanto, este é o alvo que deve ser obtido como resultado da transformação. Por outro lado, o formato ASXML da estrutura ty_result se parece com o seguinte:

       0001 Rsnast00 Uxpd_kube_kv 2 X Rainer Zufl Saída de todas as confirmações de pedidos de vendas  0002 Rbdapp01 Uxpd_edi_gut02 3 X Herbert Hurtig Notas de crédito   Teste EU   

E este é o programa XSLT que executará a transformação:

Vemos isso, basicamente, para cada desvio do &ldquo;canônico&rdquo; Representação JSON dos dados ABAP, há um modelo na transformação XSLT lidando com este desvio. Por exemplo, o tipo de nome diferente em vez de msgtype no alvo é tratado com o modelo

O ID deve ser reorganizado: de ser um atributo simples da estrutura de dados ZJOBS, ele deve ser aumentado um nível mais alto para se tornar a chave de um hash. Todos os outros atributos, exceto ID, são copiados como nós de string no resultado. Para isso, esses dois modelos são necessários:

O mapeamento do objeto de dados TY_RESULT em uma sequência JSON do formato esperado agora é realizado em ABAP com o seguinte código:

Método result_to_cdata. Dados: LO_WRITER TIPO REF para CL_SXML_STRING_WRITER. lo_writer = cl_sxml_string_writer => create (type = if_sxml => co_xt_json). Transformação de chamada ZJOBS2JSON Dados de origem = IS_RESULT RESULTO XML LO_WRITER. ev_cdata = cl_abap_codePage => convert_from (lo_writer-> get_output ()). EndMethod. "Result_to_cdata

Que&rsquo;S All: EV_CData conterá a sequência de dados JSON, a ser colocada no corpo de resposta HTTP.

Resumo

Eu descrevi alguns tópicos típicos relativos à implementação de APIs REST em ABAP. É possível manter preocupações separadas em classes separadas (locais ou globais) aplicando padrões como estratégia. É assim que a classe ZCL_JOB_DATA, servindo minha API REST Demo, é organizada (as idéias básicas foram discutidas neste blog):

Como criar uma API REST com SAP ABAP e aplicar o roteamento MVC1

Nesta postagem do blog, gostaria de mostrar como criar API REST e como aplicar o roteamento MVC1 para lidar com uma solicitação diferente simplesmente de uma classe de controlador.

Para isso, primeiro criaremos a classe Handler e Controller para a estrutura de descanso. Em seguida, adicionaremos a classe do controlador MVC1 e a classe modelo para processar a lógica de negócios.

E finalmente criaremos um serviço para lidar com solicitações de descanso.

No final da posta.

Para ler mais sobre o SAP REST, dê uma olhada no tutorial de descanso.

Criar estruturas a seguir;

  • Zrest_s_resp_state
  • Zrest_s_Response
  • Zrest_s_request
  • Zrest_s_resp_state

  • Zrest_s_Response

Agora vamos criar classes.

Ligue para uma chamada para uma chamada

  • Zrest_cl_defs
  • Zrest_cl_model
  • Zrest_cl_req_controller
  • Zrest_cl_req_http_handler
  • Zrest_cl_http_handler

Classe zrest_cl_defs definição pública Crie público . Seção pública. Constantes C_State_success Type Char1 Value 'S' ## NO_TEXT. Constantes c_state_warning tipo char1 valor 'w' ## no_text. Constantes c_state_error tipo char1 valor 'e' ## no_text. Seção protegida. Seção particular. Endclass. Classe ZREST_CL_DEFS Implementação. Endclass.

Classe zrest_cl_model Definição public final Crie público . Seção pública. Métodos Get_DateTime Exporting !Response_body Type ZREST_S_RESPONSECO-BODO !Tipo de estado ZREST_S_RESPONSETO . Seção protegida. Seção particular. Endclass. Classe ZREST_CL_MODEL Implementação. * ---------------------------------------------------------------------------------------+ * | Instância do método público zrest_cl_model-> get_dateTime * + ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ [Método get_dateTime. Dados: Exref Type Ref para CX_ROOT. TENTAR . TIPOS: Começam em ty_res, tipo DATETIME TIPO TZNTIMESTP, FIM DE TY_RES. Dados: res Tipo ty_res. res-DateTime = sy-datum && sy-luzeit. resposta_body = /ui2 /cl_json => serialize (exportando dados = res). estado-estado = zrest_cl_defs => c_state_success. Pegue cx_root em exref. Estado-estado = zrest_cl_defs => c_state_error. estado-state_text = exref-> get_text (). Final. EndMethod. Endclass.

Classe zrest_cl_req_controller definição public final crie public . Seção pública. Métodos Process_request Importing !Req_json tipo string exportando !String de tipo de resposta_json !Response_stc Tipo zrest_s_Response . Seção protegida. Seção particular. Constantes: C_REQ_GET_DATETIME TIPO ZREST_E_REQ_ID VALOR '1001'. Métodos conv_stc_to_json Importando resposta_stc tipo zrest_s_ropSonse Retorno Valor (resultado) Tipo String. Endclass. Classe ZREST_CL_REQ_CONTROLLER IMPLEMENTAÇÃO. * ---------------------------------------------------------------------------------------+ * | Instância Método privado Zrest_Cl_req_Controller-> conv_stc_to_json * + -------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- [--->] Response_stc Tipo zrest_s_Response * | [Método conv_stc_to_json. Dados: Exref Type Ref para CX_ROOT. TENTAR . resultado = /ui2 /cl_json => serialize (exportando dados = resposta_stc). Pegue cx_root. resultado = exref-> get_text (). Final. EndMethod. * ---------------------------------------------------------------------------------------+ * | Instância do método público zrest_cl_req_controller-> process_request * + ------------------------------------------------------------------------------------------------- + * | [--->] req_json tipo string * | [Método process_request. Dados: Exref Type Ref para CX_ROOT. TENTAR . Dados req_stc tipo zrest_s_request. /ui2/cl_json => Deserialize (exportando json = req_json alterando dados = req_stc). Se req_stc-id não for inicial. Dados (Model) = new Zrest_Cl_model (). Se req_stc-id eq c_req_get_dateTime. Model-> get_dateTime (importando resposta_body = Response_stc-body state = Response_stc-state). OUTRO. Response_stc-state-state = zrest_cl_defs => c_state_warning. Mensagem S001 (ZREST_MSG) em Response_Stc-State-State_text. FIM SE. OUTRO. "Preencha o conteúdo fictício como amostra req_stc-id = 9999999. req_stc-body = 'algum conteúdo json'. Response_stc-corpo = /ui2 /cl_json => serialize (exportando dados = req_stc). Response_stc-state-state = zrest_cl_defs => c_state_warning. Mensagem S002 (ZREST_MSG) em Response_Stc-State-State_text. FIM SE. Response_json = conv_stc_to_json (Response_stc = Response_stc). Pegue cx_root. Response_stc-state-state = zrest_cl_defs => c_state_error. Response_stc-state-state_text = exref-> get_text (). Response_json = conv_stc_to_json (Response_stc = Response_stc). Final. EndMethod. Endclass.

Classe zrest_cl_req_http_handler definição pública herdando de cl_rest_resource final criar público . Seção pública. Métodos if_rest_resource ~ Obtenha redefinição . Métodos if_rest_resource ~ Pós -redefinição . Seção protegida. Seção particular. Endclass. Classe ZREST_CL_REQ_HTTP_HANDLER IMPLEMPERENCIMENTAÇÃO. * ---------------------------------------------------------------------------------------+ * | Instância do método público zrest_cl_req_http_handler-> if_rest_resource ~ get * + -------------------------------------------------------------------------------------------------------------------------------------------------------------------------- +-------------------------------------------------------------------------------------- Método if_rest_resource ~ obtenha. Data (req_json) = mo_request-> get_uri_query_parameter (iv_name = 'req' iv_encoded = abap_false). Dados (controlador) = new Zrest_cl_req_controller (). controlador-> process_request (exportando req_json = req_json importando resposta_json = dados (resposta_json)). MO_RESPONSEIRO-> create_entity ()-> set_string_data (iv_data = resposta_json). EndMethod. * ---------------------------------------------------------------------------------------+ * | Instância do método público zrest_cl_req_http_handler-> if_rest_resource ~ post * + ----------------------------------------------------------------------------------------------- + * | [--->] io_entity tipo referência para if_rest_entity * +-------------------------------------------------------------------------------------- Método if_rest_resource ~ post. Dados (req_json) = mo_request-> get_entity ()-> get_string_data (). Dados (controlador) = new Zrest_cl_req_controller (). controlador-> process_request (exportando req_json = req_json importando resposta_json = dados (resposta_json)). MO_RESPONSEIRO-> create_entity ()-> set_string_data (iv_data = resposta_json). EndMethod. Endclass.

O CSRF está desativado no manipulador abaixo. Desativá -lo dos parâmetros de serviço da GUI não funciona. Você precisa implementar handle_csrf_token para desativá -lo para descansar.

classe zrest_cl_http_handler definição pública herdando de cl_rest_http_handler criar public . seção pública. "Fornece roteamento. Os caminhos de roteamento são atribuídos aos controladores nesse método de métodos if_rest_application ~ get_root_handler Redefinição . seção protegida. "Se você quiser desativar, redefina esse método. Apenas como um método vazio. Métodos handle_csrf_token Redefinição . Seção particular. Endclass. Classe ZREST_CL_HTTP_HANDLER Implementação. * ---------------------------------------------------------------------------------------+ * | Método protegido da instância zrest_cl_http_handler-> handle_csrf_token * + -------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- [--->] IO_CSRF_HANDLER TIPO REF para IF_REST_CSRF_HANDLER * | [--->] IO_REQUEST TIPO REF para if_rest_request * | [--->] IO_RESPONSE Tipo Ref para if_rest_Response * +-------------------------------------------------------------------------------------- Método handle_csrf_token. * Call Method Super-> handle_csrf_token * exporting * io_csrf_handler = * io_request = * io_Response = * . EndMethod. * ---------------------------------------------------------------------------------------+ * | Instância do método público zrest_cl_http_handler-> if_rest_application ~ get_root_handler * + ------------------------------------------------------------------------------------------------- + * | [Método if_rest_application ~ get_root_handler. "Fornece roteamento. "Caminho de serviço/SAP/BC/Rest" URL http: // vhcalnplci: 8000/SAP/BC/REST/ZREST/REST?SAP-Client = 001 & req = Data (root_handler) = new CL_REST_ROUTER (). root_handler-> Anexe (exportando iv_template = '/Rest' "Nome unificado para recursos iv_handler_class = 'zrest_cl_req_http_handler'" Nome do tipo de objeto). * "Você pode adicionar mais classes de manipulador de solicitação *" Caminho de serviço/SAP/BC/REST * "URL de amostra http: // vhcalnplci: 8000/SAP/BC/REST/ZREST/REST2?SAP-Client = 001 & req = * root_handler-> Anexe ( * exportando * iv_template = '/REST2' "Nome unificado para recursos * iv_handler_class = 'zrest_cl_req_http_handler2'" Nome do tipo de objeto *). ro_root_handler = root_handler. EndMethod. Endclass.

E passo final, criar um serviço.

Abra Sicf Tcode e Run.

Vá para/SAP/BC/REST e adicione novo sub -elemento

Adicionar descrição e vá para a guia Lista de manipuladores e nossa classe, zrest_cl_http_handler, como manipulador.

Ative Service. Clique com o botão direito do mouse e clique em Teste. Vai abrir o navegador. Altere o URL para lidar com solicitações de descanso /descanso.

No meu caso, http: // vhcalnplci: 8000/SAP/BC/Rest/Zrest/Descansar

Se você quiser passar alguns parâmetros no pedido de get, adicione parâmetros de sequência de consulta como &lsquo;http: // vhcalnplci: 8000/SAP/BC/REST/ZREST/REST?SAP-Client = 001 & req =&rsquo;

Se você não desativar o CSRF no manipulador, terá problemas para chamar métodos não obtidos, como métodos de post do Postman ou de um cliente diferente do próprio servidor.

Portanto, no meu exemplo, desativei o CSRF.

Exemplos de postman

Parâmetros básicos de autenticação

Obtenha exemplo e resultado

Post exemplo e resultado

Para consumir de ABAP

Usaremos um httpclient para fazer chamadas e analisaremos o JSON para a estrutura ABAP com o exemplo get_dateTime.

Código do cliente HTTP

Classe zutil_cl_rest_ws definição pública Crie público . "Wrapper Class para fazer o REST Service Chamadas de seção pública. "Constantes Constantes: c_content_type_json Tipo String Value 'Application/json; charset = utf-8', c_content_type_xml Tipo String Value 'Application/xml; charset = utf-8', c_request_method_get Tipo String Value 'Get', c_request_methPer. "Faz com que os métodos de chamada ws call_ws importando valor (i_url) digite o valor da string (i_content_type) digite string padrão c_content_type_json value (i_request_method) tipo string padrão c_request_method_get value (i_usename) tipo string) util_cl_defs => gty_state value (e_rosponse_str) string de tipo. Seção protegida. Seção particular. Dados http_client type ref para if_http_client. Métodos Fill_warning Retorno Valor (Estado) Tipo zutil_cl_defs => gty_state. Endclass. Classe ZUTIL_CL_REST_WS Implementação. * ---------------------------------------------------------------------------------------+ * | Instância do método público zutil_cl_rest_ws-> call_ws * + ----------------------------------------------------------------------------------------------- + * | [--->] i_url Tipo string * | [--->] i_content_type type string (default = c_content_type_json) * | [--->] i_request_method type string (default = c_request_method_get) * | [--->] i_username type string (opcional) * | [--->] i_password Type string (opcional) * | [--->] i_payload Tipo string (opcional) * | [Gty_state * | [Método call_ws. Dados: Exref Type Ref para CX_ROOT. TENTAR. "Usando o this create_by_url pode simplificar certos aspectos do uso desta classe http_client livre. cl_http_client => create_by_url (exportando url = i_url importar cliente = http_client Exceções argument_not_found = 1 plugin_not_active = 2 internal_error = 3 outros = 4). Se sy-subrc <> 0. e_state-state = Fill_warning (). RETORNAR. FIM SE. "Minha lógica usou originalmente put, mas você poderá mudar para postar http_client-> request-> set_method (i_request_method). http_client-> request-> set_content_type (i_content_type). "Lembre -se de autenticar se i_username não for inicial ou i_password não for inicial. http_client-> autenticate (nome de usuário = i_username senha = i_password). FIM SE. "Se existir, prepare a carga útil e atribua se i_payload não for inicial. "Converta essa carga útil em XString. Dados lv_payload_x tipo xstring. Função de chamada 'scms_string_to_xstring' exportando texto = i_payload de importação buffer = lv_payload_x exceções falhas = 1 outros = 2. Se sy-subrc <> 0. e_state-state = zutil_cl_defs => c_state_warning. e_state-state = 'Erro de codificação!'. RETORNAR. OUTRO. http_client-> request-> set_data (lv_payload_x). "Dados binários endif. FIM SE. "Enviando a solicitação http_client-> send (exceções http_communication_failure = 1 http_invalid_state = 2). Se sy-subrc <> 0. e_state-state = Fill_warning (). RETORNAR. FIM SE. "Recebendo a resposta http_client-> receba (exceções http_communication_failure = 1 http_invalid_state = 2 http_processing_failed = 3). Se sy-subrc <> 0. e_state-state = Fill_warning (). RETORNAR. FIM SE. "Verifique a resposta. Espero que você receba uma resposta JSON. e_Response_str = http_client-> resposta-> get_cdata (). Se E_RESPONSE_STR for inicial. e_Response_str = http_client-> resposta-> get_data (). FIM SE. e_state-state = zutil_cl_defs => c_state_success. e_state-state_text = 'concluído com sucesso.'. Pegue cx_root em exref. e_state-state = zutil_cl_defs => c_state_error. e_state-state_text = exref-> get_text (). Final. EndMethod. * ---------------------------------------------------------------------------------------+ * | Instância Método privado zutil_cl_rest_ws-> Fill_warning * + -------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- [Gty_state * +-------------------------------------------------------------------------------------- Método FILL_WARNING. estado-estado = zutil_cl_defs => c_state_warning. http_client-> get_last_error (importando mensagem = dados (msg) "mensagem de erro). estado-state_text = msg. EndMethod. Endclass.

Código de classe do consumidor. Primeiro desembrulham a estrutura de resposta e verifica o estado. Se for sucesso, desembrulhá a parte do corpo JSON para o resultado da ID de chamada 1001.

Classe zrest_cl_consumer Definição public final Crie público . Seção pública. Métodos get_dateTime exporting state type zutil_cl_defs => gty_state dt tipo tzntimestp. Seção protegida. Seção particular. Endclass. Classe ZREST_CL_CONSUMER Implementação. * ---------------------------------------------------------------------------------------+ * | Instância do método público zrest_cl_consumer-> get_dateTime * + ------------------------------------------------------------------------------------------------------------------------------------ [Gty_state * | [Método get_dateTime. "Método de amostra para consumir Dados da API da Web REST: Exref Type Ref para CX_ROOT. TENTAR. "Constantes: c_uname Type String Value 'Developer', C_Pass Type String Value 'Down1oad'. "Build Get Call Data: URL Tipo String Valor 'http: // vhcalnplci: 8000/SAP/BC/REST/ZREST/REST?SAP-Client = 001 '. Dados req_stc tipo zrest_s_request. req_stc-id = '1001'. Dados (req_json) = /ui2 /cl_json => serialize (exportando dados = req_stc). url = url && '& req =' && req_json. "Ligue para a Web API novo zutil_cl_rest_ws ()-> call_ws (exportando i_url = url i_username = c_uname i_password = c_pass importing e_state. Se o estado do estado eq zutil_cl_defs => c_state_success. Dados: resp_stc tipo Zrest_s_Response. /ui2/cl_json => Deserialize (exportando JSON = JSON_RESPONHO MONTANTE DE DADOS = RESP_STC). Se resp_stc-State-State Eq zutil_cl_defs => c_state_success. TIPOS: Começam em ty_res, tipo DATETIME TIPO TZNTIMESTP, FIM DE TY_RES. Dados: RESP_1001 TYPE TY_RES. /ui2/cl_json => Deserialize (exportando JSON = RESP_STC-BODY MONTAÇÃO DATOS = RESP_1001). dt = resp_1001-DateTime. FIM SE. FIM SE. Pegue cx_root em exref. Estado-estado = zutil_cl_defs => c_state_error. estado-state_text = exref-> get_text (). Final. EndMethod. Endclass.

Ligue para%20Result

Isso é tudo. Dessa forma, você pode integrar qualquer ambiente, sistema à SAP.

espero que ajude.

Obrigado pela leitura.

Links Relacionados