mirror of
https://github.com/hasura/graphql-engine.git
synced 2025-01-05 14:27:59 +03:00
0ab7844efa
https://github.com/hasura/graphql-engine-mono/pull/1688 Co-authored-by: Jesse Martin <174035+martincreative@users.noreply.github.com> Co-authored-by: Rikin Kachhia <54616969+rikinsk@users.noreply.github.com> GitOrigin-RevId: 7a585cb8473e0c5572a347926e156b6ce6f4d780
474 lines
15 KiB
ReStructuredText
474 lines
15 KiB
ReStructuredText
.. meta::
|
||
:description: Hasura GraphQL engine server logs
|
||
:keywords: hasura, docs, deployment, logs, server logs
|
||
|
||
.. _hge_logs:
|
||
|
||
Hasura GraphQL engine logs
|
||
==========================
|
||
|
||
.. contents:: Table of contents
|
||
:backlinks: none
|
||
:depth: 1
|
||
:local:
|
||
|
||
Accessing logs
|
||
--------------
|
||
|
||
Based on your deployment method, the Hasura GraphQL engine logs can be accessed as follows:
|
||
|
||
- :ref:`On Heroku <heroku_logs>`
|
||
- :ref:`On Docker <docker_logs>`
|
||
- :ref:`On Kubernetes <kubernetes_logs>`
|
||
- :ref:`On Digital Ocean <do_logs>`
|
||
- :ref:`On Azure <azure_logs>`
|
||
- :ref:`On Google Cloud & Kubernetes <gc_kubernetes_logs>`
|
||
|
||
.. admonition:: Detailed Logging in Hasura Cloud
|
||
:class: dhc
|
||
|
||
If you’re looking for more in-depth logging information, along with a console for browsing logs, please see :ref:`Metrics with Hasura Cloud <metrics>`.
|
||
|
||
|
||
.. _log-types:
|
||
|
||
Different log-types
|
||
-------------------
|
||
|
||
The Hasura GraphQL engine has different kind of log-types depending on the sub-system or the layer. A log-type is simply the ``type`` field in a log line, which indicates which sub-system the log comes from.
|
||
|
||
For example, the HTTP webserver logs incoming requests as an access log and is called ``http-log``. Similarly logs from the websocket layer are called ``websocket-log``, logs from the event trigger system are called ``event-trigger`` etc.
|
||
|
||
|
||
You can configure the GraphQL engine to enable/disable certain log-types using the the ``--enabled-log-types`` flag or the ``HASURA_GRAPHQL_ENABLED_LOG_TYPES`` env var. See :ref:`server_flag_reference`
|
||
|
||
Default enabled log-types are: ``startup, http-log, webhook-log, websocket-log``
|
||
|
||
All the log-types that can be enabled/disabled are:
|
||
|
||
.. list-table:: Configurable log-types
|
||
:header-rows: 1
|
||
:widths: 10 25 10
|
||
|
||
* - Log type
|
||
- Description
|
||
- Log Level
|
||
|
||
* - ``startup``
|
||
- Information that is logged during startup
|
||
- ``info``
|
||
|
||
* - ``query-log``
|
||
- Logs: the entire GraphQL query with variables, generated SQL statements
|
||
(only for database queries, not for mutations/subscriptions or remote
|
||
schema and action queries), the operation name (if provided in the
|
||
GraphQL request)
|
||
- ``info``
|
||
|
||
* - ``http-log``
|
||
- Http access and error logs at the webserver layer (handling GraphQL and metadata requests)
|
||
- ``info`` and ``error``
|
||
|
||
* - ``websocket-log``
|
||
- Websocket events and error logs at the websocket server layer (handling GraphQL requests)
|
||
- ``info`` and ``error``
|
||
|
||
* - ``webhook-log``
|
||
- Logs responses and errors from the authorization webhook (if setup)
|
||
- ``info`` and ``error``
|
||
|
||
|
||
Apart from the above, there are other internal log-types which cannot be configured:
|
||
|
||
.. list-table:: Internal log-types
|
||
:header-rows: 1
|
||
:widths: 10 25 10
|
||
|
||
* - Log type
|
||
- Description
|
||
- Log Level
|
||
|
||
* - ``pg-client``
|
||
- Logs from the postgres client library
|
||
- ``warn``
|
||
|
||
* - ``metadata``
|
||
- Logs inconsistent metadata items
|
||
- ``warn``
|
||
|
||
* - ``jwk-refresh-log``
|
||
- Logs information and errors about periodic refreshing of JWK
|
||
- ``info`` and ``error``
|
||
|
||
* - ``telemetry-log``
|
||
- Logs error (if any) while sending out telemetry data
|
||
- ``info``
|
||
|
||
* - ``event-trigger``
|
||
- Logs HTTP responses from the webhook, HTTP exceptions and internal
|
||
errors
|
||
- ``info`` and ``error``
|
||
|
||
* - ``ws-server``
|
||
- Debug logs from the websocket server, mostly used internally for debugging
|
||
- ``debug``
|
||
|
||
* - ``schema-sync-thread``
|
||
- Logs internal events, when it detects schema has changed on Postgres and
|
||
when it reloads the schema
|
||
- ``info`` and ``error``
|
||
|
||
Logging levels
|
||
--------------
|
||
|
||
You can set the desired logging level on the server using the ``log-level`` flag or the ``HASURA_GRAPHQL_LOG_LEVEL`` env var. See :ref:`server_flag_reference`.
|
||
|
||
The default log-level is ``info``.
|
||
|
||
Setting a log-level will print all logs of priority greater than the set level. The log-level hierarchy is: ``debug > info > warn > error``
|
||
|
||
For example, setting ``--log-level=warn``, will enable all warn and error level logs only. So even if you have enabled ``query-log`` it won't be printed as the level of ``query-log`` is ``info``.
|
||
|
||
See :ref:`log-types <log-types>` for more details on log-level of each log-type.
|
||
|
||
Log structure and metrics
|
||
-------------------------
|
||
|
||
All requests are identified by a request id. If the client sends a ``x-request-id`` header then that is used, otherwise a request id is generated for each request. This is also sent back to the client as a response header (``x-request-id``). This is useful to correlate logs from the server and the client.
|
||
|
||
**query-log** structure
|
||
^^^^^^^^^^^^^^^^^^^^^^^
|
||
|
||
On enabling verbose logging, i.e. enabling ``query-log``,
|
||
GraphQL engine will log the full GraphQL query object on each request.
|
||
|
||
It will also log the generated SQL for GraphQL queries (but not mutations and
|
||
subscriptions).
|
||
|
||
.. code-block:: json
|
||
|
||
{
|
||
"timestamp": "2019-06-03T13:25:10.915+0530",
|
||
"level": "info",
|
||
"type": "query-log",
|
||
"detail": {
|
||
"kind": "database",
|
||
"request_id": "840f952d-c489-4d21-a87a-cc23ad17926a",
|
||
"query": {
|
||
"variables": {
|
||
"limit": 10
|
||
},
|
||
"operationName": "getProfile",
|
||
"query": "query getProfile($limit: Int!) {\n profile(limit: $limit, where: {username: {_like: \"%a%\"}}) {\n username\n }\n myusername: profile (where: {username: {_eq: \"foobar\"}}) {\n username\n }\n}\n"
|
||
},
|
||
"generated_sql": {
|
||
"profile": {
|
||
"prepared_arguments": ["{\"x-hasura-role\":\"admin\"}", "%a%"],
|
||
"query": "SELECT coalesce(json_agg(\"root\" ), '[]' ) AS \"root\" FROM (SELECT row_to_json((SELECT \"_1_e\" FROM (SELECT \"_0_root.base\".\"username\" AS \"username\" ) AS \"_1_e\" ) ) AS \"root\" FROM (SELECT * FROM \"public\".\"profile\" WHERE ((\"public\".\"profile\".\"username\") LIKE ($2)) ) AS \"_0_root.base\" LIMIT 10 ) AS \"_2_root\" "
|
||
},
|
||
"myusername": {
|
||
"prepared_arguments": ["{\"x-hasura-role\":\"admin\"}", "foobar"],
|
||
"query": "SELECT coalesce(json_agg(\"root\" ), '[]' ) AS \"root\" FROM (SELECT row_to_json((SELECT \"_1_e\" FROM (SELECT \"_0_root.base\".\"username\" AS \"username\" ) AS \"_1_e\" ) ) AS \"root\" FROM (SELECT * FROM \"public\".\"profile\" WHERE ((\"public\".\"profile\".\"username\") = ($2)) ) AS \"_0_root.base\" ) AS \"_2_root\" "
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
|
||
The ``type`` of in the log with be ``query-log``. All the details are nested
|
||
under the ``detail`` key.
|
||
|
||
This log contains 3 important fields:
|
||
|
||
- ``kind``: indicates the type or kind of operation. ``kind`` can be
|
||
``database``, ``action``, ``remote-schema``, ``cached`` or ``introspection``
|
||
|
||
- ``request_id``: A unique ID for each request. If the client sends a
|
||
``x-request-id`` header then that is respected, otherwise a UUID is generated
|
||
for each request. This is useful to correlate between ``http-log`` and
|
||
``query-log``.
|
||
|
||
- ``query``: Contains the full GraphQL request including the variables and
|
||
operation name.
|
||
|
||
- ``generated_sql``: this contains the generated SQL for GraphQL queries. For
|
||
mutations and subscriptions this field will be ``null``.
|
||
|
||
|
||
**http-log** structure
|
||
^^^^^^^^^^^^^^^^^^^^^^
|
||
|
||
This is how the HTTP access logs look like:
|
||
|
||
- On success response:
|
||
|
||
.. code-block:: json
|
||
|
||
{
|
||
"timestamp": "2019-05-30T23:40:24.654+0530",
|
||
"level": "info",
|
||
"type": "http-log",
|
||
"detail": {
|
||
"request_id": "072b3617-6653-4fd5-b5ee-580e9d098c3d",
|
||
"operation": {
|
||
"query_execution_time": 0.009240042,
|
||
"user_vars": {
|
||
"x-hasura-role": "user"
|
||
},
|
||
"error": null,
|
||
"request_id": "072b3617-6653-4fd5-b5ee-580e9d098c3d",
|
||
"parameterized_query_hash": "7116865cef017c3b09e5c9271b0e182a6dcf4c01",
|
||
"response_size": 105,
|
||
"query": null
|
||
},
|
||
"http_info": {
|
||
"status": 200,
|
||
"http_version": "HTTP/1.1",
|
||
"url": "/v1/graphql",
|
||
"ip": "127.0.0.1",
|
||
"method": "POST"
|
||
}
|
||
}
|
||
}
|
||
|
||
|
||
- On error response:
|
||
|
||
.. code-block:: json
|
||
|
||
{
|
||
"timestamp": "2019-05-29T15:22:37.834+0530",
|
||
"level": "error",
|
||
"type": "http-log",
|
||
"detail": {
|
||
"operation": {
|
||
"query_execution_time": 0.000656144,
|
||
"user_vars": {
|
||
"x-hasura-role": "user",
|
||
"x-hasura-user-id": "1"
|
||
},
|
||
"error": {
|
||
"path": "$.selectionSet.profile.selectionSet.usernamex",
|
||
"error": "field \"usernamex\" not found in type: 'profile'",
|
||
"code": "validation-failed"
|
||
},
|
||
"request_id": "072b3617-6653-4fd5-b5ee-580e9d098c3d",
|
||
"response_size": 142,
|
||
"query": {
|
||
"variables": {
|
||
"limit": 10
|
||
},
|
||
"operationName": "getProfile",
|
||
"query": "query getProfile($limit: Int!) { profile(limit: $limit, where:{username: {_like: \"%a%\"}}) { usernamex} }"
|
||
}
|
||
},
|
||
"http_info": {
|
||
"status": 200,
|
||
"http_version": "HTTP/1.1",
|
||
"url": "/v1/graphql",
|
||
"ip": "127.0.0.1",
|
||
"method": "POST"
|
||
}
|
||
}
|
||
|
||
The ``type`` in the log will be ``http-log`` for HTTP access/error log. This
|
||
log contains basic information about the HTTP request and the GraphQL operation.
|
||
|
||
It has two important "keys" under the ``detail`` section - ``operation`` and ``http_info``.
|
||
|
||
``http_info`` lists various information regarding the HTTP request, e.g. IP
|
||
address, URL path, HTTP status code etc.
|
||
|
||
``operation`` lists various information regarding the GraphQL query/operation.
|
||
|
||
- ``query_execution_time``: the time taken to parse the GraphQL query (from JSON
|
||
request), compile it to SQL with permissions and user session variables, and
|
||
then executing it and fetching the results back from Postgres. The unit is in
|
||
seconds.
|
||
|
||
- ``user_vars``: contains the user session variables. Or the ``x-hasura-*``
|
||
session variables inferred from the authorization mode.
|
||
|
||
- ``request_id``: A unique ID for each request. If the client sends a
|
||
``x-request-id`` header then that is respected, otherwise a UUID is generated
|
||
for each request.
|
||
|
||
- ``response_size``: Size of the response in bytes.
|
||
|
||
- ``error``: *optional*. Will contain the error object when there is an error,
|
||
otherwise this will be ``null``. This key can be used to detect if there is an
|
||
error in the request. The status code for error requests will be ``200`` on
|
||
the ``v1/graphql`` endpoint.
|
||
|
||
- ``query``: *optional*. This will contain the GraphQL query object only when
|
||
there is an error. On successful response this will be ``null``.
|
||
|
||
- ``parametrized_query_hash`` (*): Hash of the incoming GraphQL query after resolving variables
|
||
with all the leaf nodes (i.e. scalar values) discarded. This value will only be logged when
|
||
the request is successful. For example, all the queries mentioned
|
||
in the below snippet will compute the same parametrized query hash.
|
||
|
||
.. code-block:: graphql
|
||
|
||
# sample query
|
||
query {
|
||
authors (where: {id: {_eq: 2}}) {
|
||
id
|
||
name
|
||
}
|
||
}
|
||
|
||
# query with a different leaf value to that of the sample query
|
||
query {
|
||
authors (where: {id: {_eq: 203943}}) {
|
||
id
|
||
name
|
||
}
|
||
}
|
||
|
||
# query with use of a variable, the value of
|
||
# the variable `id` can be anything
|
||
query {
|
||
authors (where: {id: {_eq: $id}}) {
|
||
id
|
||
name
|
||
}
|
||
}
|
||
|
||
# query with use of a boolean expression variable,
|
||
# the value when the `whereBoolExp` is in the form of
|
||
#
|
||
# {
|
||
# "id": {
|
||
# "_eq": <id>
|
||
# }
|
||
# }
|
||
|
||
query {
|
||
authors (where: $whereBoolExp) {
|
||
id
|
||
name
|
||
}
|
||
}
|
||
|
||
(*) - Supported only in Cloud and Enterprise editions only
|
||
|
||
**websocket-log** structure
|
||
^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
||
This is how the Websocket logs look like:
|
||
|
||
- On successful operation start:
|
||
|
||
.. code-block:: json
|
||
|
||
{
|
||
"timestamp": "2019-06-10T10:52:54.247+0530",
|
||
"level": "info",
|
||
"type": "websocket-log",
|
||
"detail": {
|
||
"event": {
|
||
"type": "operation",
|
||
"detail": {
|
||
"request_id": "d2ede87d-5cb7-44b6-8736-1d898117722a",
|
||
"operation_id": "1",
|
||
"query": {
|
||
"variables": {},
|
||
"query": "subscription {\n author {\n name\n }\n}\n"
|
||
},
|
||
"operation_type": {
|
||
"type": "started"
|
||
},
|
||
"operation_name": null
|
||
}
|
||
},
|
||
"connection_info": {
|
||
"websocket_id": "f590dd18-75db-4602-8693-8150239df7f7",
|
||
"jwt_expiry": null,
|
||
"msg": null
|
||
},
|
||
"user_vars": {
|
||
"x-hasura-role": "admin"
|
||
}
|
||
}
|
||
}
|
||
|
||
- On operation stop:
|
||
|
||
.. code-block:: json
|
||
|
||
{
|
||
"timestamp": "2019-06-10T11:01:40.939+0530",
|
||
"level": "info",
|
||
"type": "websocket-log",
|
||
"detail": {
|
||
"event": {
|
||
"type": "operation",
|
||
"detail": {
|
||
"request_id": null,
|
||
"operation_id": "1",
|
||
"query": null,
|
||
"operation_type": {
|
||
"type": "stopped"
|
||
},
|
||
"operation_name": null
|
||
}
|
||
},
|
||
"connection_info": {
|
||
"websocket_id": "7f782190-fd58-4305-a83f-8e17177b204e",
|
||
"jwt_expiry": null,
|
||
"msg": null
|
||
},
|
||
"user_vars": {
|
||
"x-hasura-role": "admin"
|
||
}
|
||
}
|
||
}
|
||
|
||
- On error:
|
||
|
||
.. code-block:: json
|
||
|
||
{
|
||
"timestamp": "2019-06-10T10:55:20.650+0530",
|
||
"level": "error",
|
||
"type": "websocket-log",
|
||
"detail": {
|
||
"event": {
|
||
"type": "operation",
|
||
"detail": {
|
||
"request_id": "150e3e6a-e1a7-46ba-a9d4-da6b192a4005",
|
||
"operation_id": "1",
|
||
"query": {
|
||
"variables": {},
|
||
"query": "subscription {\n author {\n namex\n }\n}\n"
|
||
},
|
||
"operation_type": {
|
||
"type": "query_err",
|
||
"detail": {
|
||
"path": "$.selectionSet.author.selectionSet.namex",
|
||
"error": "field \"namex\" not found in type: 'author'",
|
||
"code": "validation-failed"
|
||
}
|
||
},
|
||
"operation_name": null
|
||
}
|
||
},
|
||
"connection_info": {
|
||
"websocket_id": "49932ddf-e54d-42c6-bffb-8a57a1c6dcbe",
|
||
"jwt_expiry": null,
|
||
"msg": null
|
||
},
|
||
"user_vars": {
|
||
"x-hasura-role": "admin"
|
||
}
|
||
}
|
||
}
|
||
|
||
Monitoring frameworks
|
||
---------------------
|
||
|
||
You can integrate the logs emitted by Hasura GraphQL with external monitoring tools for better visibility as per
|
||
your convenience.
|
||
|
||
For some examples, see :ref:`guides_monitoring_frameworks`.
|