So i got a test to make a Flutter app that sent requests to a rest API. But the api requires a signature that is generated by combining path + request body + timestamp + token then encrypt
it with HMAC-Sha256
and then base64-encoded
it.
The problem is there is one POST
endpoint that needs a file. But, there is no clear way how to get the raw string of the MultipartRequest
.
I tried to use the await request.finalize().toBytes()
and then wrapped it in utf.decode()
, i got and error FormatException: invalid utf-8
.
And also I have tried to used FormData
class from dio
package. Still unclear how to get the raw string that being sent if it has file in it.
I tried the same thing i did which is use functions that is equivalent to get the bytes and utf8.decode
it. Still go the same error.
So does anyone know how to get the raw string by MultipartRequest
, FormData
or by any class that can send file over http?
Edit 08-12-2023:
Since people are requested to add the spec of the payload to generate the signature, here they are:
The payload consists of a path, verb, token, timestamp, and request body.
Example:
path=/ping&verb=GET&token=Bearer R04XSUbnm1GXNmDiXx9ysWMpFWBr××tamp=2019-01-02T13:14:15.678Z&body=
Detail every element of the payload:
- Path
The value in the path is the URL after the hostname, port, and /api
(if any) without Query Parameters.
example:
from {URL}/api/ping?version=1
becomes /ping
- Verb
is HTTP methods that are in uppercase.
example:
GET, POST, PUT, PATCH, and DELETE.
- Token
The token that is used for the Authorization header. (starts with 'Bearer').
example:
Bearer R04XSUbnm1GXNmDiXx9ysWMpFWBr
Note: Don't forget to add the Bearer before the Token
- Timestamp
Timestamp when you call the API. The Timestamp format must follow the ISO8601 format (yyyy-MM-ddTHH:mm:ss.SSSZ) and must be in zero UTC offset.
example:
2022-11-11T07:13:06.648Z
Note: Timestamp must be in UTC or GMT +0 timezone format.
- body
Request body sent for API Call.
Example:
&body={"hello":"world"} If there is no request body, such as a GET hit, then leave blank. &body=
Note: Be aware that you must enter the exact body you sent. Because differences in letters, spaces, and lines can cause differences in the signature.
It's actually really easy to get the body of a multipart request, using the same code that the real multipart request does. (See http's multipart_request.dart.)
Unfortunately, you need to copy some boilerplate into your own code:
Which you now use like this:
Hold onto those bytes and content type. Use the bytes in your hash and the bytes and content type in your actual request. You must set the content type of your actual request with:
as that has the randomly generated boundary chars saved in it.
However - and this is a big one - the bytes will contain binary, non-printable chars if the file(s) you include are not just all ascii.
You didn't include all of the spec in your edited question. You are meant to minify and hash the body as follows:
Lower case is easy, hex encode is easy, sha-256 is easy, but what does
minify
do - and can it accept binary chars?