Compare commits

...

23 Commits

Author SHA1 Message Date
nivedin
7411e36880 chore: remove readonly type 2024-04-24 21:13:58 +05:30
nivedin
05ad84f372 chore: add fallback url and add reqIndex for graphql saving 2024-04-24 21:13:55 +05:30
nivedin
0c993d0e90 chore: add i18n and readonly to envinput 2024-04-24 21:12:28 +05:30
nivedin
10cb900bd7 fix: revert text update in envinput for empty state 2024-04-24 21:12:28 +05:30
nivedin
8fd6b2ffb0 chore: update default tab to have empty endpoint 2024-04-24 21:12:28 +05:30
nivedin
e6e300ca86 chore: add fallback url if endpoint is empty while saving and sharing req 2024-04-24 21:12:28 +05:30
nivedin
5bac6222a0 chore: update URL input 2024-04-24 21:12:28 +05:30
nivedin
cf37fbd610 feat: hover placeholder transition for smartenvinput 2024-04-24 21:12:27 +05:30
Andrew Bastin
844eee0fa4 chore: update @hoppscotch/cli README 2024-04-22 21:12:27 +05:30
Andrew Bastin
d21bb65511 chore: bump cli version to 0.8 2024-04-22 20:29:03 +05:30
Andrew Bastin
4f614f7257 chore: bump version to 2024.3.1 2024-04-22 20:26:25 +05:30
Nivedin
0e2887b4e9 feat: first time user spotlight animation (#3977) 2024-04-22 12:21:30 +05:30
Andrew Bastin
18652ce400 fix: update prod.Dockerfile to add additional required deps during base build 2024-04-20 01:54:34 +05:30
Eduardo San Martin Morote
08c655235d fix: use pressed key rather than its code (#3978)
Co-authored-by: Joel Jacob Stephen <70131076+JoelJacobStephen@users.noreply.github.com>
2024-04-19 22:35:13 +05:30
Vincent M
51412549e8 chore(i18n): french lang translation (#3986) 2024-04-19 21:21:15 +05:30
James George
22c6eabd13 chore: migrate Node.js implementation for js-sandbox to isolated-vm (#3973)
Co-authored-by: Andrew Bastin <andrewbastin.k@gmail.com>
2024-04-19 21:08:46 +05:30
Mir Arif Hasan
a079e0f04b refactor: infra-config code refactor (#3982)
* chore: add getMissingInfraConfigEntries to avoid code duplication

* feat: isServiceConfigured modified

* docs: add missing function doc

* feat: argon2 version updated and removed types-argon2 depricated package

* Revert "feat: argon2 version updated and removed types-argon2 depricated package"

This reverts commit b99f3c5aae.

* Revert "feat: isServiceConfigured modified"

This reverts commit eaa6f105bb.
2024-04-19 12:43:43 +05:30
jamesgeorge007
375d53263a test: fix failing tests 2024-04-16 23:55:07 +05:30
Thomas Bonnet
57ef3e085f chore(i18n): Updating the packages/hoppscotch-common/locales/fr.json file. (#3555) 2024-04-16 18:15:38 +05:30
Timotej
9fb6e59e36 fix(desktop): set window caption color if Windows version >= 11 (#3939) 2024-04-16 17:51:26 +05:30
Sawako
1b0802b0e6 fix: spanish lang translation messages (#3950) 2024-04-16 17:47:10 +05:30
krisztianbarat
fb45fe4627 chore(i18n): update locale hu (#3875) 2024-04-16 17:45:55 +05:30
Akash K
0f592d1789 fix: use proper values for addTo field when auth type is api-key (#3966) 2024-04-16 17:44:28 +05:30
104 changed files with 17460 additions and 14041 deletions

View File

@@ -17,22 +17,21 @@ jobs:
steps:
- name: Checkout
uses: actions/checkout@v3
uses: actions/checkout@v4
- name: Setup environment
run: mv .env.example .env
- name: Setup node
uses: actions/setup-node@v4
with:
node-version: ${{ matrix.node-version }}
- name: Setup pnpm
uses: pnpm/action-setup@v2.2.4
uses: pnpm/action-setup@v3
with:
version: 8
run_install: true
- name: Setup node
uses: actions/setup-node@v3
with:
node-version: ${{ matrix.node }}
cache: pnpm
- name: Run tests
run: pnpm test

View File

@@ -1,6 +1,6 @@
{
"name": "hoppscotch-backend",
"version": "2024.3.0",
"version": "2024.3.1",
"description": "",
"author": "",
"private": true,
@@ -121,4 +121,4 @@
"^src/(.*)$": "<rootDir>/$1"
}
}
}
}

View File

@@ -156,6 +156,25 @@ export async function getDefaultInfraConfigs(): Promise<
return infraConfigDefaultObjs;
}
/**
* Get the missing entries in the 'infra_config' table
* @returns Array of InfraConfig
*/
export async function getMissingInfraConfigEntries() {
const prisma = new PrismaService();
const [dbInfraConfigs, infraConfigDefaultObjs] = await Promise.all([
prisma.infraConfig.findMany(),
getDefaultInfraConfigs(),
]);
const missingEntries = infraConfigDefaultObjs.filter(
(config) =>
!dbInfraConfigs.some((dbConfig) => dbConfig.name === config.name),
);
return missingEntries;
}
/**
* Verify if 'infra_config' table is loaded with all entries
* @returns boolean
@@ -163,12 +182,7 @@ export async function getDefaultInfraConfigs(): Promise<
export async function isInfraConfigTablePopulated(): Promise<boolean> {
const prisma = new PrismaService();
try {
const dbInfraConfigs = await prisma.infraConfig.findMany();
const infraConfigDefaultObjs = await getDefaultInfraConfigs();
const propsRemainingToInsert = infraConfigDefaultObjs.filter(
(p) => !dbInfraConfigs.find((e) => e.name === p.name),
);
const propsRemainingToInsert = await getMissingInfraConfigEntries();
if (propsRemainingToInsert.length > 0) {
console.log(

View File

@@ -21,7 +21,12 @@ import {
validateUrl,
} from 'src/utils';
import { ConfigService } from '@nestjs/config';
import { ServiceStatus, getDefaultInfraConfigs, stopApp } from './helper';
import {
ServiceStatus,
getDefaultInfraConfigs,
getMissingInfraConfigEntries,
stopApp,
} from './helper';
import { EnableAndDisableSSOArgs, InfraConfigArgs } from './input-args';
import { AuthProvider } from 'src/auth/helper';
@@ -56,14 +61,7 @@ export class InfraConfigService implements OnModuleInit {
*/
async initializeInfraConfigTable() {
try {
// Fetch the default values (value in .env) for configs to be saved in 'infra_config' table
const infraConfigDefaultObjs = await getDefaultInfraConfigs();
// Eliminate the rows (from 'infraConfigDefaultObjs') that are already present in the database table
const dbInfraConfigs = await this.prisma.infraConfig.findMany();
const propsToInsert = infraConfigDefaultObjs.filter(
(p) => !dbInfraConfigs.find((e) => e.name === p.name),
);
const propsToInsert = await getMissingInfraConfigEntries();
if (propsToInsert.length > 0) {
await this.prisma.infraConfig.createMany({ data: propsToInsert });
@@ -285,6 +283,7 @@ export class InfraConfigService implements OnModuleInit {
/**
* Get InfraConfigs by names
* @param names Names of the InfraConfigs
* @param checkDisallowedKeys If true, check if the names are allowed to fetch by client
* @returns InfraConfig model
*/
async getMany(names: InfraConfigEnum[], checkDisallowedKeys: boolean = true) {

View File

@@ -52,11 +52,34 @@ hopp [options or commands] arguments
Taking the above example, `pw.env.get("ENV1")` will return `"value1"`
## Install
- Before you install Hoppscotch CLI you need to make sure you have the dependencies it requires to run.
- **Windows & macOS**: You will need `node-gyp` installed. Find instructions here: https://github.com/nodejs/node-gyp
- **Debian/Ubuntu derivatives**:
```sh
sudo apt-get install python g++ build-essential
```
- **Alpine Linux**:
```sh
sudo apk add python3 make g++
```
- **Amazon Linux (AMI)**
```sh
sudo yum install gcc72 gcc72-c++
```
- **Arch Linux**
```sh
sudo pacman -S make gcc python
```
- **RHEL/Fedora derivatives**:
```sh
sudo dnf install python3 make gcc gcc-c++ zlib-devel brotli-devel openssl-devel libuv-devel
```
Install [@hoppscotch/cli](https://www.npmjs.com/package/@hoppscotch/cli) from npm by running:
```
npm i -g @hoppscotch/cli
```
- Once the dependencies are installed, install [@hoppscotch/cli](https://www.npmjs.com/package/@hoppscotch/cli) from npm by running:
```
npm i -g @hoppscotch/cli
```
## **Developing:**

View File

@@ -1,6 +1,31 @@
#!/usr/bin/env node
// * The entry point of the CLI
// @ts-check
import { cli } from "../dist/index.js";
cli(process.argv);
import { spawnSync } from "child_process";
import { cloneDeep } from "lodash-es";
const nodeVersion = parseInt(process.versions.node.split(".")[0]);
// As per isolated-vm documentation, we need to supply `--no-node-snapshot` for node >= 20
// src: https://github.com/laverdet/isolated-vm?tab=readme-ov-file#requirements
if (nodeVersion >= 20 && !process.execArgv.includes("--no-node-snapshot")) {
const argCopy = cloneDeep(process.argv);
// Replace first argument with --no-node-snapshot
// We can get argv[0] from process.argv0
argCopy[0] = "--no-node-snapshot";
const result = spawnSync(
process.argv0,
argCopy,
{ stdio: "inherit" }
);
// Exit with the same status code as the spawned process
process.exit(result.status ?? 0);
} else {
cli(process.argv);
}

View File

@@ -1,6 +1,6 @@
{
"name": "@hoppscotch/cli",
"version": "0.7.0",
"version": "0.8.0",
"description": "A CLI to run Hoppscotch test scripts in CI environments.",
"homepage": "https://hoppscotch.io",
"type": "module",
@@ -44,6 +44,7 @@
"axios": "1.6.7",
"chalk": "5.3.0",
"commander": "11.1.0",
"isolated-vm": "4.7.2",
"lodash-es": "4.17.21",
"qs": "6.11.2",
"verzod": "0.2.2",
@@ -65,4 +66,4 @@
"tsup": "8.0.2",
"typescript": "5.3.3"
}
}
}

View File

@@ -224,7 +224,7 @@ describe("Test `hopp test <file> --env <file>` command:", () => {
});
describe("Secret environment variables", () => {
jest.setTimeout(10000);
jest.setTimeout(100000);
// Reads secret environment values from system environment
test("Successfully picks the values for secret environment variables from `process.env` and persists the variables set from the pre-request script", async () => {

View File

@@ -1,27 +1,55 @@
{
"v": 1,
"name": "coll-v1",
"folders": [],
"requests": [
"v": 1,
"name": "coll-v1",
"folders": [
{
"v": 1,
"name": "coll-v1-child",
"folders": [],
"requests": [
{
"url": "https://httpbin.org",
"path": "/get",
"headers": [
{ "key": "Inactive-Header", "value": "Inactive Header", "active": false },
{ "key": "Authorization", "value": "Bearer token123", "active": true }
],
"params": [
{ "key": "key", "value": "value", "active": true },
{ "key": "inactive-key", "value": "inactive-param", "active": false }
],
"name": "req-v0",
"method": "GET",
"preRequestScript": "",
"testScript": "pw.test(\"Asserts request params\", () => {\n pw.expect(pw.response.body.args.key).toBe(\"value\")\n pw.expect(pw.response.body.args[\"inactive-key\"]).toBe(undefined)\n})\n\npw.test(\"Asserts request headers\", () => {\n pw.expect(pw.response.body.headers[\"Authorization\"]).toBe(\"Bearer token123\")\n pw.expect(pw.response.body.headers[\"Inactive-Header\"]).toBe(undefined)\n})",
"contentType": "application/json",
"body": "",
"auth": "Bearer Token",
"bearerToken": "token123"
}
]
}
"url": "https://echo.hoppscotch.io",
"path": "/get",
"headers": [
{ "key": "Inactive-Header", "value": "Inactive Header", "active": false },
{ "key": "Authorization", "value": "Bearer token123", "active": true }
],
"params": [
{ "key": "key", "value": "value", "active": true },
{ "key": "inactive-key", "value": "inactive-param", "active": false }
],
"name": "req-v0-II",
"method": "GET",
"preRequestScript": "",
"testScript": "pw.test(\"Asserts request params\", () => {\n pw.expect(pw.response.body.args.key).toBe(\"value\")\n pw.expect(pw.response.body.args[\"inactive-key\"]).toBe(undefined)\n})\n\npw.test(\"Asserts request headers\", () => {\n pw.expect(pw.response.body.headers[\"authorization\"]).toBe(\"Bearer token123\")\n pw.expect(pw.response.body.headers[\"inactive-header\"]).toBe(undefined)\n})",
"contentType": "application/json",
"body": "",
"auth": "Bearer Token",
"bearerToken": "token123"
}
]
}
],
"requests": [
{
"url": "https://echo.hoppscotch.io",
"path": "/get",
"headers": [
{ "key": "Inactive-Header", "value": "Inactive Header", "active": false },
{ "key": "Authorization", "value": "Bearer token123", "active": true }
],
"params": [
{ "key": "key", "value": "value", "active": true },
{ "key": "inactive-key", "value": "inactive-param", "active": false }
],
"name": "req-v0",
"method": "GET",
"preRequestScript": "",
"testScript": "pw.test(\"Asserts request params\", () => {\n pw.expect(pw.response.body.args.key).toBe(\"value\")\n pw.expect(pw.response.body.args[\"inactive-key\"]).toBe(undefined)\n})\n\npw.test(\"Asserts request headers\", () => {\n pw.expect(pw.response.body.headers[\"authorization\"]).toBe(\"Bearer token123\")\n pw.expect(pw.response.body.headers[\"inactive-header\"]).toBe(undefined)\n})",
"contentType": "application/json",
"body": "",
"auth": "Bearer Token",
"bearerToken": "token123"
}
]
}

View File

@@ -1,11 +1,60 @@
{
"v": 1,
"name": "coll-v1",
"folders": [],
"folders": [
{
"v": 1,
"name": "coll-v1-child",
"folders": [],
"requests": [
{
"v": "1",
"endpoint": "https://echo.hoppscotch.io",
"headers": [
{
"key": "Inactive-Header",
"value": "Inactive Header",
"active": false
},
{
"key": "Authorization",
"value": "Bearer token123",
"active": true
}
],
"params": [
{
"key": "key",
"value": "value",
"active": true
},
{
"key": "inactive-key",
"value": "inactive-param",
"active": false
}
],
"name": "req-v1-II",
"method": "GET",
"preRequestScript": "",
"testScript": "pw.test(\"Asserts request params\", () => {\n pw.expect(pw.response.body.args.key).toBe(\"value\")\n pw.expect(pw.response.body.args[\"inactive-key\"]).toBe(undefined)\n})\n\npw.test(\"Asserts request headers\", () => {\n pw.expect(pw.response.body.headers[\"authorization\"]).toBe(\"Bearer token123\")\n pw.expect(pw.response.body.headers[\"inactive-header\"]).toBe(undefined)\n})",
"body": {
"contentType": null,
"body": null
},
"auth": {
"authType": "bearer",
"authActive": true,
"token": "token123"
}
}
]
}
],
"requests": [
{
"v": "1",
"endpoint": "https://httpbin.org/get",
"endpoint": "https://echo.hoppscotch.io",
"headers": [
{
"key": "Inactive-Header",
@@ -33,7 +82,7 @@
"name": "req-v1",
"method": "GET",
"preRequestScript": "",
"testScript": "pw.test(\"Asserts request params\", () => {\n pw.expect(pw.response.body.args.key).toBe(\"value\")\n pw.expect(pw.response.body.args[\"inactive-key\"]).toBe(undefined)\n})\n\npw.test(\"Asserts request headers\", () => {\n pw.expect(pw.response.body.headers[\"Authorization\"]).toBe(\"Bearer token123\")\n pw.expect(pw.response.body.headers[\"Inactive-Header\"]).toBe(undefined)\n})",
"testScript": "pw.test(\"Asserts request params\", () => {\n pw.expect(pw.response.body.args.key).toBe(\"value\")\n pw.expect(pw.response.body.args[\"inactive-key\"]).toBe(undefined)\n})\n\npw.test(\"Asserts request headers\", () => {\n pw.expect(pw.response.body.headers[\"authorization\"]).toBe(\"Bearer token123\")\n pw.expect(pw.response.body.headers[\"inactive-header\"]).toBe(undefined)\n})",
"body": {
"contentType": null,
"body": null

View File

@@ -1,11 +1,66 @@
{
"v": 2,
"name": "coll-v2",
"folders": [],
"folders": [
{
"v": 2,
"name": "coll-v2-child",
"folders": [],
"requests": [
{
"v": "2",
"endpoint": "https://echo.hoppscotch.io",
"headers": [
{
"key": "Inactive-Header",
"value": "Inactive Header",
"active": false
},
{
"key": "Authorization",
"value": "Bearer token123",
"active": true
}
],
"params": [
{
"key": "key",
"value": "value",
"active": true
},
{
"key": "inactive-key",
"value": "inactive-param",
"active": false
}
],
"name": "req-v2-II",
"method": "GET",
"preRequestScript": "",
"testScript": "pw.test(\"Asserts request params\", () => {\n pw.expect(pw.response.body.args.key).toBe(\"value\")\n pw.expect(pw.response.body.args[\"inactive-key\"]).toBe(undefined)\n})\n\npw.test(\"Asserts request headers\", () => {\n pw.expect(pw.response.body.headers[\"authorization\"]).toBe(\"Bearer token123\")\n pw.expect(pw.response.body.headers[\"inactive-header\"]).toBe(undefined)\n})",
"body": {
"contentType": null,
"body": null
},
"auth": {
"authType": "bearer",
"authActive": true,
"token": "token123"
},
"requestVariables": []
}
],
"auth": {
"authType": "inherit",
"authActive": true
},
"headers": []
}
],
"requests": [
{
"v": "2",
"endpoint": "https://httpbin.org/get",
"endpoint": "https://echo.hoppscotch.io",
"headers": [
{
"key": "Inactive-Header",
@@ -33,7 +88,7 @@
"name": "req-v2",
"method": "GET",
"preRequestScript": "",
"testScript": "pw.test(\"Asserts request params\", () => {\n pw.expect(pw.response.body.args.key).toBe(\"value\")\n pw.expect(pw.response.body.args[\"inactive-key\"]).toBe(undefined)\n})\n\npw.test(\"Asserts request headers\", () => {\n pw.expect(pw.response.body.headers[\"Authorization\"]).toBe(\"Bearer token123\")\n pw.expect(pw.response.body.headers[\"Inactive-Header\"]).toBe(undefined)\n})",
"testScript": "pw.test(\"Asserts request params\", () => {\n pw.expect(pw.response.body.args.key).toBe(\"value\")\n pw.expect(pw.response.body.args[\"inactive-key\"]).toBe(undefined)\n})\n\npw.test(\"Asserts request headers\", () => {\n pw.expect(pw.response.body.headers[\"authorization\"]).toBe(\"Bearer token123\")\n pw.expect(pw.response.body.headers[\"inactive-header\"]).toBe(undefined)\n})",
"body": {
"contentType": null,
"body": null

View File

@@ -1,11 +1,66 @@
{
"v": 2,
"name": "coll-v2",
"folders": [],
"folders": [
{
"v": 2,
"name": "coll-v2-child",
"folders": [],
"requests": [
{
"v": "3",
"endpoint": "https://echo.hoppscotch.io",
"headers": [
{
"key": "Inactive-Header",
"value": "Inactive Header",
"active": false
},
{
"key": "Authorization",
"value": "Bearer token123",
"active": true
}
],
"params": [
{
"key": "key",
"value": "value",
"active": true
},
{
"key": "inactive-key",
"value": "inactive-param",
"active": false
}
],
"name": "req-v3-II",
"method": "GET",
"preRequestScript": "",
"testScript": "pw.test(\"Asserts request params\", () => {\n pw.expect(pw.response.body.args.key).toBe(\"value\")\n pw.expect(pw.response.body.args[\"inactive-key\"]).toBe(undefined)\n})\n\npw.test(\"Asserts request headers\", () => {\n pw.expect(pw.response.body.headers[\"authorization\"]).toBe(\"Bearer token123\")\n pw.expect(pw.response.body.headers[\"inactive-header\"]).toBe(undefined)\n})",
"body": {
"contentType": null,
"body": null
},
"auth": {
"authType": "bearer",
"authActive": true,
"token": "token123"
},
"requestVariables": []
}
],
"auth": {
"authType": "inherit",
"authActive": true
},
"headers": []
}
],
"requests": [
{
"v": "3",
"endpoint": "https://httpbin.org/get",
"endpoint": "https://echo.hoppscotch.io",
"headers": [
{
"key": "Inactive-Header",
@@ -33,7 +88,7 @@
"name": "req-v3",
"method": "GET",
"preRequestScript": "",
"testScript": "pw.test(\"Asserts request params\", () => {\n pw.expect(pw.response.body.args.key).toBe(\"value\")\n pw.expect(pw.response.body.args[\"inactive-key\"]).toBe(undefined)\n})\n\npw.test(\"Asserts request headers\", () => {\n pw.expect(pw.response.body.headers[\"Authorization\"]).toBe(\"Bearer token123\")\n pw.expect(pw.response.body.headers[\"Inactive-Header\"]).toBe(undefined)\n})",
"testScript": "pw.test(\"Asserts request params\", () => {\n pw.expect(pw.response.body.args.key).toBe(\"value\")\n pw.expect(pw.response.body.args[\"inactive-key\"]).toBe(undefined)\n})\n\npw.test(\"Asserts request headers\", () => {\n pw.expect(pw.response.body.headers[\"authorization\"]).toBe(\"Bearer token123\")\n pw.expect(pw.response.body.headers[\"inactive-header\"]).toBe(undefined)\n})",
"body": {
"contentType": null,
"body": null

View File

@@ -75,7 +75,7 @@
"auth": {
"authType": "api-key",
"authActive": true,
"addTo": "Headers",
"addTo": "HEADERS",
"key": "key",
"value": "test-key"
},

View File

@@ -5,8 +5,14 @@
"requests": [
{
"v": "3",
"auth": { "authType": "none", "authActive": true },
"body": { "body": null, "contentType": null },
"auth": {
"authType": "none",
"authActive": true
},
"body": {
"body": null,
"contentType": null
},
"name": "test-secret-headers",
"method": "GET",
"params": [],
@@ -18,13 +24,16 @@
}
],
"requestVariables": [],
"endpoint": "<<baseURL>>/headers",
"testScript": "pw.test(\"Successfully parses secret variable holding the header value\", () => {\n const secretHeaderValue = pw.env.get(\"secretHeaderValue\")\n pw.expect(secretHeaderValue).toBe(\"secret-header-value\")\n \n if (secretHeaderValue) {\n pw.expect(pw.response.body.headers[\"Secret-Header-Key\"]).toBe(secretHeaderValue)\n }\n\n pw.expect(pw.env.get(\"secretHeaderValueFromPreReqScript\")).toBe(\"secret-header-value\")\n})",
"endpoint": "<<echoHoppBaseURL>>/headers",
"testScript": "pw.test(\"Successfully parses secret variable holding the header value\", () => {\n const secretHeaderValue = pw.env.get(\"secretHeaderValue\")\n pw.expect(secretHeaderValue).toBe(\"secret-header-value\")\n \n if (secretHeaderValue) {\n pw.expect(pw.response.body.headers[\"secret-header-key\"]).toBe(secretHeaderValue)\n }\n\n pw.expect(pw.env.get(\"secretHeaderValueFromPreReqScript\")).toBe(\"secret-header-value\")\n})",
"preRequestScript": "const secretHeaderValueFromPreReqScript = pw.env.get(\"secretHeaderValue\")\npw.env.set(\"secretHeaderValueFromPreReqScript\", secretHeaderValueFromPreReqScript)"
},
{
"v": "3",
"auth": { "authType": "none", "authActive": true },
"auth": {
"authType": "none",
"authActive": true
},
"body": {
"body": "{\n \"secretBodyKey\": \"<<secretBodyValue>>\"\n}",
"contentType": "application/json"
@@ -34,14 +43,20 @@
"params": [],
"headers": [],
"requestVariables": [],
"endpoint": "<<baseURL>>/post",
"testScript": "pw.test(\"Successfully parses secret variable holding the request body value\", () => {\n const secretBodyValue = pw.env.get(\"secretBodyValue\")\n pw.expect(secretBodyValue).toBe(\"secret-body-value\")\n \n if (secretBodyValue) {\n pw.expect(pw.response.body.json.secretBodyKey).toBe(secretBodyValue)\n }\n\n pw.expect(pw.env.get(\"secretBodyValueFromPreReqScript\")).toBe(\"secret-body-value\")\n})",
"endpoint": "<<echoHoppBaseURL>>/post",
"testScript": "pw.test(\"Successfully parses secret variable holding the request body value\", () => {\n const secretBodyValue = pw.env.get(\"secretBodyValue\")\n pw.expect(secretBodyValue).toBe(\"secret-body-value\")\n \n if (secretBodyValue) {\n pw.expect(JSON.parse(pw.response.body.data).secretBodyKey).toBe(secretBodyValue)\n }\n\n pw.expect(pw.env.get(\"secretBodyValueFromPreReqScript\")).toBe(\"secret-body-value\")\n})",
"preRequestScript": "const secretBodyValueFromPreReqScript = pw.env.get(\"secretBodyValue\")\npw.env.set(\"secretBodyValueFromPreReqScript\", secretBodyValueFromPreReqScript)"
},
{
"v": "3",
"auth": { "authType": "none", "authActive": true },
"body": { "body": null, "contentType": null },
"auth": {
"authType": "none",
"authActive": true
},
"body": {
"body": null,
"contentType": null
},
"name": "test-secret-query-params",
"method": "GET",
"params": [
@@ -53,7 +68,7 @@
],
"headers": [],
"requestVariables": [],
"endpoint": "<<baseURL>>/get",
"endpoint": "<<echoHoppBaseURL>>",
"testScript": "pw.test(\"Successfully parses secret variable holding the query param value\", () => {\n const secretQueryParamValue = pw.env.get(\"secretQueryParamValue\")\n pw.expect(secretQueryParamValue).toBe(\"secret-query-param-value\")\n \n if (secretQueryParamValue) {\n pw.expect(pw.response.body.args.secretQueryParamKey).toBe(secretQueryParamValue)\n }\n\n pw.expect(pw.env.get(\"secretQueryParamValueFromPreReqScript\")).toBe(\"secret-query-param-value\")\n})",
"preRequestScript": "const secretQueryParamValueFromPreReqScript = pw.env.get(\"secretQueryParamValue\")\npw.env.set(\"secretQueryParamValueFromPreReqScript\", secretQueryParamValueFromPreReqScript)"
},
@@ -65,14 +80,17 @@
"username": "<<secretBasicAuthUsername>>",
"authActive": true
},
"body": { "body": null, "contentType": null },
"body": {
"body": null,
"contentType": null
},
"name": "test-secret-basic-auth",
"method": "GET",
"params": [],
"headers": [],
"requestVariables": [],
"endpoint": "<<baseURL>>/basic-auth/<<secretBasicAuthUsername>>/<<secretBasicAuthPassword>>",
"testScript": "pw.test(\"Successfully parses secret variables holding basic auth credentials\", () => {\n\tconst secretBasicAuthUsername = pw.env.get(\"secretBasicAuthUsername\")\n \tconst secretBasicAuthPassword = pw.env.get(\"secretBasicAuthPassword\")\n\n pw.expect(secretBasicAuthUsername).toBe(\"test-user\")\n pw.expect(secretBasicAuthPassword).toBe(\"test-pass\")\n\n if (secretBasicAuthUsername && secretBasicAuthPassword) {\n const { authenticated, user } = pw.response.body\n pw.expect(authenticated).toBe(true)\n pw.expect(user).toBe(secretBasicAuthUsername)\n }\n});",
"endpoint": "<<httpbinBaseURL>>/basic-auth/<<secretBasicAuthUsername>>/<<secretBasicAuthPassword>>",
"testScript": "pw.test(\"Successfully parses secret variables holding basic auth credentials\", () => {\n\tconst secretBasicAuthUsername = pw.env.get(\"secretBasicAuthUsername\")\n \tconst secretBasicAuthPassword = pw.env.get(\"secretBasicAuthPassword\")\n\n pw.expect(secretBasicAuthUsername).toBe(\"test-user\")\n pw.expect(secretBasicAuthPassword).toBe(\"test-pass\")\n\n // The endpoint at times results in a `502` bad gateway\n if (pw.response.status !== 200) {\n return\n }\n\n if (secretBasicAuthUsername && secretBasicAuthPassword) {\n const { authenticated, user } = pw.response.body\n pw.expect(authenticated).toBe(true)\n pw.expect(user).toBe(secretBasicAuthUsername)\n }\n});",
"preRequestScript": ""
},
{
@@ -84,30 +102,42 @@
"username": "testuser",
"authActive": true
},
"body": { "body": null, "contentType": null },
"body": {
"body": null,
"contentType": null
},
"name": "test-secret-bearer-auth",
"method": "GET",
"params": [],
"headers": [],
"requestVariables": [],
"endpoint": "<<baseURL>>/bearer",
"testScript": "pw.test(\"Successfully parses secret variable holding the bearer token\", () => {\n const secretBearerToken = pw.env.get(\"secretBearerToken\")\n const preReqSecretBearerToken = pw.env.get(\"preReqSecretBearerToken\")\n\n pw.expect(secretBearerToken).toBe(\"test-token\")\n\n if (secretBearerToken) { \n pw.expect(pw.response.body.token).toBe(secretBearerToken)\n pw.expect(preReqSecretBearerToken).toBe(\"test-token\")\n }\n});",
"endpoint": "<<httpbinBaseURL>>/bearer",
"testScript": "pw.test(\"Successfully parses secret variable holding the bearer token\", () => {\n const secretBearerToken = pw.env.get(\"secretBearerToken\")\n const preReqSecretBearerToken = pw.env.get(\"preReqSecretBearerToken\")\n\n pw.expect(secretBearerToken).toBe(\"test-token\")\n\n // Safeguard to prevent test failures due to the endpoint\n if (pw.response.status !== 200) {\n return\n }\n\n if (secretBearerToken) { \n pw.expect(pw.response.body.token).toBe(secretBearerToken)\n pw.expect(preReqSecretBearerToken).toBe(\"test-token\")\n }\n});",
"preRequestScript": "const secretBearerToken = pw.env.get(\"secretBearerToken\")\npw.env.set(\"preReqSecretBearerToken\", secretBearerToken)"
},
{
"v": "3",
"auth": { "authType": "none", "authActive": true },
"body": { "body": null, "contentType": null },
"auth": {
"authType": "none",
"authActive": true
},
"body": {
"body": null,
"contentType": null
},
"name": "test-secret-fallback",
"method": "GET",
"params": [],
"headers": [],
"requestVariables": [],
"endpoint": "<<baseURL>>",
"endpoint": "<<echoHoppBaseURL>>",
"testScript": "pw.test(\"Returns an empty string if the value for a secret environment variable is not found in the system environment\", () => {\n pw.expect(pw.env.get(\"nonExistentValueInSystemEnv\")).toBe(\"\")\n})",
"preRequestScript": ""
}
],
"auth": { "authType": "inherit", "authActive": false },
"auth": {
"authType": "inherit",
"authActive": false
},
"headers": []
}
}

View File

@@ -1,6 +1,6 @@
{
"v": 2,
"name": "secret-envs-setters-coll",
"name": "secret-envs-persistence-coll",
"folders": [],
"requests": [
{
@@ -24,8 +24,8 @@
"active": true
}
],
"endpoint": "<<baseURL>>/headers",
"testScript": "pw.test(\"Successfully parses secret variable holding the header value\", () => {\n const secretHeaderValue = pw.env.getResolve(\"secretHeaderValue\")\n pw.expect(secretHeaderValue).toBe(\"secret-header-value\")\n \n if (secretHeaderValue) {\n pw.expect(pw.response.body.headers[\"Secret-Header-Key\"]).toBe(secretHeaderValue)\n }\n\n pw.expect(pw.env.getResolve(\"secretHeaderValueFromPreReqScript\")).toBe(\"secret-header-value\")\n})",
"endpoint": "<<echoHoppBaseURL>>",
"testScript": "pw.test(\"Successfully parses secret variable holding the header value\", () => {\n const secretHeaderValue = pw.env.getResolve(\"secretHeaderValue\")\n pw.expect(secretHeaderValue).toBe(\"secret-header-value\")\n \n if (secretHeaderValue) {\n pw.expect(pw.response.body.headers[\"secret-header-key\"]).toBe(secretHeaderValue)\n }\n\n pw.expect(pw.env.getResolve(\"secretHeaderValueFromPreReqScript\")).toBe(\"secret-header-value\")\n})",
"preRequestScript": "pw.env.set(\"secretHeaderValue\", \"secret-header-value\")\n\nconst secretHeaderValueFromPreReqScript = pw.env.getResolve(\"secretHeaderValue\")\npw.env.set(\"secretHeaderValueFromPreReqScript\", secretHeaderValueFromPreReqScript)"
},
{
@@ -49,8 +49,8 @@
"active": true
}
],
"endpoint": "<<baseURL>>/headers",
"testScript": "pw.test(\"Value set at the pre-request script takes precedence\", () => {\n const secretHeaderValue = pw.env.getResolve(\"secretHeaderValue\")\n pw.expect(secretHeaderValue).toBe(\"secret-header-value-overriden\")\n \n if (secretHeaderValue) {\n pw.expect(pw.response.body.headers[\"Secret-Header-Key\"]).toBe(secretHeaderValue)\n }\n\n pw.expect(pw.env.getResolve(\"secretHeaderValueFromPreReqScript\")).toBe(\"secret-header-value-overriden\")\n})",
"endpoint": "<<echoHoppBaseURL>>",
"testScript": "pw.test(\"Value set at the pre-request script takes precedence\", () => {\n const secretHeaderValue = pw.env.getResolve(\"secretHeaderValue\")\n pw.expect(secretHeaderValue).toBe(\"secret-header-value-overriden\")\n \n if (secretHeaderValue) {\n pw.expect(pw.response.body.headers[\"secret-header-key\"]).toBe(secretHeaderValue)\n }\n\n pw.expect(pw.env.getResolve(\"secretHeaderValueFromPreReqScript\")).toBe(\"secret-header-value-overriden\")\n})",
"preRequestScript": "pw.env.set(\"secretHeaderValue\", \"secret-header-value-overriden\")\n\nconst secretHeaderValueFromPreReqScript = pw.env.getResolve(\"secretHeaderValue\")\npw.env.set(\"secretHeaderValueFromPreReqScript\", secretHeaderValueFromPreReqScript)"
},
{
@@ -68,8 +68,8 @@
"params": [],
"requestVariables": [],
"headers": [],
"endpoint": "<<baseURL>>/post",
"testScript": "pw.test(\"Successfully parses secret variable holding the request body value\", () => {\n const secretBodyValue = pw.env.get(\"secretBodyValue\")\n pw.expect(secretBodyValue).toBe(\"secret-body-value\")\n \n if (secretBodyValue) {\n pw.expect(pw.response.body.json.secretBodyKey).toBe(secretBodyValue)\n }\n\n pw.expect(pw.env.get(\"secretBodyValueFromPreReqScript\")).toBe(\"secret-body-value\")\n})",
"endpoint": "<<echoHoppBaseURL>>/post",
"testScript": "pw.test(\"Successfully parses secret variable holding the request body value\", () => {\n const secretBodyValue = pw.env.get(\"secretBodyValue\")\n pw.expect(secretBodyValue).toBe(\"secret-body-value\")\n \n if (secretBodyValue) {\n pw.expect(JSON.parse(pw.response.body.data).secretBodyKey).toBe(secretBodyValue)\n }\n\n pw.expect(pw.env.get(\"secretBodyValueFromPreReqScript\")).toBe(\"secret-body-value\")\n})",
"preRequestScript": "const secretBodyValue = pw.env.get(\"secretBodyValue\")\n\nif (!secretBodyValue) { \n pw.env.set(\"secretBodyValue\", \"secret-body-value\")\n}\n\nconst secretBodyValueFromPreReqScript = pw.env.get(\"secretBodyValue\")\npw.env.set(\"secretBodyValueFromPreReqScript\", secretBodyValueFromPreReqScript)"
},
{
@@ -93,7 +93,7 @@
],
"requestVariables": [],
"headers": [],
"endpoint": "<<baseURL>>/get",
"endpoint": "<<echoHoppBaseURL>>",
"testScript": "pw.test(\"Successfully parses secret variable holding the query param value\", () => {\n const secretQueryParamValue = pw.env.get(\"secretQueryParamValue\")\n pw.expect(secretQueryParamValue).toBe(\"secret-query-param-value\")\n \n if (secretQueryParamValue) {\n pw.expect(pw.response.body.args.secretQueryParamKey).toBe(secretQueryParamValue)\n }\n\n pw.expect(pw.env.get(\"secretQueryParamValueFromPreReqScript\")).toBe(\"secret-query-param-value\")\n})",
"preRequestScript": "const secretQueryParamValue = pw.env.get(\"secretQueryParamValue\")\n\nif (!secretQueryParamValue) {\n pw.env.set(\"secretQueryParamValue\", \"secret-query-param-value\")\n}\n\nconst secretQueryParamValueFromPreReqScript = pw.env.get(\"secretQueryParamValue\")\npw.env.set(\"secretQueryParamValueFromPreReqScript\", secretQueryParamValueFromPreReqScript)"
},
@@ -114,8 +114,8 @@
"params": [],
"requestVariables": [],
"headers": [],
"endpoint": "<<baseURL>>/basic-auth/<<secretBasicAuthUsername>>/<<secretBasicAuthPassword>>",
"testScript": "pw.test(\"Successfully parses secret variables holding basic auth credentials\", () => {\n\tconst secretBasicAuthUsername = pw.env.get(\"secretBasicAuthUsername\")\n \tconst secretBasicAuthPassword = pw.env.get(\"secretBasicAuthPassword\")\n\n pw.expect(secretBasicAuthUsername).toBe(\"test-user\")\n pw.expect(secretBasicAuthPassword).toBe(\"test-pass\")\n\n if (secretBasicAuthUsername && secretBasicAuthPassword) {\n const { authenticated, user } = pw.response.body\n pw.expect(authenticated).toBe(true)\n pw.expect(user).toBe(secretBasicAuthUsername)\n }\n});",
"endpoint": "<<httpbinBaseURL>>/basic-auth/<<secretBasicAuthUsername>>/<<secretBasicAuthPassword>>",
"testScript": "pw.test(\"Successfully parses secret variables holding basic auth credentials\", () => {\n\tconst secretBasicAuthUsername = pw.env.get(\"secretBasicAuthUsername\")\n \tconst secretBasicAuthPassword = pw.env.get(\"secretBasicAuthPassword\")\n\n pw.expect(secretBasicAuthUsername).toBe(\"test-user\")\n pw.expect(secretBasicAuthPassword).toBe(\"test-pass\")\n\n // The endpoint at times results in a `502` bad gateway\n if (pw.response.status !== 200) {\n return\n }\n\n if (secretBasicAuthUsername && secretBasicAuthPassword) {\n const { authenticated, user } = pw.response.body\n pw.expect(authenticated).toBe(true)\n pw.expect(user).toBe(secretBasicAuthUsername)\n }\n});",
"preRequestScript": "let secretBasicAuthUsername = pw.env.get(\"secretBasicAuthUsername\")\n\nlet secretBasicAuthPassword = pw.env.get(\"secretBasicAuthPassword\")\n\nif (!secretBasicAuthUsername) {\n pw.env.set(\"secretBasicAuthUsername\", \"test-user\")\n}\n\nif (!secretBasicAuthPassword) {\n pw.env.set(\"secretBasicAuthPassword\", \"test-pass\")\n}"
},
{
@@ -136,8 +136,8 @@
"params": [],
"requestVariables": [],
"headers": [],
"endpoint": "<<baseURL>>/bearer",
"testScript": "pw.test(\"Successfully parses secret variable holding the bearer token\", () => {\n const secretBearerToken = pw.env.resolve(\"<<secretBearerToken>>\")\n const preReqSecretBearerToken = pw.env.resolve(\"<<preReqSecretBearerToken>>\")\n\n pw.expect(secretBearerToken).toBe(\"test-token\")\n\n if (secretBearerToken) { \n pw.expect(pw.response.body.token).toBe(secretBearerToken)\n pw.expect(preReqSecretBearerToken).toBe(\"test-token\")\n }\n});",
"endpoint": "<<httpbinBaseURL>>/bearer",
"testScript": "pw.test(\"Successfully parses secret variable holding the bearer token\", () => {\n const secretBearerToken = pw.env.resolve(\"<<secretBearerToken>>\")\n const preReqSecretBearerToken = pw.env.resolve(\"<<preReqSecretBearerToken>>\")\n\n pw.expect(secretBearerToken).toBe(\"test-token\")\n\n // Safeguard to prevent test failures due to the endpoint\n if (pw.response.status !== 200) {\n return\n }\n\n if (secretBearerToken) { \n pw.expect(pw.response.body.token).toBe(secretBearerToken)\n pw.expect(preReqSecretBearerToken).toBe(\"test-token\")\n }\n});",
"preRequestScript": "let secretBearerToken = pw.env.resolve(\"<<secretBearerToken>>\")\n\nif (!secretBearerToken) {\n pw.env.set(\"secretBearerToken\", \"test-token\")\n secretBearerToken = pw.env.resolve(\"<<secretBearerToken>>\")\n}\n\npw.env.set(\"preReqSecretBearerToken\", secretBearerToken)"
}
],
@@ -146,4 +146,4 @@
"authActive": false
},
"headers": []
}
}

View File

@@ -5,7 +5,7 @@
"requests": [
{
"v": "3",
"endpoint": "https://httpbin.org/post",
"endpoint": "https://echo.hoppscotch.io/post",
"name": "req",
"params": [],
"headers": [
@@ -18,7 +18,7 @@
"method": "POST",
"auth": { "authType": "none", "authActive": true },
"preRequestScript": "pw.env.set(\"preReqVarOne\", \"pre-req-value-one\")\n\npw.env.set(\"preReqVarTwo\", \"pre-req-value-two\")\n\npw.env.set(\"customHeaderValueFromSecretVar\", \"custom-header-secret-value\")\n\npw.env.set(\"customBodyValue\", \"custom-body-value\")",
"testScript": "pw.test(\"Secret environment value set from the pre-request script takes precedence\", () => {\n pw.expect(pw.env.get(\"preReqVarOne\")).toBe(\"pre-req-value-one\")\n})\n\npw.test(\"Successfully sets initial value for the secret variable from the pre-request script\", () => {\n pw.env.set(\"postReqVarTwo\", \"post-req-value-two\")\n pw.expect(pw.env.get(\"postReqVarTwo\")).toBe(\"post-req-value-two\")\n})\n\npw.test(\"Successfully resolves secret variable values referred in request headers that are set in pre-request sccript\", () => {\n pw.expect(pw.response.body.headers[\"Custom-Header\"]).toBe(\"custom-header-secret-value\")\n})\n\npw.test(\"Successfully resolves secret variable values referred in request body that are set in pre-request sccript\", () => {\n pw.expect(pw.response.body.json.key).toBe(\"custom-body-value\")\n})\n\npw.test(\"Secret environment variable set from the post-request script takes precedence\", () => {\n pw.env.set(\"postReqVarOne\", \"post-req-value-one\")\n pw.expect(pw.env.get(\"postReqVarOne\")).toBe(\"post-req-value-one\")\n})\n\npw.test(\"Successfully sets initial value for the secret variable from the post-request script\", () => {\n pw.env.set(\"postReqVarTwo\", \"post-req-value-two\")\n pw.expect(pw.env.get(\"postReqVarTwo\")).toBe(\"post-req-value-two\")\n})\n\npw.test(\"Successfully removes environment variables via the pw.env.unset method\", () => {\n pw.env.unset(\"preReqVarOne\")\n pw.env.unset(\"postReqVarTwo\")\n\n pw.expect(pw.env.get(\"preReqVarOne\")).toBe(undefined)\n pw.expect(pw.env.get(\"postReqVarTwo\")).toBe(undefined)\n})",
"testScript": "pw.test(\"Secret environment value set from the pre-request script takes precedence\", () => {\n pw.expect(pw.env.get(\"preReqVarOne\")).toBe(\"pre-req-value-one\")\n})\n\npw.test(\"Successfully sets initial value for the secret variable from the pre-request script\", () => {\n pw.env.set(\"postReqVarTwo\", \"post-req-value-two\")\n pw.expect(pw.env.get(\"postReqVarTwo\")).toBe(\"post-req-value-two\")\n})\n\npw.test(\"Successfully resolves secret variable values referred in request headers that are set in pre-request script\", () => {\n pw.expect(pw.response.body.headers[\"custom-header\"]).toBe(\"custom-header-secret-value\")\n})\n\npw.test(\"Successfully resolves secret variable values referred in request body that are set in pre-request script\", () => {\n pw.expect(JSON.parse(pw.response.body.data).key).toBe(\"custom-body-value\")\n})\n\npw.test(\"Secret environment variable set from the post-request script takes precedence\", () => {\n pw.env.set(\"postReqVarOne\", \"post-req-value-one\")\n pw.expect(pw.env.get(\"postReqVarOne\")).toBe(\"post-req-value-one\")\n})\n\npw.test(\"Successfully sets initial value for the secret variable from the post-request script\", () => {\n pw.env.set(\"postReqVarTwo\", \"post-req-value-two\")\n pw.expect(pw.env.get(\"postReqVarTwo\")).toBe(\"post-req-value-two\")\n})\n\npw.test(\"Successfully removes environment variables via the pw.env.unset method\", () => {\n pw.env.unset(\"preReqVarOne\")\n pw.env.unset(\"postReqVarTwo\")\n\n pw.expect(pw.env.get(\"preReqVarOne\")).toBe(undefined)\n pw.expect(pw.env.get(\"postReqVarTwo\")).toBe(undefined)\n})",
"body": {
"contentType": "application/json",
"body": "{\n \"key\": \"<<customBodyValue>>\"\n}"

View File

@@ -32,7 +32,12 @@
"secret": true
},
{
"key": "baseURL",
"key": "echoHoppBaseURL",
"value": "https://echo.hoppscotch.io",
"secret": false
},
{
"key": "httpbinBaseURL",
"value": "https://httpbin.org",
"secret": false
}

View File

@@ -38,7 +38,12 @@
"secret": true
},
{
"key": "baseURL",
"key": "echoHoppBaseURL",
"value": "https://echo.hoppscotch.io",
"secret": false
},
{
"key": "httpbinBaseURL",
"value": "https://httpbin.org",
"secret": false
}

View File

@@ -3,15 +3,16 @@ import { resolve } from "path";
import { ExecResponse } from "./types";
export const runCLI = (args: string, options = {}): Promise<ExecResponse> =>
{
const CLI_PATH = resolve(__dirname, "../../bin/hopp");
const command = `node ${CLI_PATH} ${args}`
export const runCLI = (args: string, options = {}): Promise<ExecResponse> => {
const CLI_PATH = resolve(__dirname, "../../bin/hopp.js");
const command = `node ${CLI_PATH} ${args}`;
return new Promise((resolve) =>
exec(command, options, (error, stdout, stderr) => resolve({ error, stdout, stderr }))
);
}
return new Promise((resolve) =>
exec(command, options, (error, stdout, stderr) =>
resolve({ error, stdout, stderr })
)
);
};
export const trimAnsi = (target: string) => {
const ansiRegex =
@@ -25,12 +26,18 @@ export const getErrorCode = (out: string) => {
return ansiTrimmedStr.split(" ")[0];
};
export const getTestJsonFilePath = (file: string, kind: "collection" | "environment") => {
export const getTestJsonFilePath = (
file: string,
kind: "collection" | "environment"
) => {
const kindDir = {
collection: "collections",
environment: "environments",
}[kind];
const filePath = resolve(__dirname, `../../src/__tests__/samples/${kindDir}/${file}`);
const filePath = resolve(
__dirname,
`../../src/__tests__/samples/${kindDir}/${file}`
);
return filePath;
};

View File

@@ -1,6 +1,7 @@
import chalk from "chalk";
import { Command } from "commander";
import * as E from "fp-ts/Either";
import { version } from "../package.json";
import { test } from "./commands/test";
import { handleError } from "./handlers/error";
@@ -20,7 +21,7 @@ const CLI_AFTER_ALL_TXT = `\nFor more help, head on to ${accent(
"https://docs.hoppscotch.io/documentation/clients/cli"
)}`;
const program = new Command()
const program = new Command();
program
.name("hopp")

View File

@@ -7,6 +7,41 @@ import { error } from "../types/errors";
import { FormDataEntry } from "../types/request";
import { isHoppErrnoException } from "./checks";
const getValidRequests = (
collections: HoppCollection[],
collectionFilePath: string
) => {
return collections.map((collection) => {
// Validate requests using zod schema
const requestSchemaParsedResult = z
.array(entityReference(HoppRESTRequest))
.safeParse(collection.requests);
// Handle validation errors
if (!requestSchemaParsedResult.success) {
throw error({
code: "MALFORMED_COLLECTION",
path: collectionFilePath,
data: "Please check the collection data.",
});
}
// Recursively validate requests in nested folders
if (collection.folders.length > 0) {
collection.folders = getValidRequests(
collection.folders,
collectionFilePath
);
}
// Return validated collection
return {
...collection,
requests: requestSchemaParsedResult.data,
};
});
};
/**
* Parses array of FormDataEntry to FormData.
* @param values Array of FormDataEntry.
@@ -82,22 +117,5 @@ export async function parseCollectionData(
});
}
return collectionSchemaParsedResult.data.map((collection) => {
const requestSchemaParsedResult = z
.array(entityReference(HoppRESTRequest))
.safeParse(collection.requests);
if (!requestSchemaParsedResult.success) {
throw error({
code: "MALFORMED_COLLECTION",
path,
data: "Please check the collection data.",
});
}
return {
...collection,
requests: requestSchemaParsedResult.data,
};
});
return getValidRequests(collectionSchemaParsedResult.data, path);
}

View File

@@ -136,13 +136,13 @@ export function getEffectiveRESTRequest(
}
} else if (request.auth.authType === "api-key") {
const { key, value, addTo } = request.auth;
if (addTo === "Headers") {
if (addTo === "HEADERS") {
effectiveFinalHeaders.push({
active: true,
key: parseTemplateString(key, envVariables),
value: parseTemplateString(value, envVariables),
});
} else if (addTo === "Query params") {
} else if (addTo === "QUERY_PARAMS") {
effectiveFinalParams.push({
active: true,
key: parseTemplateString(key, envVariables),

View File

@@ -162,6 +162,8 @@
"label_client_credentials": "Client Credentials"
},
"pass_key_by": "Pass by",
"pass_by_query_params_label": "Query Parameters",
"pass_by_headers_label": "Headers",
"password": "Password",
"save_to_inherit": "Please save this request in any collection to inherit the authorization",
"token": "Token",
@@ -173,6 +175,7 @@
"different_parent": "Cannot reorder collection with different parent",
"edit": "Edit Collection",
"import_or_create": "Import or create a collection",
"import_collection":"Import Collection",
"invalid_name": "Please provide a name for the collection",
"invalid_root_move": "Collection already in the root",
"moved": "Moved Successfully",
@@ -565,7 +568,9 @@
"generated_code": "Generated code",
"go_to_authorization_tab": "Go to Authorization tab",
"go_to_body_tab": "Go to Body tab",
"graphql_placeholder": "Enter a URL",
"header_list": "Header List",
"http_placeholder":"Enter a URL or cURL command",
"invalid_name": "Please provide a name for the request",
"method": "Method",
"moved": "Request moved",
@@ -847,6 +852,13 @@
"new": "Create new workspace",
"switch_to_personal": "Switch to your personal workspace",
"title": "Workspaces"
},
"phrases":{
"try": "Try",
"import_collections": "Import collections",
"create_environment": "Create environment",
"create_workspace": "Create workspace",
"share_request": "Share request"
}
},
"sse": {

View File

@@ -32,8 +32,8 @@
"no": "No",
"open_workspace": "Abrir espacio de trabajo",
"paste": "Pegar",
"prettify": "Embellecer",
"properties": "Properties",
"prettify": "Formatear",
"properties": "Propiedades",
"remove": "Eliminar",
"rename": "Rename",
"restore": "Restaurar",
@@ -63,7 +63,7 @@
"contact_us": "Contáctanos",
"cookies": "Cookies",
"copy": "Copiar",
"copy_interface_type": "Copy interface type",
"copy_interface_type": "Copiar tipo de interfaz",
"copy_user_id": "Copiar token de autenticación de usuario",
"developer_option": "Opciones para desarrolladores",
"developer_option_description": "Herramientas para desarrolladores que ayudan en el desarrollo y mantenimiento de Hoppscotch.",
@@ -80,14 +80,14 @@
"name": "Hoppscotch",
"new_version_found": "Se ha encontrado una nueva versión. Recarga la página para usarla.",
"open_in_hoppscotch": "Open in Hoppscotch",
"options": "Options",
"options": "Opciones",
"proxy_privacy_policy": "Política de privacidad de proxy",
"reload": "Recargar",
"search": "Buscar",
"share": "Compartir",
"shortcuts": "Atajos",
"social_description": "Follow us on social media to stay updated with the latest news, updates and releases.",
"social_links": "Social links",
"social_description": "Síguenos en redes sociales para estar al día de las últimas noticias, actualizaciones y lanzamientos.",
"social_links": "Redes sociales",
"spotlight": "Destacar",
"status": "Estado",
"status_description": "Comprobar el estado del sitio web",
@@ -119,27 +119,27 @@
},
"authorization": {
"generate_token": "Generar token",
"graphql_headers": "Authorization Headers are sent as part of the payload to connection_init",
"graphql_headers": "Las cabeceras de autorización se envían como parte de la carga útil de connection_init",
"include_in_url": "Incluir en la URL",
"inherited_from": "Inherited from {auth} from Parent Collection {collection} ",
"inherited_from": "Heredado {auth} de colección padre {collection} ",
"learn": "Aprender",
"oauth": {
"redirect_auth_server_returned_error": "Auth Server returned an error state",
"redirect_auth_token_request_failed": "Request to get the auth token failed",
"redirect_auth_token_request_invalid_response": "Invalid Response from the Token Endpoint when requesting for an auth token",
"redirect_invalid_state": "Invalid State value present in the redirect",
"redirect_no_auth_code": "No Authorization Code present in the redirect",
"redirect_no_client_id": "No Client ID defined",
"redirect_no_client_secret": "No Client Secret Defined",
"redirect_no_code_verifier": "No Code Verifier Defined",
"redirect_no_token_endpoint": "No Token Endpoint Defined",
"something_went_wrong_on_oauth_redirect": "Something went wrong during OAuth Redirect",
"something_went_wrong_on_token_generation": "Something went wrong on token generation",
"token_generation_oidc_discovery_failed": "Failure on token generation: OpenID Connect Discovery Failed"
"redirect_auth_server_returned_error": "El servidor de autenticación ha devuelto un estado de error",
"redirect_auth_token_request_failed": "Fallo en la solicitud de token de autentificación",
"redirect_auth_token_request_invalid_response": "Respuesta no válida del punto final de Token al solicitar un token de autentificación",
"redirect_invalid_state": "Valor de estado no válido presente en la redirección",
"redirect_no_auth_code": "No hay código de autorización en la redirección",
"redirect_no_client_id": "No se ha definido el ID de cliente",
"redirect_no_client_secret": "No se ha definido ningún ID secreto de cliente",
"redirect_no_code_verifier": "No se ha definido ningún verificador de códigos",
"redirect_no_token_endpoint": "No se ha definido ningún punto final de token",
"something_went_wrong_on_oauth_redirect": "Algo ha ido mal durante la redirección OAuth",
"something_went_wrong_on_token_generation": "Algo salió mal en la generación del token",
"token_generation_oidc_discovery_failed": "Fallo en la generación del token: OpenID Connect Discovery Failed"
},
"pass_key_by": "Pasar por",
"password": "Contraseña",
"save_to_inherit": "Please save this request in any collection to inherit the authorization",
"save_to_inherit": "Por favor, guarda esta solicitud en cualquier colección para heredar la autorización",
"token": "Token",
"type": "Tipo de autorización",
"username": "Nombre de usuario"
@@ -148,7 +148,7 @@
"created": "Colección creada",
"different_parent": "No se puede reordenar la colección con un padre diferente",
"edit": "Editar colección",
"import_or_create": "Import or create a collection",
"import_or_create": "Importar o crear una colección",
"invalid_name": "Proporciona un nombre válido para la colección.",
"invalid_root_move": "La colección ya está en la raíz",
"moved": "Movido con éxito",
@@ -157,20 +157,20 @@
"name_length_insufficient": "El nombre de la colección debe tener al menos 3 caracteres",
"new": "Nueva colección",
"order_changed": "Orden de colección actualizada",
"properties": "Collection Properties",
"properties_updated": "Collection Properties Updated",
"properties": "Propiedades de la colección",
"properties_updated": "Propiedades de la colección actualizadas",
"renamed": "Colección renombrada",
"request_in_use": "Solicitud en uso",
"save_as": "Guardar como",
"save_to_collection": "Save to Collection",
"save_to_collection": "Guardar en la colección",
"select": "Seleccionar colección",
"select_location": "Seleccionar ubicación",
"select_team": "Seleccionar equipo",
"team_collections": "Colecciones de equipos"
},
"confirm": {
"close_unsaved_tab": "Are you sure you want to close this tab?",
"close_unsaved_tabs": "Are you sure you want to close all tabs? {count} unsaved tabs will be lost.",
"close_unsaved_tab": "¿Seguro que quieres cerrar esta pestaña?",
"close_unsaved_tabs": "¿Estás seguro de que quieres cerrar todas las pestañas? {count} pestañas no guardadas se perderán.",
"exit_team": "¿Estás seguro de que quieres dejar este equipo?",
"logout": "¿Estás seguro de que deseas cerrar la sesión?",
"remove_collection": "¿Estás seguro de que deseas eliminar esta colección de forma permanente?",
@@ -178,7 +178,7 @@
"remove_folder": "¿Estás seguro de que deseas eliminar esta carpeta de forma permanente?",
"remove_history": "¿Estás seguro de que deseas eliminar todo el historial de forma permanente?",
"remove_request": "¿Estás seguro de que deseas eliminar esta solicitud de forma permanente?",
"remove_shared_request": "Are you sure you want to permanently delete this shared request?",
"remove_shared_request": "¿Estás seguro de que quieres eliminar definitivamente esta solicitud compartida?",
"remove_team": "¿Estás seguro de que deseas eliminar este equipo?",
"remove_telemetry": "¿Estás seguro de que deseas darse de baja de la telemetría?",
"request_change": "¿Estás seguro de que deseas descartar la solicitud actual, los cambios no guardados se perderán.",
@@ -186,35 +186,35 @@
"sync": "¿Estás seguro de que deseas sincronizar este espacio de trabajo?"
},
"context_menu": {
"add_parameters": "Add to parameters",
"open_request_in_new_tab": "Open request in new tab",
"set_environment_variable": "Set as variable"
"add_parameters": "Añadir a parámetros",
"open_request_in_new_tab": "Abrir solicitud en una nueva pestaña",
"set_environment_variable": "Establecer como variable"
},
"cookies": {
"modal": {
"cookie_expires": "Expires",
"cookie_name": "Name",
"cookie_path": "Path",
"cookie_string": "Cookie string",
"cookie_value": "Value",
"empty_domain": "Domain is empty",
"empty_domains": "Domain list is empty",
"enter_cookie_string": "Enter cookie string",
"interceptor_no_support": "Your currently selected interceptor does not support cookies. Select a different Interceptor and try again.",
"managed_tab": "Managed",
"new_domain_name": "New domain name",
"no_cookies_in_domain": "No cookies set for this domain",
"raw_tab": "Raw",
"set": "Set a cookie"
"cookie_expires": "Expira en",
"cookie_name": "Nombre",
"cookie_path": "Ruta",
"cookie_string": "Cookies",
"cookie_value": "Valor",
"empty_domain": "Dominio vacio",
"empty_domains": "No hay dominios",
"enter_cookie_string": "Introducir cookies",
"interceptor_no_support": "El interceptor seleccionado actualmente no admite cookies. Seleccione otro interceptor e inténtelo de nuevo.",
"managed_tab": "Gestionado",
"new_domain_name": "Nuevo nombre de dominio",
"no_cookies_in_domain": "No hay cookies para este dominio",
"raw_tab": "Sin procesar",
"set": "Establecer una cookie"
}
},
"count": {
"header": "Encabezado {count}",
"message": "Mensaje {count}",
"parameter": "Parámetro {count}",
"protocol": "Protocolo {count}",
"value": "Valor {cuenta}",
"variable": "Variable {count}"
"header": "{count} encabezado(s)",
"message": "{count} mensaje(s)",
"parameter": "{count} parámetro(s)",
"protocol": "{count} protocolo(s)",
"value": "{cuenta} valor(es)",
"variable": "{count} variable(es)"
},
"documentation": {
"generate": "Generar documentación",
@@ -223,26 +223,26 @@
"empty": {
"authorization": "Esta solicitud no utiliza ninguna autorización.",
"body": "Esta solicitud no tiene cuerpo",
"collection": "La colección está vacía",
"collections": "Las colecciones están vacías",
"collection": "Colección vacía",
"collections": "No hay colecciones",
"documentation": "Conectarse a un punto final de GraphQL para ver la documentación",
"endpoint": "El punto final no puede estar vacío",
"environments": "Los entornos están vacíos",
"folder": "La carpeta está vacía",
"environments": "No hay entornos",
"folder": "Carpeta vacía",
"headers": "Esta solicitud no tiene encabezados",
"history": "El historial está vacío",
"invites": "La lista de invitados está vacía",
"members": "El equipo está vacío",
"history": "No hay historial",
"invites": "Lista de invitados vacía",
"members": "No hay miembros en el equipo",
"parameters": "Esta solicitud no tiene ningún parámetro",
"pending_invites": "No hay invitaciones pendientes para este equipo",
"profile": "Iniciar sesión para ver tu perfil",
"protocols": "Los protocolos están vacíos",
"protocols": "No hay protocolos",
"schema": "Conectarse a un punto final de GraphQL",
"shared_requests": "Shared requests are empty",
"shared_requests_logout": "Login to view your shared requests or create a new one",
"subscription": "Subscriptions are empty",
"shared_requests": "No hay solicitudes compartidas",
"shared_requests_logout": "Iniciar sesión para ver sus solicitudes compartidas o crear una nueva",
"subscription": "No hay suscripciones",
"team_name": "Nombre del equipo vacío",
"teams": "Los equipos están vacíos",
"teams": "No hay equipos",
"tests": "No hay pruebas para esta solicitud",
"shortcodes": "Aún no se han creado Shortcodes"
},
@@ -250,41 +250,41 @@
"add_to_global": "Añadir a Global",
"added": "Adición al entorno",
"create_new": "Crear un nuevo entorno",
"created": "Environment created",
"created": "Entorno creado",
"deleted": "Eliminar el entorno",
"duplicated": "Environment duplicated",
"duplicated": "Entorno duplicado",
"edit": "Editar entorno",
"empty_variables": "No variables",
"empty_variables": "No hay variables",
"global": "Global",
"global_variables": "Global variables",
"import_or_create": "Import or create a environment",
"global_variables": "Variables globales",
"import_or_create": "Importar o crear un entorno",
"invalid_name": "Proporciona un nombre válido para el entorno.",
"list": "Environment variables",
"list": "Variables de entorno",
"my_environments": "Mis entornos",
"name": "Name",
"name": "Nombre",
"nested_overflow": "las variables de entorno anidadas están limitadas a 10 niveles",
"new": "Nuevo entorno",
"no_active_environment": "No active environment",
"no_active_environment": "Ningún entorno activo",
"no_environment": "Sin entorno",
"no_environment_description": "No se ha seleccionado ningún entorno. Elije qué hacer con las siguientes variables.",
"quick_peek": "Environment Quick Peek",
"replace_with_variable": "Replace with variable",
"scope": "Scope",
"quick_peek": "Vistazo rápido al entorno",
"replace_with_variable": "Sustituir por variable",
"scope": "Ámbito",
"select": "Seleccionar entorno",
"set": "Set environment",
"set_as_environment": "Set as environment",
"set": "Establecer entorno",
"set_as_environment": "Establecer como entorno",
"team_environments": "Entornos de trabajo en equipo",
"title": "Entornos",
"updated": "Entorno actualizado",
"value": "Value",
"value": "Valor",
"variable": "Variable",
"variable_list": "Lista de variables"
},
"error": {
"authproviders_load_error": "Unable to load auth providers",
"authproviders_load_error": "No se han podido cargar los proveedores de autenticación",
"browser_support_sse": "Este navegador no parece ser compatible con los eventos enviados por el servidor.",
"check_console_details": "Consulta el registro de la consola para obtener más detalles.",
"check_how_to_add_origin": "Check how you can add an origin",
"check_how_to_add_origin": "Comprueba cómo puede añadir un origen",
"curl_invalid_format": "cURL no está formateado correctamente",
"danger_zone": "Zona de peligro",
"delete_account": "Tu cuenta es actualmente propietaria en estos equipos:",
@@ -300,13 +300,13 @@
"json_prettify_invalid_body": "No se puede aplicar embellecedor a un cuerpo inválido, resuelve errores de sintaxis json y vuelve a intentarlo",
"network_error": "Parece que hay un error de red. Por favor, inténtalo de nuevo.",
"network_fail": "No se pudo enviar la solicitud",
"no_collections_to_export": "No collections to export. Please create a collection to get started.",
"no_collections_to_export": "No hay colecciones para exportar. Crea una colección para empezar.",
"no_duration": "Sin duración",
"no_environments_to_export": "No environments to export. Please create an environment to get started.",
"no_environments_to_export": "No hay entornos para exportar. Por favor, crea un entorno para empezar.",
"no_results_found": "No se han encontrado coincidencias",
"page_not_found": "No se ha podido encontrar esta página",
"please_install_extension": "Please install the extension and add origin to the extension.",
"proxy_error": "Proxy error",
"proxy_error": "Error de proxy",
"script_fail": "No se pudo ejecutar el script de solicitud previa",
"something_went_wrong": "Algo salió mal",
"test_script_fail": "No se ha podido ejecutar la secuencia de comandos posterior a la solicitud"
@@ -314,15 +314,15 @@
"export": {
"as_json": "Exportar como JSON",
"create_secret_gist": "Crear un Gist secreto",
"failed": "Something went wrong while exporting",
"failed": "Algo ha ido mal al exportar",
"gist_created": "Gist creado",
"require_github": "Iniciar sesión con GitHub para crear un Gist secreto",
"title": "Exportar"
},
"filter": {
"all": "All",
"none": "None",
"starred": "Starred"
"all": "Todos",
"none": "Ninguno",
"starred": "Destacado"
},
"folder": {
"created": "Carpeta creada",
@@ -333,16 +333,16 @@
"renamed": "Carpeta renombrada"
},
"graphql": {
"connection_switch_confirm": "Do you want to connect with the latest GraphQL endpoint?",
"connection_switch_new_url": "Switching to a tab will disconnected you from the active GraphQL connection. New connection URL is",
"connection_switch_url": "You're connected to a GraphQL endpoint the connection URL is",
"connection_switch_confirm": "¿Deseas conectarte con el endpoint GraphQL más reciente?",
"connection_switch_new_url": "Al cambiar a una pestaña se desconectará de la conexión GraphQL activa. La nueva URL de conexión es",
"connection_switch_url": "Estás conectado a un endpoint GraphQL cuya URL de conexión es",
"mutations": "Mutaciones",
"schema": "Esquema",
"subscriptions": "Suscripciones",
"switch_connection": "Switch connection"
"switch_connection": "Cambiar conexión"
},
"graphql_collections": {
"title": "GraphQL Collections"
"title": "Colecciones de GraphQL"
},
"group": {
"time": "Tiempo",
@@ -355,8 +355,8 @@
},
"helpers": {
"authorization": "El encabezado de autorización se generará automáticamente cuando se envía la solicitud.",
"collection_properties_authorization": " This authorization will be set for every request in this collection.",
"collection_properties_header": "This header will be set for every request in this collection.",
"collection_properties_authorization": " Esta autorización se establecerá para cada solicitud de esta colección.",
"collection_properties_header": "Este encabezado se establecerá para cada solicitud de esta colección.",
"generate_documentation_first": "Generar la documentación primero",
"network_fail": "No se puede acceder a la API. Comprueba tu conexión de red y vuelve a intentarlo.",
"offline": "Parece estar desconectado. Es posible que los datos de este espacio de trabajo no estén actualizados.",
@@ -376,10 +376,10 @@
"import": {
"collections": "Importar colecciones",
"curl": "Importar cURL",
"environments_from_gist": "Import From Gist",
"environments_from_gist_description": "Import Hoppscotch Environments From Gist",
"environments_from_gist": "Importar desde Gist",
"environments_from_gist_description": "Importar entornos Hoppscotch desde Gist",
"failed": "Importación fallida",
"from_file": "Import from File",
"from_file": "Importar desde archivo",
"from_gist": "Importar desde Gist",
"from_gist_description": "Importar desde URL de Gist",
"from_insomnia": "Importar desde Insomnia",
@@ -394,41 +394,41 @@
"from_postman_description": "Importar desde una colección de Postman",
"from_url": "Importar desde una URL",
"gist_url": "Introduce la URL de Gist",
"gql_collections_from_gist_description": "Import GraphQL Collections From Gist",
"hoppscotch_environment": "Hoppscotch Environment",
"hoppscotch_environment_description": "Import Hoppscotch Environment JSON file",
"import_from_url_invalid_fetch": "Couldn't get data from the url",
"import_from_url_invalid_file_format": "Error while importing collections",
"import_from_url_invalid_type": "Unsupported type. accepted values are 'hoppscotch', 'openapi', 'postman', 'insomnia'",
"import_from_url_success": "Collections Imported",
"insomnia_environment_description": "Import Insomnia Environment from a JSON/YAML file",
"gql_collections_from_gist_description": "Importar colecciones GraphQL desde Gist",
"hoppscotch_environment": "Entorno de Hoppscotch",
"hoppscotch_environment_description": "Importar archivo JSON del entorno de Hoppscotch",
"import_from_url_invalid_fetch": "No se han podido obtener datos de la url",
"import_from_url_invalid_file_format": "Error al importar colecciones",
"import_from_url_invalid_type": "Tipo no admitido. Los valores aceptados son \"hoppscotch\", \"openapi\", \"postman\", \"insomnia\".",
"import_from_url_success": "Colecciones Importadas",
"insomnia_environment_description": "Importar el entorno de Insomnia desde un archivo JSON/YAML",
"json_description": "Importar colecciones desde un archivo JSON de colecciones de Hoppscotch",
"postman_environment": "Postman Environment",
"postman_environment_description": "Import Postman Environment from a JSON file",
"postman_environment": "Entorno de Postman",
"postman_environment_description": "Importar entorno de Postman desde un archivo JSON",
"title": "Importar"
},
"inspections": {
"description": "Inspect possible errors",
"description": "Inspeccionar posibles errores",
"environment": {
"add_environment": "Add to Environment",
"not_found": "Environment variable “{environment}” not found."
"add_environment": "Añadir al Entorno",
"not_found": "No se ha encontrado la variable de entorno \"{environment}\"."
},
"header": {
"cookie": "The browser doesn't allow Hoppscotch to set the Cookie Header. While we're working on the Hoppscotch Desktop App (coming soon), please use the Authorization Header instead."
"cookie": "El navegador no permite que Hoppscotch establezca el encabezado Cookie. Mientras trabajamos en la aplicación de escritorio de Hoppscotch (próximamente), por favor utilice el encabezado de autorización en su lugar."
},
"response": {
"401_error": "Please check your authentication credentials.",
"404_error": "Please check your request URL and method type.",
"cors_error": "Please check your Cross-Origin Resource Sharing configuration.",
"default_error": "Please check your request.",
"network_error": "Please check your network connection."
"401_error": "Compruebe tus credenciales de autenticación.",
"404_error": "Compruebe la URL de su solicitud y el tipo de método.",
"cors_error": "Por favor, comprueba tu configuración de Compartición de Recursos \"Cross-Origin\".",
"default_error": "Por favor, comprueba tu solicitud.",
"network_error": "Comprueba tu conexión de red."
},
"title": "Inspector",
"title": "Inspeccionador",
"url": {
"extension_not_installed": "Extension not installed.",
"extension_unknown_origin": "Make sure you've added the API endpoint's origin to the Hoppscotch Browser Extension list.",
"extention_enable_action": "Enable Browser Extension",
"extention_not_enabled": "Extension not enabled."
"extension_not_installed": "Extensión no instalada.",
"extension_unknown_origin": "Asegúrate de haber agregado el origen del punto final de la API a la lista de Extensiones del Navegador Hoppscotch.",
"extention_enable_action": "Activar la extensión del navegador",
"extention_not_enabled": "Extensión no habilitada."
}
},
"layout": {
@@ -442,10 +442,10 @@
"close_unsaved_tab": "Tienes cambios sin guardar",
"collections": "Colecciones",
"confirm": "Confirmar",
"customize_request": "Customize Request",
"customize_request": "Personalizar solicitud",
"edit_request": "Editar solicitud",
"import_export": "Importación y exportación",
"share_request": "Share Request"
"share_request": "Compartir solicitud"
},
"mqtt": {
"already_subscribed": "Ya estás suscrito a este tema.",
@@ -493,7 +493,7 @@
},
"profile": {
"app_settings": "Ajustes de la aplicación",
"default_hopp_displayname": "Unnamed User",
"default_hopp_displayname": "Usuario anónimo",
"editor": "Editor",
"editor_description": "Los editores pueden añadir, editar y eliminar solicitudes.",
"email_verification_mail": "Se ha enviado un correo electrónico de verificación a tu dirección de correo electrónico. Haz clic en el enlace para verificar tu dirección de correo electrónico.",
@@ -526,12 +526,12 @@
"enter_curl": "Ingrese cURL",
"generate_code": "Generar código",
"generated_code": "Código generado",
"go_to_authorization_tab": "Go to Authorization tab",
"go_to_body_tab": "Go to Body tab",
"go_to_authorization_tab": "Ir a la pestaña Autorización",
"go_to_body_tab": "Ir a la pestaña de cuerpo",
"header_list": "Lista de encabezados",
"invalid_name": "Proporciona un nombre para la solicitud.",
"method": "Método",
"moved": "Request moved",
"moved": "Solicitud movida",
"name": "Nombre de solicitud",
"new": "Nueva solicitud",
"order_changed": "Orden de solicitudes actualizadas",
@@ -543,8 +543,8 @@
"path": "Ruta",
"payload": "Carga útil",
"query": "Consulta",
"raw_body": "Cuerpo de solicitud sin procesar",
"rename": "Rename Request",
"raw_body": "cuerpo sin procesar",
"rename": "Renombrar solicitud",
"renamed": "Solicitud renombrada",
"run": "Ejecutar",
"save": "Guardar",
@@ -552,8 +552,8 @@
"saved": "Solicitud guardada",
"share": "Compartir",
"share_description": "Comparte Hoppscotch con tus amigos",
"share_request": "Share Request",
"stop": "Stop",
"share_request": "Compartir solicitud",
"stop": "Detener",
"title": "Solicitud",
"type": "Tipo de solicitud",
"url": "URL",
@@ -571,7 +571,7 @@
"json": "JSON",
"pdf": "PDF",
"preview_html": "Vista previa de HTML",
"raw": "Crudo",
"raw": "Sin procesar",
"size": "Tamaño",
"status": "Estado",
"time": "Tiempo",
@@ -635,29 +635,29 @@
"verify_email": "Verificar correo electrónico"
},
"shared_requests": {
"button": "Button",
"button_info": "Create a 'Run in Hoppscotch' button for your website, blog or a README.",
"copy_html": "Copy HTML",
"copy_link": "Copy Link",
"copy_markdown": "Copy Markdown",
"creating_widget": "Creating widget",
"customize": "Customize",
"deleted": "Shared request deleted",
"description": "Select a widget, you can change and customize this later",
"embed": "Embed",
"embed_info": "Add a mini 'Hoppscotch API Playground' to your website, blog or documentation.",
"link": "Link",
"link_info": "Create a shareable link to share with anyone on the internet with view access.",
"modified": "Shared request modified",
"not_found": "Shared request not found",
"open_new_tab": "Open in new tab",
"preview": "Preview",
"run_in_hoppscotch": "Run in Hoppscotch",
"button": "Botón",
"button_info": "Crea un botón \"Ejecutar en Hoppscotch\" para tu página web, blog o un README.",
"copy_html": "Copiar HTML",
"copy_link": "Copiar enlace",
"copy_markdown": "Copiar Markdown",
"creating_widget": "Crear widget",
"customize": "Personalizar",
"deleted": "Solicitud compartida eliminada",
"description": "Selecciona un widget, puedes cambiarlo y personalizarlo más tarde",
"embed": "Incrustar",
"embed_info": "Añada un mini \"Hoppscotch API Playground\" a tu sitio web, blog o documentación.",
"link": "Enlace",
"link_info": "Crea un enlace compartible para compartirlo con cualquier persona en Internet con acceso de visualización.",
"modified": "Solicitud compartida modificada",
"not_found": "Solicitud compartida no encontrada",
"open_new_tab": "Abrir en una nueva pestaña",
"preview": "Vista previa",
"run_in_hoppscotch": "Correr en Hoppscotch",
"theme": {
"dark": "Dark",
"light": "Light",
"system": "System",
"title": "Theme"
"dark": "Oscuro",
"light": "Claro",
"system": "Sistema",
"title": "Tema"
}
},
"shortcut": {
@@ -684,8 +684,8 @@
"title": "Navegación"
},
"others": {
"prettify": "Prettify Editor's Content",
"title": "Others"
"prettify": "Formatear el contenido del editor",
"title": "Otros"
},
"request": {
"delete_method": "Seleccionar método DELETE",
@@ -697,13 +697,13 @@
"post_method": "Seleccionar método POST",
"previous_method": "Seleccionar método anterior",
"put_method": "Seleccionar método PUT",
"rename": "Rename Request",
"rename": "Renombrar solicitud",
"reset_request": "Solicitud de reinicio",
"save_request": "Save Request",
"save_request": "Guardar solicitud",
"save_to_collections": "Guardar en colecciones",
"send_request": "Enviar solicitud",
"share_request": "Share Request",
"show_code": "Generate code snippet",
"share_request": "Compartir solicitud",
"show_code": "Generar fragmento de código",
"title": "Solicitud",
"copy_request_link": "Copiar enlace de solicitud"
},
@@ -722,95 +722,95 @@
},
"show": {
"code": "Mostrar código",
"collection": "Expand Collection Panel",
"collection": "Ampliar el panel de colecciones",
"more": "Mostrar más",
"sidebar": "Mostrar barra lateral"
},
"socketio": {
"communication": "Comunicación",
"connection_not_authorized": "This SocketIO connection does not use any authentication.",
"connection_not_authorized": "Esta conexión SocketIO no utiliza ningún tipo de autenticación.",
"event_name": "Nombre del evento",
"events": "Eventos",
"log": "Registro",
"url": "URL"
},
"spotlight": {
"change_language": "Change Language",
"change_language": "Cambiar idioma",
"environments": {
"delete": "Delete current environment",
"duplicate": "Duplicate current environment",
"duplicate_global": "Duplicate global environment",
"edit": "Edit current environment",
"edit_global": "Edit global environment",
"new": "Create new environment",
"new_variable": "Create a new environment variable",
"title": "Environments"
"delete": "Borrar el entorno actual",
"duplicate": "Duplicar el entorno actual",
"duplicate_global": "Entorno global duplicado",
"edit": "Editar el entorno actual",
"edit_global": "Editar el entorno global",
"new": "Crear un nuevo entorno",
"new_variable": "Crear una nueva variable de entorno",
"title": "Entornos"
},
"general": {
"chat": "Chat with support",
"help_menu": "Help and support",
"open_docs": "Read Documentation",
"open_github": "Open GitHub repository",
"open_keybindings": "Keyboard shortcuts",
"chat": "Chatear con el servicio de asistencia",
"help_menu": "Ayuda y asistencia",
"open_docs": "Leer la documentación",
"open_github": "Abrir repositorio de GitHub",
"open_keybindings": "Atajos de teclado",
"social": "Social",
"title": "General"
},
"graphql": {
"connect": "Connect to server",
"disconnect": "Disconnect from server"
"connect": "Conectarse al servidor",
"disconnect": "Desconectarse del servidor"
},
"miscellaneous": {
"invite": "Invite your friends to Hoppscotch",
"title": "Miscellaneous"
"invite": "Invita a tus amigos a Hoppscotch",
"title": "Varios"
},
"request": {
"save_as_new": "Save as new request",
"select_method": "Select method",
"switch_to": "Switch to",
"tab_authorization": "Authorization tab",
"tab_body": "Body tab",
"tab_headers": "Headers tab",
"tab_parameters": "Parameters tab",
"tab_pre_request_script": "Pre-request script tab",
"tab_query": "Query tab",
"tab_tests": "Tests tab",
"tab_variables": "Variables tab"
"save_as_new": "Guardar como nueva solicitud",
"select_method": "Seleccionar método",
"switch_to": "Cambiar a",
"tab_authorization": "Pestaña de autorización",
"tab_body": "Pestaña de cuerpo",
"tab_headers": "Pestaña de encabezados",
"tab_parameters": "Pestaña de parámetros",
"tab_pre_request_script": "Pestaña del script de pre-solicitud",
"tab_query": "Pestaña de consulta",
"tab_tests": "Pestaña de pruebas",
"tab_variables": "Pestaña de variables"
},
"response": {
"copy": "Copy response",
"download": "Download response as file",
"title": "Response"
"copy": "Copiar respuesta",
"download": "Descargar la respuesta como archivo",
"title": "Respuesta"
},
"section": {
"interceptor": "Interceptor",
"interface": "Interface",
"theme": "Theme",
"user": "User"
"interface": "Interfaz",
"theme": "Tema",
"user": "Usuario"
},
"settings": {
"change_interceptor": "Change Interceptor",
"change_language": "Change Language",
"change_interceptor": "Cambiar Interceptor",
"change_language": "Cambiar idioma",
"theme": {
"black": "Black",
"dark": "Dark",
"light": "Light",
"system": "System preference"
"black": "Negro",
"dark": "Oscuro",
"light": "Claro",
"system": "Preferencia del sistema"
}
},
"tab": {
"close_current": "Close current tab",
"close_others": "Close all other tabs",
"duplicate": "Duplicate current tab",
"new_tab": "Open a new tab",
"title": "Tabs"
"close_current": "Cerrar la pestaña actual",
"close_others": "Cerrar todas las demás pestañas",
"duplicate": "Duplicar pestaña actual",
"new_tab": "Abrir una nueva pestaña",
"title": "Pestañas"
},
"workspace": {
"delete": "Delete current team",
"edit": "Edit current team",
"invite": "Invite people to team",
"new": "Create new team",
"switch_to_personal": "Switch to your personal workspace",
"title": "Teams"
"delete": "Borrar el equipo actual",
"edit": "Editar el equipo actual",
"invite": "Invitar al equipo",
"new": "Crear un nuevo equipo",
"switch_to_personal": "Cambia a tu espacio de trabajo personal",
"title": "Equipos"
}
},
"sse": {
@@ -825,10 +825,10 @@
"connected": "Conectado",
"connected_to": "Conectado a {name}",
"connecting_to": "Conectando con {name}...",
"connection_error": "Failed to connect",
"connection_failed": "Error de conexión",
"connection_error": "Error de conexión",
"connection_failed": "Conexión fallida",
"connection_lost": "Conexión perdida",
"copied_interface_to_clipboard": "Copied {language} interface type to clipboard",
"copied_interface_to_clipboard": "Copiado tipo de interfaz {language} al portapapeles",
"copied_to_clipboard": "Copiado al portapapeles",
"deleted": "Eliminado",
"deprecated": "OBSOLETO",
@@ -836,21 +836,21 @@
"disconnected": "Desconectado",
"disconnected_from": "Desconectado de {name}",
"docs_generated": "Documentación generada",
"download_failed": "Download failed",
"download_failed": "Descarga fallida",
"download_started": "Descarga iniciada",
"enabled": "Activado",
"file_imported": "Archivo importado",
"finished_in": "Terminado en {duration} ms",
"hide": "Hide",
"finished_in": "Terminado en {duration}ms",
"hide": "Ocultar",
"history_deleted": "Historial eliminado",
"linewrap": "Envolver líneas",
"loading": "Cargando...",
"message_received": "Mensaje: {mensaje} llegó sobre el tema: {topic}",
"message_received": "Mensaje: llegó {message} al: {topic}",
"mqtt_subscription_failed": "Algo ha ido mal al suscribirse al tema: {topic}",
"none": "Ninguno",
"nothing_found": "Nada encontrado para",
"published_error": "Algo ha ido mal al publicar el mensaje: {topic} al tema: {message}",
"published_message": "Mensaje publicado: {mensaje} al tema: {topic}",
"published_error": "Algo ha ido mal al publicar el mensaje: {message} al tema: {topic}",
"published_message": "Mensaje publicado: {message} al tema: {topic}",
"reconnection_error": "Fallo en la reconexión",
"show": "Show",
"subscribed_failed": "Error al suscribirse al tema: {topic}",
@@ -874,12 +874,12 @@
"tab": {
"authorization": "Autorización",
"body": "Cuerpo",
"close": "Close Tab",
"close_others": "Close other Tabs",
"close": "Cerrar pestaña",
"close_others": "Cerrar otras pestañas",
"collections": "Colecciones",
"documentation": "Documentación",
"duplicate": "Duplicate Tab",
"environments": "Environments",
"duplicate": "Duplicar pestaña",
"environments": "Entornos",
"headers": "Encabezados",
"history": "Historial",
"mqtt": "MQTT",
@@ -888,7 +888,7 @@
"queries": "Consultas",
"query": "Consulta",
"schema": "Esquema",
"shared_requests": "Shared Requests",
"shared_requests": "Solicitudes compartidas",
"socketio": "Socket.IO",
"sse": "SSE",
"tests": "Pruebas",
@@ -905,7 +905,7 @@
"email_do_not_match": "El correo electrónico no coincide con los datos de tu cuenta. Ponte en contacto con el propietario de tu equipo.",
"exit": "Salir del equipo",
"exit_disabled": "Solo el propietario puede salir del equipo",
"failed_invites": "Failed invites",
"failed_invites": "Invitaciones fallidas",
"invalid_coll_id": "Identificador de colección no válido",
"invalid_email_format": "El formato de correo electrónico no es válido",
"invalid_id": "Identificador de equipo inválido. Ponte en contacto con el propietario de tu equipo.",
@@ -915,7 +915,7 @@
"invite": "Invitar",
"invite_more": "Invitar a más",
"invite_tooltip": "Invite a personas a este espacio de trabajo",
"invited_to_team": "{owner} te ha invitado a unirte al {tema}",
"invited_to_team": "{owner} te ha invitado al equipo {team}",
"join": "Invitación aceptada",
"join_beta": "Únete al programa beta para acceder a los equipos.",
"join_team": "Entrar a {team}",
@@ -930,7 +930,7 @@
"member_removed": "Usuario eliminado",
"member_role_updated": "Funciones de usuario actualizadas",
"members": "Miembros",
"more_members": "+{count} more",
"more_members": "+{count} más",
"name_length_insufficient": "El nombre del equipo debe tener al menos 6 caracteres",
"name_updated": "Nombre de equipo actualizado",
"new": "Nuevo equipo",
@@ -944,13 +944,13 @@
"parent_coll_move": "No se puede mover la colección a una colección hija",
"pending_invites": "Invitaciones pendientes",
"permissions": "Permisos",
"same_target_destination": "Same target and destination",
"same_target_destination": "Mismo objetivo y destino",
"saved": "Equipo guardado",
"select_a_team": "Seleccionar un equipo",
"success_invites": "Success invites",
"success_invites": "Invitaciones con éxito",
"title": "Equipos",
"we_sent_invite_link": "¡Hemos enviado un enlace de invitación a todos los invitados!",
"we_sent_invite_link_description": "Pide a todos los invitados que revisen tu bandeja de entrada. Haz clic en el enlace para unirse al equipo."
"we_sent_invite_link_description": "Pide a todos los invitados que revisen su bandeja de entrada. Tienen que hacer clic en el enlace para unirse al equipo."
},
"team_environment": {
"deleted": "Entorno eliminado",

View File

@@ -1,15 +1,15 @@
{
"action": {
"add": "Add",
"autoscroll": "Autoscroll",
"add": "Ajouter",
"autoscroll": "Auto-scroll",
"cancel": "Annuler",
"choose_file": "Choisir un fichier",
"clear": "Effacer",
"clear_all": "Tout effacer",
"clear_history": "Clear all History",
"close": "Close",
"clear_history": "Effacer tout l'historique",
"close": "Fermer",
"connect": "Connecter",
"connecting": "Connecting",
"connecting": "Connexion",
"copy": "Copier",
"create": "Create",
"delete": "Supprimer",
@@ -22,8 +22,8 @@
"edit": "Éditer",
"filter": "Filter",
"go_back": "Retour",
"go_forward": "Go forward",
"group_by": "Group by",
"go_forward": "Avancer",
"group_by": "Grouper par",
"label": "Étiqueter",
"learn_more": "En savoir plus",
"less": "Moins",
@@ -35,16 +35,16 @@
"prettify": "Formater",
"properties": "Properties",
"remove": "Supprimer",
"rename": "Rename",
"rename": "Renommer",
"restore": "Restaurer",
"save": "Sauvegarder",
"scroll_to_bottom": "Scroll to bottom",
"scroll_to_top": "Scroll to top",
"scroll_to_bottom": "Défiler vers le bas",
"scroll_to_top": "Défiler vers le haut",
"search": "Chercher",
"send": "Envoyer",
"share": "Share",
"start": "Démarrer",
"starting": "Starting",
"starting": "Démarrage",
"stop": "Arrêter",
"to_close": "pour fermer",
"to_navigate": "pour naviguer",
@@ -58,6 +58,24 @@
"new": "Ajouter un nouveau",
"star": "Ajouter une étoile"
},
"cookies": {
"modal": {
"new_domain_name": "Nouveau nom de domaine",
"set": "Définir un cookie",
"cookie_string": "Chaîne de caractères de cookie",
"enter_cookie_string": "Saisir la chaîne de caractères du cookie",
"cookie_name": "Nom",
"cookie_value": "Valeur",
"cookie_path": "Chemin d'accès",
"cookie_expires": "Expiration",
"managed_tab": "Gestion",
"raw_tab": "Brut",
"interceptor_no_support": "L'intercepteur que vous avez sélectionné ne prend pas en charge les cookies. Sélectionnez un autre intercepteur et réessayez.",
"empty_domains": "La liste des domaines est vide",
"empty_domain": "Le domaine est vide",
"no_cookies_in_domain": "Aucun cookie n'est défini pour ce domaine"
}
},
"app": {
"chat_with_us": "Discuter avec nous",
"contact_us": "Nous contacter",
@@ -86,8 +104,8 @@
"search": "Chercher",
"share": "Partager",
"shortcuts": "Raccourcis",
"social_description": "Follow us on social media to stay updated with the latest news, updates and releases.",
"social_links": "Social links",
"social_description": "Suivez-nous sur les médias sociaux pour rester informé des dernières nouvelles, mises à jour et communiqués.",
"social_links": "Liens sociaux",
"spotlight": "Projecteur",
"status": "Statut",
"status_description": "Vérifier l'état du site web",
@@ -95,7 +113,7 @@
"twitter": "Twitter",
"type_a_command_search": "Tapez une commande ou recherchez…",
"we_use_cookies": "Nous utilisons des cookies",
"whats_new": "Quoi de neuf?",
"whats_new": "Quoi de neuf ?",
"wiki": "Wiki"
},
"auth": {
@@ -119,39 +137,38 @@
},
"authorization": {
"generate_token": "Générer un jeton",
"graphql_headers": "Authorization Headers are sent as part of the payload to connection_init",
"graphql_headers": "Les en-têtes d'autorisation sont envoyés en tant que partie de la charge utile de connection_init.",
"include_in_url": "Inclure dans l'URL",
"inherited_from": "Inherited from {auth} from Parent Collection {collection} ",
"learn": "Apprendre comment",
"oauth": {
"redirect_auth_server_returned_error": "Auth Server returned an error state",
"redirect_auth_token_request_failed": "Request to get the auth token failed",
"redirect_auth_token_request_invalid_response": "Invalid Response from the Token Endpoint when requesting for an auth token",
"redirect_invalid_state": "Invalid State value present in the redirect",
"redirect_no_auth_code": "No Authorization Code present in the redirect",
"redirect_no_client_id": "No Client ID defined",
"redirect_no_client_secret": "No Client Secret Defined",
"redirect_no_code_verifier": "No Code Verifier Defined",
"redirect_no_token_endpoint": "No Token Endpoint Defined",
"something_went_wrong_on_oauth_redirect": "Something went wrong during OAuth Redirect",
"something_went_wrong_on_token_generation": "Something went wrong on token generation",
"token_generation_oidc_discovery_failed": "Failure on token generation: OpenID Connect Discovery Failed"
},
"pass_key_by": "Pass by",
"password": "Mot de passe",
"save_to_inherit": "Please save this request in any collection to inherit the authorization",
"token": "Jeton",
"type": "Type d'autorisation",
"username": "Nom d'utilisateur"
"username": "Nom d'utilisateur",
"oauth": {
"something_went_wrong_on_token_generation": "Un problème s'est produit lors de la génération des jetons",
"redirect_auth_server_returned_error": "Le serveur d'authentification a renvoyé un état d'erreur",
"redirect_no_auth_code": "Pas de code d'autorisation dans la redirection",
"redirect_invalid_state": "Valeur d'état non valide présente dans la redirection",
"redirect_no_token_endpoint": "Aucun point de terminaison de jeton n'est défini",
"redirect_no_client_id": "Pas d'ID client défini",
"redirect_no_client_secret": "Pas de secret client défini",
"redirect_no_code_verifier": "Pas de vérificateur de code défini",
"redirect_auth_token_request_failed": "La demande d'obtention du jeton d'authentification a échoué",
"redirect_auth_token_request_invalid_response": "Réponse invalide du point de terminaison Token lors de la demande d'un jeton d'authentification",
"something_went_wrong_on_oauth_redirect": "Quelque chose s'est mal passé lors de la redirection OAuth"
}
},
"collection": {
"created": "Collection créée",
"different_parent": "Cannot reorder collection with different parent",
"different_parent": "Impossible de réorganiser une collection dont le parent est différent",
"edit": "Modifier la collection",
"import_or_create": "Import or create a collection",
"import_or_create": "Importer ou créer une collection",
"invalid_name": "Veuillez fournir un nom valide pour la collection",
"invalid_root_move": "Collection already in the root",
"moved": "Moved Successfully",
"invalid_root_move": "Collection déjà présente dans la racine",
"moved": "Déplacement réussi",
"my_collections": "Mes collections",
"name": "Ma nouvelle collection",
"name_length_insufficient": "Le nom de la collection doit comporter au moins 3 caractères",
@@ -162,16 +179,16 @@
"renamed": "Collection renommée",
"request_in_use": "Demande en cours d'utilisation",
"save_as": "Enregistrer sous",
"save_to_collection": "Save to Collection",
"save_to_collection": "Enregistrer dans la collection",
"select": "Sélectionnez une collection",
"select_location": "Sélectionnez l'emplacement",
"select_team": "Sélectionnez une équipe",
"team_collections": "Collections de l'équipe"
},
"confirm": {
"close_unsaved_tab": "Are you sure you want to close this tab?",
"close_unsaved_tabs": "Are you sure you want to close all tabs? {count} unsaved tabs will be lost.",
"exit_team": "Are you sure you want to leave this team?",
"close_unsaved_tab": "Êtes-vous sûr de vouloir fermer cet onglet ?",
"close_unsaved_tabs": "Êtes-vous sûr de vouloir fermer tous les onglets ? {Les onglets non enregistrés seront perdus.",
"exit_team": "Êtes-vous sûr de vouloir quitter cette équipe ?",
"logout": "Êtes-vous sûr de vouloir vous déconnecter?",
"remove_collection": "Voulez-vous vraiment supprimer définitivement cette collection ?",
"remove_environment": "Voulez-vous vraiment supprimer définitivement cet environnement ?",
@@ -181,32 +198,14 @@
"remove_shared_request": "Are you sure you want to permanently delete this shared request?",
"remove_team": "Voulez-vous vraiment supprimer cette équipe ?",
"remove_telemetry": "Êtes-vous sûr de vouloir désactiver la télémétrie ?",
"request_change": "Are you sure you want to discard current request, unsaved changes will be lost.",
"save_unsaved_tab": "Do you want to save changes made in this tab?",
"request_change": "Êtes-vous sûr de vouloir rejeter la demande en cours ? Les modifications non enregistrées seront perdues.",
"save_unsaved_tab": "Souhaitez-vous enregistrer les modifications apportées dans cet onglet ?",
"sync": "Voulez-vous vraiment synchroniser cet espace de travail ?"
},
"context_menu": {
"add_parameters": "Add to parameters",
"open_request_in_new_tab": "Open request in new tab",
"set_environment_variable": "Set as variable"
},
"cookies": {
"modal": {
"cookie_expires": "Expires",
"cookie_name": "Name",
"cookie_path": "Path",
"cookie_string": "Cookie string",
"cookie_value": "Value",
"empty_domain": "Domain is empty",
"empty_domains": "Domain list is empty",
"enter_cookie_string": "Enter cookie string",
"interceptor_no_support": "Your currently selected interceptor does not support cookies. Select a different Interceptor and try again.",
"managed_tab": "Managed",
"new_domain_name": "New domain name",
"no_cookies_in_domain": "No cookies set for this domain",
"raw_tab": "Raw",
"set": "Set a cookie"
}
"add_parameters": "Ajouter aux paramètres",
"open_request_in_new_tab": "Ouvrir la demande dans un nouvel onglet",
"set_environment_variable": "Définir comme variable"
},
"count": {
"header": "En-tête {count}",
@@ -238,7 +237,7 @@
"profile": "Connectez-vous pour voir votre profil",
"protocols": "Les protocoles sont vides",
"schema": "Se connecter à un point de terminaison GraphQL",
"shared_requests": "Shared requests are empty",
"shared_requests": "Il n'y a pas de requêtes partagées",
"shared_requests_logout": "Login to view your shared requests or create a new one",
"subscription": "Subscriptions are empty",
"team_name": "Nom de l'équipe vide",
@@ -252,15 +251,15 @@
"create_new": "Créer un nouvel environnement",
"created": "Environnement créé",
"deleted": "Environnement supprimé",
"duplicated": "Environment duplicated",
"duplicated": "Environnement dupliqué",
"edit": "Modifier l'environnement",
"empty_variables": "No variables",
"global": "Global",
"global_variables": "Global variables",
"import_or_create": "Import or create a environment",
"global_variables": "Variables globales",
"import_or_create": "Importer ou créer un environnement",
"invalid_name": "Veuillez fournir un nom valide pour l'environnement",
"list": "Environment variables",
"my_environments": "My Environments",
"list": "Variables d'environnement",
"my_environments": "Mes environnements",
"name": "Name",
"nested_overflow": "les variables d'environnement imbriquées sont limitées à 10 niveaux",
"new": "Nouvel environnement",
@@ -284,11 +283,11 @@
"authproviders_load_error": "Unable to load auth providers",
"browser_support_sse": "Ce navigateur ne semble pas prendre en charge les événements envoyés par le serveur.",
"check_console_details": "Consultez le journal de la console pour plus de détails.",
"check_how_to_add_origin": "Check how you can add an origin",
"check_how_to_add_origin": "Vérifiez comment vous pouvez ajouter une origine",
"curl_invalid_format": "cURL n'est pas formaté correctement",
"danger_zone": "Danger zone",
"delete_account": "Your account is currently an owner in these teams:",
"delete_account_description": "You must either remove yourself, transfer ownership, or delete these teams before you can delete your account.",
"danger_zone": "Zone de danger",
"delete_account": "Votre compte est actuellement propriétaire de ces équipes :",
"delete_account_description": "Vous devez vous retirer, transférer la propriété ou supprimer ces équipes avant de pouvoir supprimer votre compte.",
"empty_req_name": "Nom de la requête vide",
"f12_details": "(F12 pour les détails)",
"gql_prettify_invalid_query": "Impossible de formater une requête non valide, résolvez les erreurs de syntaxe de la requête et réessayer",
@@ -300,13 +299,13 @@
"json_prettify_invalid_body": "Impossible de formater un corps non valide, résolvez les erreurs de syntaxe json et réessayez",
"network_error": "Il semble y avoir une erreur de réseau. Veuillez réessayer.",
"network_fail": "Impossible d'envoyer la requête",
"no_collections_to_export": "No collections to export. Please create a collection to get started.",
"no_collections_to_export": "Aucune collection à exporter. Veuillez créer une collection pour commencer.",
"no_duration": "Pas de durée",
"no_environments_to_export": "No environments to export. Please create an environment to get started.",
"no_environments_to_export": "Aucun environnement à exporter. Veuillez créer un environnement pour commencer.",
"no_results_found": "Aucune correspondance trouvée",
"page_not_found": "Cette page n'a pas pu être trouvée",
"please_install_extension": "Please install the extension and add origin to the extension.",
"proxy_error": "Proxy error",
"please_install_extension": "Veuillez installer l'extension et ajouter l'origine à l'extension.",
"proxy_error": "Erreur de proxy",
"script_fail": "Impossible d'exécuter le script de pré-requête",
"something_went_wrong": "Quelque chose s'est mal passé",
"test_script_fail": "Impossible d'exécuter le script post-requête"
@@ -320,9 +319,9 @@
"title": "Exportation"
},
"filter": {
"all": "All",
"none": "None",
"starred": "Starred"
"all": "Tout",
"none": "Aucun",
"starred": "Étoilé"
},
"folder": {
"created": "Dossier créé",
@@ -333,19 +332,19 @@
"renamed": "Dossier renommé"
},
"graphql": {
"connection_switch_confirm": "Do you want to connect with the latest GraphQL endpoint?",
"connection_switch_new_url": "Switching to a tab will disconnected you from the active GraphQL connection. New connection URL is",
"connection_switch_url": "You're connected to a GraphQL endpoint the connection URL is",
"connection_switch_confirm": "Voulez-vous vous connecter avec le dernier point de terminaison GraphQL ?",
"connection_switch_new_url": "Le passage à un autre onglet vous déconnectera de la connexion GraphQL active. La nouvelle URL de connexion est",
"connection_switch_url": "Vous êtes connecté à un point de terminaison GraphQL dont l'URL de connexion est",
"mutations": "Mutations",
"schema": "Schéma",
"subscriptions": "Abonnements",
"switch_connection": "Switch connection"
"switch_connection": "Changer de connexion"
},
"graphql_collections": {
"title": "GraphQL Collections"
},
"group": {
"time": "Time",
"time": "Temps",
"url": "URL"
},
"header": {
@@ -376,7 +375,7 @@
"import": {
"collections": "Importer des collections",
"curl": "Importer en cURL",
"environments_from_gist": "Import From Gist",
"environments_from_gist": "Importer depuis Gist",
"environments_from_gist_description": "Import Hoppscotch Environments From Gist",
"failed": "Échec de l'importation",
"from_file": "Import from File",
@@ -408,27 +407,27 @@
"title": "Importer"
},
"inspections": {
"description": "Inspect possible errors",
"description": "Inspecter les erreurs possibles",
"environment": {
"add_environment": "Add to Environment",
"not_found": "Environment variable “{environment} not found."
"add_environment": "Ajouter à l'environnement",
"not_found": "La variable d'environnement “{environnement} n'a pas été trouvée."
},
"header": {
"cookie": "The browser doesn't allow Hoppscotch to set the Cookie Header. While we're working on the Hoppscotch Desktop App (coming soon), please use the Authorization Header instead."
"cookie": "Le navigateur ne permet pas à Hoppscotch de définir l'en-tête Cookie. Pendant que nous travaillons sur l'application de bureau Hoppscotch (bientôt disponible), veuillez utiliser l'en-tête d'autorisation à la place."
},
"response": {
"401_error": "Please check your authentication credentials.",
"404_error": "Please check your request URL and method type.",
"cors_error": "Please check your Cross-Origin Resource Sharing configuration.",
"default_error": "Please check your request.",
"network_error": "Please check your network connection."
"401_error": "Veuillez vérifier vos informations d'authentification.",
"404_error": "Veuillez vérifier l'URL de votre demande et le type de méthode.",
"cors_error": "Veuillez vérifier la configuration du partage des ressources entre les origines.",
"default_error": "Veuillez vérifier votre demande.",
"network_error": "Veuillez vérifier votre connexion réseau."
},
"title": "Inspector",
"title": "Inspecteur",
"url": {
"extension_not_installed": "Extension not installed.",
"extension_unknown_origin": "Make sure you've added the API endpoint's origin to the Hoppscotch Browser Extension list.",
"extention_enable_action": "Enable Browser Extension",
"extention_not_enabled": "Extension not enabled."
"extension_not_installed": "L'extension n'est pas installée.",
"extension_unknown_origin": "Assurez-vous d'avoir ajouté l'origine du point de terminaison de l'API à la liste des extensions du navigateur Hoppscotch.",
"extention_enable_action": "Activer l'extension du navigateur",
"extention_not_enabled": "L'extension n'est pas activée."
}
},
"layout": {
@@ -439,25 +438,25 @@
"row": "Disposition horizontale"
},
"modal": {
"close_unsaved_tab": "You have unsaved changes",
"close_unsaved_tab": "Vous avez des modifications non enregistrées",
"collections": "Collections",
"confirm": "Confirmer",
"customize_request": "Customize Request",
"edit_request": "Modifier la requête",
"import_export": "Importer / Exporter",
"share_request": "Share Request"
"share_request": "Partager une requête"
},
"mqtt": {
"already_subscribed": "You are already subscribed to this topic.",
"clean_session": "Clean Session",
"clear_input": "Clear input",
"clear_input_on_send": "Clear input on send",
"already_subscribed": "Vous êtes déjà abonné à ce sujet.",
"clean_session": "Effacer la Session",
"clear_input": "Effacer la saisie",
"clear_input_on_send": "Effacer la saisie lors de l'envoi",
"client_id": "Client ID",
"color": "Pick a color",
"color": "Choisir la couleur",
"communication": "Communication",
"connection_config": "Connection Config",
"connection_not_authorized": "This MQTT connection does not use any authentication.",
"invalid_topic": "Please provide a topic for the subscription",
"connection_not_authorized": "Cette connexion MQTT n'utilise pas d'authentification.",
"invalid_topic": "Veuillez fournir un sujet pour l'abonnement",
"keep_alive": "Keep Alive",
"log": "Infos",
"lw_message": "Last-Will Message",
@@ -466,7 +465,7 @@
"lw_topic": "Last-Will Topic",
"message": "Message",
"new": "New Subscription",
"not_connected": "Please start a MQTT connection first.",
"not_connected": "Veuillez d'abord établir une connexion MQTT.",
"publish": "Publier",
"qos": "QoS",
"ssl": "SSL",
@@ -480,7 +479,7 @@
"navigation": {
"doc": "Documents",
"graphql": "GraphQL",
"profile": "Profile",
"profile": "Profil",
"realtime": "Temps réel",
"rest": "REST",
"settings": "Paramètres"
@@ -493,7 +492,7 @@
},
"profile": {
"app_settings": "Réglages de l'application",
"default_hopp_displayname": "Unnamed User",
"default_hopp_displayname": "Utilisateur anonyme",
"editor": "Éditeur",
"editor_description": "Les éditeurs peuvent ajouter, modifier et supprimer des demandes.",
"email_verification_mail": "Un e-mail de vérification a été envoyé à votre adresse e-mail. Veuillez cliquer sur le lien pour vérifier votre adresse électronique.",
@@ -526,17 +525,17 @@
"enter_curl": "Entrer cURL",
"generate_code": "Générer le code",
"generated_code": "Code généré",
"go_to_authorization_tab": "Go to Authorization tab",
"go_to_body_tab": "Go to Body tab",
"go_to_authorization_tab": "Aller à l'autorisation",
"header_list": "Liste des en-têtes",
"invalid_name": "Veuillez fournir un nom pour la requête",
"method": "Méthode",
"moved": "Request moved",
"name": "Nom de la requête",
"new": "Nouvelle requête",
"order_changed": "Request Order Updated",
"order_changed": "Demande de commande Mise à jour",
"override": "Remplacer",
"override_help": "Set <xmp>Content-Type</xmp> in Headers",
"override_help": "Définir <xmp>Content-Type</xmp> dans les en-têtes",
"overriden": "Remplacé",
"parameter_list": "Paramètres de requête",
"parameters": "Paramètres",
@@ -544,7 +543,7 @@
"payload": "Charge utile",
"query": "Requête",
"raw_body": "Corps de requête brut",
"rename": "Rename Request",
"rename": "Demande de renommage",
"renamed": "Requête renommée",
"run": "Lancer",
"save": "Sauvegarder",
@@ -564,7 +563,7 @@
"response": {
"audio": "Audio",
"body": "Corps de réponse",
"filter_response_body": "Filter JSON response body (uses JSONPath syntax)",
"filter_response_body": "Filtrer le corps de la réponse JSON (utilise la syntaxe JSONPath)",
"headers": "En-têtes",
"html": "HTML",
"image": "Image",
@@ -576,14 +575,14 @@
"status": "Statut",
"time": "Temps",
"title": "Réponse",
"video": "Video",
"video": "Vidéo",
"waiting_for_connection": "En attente de connexion",
"xml": "XML"
},
"settings": {
"accent_color": "Couleur d'accent",
"account": "Compte",
"account_deleted": "Your account has been deleted",
"account_deleted": "Votre compte a été supprimé",
"account_description": "Personnalisez les paramètres de votre compte.",
"account_email_description": "Votre adresse e-mail principale.",
"account_name_description": "Ceci est votre nom d'affichage.",
@@ -592,8 +591,8 @@
"black_mode": "Noir",
"choose_language": "Choisissez la langue",
"dark_mode": "Sombre",
"delete_account": "Delete account",
"delete_account_description": "Once you delete your account, all your data will be permanently deleted. This action cannot be undone.",
"delete_account": "Supprimer le compte",
"delete_account_description": "Lorsque vous supprimez votre compte, toutes vos données sont définitivement effacées. Cette action ne peut être annulée.",
"expand_navigation": "Expand navigation",
"experiments": "Expériences",
"experiments_notice": "Il s'agit d'une collection d'expériences sur lesquelles nous travaillons et qui pourraient s'avérer utiles, amusantes, les deux ou aucune. Ils ne sont pas définitifs et peuvent ne pas être stables, donc si quelque chose de trop étrange se produit, ne paniquez pas. Il suffit d'éteindre le truc. Blague à part,",
@@ -601,7 +600,7 @@
"extension_version": "Version d'extension",
"extensions": "Extensions",
"extensions_use_toggle": "Utilisez l'extension de navigateur pour envoyer des requêtes (le cas échéant)",
"follow": "Follow Us",
"follow": "Suivez-nous",
"interceptor": "Intercepteur",
"interceptor_description": "Middleware entre l'application et les API.",
"language": "Langue",
@@ -631,7 +630,7 @@
"theme_description": "Personnalisez le thème de votre application.",
"use_experimental_url_bar": "Utiliser la barre d'URL expérimentale avec mise en évidence de l'environnement",
"user": "Utilisateur",
"verified_email": "Verified email",
"verified_email": "E-mail vérifié",
"verify_email": "Vérifier l'email"
},
"shared_requests": {
@@ -684,20 +683,20 @@
"title": "Navigation"
},
"others": {
"prettify": "Prettify Editor's Content",
"title": "Others"
"prettify": "Améliorer le contenu de l'éditeur",
"title": "Autres"
},
"request": {
"delete_method": "Sélectionnez la méthode DELETE",
"get_method": "Sélectionnez la méthode GET",
"head_method": "Sélectionnez la méthode HEAD",
"import_curl": "Import cURL",
"import_curl": "Importer cURL",
"method": "Méthode",
"next_method": "Sélectionnez la méthode suivante",
"post_method": "Sélectionnez la méthode POST",
"previous_method": "Sélectionnez la méthode précédente",
"put_method": "Sélectionnez la méthode PUT",
"rename": "Rename Request",
"rename": "Demande de renommage",
"reset_request": "Réinitialiser la requête",
"save_request": "Save Request",
"save_to_collections": "Enregistrer dans les collections",
@@ -728,89 +727,89 @@
},
"socketio": {
"communication": "Communication",
"connection_not_authorized": "This SocketIO connection does not use any authentication.",
"connection_not_authorized": "Cette connexion SocketIO n'utilise pas d'authentification.",
"event_name": "Nom de l'événement",
"events": "Événements",
"log": "Infos",
"url": "URL"
},
"spotlight": {
"change_language": "Change Language",
"change_language": "Changer de langue",
"environments": {
"delete": "Delete current environment",
"duplicate": "Duplicate current environment",
"duplicate_global": "Duplicate global environment",
"edit": "Edit current environment",
"edit_global": "Edit global environment",
"new": "Create new environment",
"new_variable": "Create a new environment variable",
"delete": "Supprimer l'environnement actuel",
"duplicate": "Dupliquer l'environnement actuel",
"duplicate_global": "Duplication de l'environnement global",
"edit": "Modifier l'environnement actuel",
"edit_global": "Modifier l'environnement mondial",
"new": "Créer un nouvel environnement",
"new_variable": "Créer une nouvelle variable d'environnement",
"title": "Environments"
},
"general": {
"chat": "Chat with support",
"help_menu": "Help and support",
"open_docs": "Read Documentation",
"open_github": "Open GitHub repository",
"open_keybindings": "Keyboard shortcuts",
"chat": "Chat avec le support",
"help_menu": "Aide et assistance",
"open_docs": "Lire la documentation",
"open_github": "Ouvrir le dépôt GitHub",
"open_keybindings": "Raccourcis clavier",
"social": "Social",
"title": "General"
"title": "Général"
},
"graphql": {
"connect": "Connect to server",
"disconnect": "Disconnect from server"
"connect": "Connexion au serveur",
"disconnect": "Déconnexion du serveur"
},
"miscellaneous": {
"invite": "Invite your friends to Hoppscotch",
"title": "Miscellaneous"
"invite": "Invitez vos amis à Hoppscotch",
"title": "Divers"
},
"request": {
"save_as_new": "Save as new request",
"select_method": "Select method",
"switch_to": "Switch to",
"tab_authorization": "Authorization tab",
"tab_body": "Body tab",
"tab_headers": "Headers tab",
"tab_parameters": "Parameters tab",
"tab_pre_request_script": "Pre-request script tab",
"tab_query": "Query tab",
"tab_tests": "Tests tab",
"tab_variables": "Variables tab"
"save_as_new": "Sauvegarder comme nouvelle demande",
"select_method": "Sélectionner la méthode",
"switch_to": "Basculer vers",
"tab_authorization": "Onglet Autorisation",
"tab_body": "Onglet du corps",
"tab_headers": "Onglet En-têtes",
"tab_parameters": "Onglet Paramètres",
"tab_pre_request_script": "Onglet script de pré-demande",
"tab_query": "Onglet Requête",
"tab_tests": "Onglet Tests",
"tab_variables": "Onglet Variables"
},
"response": {
"copy": "Copy response",
"download": "Download response as file",
"title": "Response"
"copy": "Copier la réponse",
"download": "Télécharger la réponse sous forme de fichier",
"title": "Réponse"
},
"section": {
"interceptor": "Interceptor",
"interceptor": "Intercepteur",
"interface": "Interface",
"theme": "Theme",
"user": "User"
"theme": "Thème",
"user": "Utilisateur"
},
"settings": {
"change_interceptor": "Change Interceptor",
"change_language": "Change Language",
"change_interceptor": "Changer d'intercepteur",
"change_language": "Changer de langue",
"theme": {
"black": "Black",
"dark": "Dark",
"light": "Light",
"system": "System preference"
"black": "Noir",
"dark": "Sombre",
"light": "Clair",
"system": "Préférence du système"
}
},
"tab": {
"close_current": "Close current tab",
"close_others": "Close all other tabs",
"duplicate": "Duplicate current tab",
"new_tab": "Open a new tab",
"title": "Tabs"
"close_current": "Fermer l'onglet actuel",
"close_others": "Fermer tous les autres onglets",
"duplicate": "Dupliquer l'onglet actuel",
"new_tab": "Ouvrir un nouvel onglet",
"title": "Onglets"
},
"workspace": {
"delete": "Delete current team",
"edit": "Edit current team",
"invite": "Invite people to team",
"new": "Create new team",
"switch_to_personal": "Switch to your personal workspace",
"title": "Teams"
"delete": "Supprimer l'équipe actuelle",
"edit": "Modifier l'équipe actuelle",
"invite": "Inviter les gens à rejoindre l'équipe",
"new": "Créer une nouvelle équipe",
"switch_to_personal": "Passez à votre espace de travail personnel",
"title": "Les équipes"
}
},
"sse": {
@@ -836,12 +835,12 @@
"disconnected": "Déconnecté",
"disconnected_from": "Déconnecté de {name}",
"docs_generated": "Documentation générée",
"download_failed": "Download failed",
"download_started": "Téléchargement commencé",
"download_failed": "Téléchargement échoué",
"enabled": "Active",
"file_imported": "Fichier importé",
"finished_in": "Terminé en {duration} ms",
"hide": "Hide",
"hide": "Cacher",
"history_deleted": "Historique supprimé",
"linewrap": "Retour à la ligne",
"loading": "Chargement...",
@@ -852,7 +851,7 @@
"published_error": "Quelque chose s'est mal passé lors de la publication du message : {topic} dans le sujet : {message}",
"published_message": "Message publié : {message} au sujet : {topic}",
"reconnection_error": "Échec de la reconnexion",
"show": "Show",
"show": "Afficher",
"subscribed_failed": "Échec de l'inscription au sujet : {topic}",
"subscribed_success": "Inscription réussie au sujet : {topic}",
"unsubscribed_failed": "Échec de la désinscription du sujet : {topic}",
@@ -861,7 +860,7 @@
},
"support": {
"changelog": "En savoir plus sur les dernières versions",
"chat": "Des questions? Discutez avec nous!",
"chat": "Des questions ? Discutez avec nous!",
"community": "Posez des questions et aidez les autres",
"documentation": "En savoir plus sur Hoppscotch",
"forum": "Posez des questions et obtenez des réponses",
@@ -874,21 +873,21 @@
"tab": {
"authorization": "Autorisation",
"body": "Corps",
"close": "Close Tab",
"close_others": "Close other Tabs",
"close": "Fermer l'onglet",
"close_others": "Fermer les autres onglets",
"collections": "Collections",
"documentation": "Documentation",
"duplicate": "Duplicate Tab",
"duplicate": "Dupliquer l'onglet",
"environments": "Environments",
"headers": "En-têtes",
"history": "Histoire",
"history": "Historique",
"mqtt": "MQTT",
"parameters": "Paramètres",
"pre_request_script": "Script de pré-requête",
"queries": "Requêtes",
"query": "Requête",
"schema": "Schema",
"shared_requests": "Shared Requests",
"shared_requests": "Requêtes partagées",
"socketio": "Socket.IO",
"sse": "ESS",
"tests": "Tests",
@@ -905,7 +904,6 @@
"email_do_not_match": "L'email ne correspond pas aux détails de votre compte. Contactez le propriétaire de votre équipe.",
"exit": "Quitter l'équipe",
"exit_disabled": "Seul le propriétaire ne peut pas quitter l'équipe",
"failed_invites": "Failed invites",
"invalid_coll_id": "Invalid collection ID",
"invalid_email_format": "Le format de l'e-mail n'est pas valide",
"invalid_id": "L'email ne correspond pas aux détails de votre compte. Contactez le propriétaire de votre équipe.",
@@ -941,21 +939,22 @@
"no_request_found": "Request not found.",
"not_found": "Équipe non trouvée. Contactez le propriétaire de votre équipe.",
"not_valid_viewer": "Vous n'êtes pas un visionneur valide. Contactez le propriétaire de votre équipe.",
"parent_coll_move": "Cannot move collection to a child collection",
"parent_coll_move": "Impossible de déplacer une collection vers une collection enfant",
"success_invites": "Les invitations réussites",
"pending_invites": "Invitations en attente",
"failed_invites": "Échec des invitations",
"permissions": "Autorisations",
"same_target_destination": "Same target and destination",
"same_target_destination": "me destinataire et même cible",
"saved": "Équipe enregistrée",
"select_a_team": "Choisir une équipe",
"success_invites": "Success invites",
"title": "Équipes",
"we_sent_invite_link": "Nous avons envoyé un lien d'invitation à tous les invités !",
"we_sent_invite_link_description": "Demandez à tous les invités de vérifier leur boîte de réception. Cliquez sur le lien pour rejoindre l'équipe."
},
"team_environment": {
"deleted": "Environment Deleted",
"duplicate": "Environment Duplicated",
"not_found": "Environment not found."
"deleted": "Environment supprimé",
"duplicate": "Environment dupliqué",
"not_found": "Environment non trouvé"
},
"test": {
"failed": "Test échoué",
@@ -975,10 +974,10 @@
"url": "URL"
},
"workspace": {
"change": "Change workspace",
"personal": "My Workspace",
"team": "Team Workspace",
"title": "Workspaces"
"change": "Changer d'espace de travail",
"personal": "Mon espace de travail",
"team": "Espace de travail de l'équipe",
"title": "Espaces de travail"
},
"shortcodes": {
"actions": "Actions",

View File

@@ -1,17 +1,17 @@
{
"action": {
"add": "Add",
"add": "Hozzáadás",
"autoscroll": "Automatikus görgetés",
"cancel": "Mégse",
"choose_file": "Válasszon egy fájlt",
"clear": "Törlés",
"clear_all": "Összes törlése",
"clear_history": "Clear all History",
"clear_history": "Összes előzmény törlése",
"close": "Bezárás",
"connect": "Kapcsolódás",
"connecting": "Kapcsolódás",
"copy": "Másolás",
"create": "Create",
"create": "Létrehozás",
"delete": "Törlés",
"disconnect": "Leválasztás",
"dismiss": "Eltüntetés",
@@ -33,16 +33,16 @@
"open_workspace": "Munkaterület megnyitása",
"paste": "Beillesztés",
"prettify": "Csinosítás",
"properties": "Properties",
"properties": "Tulajdonságok",
"remove": "Eltávolítás",
"rename": "Rename",
"rename": "Átnevezés",
"restore": "Visszaállítás",
"save": "Mentés",
"scroll_to_bottom": "Görgetés az aljára",
"scroll_to_top": "Görgetés a tetejére",
"search": "Keresés",
"send": "Küldés",
"share": "Share",
"share": "Megosztás",
"start": "Indítás",
"starting": "Indítás",
"stop": "Leállítás",
@@ -61,9 +61,9 @@
"app": {
"chat_with_us": "Csevegjen velünk",
"contact_us": "Lépjen kapcsolatba velünk",
"cookies": "Cookies",
"cookies": "Sütik",
"copy": "Másolás",
"copy_interface_type": "Copy interface type",
"copy_interface_type": "Interface típusának másolása",
"copy_user_id": "Felhasználó-hitelesítési token másolása",
"developer_option": "Fejlesztői beállítások",
"developer_option_description": "Fejlesztői eszközök, amelyek segítenek a Hoppscotch fejlesztésében és karbantartásában.",
@@ -79,15 +79,15 @@
"keyboard_shortcuts": "Gyorsbillentyűk",
"name": "Hoppscotch",
"new_version_found": "Új verzió található. Töltse újra az oldalt a frissítéshez.",
"open_in_hoppscotch": "Open in Hoppscotch",
"open_in_hoppscotch": "Megnyitás Hoppscotch-ban.",
"options": "Beállítások",
"proxy_privacy_policy": "Proxy adatvédelmi irányelvei",
"reload": "Újratöltés",
"search": "Keresés",
"share": "Megosztás",
"shortcuts": "Gyorsbillentyűk",
"social_description": "Follow us on social media to stay updated with the latest news, updates and releases.",
"social_links": "Social links",
"social_description": "Kövess minket a közösségi médiában, hogy ne maradj le a hírekről, frissítésekről és új kiadásokról.",
"social_links": "Közösségi média linkek",
"spotlight": "Reflektorfény",
"status": "Állapot",
"status_description": "A weboldal állapotának ellenőrzése",
@@ -119,27 +119,27 @@
},
"authorization": {
"generate_token": "Token előállítása",
"graphql_headers": "Authorization Headers are sent as part of the payload to connection_init",
"graphql_headers": "Azonosító fejléc connection_init tartalmaként elküldve",
"include_in_url": "Felvétel az URL-be",
"inherited_from": "Inherited from {auth} from Parent Collection {collection} ",
"inherited_from": "Örökölt a(z) {auth}-tól, a(z) {collection} gyűjteményből ",
"learn": "Tudja meg, hogyan",
"oauth": {
"redirect_auth_server_returned_error": "Auth Server returned an error state",
"redirect_auth_token_request_failed": "Request to get the auth token failed",
"redirect_auth_token_request_invalid_response": "Invalid Response from the Token Endpoint when requesting for an auth token",
"redirect_invalid_state": "Invalid State value present in the redirect",
"redirect_no_auth_code": "No Authorization Code present in the redirect",
"redirect_no_client_id": "No Client ID defined",
"redirect_no_client_secret": "No Client Secret Defined",
"redirect_no_code_verifier": "No Code Verifier Defined",
"redirect_no_token_endpoint": "No Token Endpoint Defined",
"something_went_wrong_on_oauth_redirect": "Something went wrong during OAuth Redirect",
"something_went_wrong_on_token_generation": "Something went wrong on token generation",
"token_generation_oidc_discovery_failed": "Failure on token generation: OpenID Connect Discovery Failed"
"redirect_auth_server_returned_error": "Az Auth szerver hibás állapottal tért vissza",
"redirect_auth_token_request_failed": "Kérés az auth token lekéréséhez sikertelen",
"redirect_auth_token_request_invalid_response": "Érvénytelen válasz a Token Endpoint-tól, az auth token lekérésekpr",
"redirect_invalid_state": "Érvénytelen állapotérték az átirányításban",
"redirect_no_auth_code": "Nincs azonosítás az átirányításban",
"redirect_no_client_id": "Nincs felhasználó azonosító",
"redirect_no_client_secret": "Nincs felhasználó jelszó",
"redirect_no_code_verifier": "Nincs kódellenőrző",
"redirect_no_token_endpoint": "Nincs \"Token Endpoint\"",
"something_went_wrong_on_oauth_redirect": "Valami rosszul sikerült az OAuth átirányításakor",
"something_went_wrong_on_token_generation": "Valami rosszul sikerült a token generálásakor",
"token_generation_oidc_discovery_failed": "Hiba a token generálásakor: OpenID Connect Discovery hiba"
},
"pass_key_by": "Átadta",
"password": "Jelszó",
"save_to_inherit": "Please save this request in any collection to inherit the authorization",
"save_to_inherit": "Kérjük, mentse el ezt kérést bármelyik gyűjteménybe, hogy az azonosítás örökölhető lehessen",
"token": "Token",
"type": "Felhatalmazás típusa",
"username": "Felhasználónév"
@@ -148,7 +148,7 @@
"created": "Gyűjtemény létrehozva",
"different_parent": "Nem lehet átrendezni a különböző szülővel rendelkező gyűjteményt",
"edit": "Gyűjtemény szerkesztése",
"import_or_create": "Import or create a collection",
"import_or_create": "Gyűjtemény importálása vagy létrehozása",
"invalid_name": "Adjon nevet a gyűjteménynek",
"invalid_root_move": "A gyűjtemény már a gyökérben van",
"moved": "Sikeresen áthelyezve",
@@ -157,20 +157,20 @@
"name_length_insufficient": "A gyűjtemény nevének legalább 3 karakter hosszúságúnak kell lennie",
"new": "Új gyűjtemény",
"order_changed": "Gyűjtemény sorrendje frissítve",
"properties": "Collection Properties",
"properties_updated": "Collection Properties Updated",
"properties": "Gyűjtemény tulajdonságok",
"properties_updated": "Gyűjtemény tulajdonságai frissítve",
"renamed": "Gyűjtemény átnevezve",
"request_in_use": "A kérés használatban",
"save_as": "Mentés másként",
"save_to_collection": "Save to Collection",
"save_to_collection": "Mentés egy gyűjteménybe",
"select": "Gyűjtemény kiválasztása",
"select_location": "Hely kiválasztása",
"select_team": "Csapat kiválasztása",
"team_collections": "Csapat gyűjteményei"
},
"confirm": {
"close_unsaved_tab": "Are you sure you want to close this tab?",
"close_unsaved_tabs": "Are you sure you want to close all tabs? {count} unsaved tabs will be lost.",
"close_unsaved_tab": "Biztos, hogy bezárja ezt a lapot?",
"close_unsaved_tabs": "Biztos, hogy bezárja az összes lapot? {count} elmentetlen lap el fog veszni.",
"exit_team": "Biztosan el szeretné hagyni ezt a csapatot?",
"logout": "Biztosan ki szeretne jelentkezni?",
"remove_collection": "Biztosan véglegesen törölni szeretné ezt a gyűjteményt?",
@@ -178,7 +178,7 @@
"remove_folder": "Biztosan véglegesen törölni szeretné ezt a mappát?",
"remove_history": "Biztosan véglegesen törölni szeretné az összes előzményt?",
"remove_request": "Biztosan véglegesen törölni szeretné ezt a kérést?",
"remove_shared_request": "Are you sure you want to permanently delete this shared request?",
"remove_shared_request": "Biztos, hogy véglegesen törölni szeretné ezt a megosztott kérést?",
"remove_team": "Biztosan törölni szeretné ezt a csapatot?",
"remove_telemetry": "Biztosan ki szeretné kapcsolni a telemetriát?",
"request_change": "Biztosan el szeretné vetni a jelenlegi kérést? Minden mentetlen változtatás el fog veszni.",
@@ -186,26 +186,26 @@
"sync": "Szeretné visszaállítani a munkaterületét a felhőből? Ez el fogja vetni a helyi folyamatát."
},
"context_menu": {
"add_parameters": "Add to parameters",
"open_request_in_new_tab": "Open request in new tab",
"set_environment_variable": "Set as variable"
"add_parameters": "Paraméterek hozzáadása",
"open_request_in_new_tab": "Kérés megnyitása új lapot",
"set_environment_variable": "Változóként való beállítás"
},
"cookies": {
"modal": {
"cookie_expires": "Expires",
"cookie_name": "Name",
"cookie_path": "Path",
"cookie_string": "Cookie string",
"cookie_value": "Value",
"empty_domain": "Domain is empty",
"empty_domains": "Domain list is empty",
"enter_cookie_string": "Enter cookie string",
"interceptor_no_support": "Your currently selected interceptor does not support cookies. Select a different Interceptor and try again.",
"managed_tab": "Managed",
"new_domain_name": "New domain name",
"no_cookies_in_domain": "No cookies set for this domain",
"raw_tab": "Raw",
"set": "Set a cookie"
"cookie_expires": "Lejárat",
"cookie_name": "Név",
"cookie_path": "Útvonal",
"cookie_string": "Süti szöveg",
"cookie_value": "Érték",
"empty_domain": "Üres domain",
"empty_domains": "Domain lista üres",
"enter_cookie_string": "Süti szövegének megadása",
"interceptor_no_support": "A kiválasztott interceptor nem támogatja a sütiket. Válasszon ki egy másik interceptor-t és próbálja újra.",
"managed_tab": "Menedzselt",
"new_domain_name": "Új domain neve",
"no_cookies_in_domain": "Nincs süti beállítva ehhez a domainhez.",
"raw_tab": "Nyers",
"set": "Süti beállítása"
}
},
"count": {
@@ -221,7 +221,7 @@
"generate_message": "Importáljon bármilyen Hoppscotch-gyűjteményt, hogy API-dokumentációt készítsen a folyamat során."
},
"empty": {
"authorization": "Ez a kérés nem használ felhatalmazást",
"authorization": "Ez a kérés nem használ azonosítást",
"body": "Ennek a kérésnek nincs törzse",
"collection": "A gyűjtemény üres",
"collections": "A gyűjtemények üresek",
@@ -252,39 +252,39 @@
"create_new": "Új környezet létrehozása",
"created": "Környezet létrehozva",
"deleted": "Környezet törlése",
"duplicated": "Environment duplicated",
"duplicated": "Környezet duplikálása",
"edit": "Környezet szerkesztése",
"empty_variables": "No variables",
"global": "Global",
"global_variables": "Global variables",
"import_or_create": "Import or create a environment",
"empty_variables": "Nincs változó",
"global": "Globális",
"global_variables": "Globális változók",
"import_or_create": "Környezet importálása vagy létrehozása",
"invalid_name": "Adjon nevet a környezetnek",
"list": "Environment variables",
"list": "Környezeti változók",
"my_environments": "Saját környezetek",
"name": "Name",
"name": "Név",
"nested_overflow": "az egymásba ágyazott környezeti változók 10 szintre vannak korlátozva",
"new": "Új környezet",
"no_active_environment": "No active environment",
"no_active_environment": "Nincs aktív környezet",
"no_environment": "Nincs környezet",
"no_environment_description": "Nem lettek környezetek kiválasztva. Válassza ki, hogy mit kell tenni a következő változókkal.",
"quick_peek": "Environment Quick Peek",
"replace_with_variable": "Replace with variable",
"scope": "Scope",
"quick_peek": "Környezet gyors megnézése",
"replace_with_variable": "Cserélje le egy változóra",
"scope": "Hatókör",
"select": "Környezet kiválasztása",
"set": "Set environment",
"set_as_environment": "Set as environment",
"set": "Környezet beállítása",
"set_as_environment": "Környezetként való beállítás",
"team_environments": "Csapatkörnyezetek",
"title": "Környezetek",
"updated": "Környezet frissítve",
"value": "Value",
"variable": "Variable",
"value": "Érték",
"variable": "Változó",
"variable_list": "Változólista"
},
"error": {
"authproviders_load_error": "Unable to load auth providers",
"authproviders_load_error": "Nem sikerült betölteni az azonosító szolgáltatókat",
"browser_support_sse": "Úgy tűnik, hogy ez a böngésző nem támogatja a kiszolgáló által küldött eseményeket.",
"check_console_details": "Nézze meg a konzolnaplót a részletekért.",
"check_how_to_add_origin": "Check how you can add an origin",
"check_how_to_add_origin": "Ellenőrizze, hogy hogyan adhat hozzá forrást",
"curl_invalid_format": "A cURL nincs megfelelően formázva",
"danger_zone": "Veszélyes zóna",
"delete_account": "Az Ön fiókja jelenleg tulajdonos ezekben a csapatokban:",
@@ -300,13 +300,13 @@
"json_prettify_invalid_body": "Nem sikerült csinosítani egy érvénytelen törzset, oldja meg a JSON szintaktikai hibáit, és próbálja újra",
"network_error": "Úgy tűnik, hogy hálózati hiba van. Próbálja újra.",
"network_fail": "Nem sikerült elküldeni a kérést",
"no_collections_to_export": "No collections to export. Please create a collection to get started.",
"no_collections_to_export": "Nincs exportálható gyűjtemény. Kérjük, hozzon létre egyet, hogy elkezdhesse.",
"no_duration": "Nincs időtartam",
"no_environments_to_export": "No environments to export. Please create an environment to get started.",
"no_environments_to_export": "Nincs exportálható környezet. Kérjük, hozzon létre egyet, hogy elkezdhesse.",
"no_results_found": "Nincs találat",
"page_not_found": "Ez az oldal nem található",
"please_install_extension": "Please install the extension and add origin to the extension.",
"proxy_error": "Proxy error",
"please_install_extension": "Kérjük telepítse a bővítményt és adja hozzá a forráshoz.",
"proxy_error": "Proxy hiba",
"script_fail": "Nem sikerült végrehajtani a kérés előtti parancsfájlt",
"something_went_wrong": "Valami elromlott",
"test_script_fail": "Nem sikerült végrehajtani a kérés utáni parancsfájlt"
@@ -314,7 +314,7 @@
"export": {
"as_json": "Exportálás JSON formátumban",
"create_secret_gist": "Titkos Gist létrehozása",
"failed": "Something went wrong while exporting",
"failed": "Valami hiba történt az exportálás közben",
"gist_created": "Gist létrehozva",
"require_github": "Jelentkezzen be GitHub használatával a titkos Gist létrehozásához",
"title": "Exportálás"
@@ -333,16 +333,16 @@
"renamed": "Mappa átnevezve"
},
"graphql": {
"connection_switch_confirm": "Do you want to connect with the latest GraphQL endpoint?",
"connection_switch_new_url": "Switching to a tab will disconnected you from the active GraphQL connection. New connection URL is",
"connection_switch_url": "You're connected to a GraphQL endpoint the connection URL is",
"connection_switch_confirm": "Szeretne csatlakozni a legújabb GraphQL végponttal?",
"connection_switch_new_url": "A tab váltása lecsatlakoztatta az aktív GraphQL kapcsolatról. Az új kapcsolat",
"connection_switch_url": "Kapcsolódott a GraphQL végponthoz. A kapcsolat",
"mutations": "Mutációk",
"schema": "Séma",
"subscriptions": "Feliratkozások",
"switch_connection": "Switch connection"
"switch_connection": "Kapcsolat váltása"
},
"graphql_collections": {
"title": "GraphQL Collections"
"title": "GraphQL gyűjtemény"
},
"group": {
"time": "Idő",
@@ -354,9 +354,9 @@
"save_workspace": "Saját munkaterület mentése"
},
"helpers": {
"authorization": "A felhatalmazási fejléc automatikusan elő lesz állítva a kérés elküldésekor.",
"collection_properties_authorization": " This authorization will be set for every request in this collection.",
"collection_properties_header": "This header will be set for every request in this collection.",
"authorization": "Az Azonosítás fejléc automatikusan elő lesz állítva a kérés elküldésekor.",
"collection_properties_authorization": " Ez az azonosítás be lesz állítva minden kéréshez ebben a gyűjteményben.",
"collection_properties_header": "Ez a fejléc be lesz állítva mint minden kéréshez ebben a gyűjteményben.",
"generate_documentation_first": "Először állítsa elő a dokumentációt",
"network_fail": "Nem lehet elérni az API-végpontot. Ellenőrizze a hálózati kapcsolatot vagy válasszon egy másik elfogót, és próbálja újra.",
"offline": "Úgy tűnik, hogy kapcsolat nélküli módban van. Előfordulhat, hogy a munkaterületen lévő adatok nem naprakészek.",
@@ -376,11 +376,11 @@
"import": {
"collections": "Gyűjtemények importálása",
"curl": "cURL importálása",
"environments_from_gist": "Import From Gist",
"environments_from_gist_description": "Import Hoppscotch Environments From Gist",
"environments_from_gist": "Importálás Gist-ből",
"environments_from_gist_description": "Hoppscotch környezetek importálása Gist-ből",
"failed": "Hiba az importálás során: a formátum nem azonosítható",
"from_file": "Import from File",
"from_gist": "Importálás Gistből",
"from_file": "Importálás fájlból",
"from_gist": "Importálás Gist-ből",
"from_gist_description": "Importálás Gist URL-ből",
"from_insomnia": "Importálás Insomniából",
"from_insomnia_description": "Importálás Insomnia-gyűjteményből",
@@ -390,45 +390,45 @@
"from_my_collections_description": "Importálás saját gyűjtemények fájlból",
"from_openapi": "Importálás OpenAPI-ból",
"from_openapi_description": "Importálás OpenAPI specifikációs fájlból (YML/JSON)",
"from_postman": "Importálás Postmanből",
"from_postman": "Importálás Postman-ből",
"from_postman_description": "Importálás Postman-gyűjteményből",
"from_url": "Importálás URL-ből",
"gist_url": "Gist URL megadása",
"gql_collections_from_gist_description": "Import GraphQL Collections From Gist",
"hoppscotch_environment": "Hoppscotch Environment",
"hoppscotch_environment_description": "Import Hoppscotch Environment JSON file",
"gql_collections_from_gist_description": "GraphQL gyűjtemények importálása Gist-ből",
"hoppscotch_environment": "Hoppscotch környezet",
"hoppscotch_environment_description": "Hoppscotch környezet importálása JSON fájlból",
"import_from_url_invalid_fetch": "Nem sikerült lekérni az adatokat az URL-ről",
"import_from_url_invalid_file_format": "Hiba a gyűjtemények importálása során",
"import_from_url_invalid_type": "Nem támogatott típus. Az elfogadott értékek: „hoppscotch”, „openapi”, „postman” vagy „insomnia”.",
"import_from_url_success": "Gyűjtemények importálva",
"insomnia_environment_description": "Import Insomnia Environment from a JSON/YAML file",
"insomnia_environment_description": "Insomnia környezet importálása JSON/YAML fájlból",
"json_description": "Gyűjtemények importálása Hoppscotch-gyűjtemények JSON-fájlból",
"postman_environment": "Postman Environment",
"postman_environment_description": "Import Postman Environment from a JSON file",
"postman_environment": "Postman környezet",
"postman_environment_description": "Postman környezet importálása JSON fájlból",
"title": "Importálás"
},
"inspections": {
"description": "Inspect possible errors",
"description": "Lehetséges hibák ellenőrzése",
"environment": {
"add_environment": "Add to Environment",
"not_found": "Environment variable “{environment}” not found."
"add_environment": "Hozzáadás a környezethez",
"not_found": "\"{environment}\" környezet nem található."
},
"header": {
"cookie": "The browser doesn't allow Hoppscotch to set the Cookie Header. While we're working on the Hoppscotch Desktop App (coming soon), please use the Authorization Header instead."
"cookie": "A böngésző nem engedélyezi a Hoppscotch-nak, hogy süti fejlécet állítson be. Amíg dolgozunk a Hoppscotch asztali alkalmazáson (hamarosan), kérjük használjon Authorization Header fejlécet."
},
"response": {
"401_error": "Please check your authentication credentials.",
"404_error": "Please check your request URL and method type.",
"cors_error": "Please check your Cross-Origin Resource Sharing configuration.",
"default_error": "Please check your request.",
"network_error": "Please check your network connection."
"401_error": "Kérjük ellenőrizze az autentikációs adatokat.",
"404_error": "Kérjük ellenőrizze a kérés URL-jét és típusát.",
"cors_error": "Kérjük ellenőrizze a Cross-Origin Resource Sharing beállítást.",
"default_error": "Kérjük ellenőrizze a kérését.",
"network_error": "Kérjük ellenőrizze a internet elérhetőségét."
},
"title": "Inspector",
"url": {
"extension_not_installed": "Extension not installed.",
"extension_unknown_origin": "Make sure you've added the API endpoint's origin to the Hoppscotch Browser Extension list.",
"extention_enable_action": "Enable Browser Extension",
"extention_not_enabled": "Extension not enabled."
"extension_not_installed": "Bővítmény nincs telepítve.",
"extension_unknown_origin": "Ellenőrizze, hogy hozzáadta az API végpont forrását Hoppscotch Browser bővítmény listájához.",
"extention_enable_action": "Bővítmény engedélyezése",
"extention_not_enabled": "Bővítmény nincs engedélyezve."
}
},
"layout": {
@@ -442,10 +442,10 @@
"close_unsaved_tab": "Elmentetlen változtatásai vannak",
"collections": "Gyűjtemények",
"confirm": "Megerősítés",
"customize_request": "Customize Request",
"customize_request": "Kérés testreszabása",
"edit_request": "Kérés szerkesztése",
"import_export": "Importálás és exportálás",
"share_request": "Share Request"
"share_request": "Kérés megosztása"
},
"mqtt": {
"already_subscribed": "Ön már feliratkozott erre a témára.",
@@ -511,7 +511,7 @@
},
"request": {
"added": "Kérés hozzáadva",
"authorization": "Felhatalmazás",
"authorization": "Azonosítás",
"body": "Kérés törzse",
"choose_language": "Nyelv kiválasztása",
"content_type": "Tartalom típusa",
@@ -526,8 +526,8 @@
"enter_curl": "cURL-parancs megadása",
"generate_code": "Kód előállítása",
"generated_code": "Előállított kód",
"go_to_authorization_tab": "Go to Authorization tab",
"go_to_body_tab": "Go to Body tab",
"go_to_authorization_tab": "Navigálás az Azonosítás lapra",
"go_to_body_tab": "Navigálás a Törzs lapra.",
"header_list": "Fejléclista",
"invalid_name": "Adjon nevet a kérésnek",
"method": "Módszer",
@@ -552,8 +552,8 @@
"saved": "Kérés elmentve",
"share": "Megosztás",
"share_description": "A Hoppscotch megosztása az ismerőseivel",
"share_request": "Share Request",
"stop": "Stop",
"share_request": "Kérés megosztása",
"stop": "Leállítás",
"title": "Kérés",
"type": "Kérés típusa",
"url": "URL",
@@ -587,7 +587,7 @@
"account_description": "A fiókbeállítások személyre szabása.",
"account_email_description": "Az Ön elsődleges e-mail-címe.",
"account_name_description": "Ez a megjelenített neve.",
"additional": "Additional Settings",
"additional": "További beállítások",
"background": "Háttér",
"black_mode": "Fekete",
"choose_language": "Nyelv kiválasztása",
@@ -635,29 +635,29 @@
"verify_email": "E-mail-cím ellenőrzése"
},
"shared_requests": {
"button": "Button",
"button_info": "Create a 'Run in Hoppscotch' button for your website, blog or a README.",
"copy_html": "Copy HTML",
"copy_link": "Copy Link",
"copy_markdown": "Copy Markdown",
"creating_widget": "Creating widget",
"customize": "Customize",
"deleted": "Shared request deleted",
"description": "Select a widget, you can change and customize this later",
"embed": "Embed",
"embed_info": "Add a mini 'Hoppscotch API Playground' to your website, blog or documentation.",
"button": "Gomb",
"button_info": "Hozza létre a 'Futtatás Hoppscotch-ban' gombot a weboldalán vagy blogján.",
"copy_html": "HTML másolása",
"copy_link": "Link másolása",
"copy_markdown": "Jelölő másolása",
"creating_widget": "Widget létrehozása",
"customize": "Testreszabás",
"deleted": "Megosztott kérés törölve",
"description": "Válasszon ki egy widgetet, ezt később módosíthatja és testreszabhatja",
"embed": "Beágyazás",
"embed_info": "Adja hozzá a \"Hoppscotch API Playground\"-ot weboldalához vagy blogjához.",
"link": "Link",
"link_info": "Create a shareable link to share with anyone on the internet with view access.",
"modified": "Shared request modified",
"not_found": "Shared request not found",
"open_new_tab": "Open in new tab",
"preview": "Preview",
"run_in_hoppscotch": "Run in Hoppscotch",
"link_info": "Link létrehozása olvasási joggal való megosztáshoz.",
"modified": "Megosztott kérés módosítva",
"not_found": "Megosztott kérés nem található.",
"open_new_tab": "Megnyitás új lapon.",
"preview": "Előnézet",
"run_in_hoppscotch": "Futtatás Hoppscotch-ban.",
"theme": {
"dark": "Dark",
"light": "Light",
"system": "System",
"title": "Theme"
"dark": "Sötét",
"light": "Világos",
"system": "Rendszer",
"title": "Téma"
}
},
"shortcut": {
@@ -669,7 +669,7 @@
"title": "Általános"
},
"miscellaneous": {
"invite": "Emberek meghívása a Hoppscotchba",
"invite": "Emberek meghívása a Hoppscotch-ba",
"title": "Egyebek"
},
"navigation": {
@@ -691,19 +691,19 @@
"delete_method": "DELETE módszer kiválasztása",
"get_method": "GET módszer kiválasztása",
"head_method": "HEAD módszer kiválasztása",
"import_curl": "Import cURL",
"import_curl": "cURL importálása",
"method": "Módszer",
"next_method": "Következő módszer kiválasztása",
"post_method": "POST módszer kiválasztása",
"previous_method": "Előző módszer kiválasztása",
"put_method": "PUT módszer kiválasztása",
"rename": "Rename Request",
"rename": "Kérés átnevezése",
"reset_request": "Kérés visszaállítása",
"save_request": "Save Request",
"save_request": "Kérés mentése",
"save_to_collections": "Mentés a gyűjteményekbe",
"send_request": "Kérés elküldése",
"share_request": "Share Request",
"show_code": "Generate code snippet",
"share_request": "Kérés megosztása",
"show_code": "Kódrészlet generálása",
"title": "Kérés",
"copy_request_link": "Kérés hivatkozásának másolása"
},
@@ -735,82 +735,82 @@
"url": "URL"
},
"spotlight": {
"change_language": "Change Language",
"change_language": "Nyelv váltása",
"environments": {
"delete": "Delete current environment",
"duplicate": "Duplicate current environment",
"duplicate_global": "Duplicate global environment",
"edit": "Edit current environment",
"edit_global": "Edit global environment",
"new": "Create new environment",
"new_variable": "Create a new environment variable",
"title": "Environments"
"delete": "Jelenlegi környezet törlése",
"duplicate": "Jelenlegi környezet duplikálása",
"duplicate_global": "Globális környezet duplikálása",
"edit": "Jelenlegi környezet szerkesztése",
"edit_global": "Globális környezet szerkesztése",
"new": "Új környezet létrehozása",
"new_variable": "Új környezeti változó létrehozása",
"title": "Környezetek"
},
"general": {
"chat": "Chat with support",
"help_menu": "Help and support",
"open_docs": "Read Documentation",
"open_github": "Open GitHub repository",
"open_keybindings": "Keyboard shortcuts",
"social": "Social",
"title": "General"
"chat": "Üzenet a supportnak",
"help_menu": "Segítség és support",
"open_docs": "Dokumentáció olvasása",
"open_github": "GitHub repository megnyitása",
"open_keybindings": "Billentyűkombinációk megnyitása",
"social": "Közösség",
"title": "Általános"
},
"graphql": {
"connect": "Connect to server",
"disconnect": "Disconnect from server"
"connect": "Csatlakozás a szerverhez",
"disconnect": "Lecsatlakozás a szerverről"
},
"miscellaneous": {
"invite": "Invite your friends to Hoppscotch",
"title": "Miscellaneous"
"invite": "Hívja meg barátait a Hoppscotch-ba",
"title": "Egyéb"
},
"request": {
"save_as_new": "Save as new request",
"select_method": "Select method",
"switch_to": "Switch to",
"tab_authorization": "Authorization tab",
"tab_body": "Body tab",
"tab_headers": "Headers tab",
"tab_parameters": "Parameters tab",
"tab_pre_request_script": "Pre-request script tab",
"tab_query": "Query tab",
"tab_tests": "Tests tab",
"tab_variables": "Variables tab"
"save_as_new": "Mentés új kérésként",
"select_method": "Módszer kiválasztása",
"switch_to": "Váltás",
"tab_authorization": "Azonosítás lap",
"tab_body": "Törzs lap",
"tab_headers": "Fejlécek lap",
"tab_parameters": "Paraméterek lap",
"tab_pre_request_script": "Előzetes script lap",
"tab_query": "Kérés lap",
"tab_tests": "Tesztek lap",
"tab_variables": "Változók lap"
},
"response": {
"copy": "Copy response",
"download": "Download response as file",
"title": "Response"
"copy": "Válasz másolása",
"download": "Válasz letöltése fájlként",
"title": "Válasz"
},
"section": {
"interceptor": "Interceptor",
"interface": "Interface",
"theme": "Theme",
"user": "User"
"theme": "Téma",
"user": "Felhasználó"
},
"settings": {
"change_interceptor": "Change Interceptor",
"change_language": "Change Language",
"change_interceptor": "Interceptor váltása",
"change_language": "Nyelv váltása",
"theme": {
"black": "Black",
"dark": "Dark",
"light": "Light",
"system": "System preference"
"black": "Fekete",
"dark": "Sötét",
"light": "Világos",
"system": "Rendszer"
}
},
"tab": {
"close_current": "Close current tab",
"close_others": "Close all other tabs",
"duplicate": "Duplicate current tab",
"new_tab": "Open a new tab",
"title": "Tabs"
"close_current": "Jelenlegi lap bezására",
"close_others": "Összes többi lap bezására",
"duplicate": "Jelenlegi lap diplikálása",
"new_tab": "Új lap megnyitása",
"title": "Lapok"
},
"workspace": {
"delete": "Delete current team",
"edit": "Edit current team",
"invite": "Invite people to team",
"new": "Create new team",
"switch_to_personal": "Switch to your personal workspace",
"title": "Teams"
"delete": "Jelenlegi csapat törlése",
"edit": "Jelenlegi csapat szerkesztése",
"invite": "Emberek meghívása a jelenlegi csapatba",
"new": "Új csapat létrehozása",
"switch_to_personal": "Váltás a személyes munkaterületére",
"title": "Csapatok"
}
},
"sse": {
@@ -828,20 +828,20 @@
"connection_error": "Nem sikerült kapcsolódni",
"connection_failed": "A kapcsolódás sikertelen",
"connection_lost": "A kapcsolat elveszett",
"copied_interface_to_clipboard": "Copied {language} interface type to clipboard",
"copied_interface_to_clipboard": "{language} interface típusa vágólapra másolva",
"copied_to_clipboard": "Vágólapra másolva",
"deleted": "Törölve",
"deprecated": "ELAVULT",
"disabled": "Letiltva",
"disconnected": "Leválasztva",
"disconnected_from": "Leválasztva innen: {name}",
"disconnected": "Lecsatlakoztatva",
"disconnected_from": "Lecsatlakoztatva innen: {name}",
"docs_generated": "Dokumentáció előállítva",
"download_failed": "Download failed",
"download_failed": "Letöltés sikertelen",
"download_started": "A letöltés elkezdődött",
"enabled": "Engedélyezve",
"file_imported": "Fájl importálva",
"finished_in": "Befejeződött {duration} ms alatt",
"hide": "Hide",
"hide": "Elrejtés",
"history_deleted": "Előzmények törölve",
"linewrap": "Sorok tördelése",
"loading": "Betöltés…",
@@ -872,13 +872,13 @@
"twitter": "Kövessen minket Twitteren"
},
"tab": {
"authorization": "Felhatalmazás",
"authorization": "Azonosítás",
"body": "Törzs",
"close": "Close Tab",
"close_others": "Close other Tabs",
"close": "Lap bezárása",
"close_others": "Többi lap bezárása",
"collections": "Gyűjtemények",
"documentation": "Dokumentáció",
"duplicate": "Duplicate Tab",
"duplicate": "Lap duplikálása",
"environments": "Környezetek",
"headers": "Fejlécek",
"history": "Előzmények",
@@ -905,7 +905,7 @@
"email_do_not_match": "Az e-mail-cím nem egyezik a fiókja részleteivel. Vegye fel a kapcsolatot a csapat tulajdonosával.",
"exit": "Kilépés a csapatból",
"exit_disabled": "Csak a tulajdonos nem léphet ki a csapatból",
"failed_invites": "Failed invites",
"failed_invites": "Hiba a meghívás közben",
"invalid_coll_id": "Érvénytelen gyűjteményazonosító",
"invalid_email_format": "Az e-mail formátuma érvénytelen",
"invalid_id": "Érvénytelen csapatazonosító. Vegye fel a kapcsolatot a csapat tulajdonosával.",

View File

@@ -1,7 +1,7 @@
{
"name": "@hoppscotch/common",
"private": true,
"version": "2024.3.0",
"version": "2024.3.1",
"scripts": {
"dev": "pnpm exec npm-run-all -p -l dev:*",
"test": "vitest --run",

View File

@@ -3,214 +3,214 @@
// @ts-nocheck
// Generated by unplugin-vue-components
// Read more: https://github.com/vuejs/core/pull/3399
import "@vue/runtime-core"
export {}
declare module "@vue/runtime-core" {
declare module 'vue' {
export interface GlobalComponents {
AppActionHandler: (typeof import("./components/app/ActionHandler.vue"))["default"]
AppBanner: (typeof import("./components/app/Banner.vue"))["default"]
AppContextMenu: (typeof import("./components/app/ContextMenu.vue"))["default"]
AppDeveloperOptions: (typeof import("./components/app/DeveloperOptions.vue"))["default"]
AppFooter: (typeof import("./components/app/Footer.vue"))["default"]
AppGitHubStarButton: (typeof import("./components/app/GitHubStarButton.vue"))["default"]
AppHeader: (typeof import("./components/app/Header.vue"))["default"]
AppInspection: (typeof import("./components/app/Inspection.vue"))["default"]
AppInterceptor: (typeof import("./components/app/Interceptor.vue"))["default"]
AppLogo: (typeof import("./components/app/Logo.vue"))["default"]
AppOptions: (typeof import("./components/app/Options.vue"))["default"]
AppPaneLayout: (typeof import("./components/app/PaneLayout.vue"))["default"]
AppShare: (typeof import("./components/app/Share.vue"))["default"]
AppShortcuts: (typeof import("./components/app/Shortcuts.vue"))["default"]
AppShortcutsEntry: (typeof import("./components/app/ShortcutsEntry.vue"))["default"]
AppShortcutsPrompt: (typeof import("./components/app/ShortcutsPrompt.vue"))["default"]
AppSidenav: (typeof import("./components/app/Sidenav.vue"))["default"]
AppSpotlight: (typeof import("./components/app/spotlight/index.vue"))["default"]
AppSpotlightEntry: (typeof import("./components/app/spotlight/Entry.vue"))["default"]
AppSpotlightEntryGQLHistory: (typeof import("./components/app/spotlight/entry/GQLHistory.vue"))["default"]
AppSpotlightEntryGQLRequest: (typeof import("./components/app/spotlight/entry/GQLRequest.vue"))["default"]
AppSpotlightEntryIconSelected: (typeof import("./components/app/spotlight/entry/IconSelected.vue"))["default"]
AppSpotlightEntryRESTHistory: (typeof import("./components/app/spotlight/entry/RESTHistory.vue"))["default"]
AppSpotlightEntryRESTRequest: (typeof import("./components/app/spotlight/entry/RESTRequest.vue"))["default"]
AppSupport: (typeof import("./components/app/Support.vue"))["default"]
Collections: (typeof import("./components/collections/index.vue"))["default"]
CollectionsAdd: (typeof import("./components/collections/Add.vue"))["default"]
CollectionsAddFolder: (typeof import("./components/collections/AddFolder.vue"))["default"]
CollectionsAddRequest: (typeof import("./components/collections/AddRequest.vue"))["default"]
CollectionsCollection: (typeof import("./components/collections/Collection.vue"))["default"]
CollectionsEdit: (typeof import("./components/collections/Edit.vue"))["default"]
CollectionsEditFolder: (typeof import("./components/collections/EditFolder.vue"))["default"]
CollectionsEditRequest: (typeof import("./components/collections/EditRequest.vue"))["default"]
CollectionsGraphql: (typeof import("./components/collections/graphql/index.vue"))["default"]
CollectionsGraphqlAdd: (typeof import("./components/collections/graphql/Add.vue"))["default"]
CollectionsGraphqlAddFolder: (typeof import("./components/collections/graphql/AddFolder.vue"))["default"]
CollectionsGraphqlAddRequest: (typeof import("./components/collections/graphql/AddRequest.vue"))["default"]
CollectionsGraphqlCollection: (typeof import("./components/collections/graphql/Collection.vue"))["default"]
CollectionsGraphqlEdit: (typeof import("./components/collections/graphql/Edit.vue"))["default"]
CollectionsGraphqlEditFolder: (typeof import("./components/collections/graphql/EditFolder.vue"))["default"]
CollectionsGraphqlEditRequest: (typeof import("./components/collections/graphql/EditRequest.vue"))["default"]
CollectionsGraphqlFolder: (typeof import("./components/collections/graphql/Folder.vue"))["default"]
CollectionsGraphqlImportExport: (typeof import("./components/collections/graphql/ImportExport.vue"))["default"]
CollectionsGraphqlRequest: (typeof import("./components/collections/graphql/Request.vue"))["default"]
CollectionsImportExport: (typeof import("./components/collections/ImportExport.vue"))["default"]
CollectionsMyCollections: (typeof import("./components/collections/MyCollections.vue"))["default"]
CollectionsProperties: (typeof import("./components/collections/Properties.vue"))["default"]
CollectionsRequest: (typeof import("./components/collections/Request.vue"))["default"]
CollectionsSaveRequest: (typeof import("./components/collections/SaveRequest.vue"))["default"]
CollectionsTeamCollections: (typeof import("./components/collections/TeamCollections.vue"))["default"]
CookiesAllModal: (typeof import("./components/cookies/AllModal.vue"))["default"]
CookiesEditCookie: (typeof import("./components/cookies/EditCookie.vue"))["default"]
Embeds: (typeof import("./components/embeds/index.vue"))["default"]
Environments: (typeof import("./components/environments/index.vue"))["default"]
EnvironmentsAdd: (typeof import("./components/environments/Add.vue"))["default"]
EnvironmentsImportExport: (typeof import("./components/environments/ImportExport.vue"))["default"]
EnvironmentsMy: (typeof import("./components/environments/my/index.vue"))["default"]
EnvironmentsMyDetails: (typeof import("./components/environments/my/Details.vue"))["default"]
EnvironmentsMyEnvironment: (typeof import("./components/environments/my/Environment.vue"))["default"]
EnvironmentsSelector: (typeof import("./components/environments/Selector.vue"))["default"]
EnvironmentsTeams: (typeof import("./components/environments/teams/index.vue"))["default"]
EnvironmentsTeamsDetails: (typeof import("./components/environments/teams/Details.vue"))["default"]
EnvironmentsTeamsEnvironment: (typeof import("./components/environments/teams/Environment.vue"))["default"]
FirebaseLogin: (typeof import("./components/firebase/Login.vue"))["default"]
FirebaseLogout: (typeof import("./components/firebase/Logout.vue"))["default"]
GraphqlAuthorization: (typeof import("./components/graphql/Authorization.vue"))["default"]
GraphqlField: (typeof import("./components/graphql/Field.vue"))["default"]
GraphqlHeaders: (typeof import("./components/graphql/Headers.vue"))["default"]
GraphqlQuery: (typeof import("./components/graphql/Query.vue"))["default"]
GraphqlRequest: (typeof import("./components/graphql/Request.vue"))["default"]
GraphqlRequestOptions: (typeof import("./components/graphql/RequestOptions.vue"))["default"]
GraphqlRequestTab: (typeof import("./components/graphql/RequestTab.vue"))["default"]
GraphqlResponse: (typeof import("./components/graphql/Response.vue"))["default"]
GraphqlSidebar: (typeof import("./components/graphql/Sidebar.vue"))["default"]
GraphqlSubscriptionLog: (typeof import("./components/graphql/SubscriptionLog.vue"))["default"]
GraphqlTabHead: (typeof import("./components/graphql/TabHead.vue"))["default"]
GraphqlType: (typeof import("./components/graphql/Type.vue"))["default"]
GraphqlTypeLink: (typeof import("./components/graphql/TypeLink.vue"))["default"]
GraphqlVariable: (typeof import("./components/graphql/Variable.vue"))["default"]
History: (typeof import("./components/history/index.vue"))["default"]
HistoryGraphqlCard: (typeof import("./components/history/graphql/Card.vue"))["default"]
HistoryRestCard: (typeof import("./components/history/rest/Card.vue"))["default"]
HoppButtonPrimary: (typeof import("@hoppscotch/ui"))["HoppButtonPrimary"]
HoppButtonSecondary: (typeof import("@hoppscotch/ui"))["HoppButtonSecondary"]
HoppSmartAnchor: (typeof import("@hoppscotch/ui"))["HoppSmartAnchor"]
HoppSmartCheckbox: (typeof import("@hoppscotch/ui"))["HoppSmartCheckbox"]
HoppSmartConfirmModal: (typeof import("@hoppscotch/ui"))["HoppSmartConfirmModal"]
HoppSmartExpand: (typeof import("@hoppscotch/ui"))["HoppSmartExpand"]
HoppSmartFileChip: (typeof import("@hoppscotch/ui"))["HoppSmartFileChip"]
HoppSmartInput: (typeof import("@hoppscotch/ui"))["HoppSmartInput"]
HoppSmartIntersection: (typeof import("@hoppscotch/ui"))["HoppSmartIntersection"]
HoppSmartItem: (typeof import("@hoppscotch/ui"))["HoppSmartItem"]
HoppSmartLink: (typeof import("@hoppscotch/ui"))["HoppSmartLink"]
HoppSmartModal: (typeof import("@hoppscotch/ui"))["HoppSmartModal"]
HoppSmartPicture: (typeof import("@hoppscotch/ui"))["HoppSmartPicture"]
HoppSmartPlaceholder: (typeof import("@hoppscotch/ui"))["HoppSmartPlaceholder"]
HoppSmartProgressRing: (typeof import("@hoppscotch/ui"))["HoppSmartProgressRing"]
HoppSmartRadio: (typeof import("@hoppscotch/ui"))["HoppSmartRadio"]
HoppSmartRadioGroup: (typeof import("@hoppscotch/ui"))["HoppSmartRadioGroup"]
HoppSmartSelectWrapper: (typeof import("@hoppscotch/ui"))["HoppSmartSelectWrapper"]
HoppSmartSlideOver: (typeof import("@hoppscotch/ui"))["HoppSmartSlideOver"]
HoppSmartSpinner: (typeof import("@hoppscotch/ui"))["HoppSmartSpinner"]
HoppSmartTab: (typeof import("@hoppscotch/ui"))["HoppSmartTab"]
HoppSmartTabs: (typeof import("@hoppscotch/ui"))["HoppSmartTabs"]
HoppSmartToggle: (typeof import("@hoppscotch/ui"))["HoppSmartToggle"]
HoppSmartTree: (typeof import("@hoppscotch/ui"))["HoppSmartTree"]
HoppSmartWindow: (typeof import("@hoppscotch/ui"))["HoppSmartWindow"]
HoppSmartWindows: (typeof import("@hoppscotch/ui"))["HoppSmartWindows"]
HttpAuthorization: (typeof import("./components/http/Authorization.vue"))["default"]
HttpAuthorizationApiKey: (typeof import("./components/http/authorization/ApiKey.vue"))["default"]
HttpAuthorizationBasic: (typeof import("./components/http/authorization/Basic.vue"))["default"]
HttpBody: (typeof import("./components/http/Body.vue"))["default"]
HttpBodyParameters: (typeof import("./components/http/BodyParameters.vue"))["default"]
HttpCodegenModal: (typeof import("./components/http/CodegenModal.vue"))["default"]
HttpHeaders: (typeof import("./components/http/Headers.vue"))["default"]
HttpImportCurl: (typeof import("./components/http/ImportCurl.vue"))["default"]
HttpOAuth2Authorization: (typeof import("./components/http/OAuth2Authorization.vue"))["default"]
HttpParameters: (typeof import("./components/http/Parameters.vue"))["default"]
HttpPreRequestScript: (typeof import("./components/http/PreRequestScript.vue"))["default"]
HttpRawBody: (typeof import("./components/http/RawBody.vue"))["default"]
HttpReqChangeConfirmModal: (typeof import("./components/http/ReqChangeConfirmModal.vue"))["default"]
HttpRequest: (typeof import("./components/http/Request.vue"))["default"]
HttpRequestOptions: (typeof import("./components/http/RequestOptions.vue"))["default"]
HttpRequestTab: (typeof import("./components/http/RequestTab.vue"))["default"]
HttpResponse: (typeof import("./components/http/Response.vue"))["default"]
HttpResponseMeta: (typeof import("./components/http/ResponseMeta.vue"))["default"]
HttpSidebar: (typeof import("./components/http/Sidebar.vue"))["default"]
HttpTabHead: (typeof import("./components/http/TabHead.vue"))["default"]
HttpTestResult: (typeof import("./components/http/TestResult.vue"))["default"]
HttpTestResultEntry: (typeof import("./components/http/TestResultEntry.vue"))["default"]
HttpTestResultEnv: (typeof import("./components/http/TestResultEnv.vue"))["default"]
HttpTestResultReport: (typeof import("./components/http/TestResultReport.vue"))["default"]
HttpTests: (typeof import("./components/http/Tests.vue"))["default"]
HttpURLEncodedParams: (typeof import("./components/http/URLEncodedParams.vue"))["default"]
IconLucideActivity: (typeof import("~icons/lucide/activity"))["default"]
IconLucideAlertTriangle: (typeof import("~icons/lucide/alert-triangle"))["default"]
IconLucideArrowLeft: (typeof import("~icons/lucide/arrow-left"))["default"]
IconLucideArrowUpRight: (typeof import("~icons/lucide/arrow-up-right"))["default"]
AppActionHandler: typeof import('./components/app/ActionHandler.vue')['default']
AppBanner: typeof import('./components/app/Banner.vue')['default']
AppContextMenu: typeof import('./components/app/ContextMenu.vue')['default']
AppDeveloperOptions: typeof import('./components/app/DeveloperOptions.vue')['default']
AppFooter: typeof import('./components/app/Footer.vue')['default']
AppGitHubStarButton: typeof import('./components/app/GitHubStarButton.vue')['default']
AppHeader: typeof import('./components/app/Header.vue')['default']
AppInspection: typeof import('./components/app/Inspection.vue')['default']
AppInterceptor: typeof import('./components/app/Interceptor.vue')['default']
AppLogo: typeof import('./components/app/Logo.vue')['default']
AppOptions: typeof import('./components/app/Options.vue')['default']
AppPaneLayout: typeof import('./components/app/PaneLayout.vue')['default']
AppShare: typeof import('./components/app/Share.vue')['default']
AppShortcuts: typeof import('./components/app/Shortcuts.vue')['default']
AppShortcutsEntry: typeof import('./components/app/ShortcutsEntry.vue')['default']
AppShortcutsPrompt: typeof import('./components/app/ShortcutsPrompt.vue')['default']
AppSidenav: typeof import('./components/app/Sidenav.vue')['default']
AppSpotlight: typeof import('./components/app/spotlight/index.vue')['default']
AppSpotlightEntry: typeof import('./components/app/spotlight/Entry.vue')['default']
AppSpotlightEntryGQLHistory: typeof import('./components/app/spotlight/entry/GQLHistory.vue')['default']
AppSpotlightEntryGQLRequest: typeof import('./components/app/spotlight/entry/GQLRequest.vue')['default']
AppSpotlightEntryIconSelected: typeof import('./components/app/spotlight/entry/IconSelected.vue')['default']
AppSpotlightEntryRESTHistory: typeof import('./components/app/spotlight/entry/RESTHistory.vue')['default']
AppSpotlightEntryRESTRequest: typeof import('./components/app/spotlight/entry/RESTRequest.vue')['default']
AppSpotlightEntryRESTTeamRequestEntry: typeof import('./components/app/spotlight/entry/RESTTeamRequestEntry.vue')['default']
AppSupport: typeof import('./components/app/Support.vue')['default']
Collections: typeof import('./components/collections/index.vue')['default']
CollectionsAdd: typeof import('./components/collections/Add.vue')['default']
CollectionsAddFolder: typeof import('./components/collections/AddFolder.vue')['default']
CollectionsAddRequest: typeof import('./components/collections/AddRequest.vue')['default']
CollectionsCollection: typeof import('./components/collections/Collection.vue')['default']
CollectionsEdit: typeof import('./components/collections/Edit.vue')['default']
CollectionsEditFolder: typeof import('./components/collections/EditFolder.vue')['default']
CollectionsEditRequest: typeof import('./components/collections/EditRequest.vue')['default']
CollectionsGraphql: typeof import('./components/collections/graphql/index.vue')['default']
CollectionsGraphqlAdd: typeof import('./components/collections/graphql/Add.vue')['default']
CollectionsGraphqlAddFolder: typeof import('./components/collections/graphql/AddFolder.vue')['default']
CollectionsGraphqlAddRequest: typeof import('./components/collections/graphql/AddRequest.vue')['default']
CollectionsGraphqlCollection: typeof import('./components/collections/graphql/Collection.vue')['default']
CollectionsGraphqlEdit: typeof import('./components/collections/graphql/Edit.vue')['default']
CollectionsGraphqlEditFolder: typeof import('./components/collections/graphql/EditFolder.vue')['default']
CollectionsGraphqlEditRequest: typeof import('./components/collections/graphql/EditRequest.vue')['default']
CollectionsGraphqlFolder: typeof import('./components/collections/graphql/Folder.vue')['default']
CollectionsGraphqlImportExport: typeof import('./components/collections/graphql/ImportExport.vue')['default']
CollectionsGraphqlRequest: typeof import('./components/collections/graphql/Request.vue')['default']
CollectionsImportExport: typeof import('./components/collections/ImportExport.vue')['default']
CollectionsMyCollections: typeof import('./components/collections/MyCollections.vue')['default']
CollectionsProperties: typeof import('./components/collections/Properties.vue')['default']
CollectionsRequest: typeof import('./components/collections/Request.vue')['default']
CollectionsSaveRequest: typeof import('./components/collections/SaveRequest.vue')['default']
CollectionsTeamCollections: typeof import('./components/collections/TeamCollections.vue')['default']
CookiesAllModal: typeof import('./components/cookies/AllModal.vue')['default']
CookiesEditCookie: typeof import('./components/cookies/EditCookie.vue')['default']
Embeds: typeof import('./components/embeds/index.vue')['default']
Environments: typeof import('./components/environments/index.vue')['default']
EnvironmentsAdd: typeof import('./components/environments/Add.vue')['default']
EnvironmentsImportExport: typeof import('./components/environments/ImportExport.vue')['default']
EnvironmentsMy: typeof import('./components/environments/my/index.vue')['default']
EnvironmentsMyDetails: typeof import('./components/environments/my/Details.vue')['default']
EnvironmentsMyEnvironment: typeof import('./components/environments/my/Environment.vue')['default']
EnvironmentsSelector: typeof import('./components/environments/Selector.vue')['default']
EnvironmentsTeams: typeof import('./components/environments/teams/index.vue')['default']
EnvironmentsTeamsDetails: typeof import('./components/environments/teams/Details.vue')['default']
EnvironmentsTeamsEnvironment: typeof import('./components/environments/teams/Environment.vue')['default']
FirebaseLogin: typeof import('./components/firebase/Login.vue')['default']
FirebaseLogout: typeof import('./components/firebase/Logout.vue')['default']
GraphqlAuthorization: typeof import('./components/graphql/Authorization.vue')['default']
GraphqlField: typeof import('./components/graphql/Field.vue')['default']
GraphqlHeaders: typeof import('./components/graphql/Headers.vue')['default']
GraphqlQuery: typeof import('./components/graphql/Query.vue')['default']
GraphqlRequest: typeof import('./components/graphql/Request.vue')['default']
GraphqlRequestOptions: typeof import('./components/graphql/RequestOptions.vue')['default']
GraphqlRequestTab: typeof import('./components/graphql/RequestTab.vue')['default']
GraphqlResponse: typeof import('./components/graphql/Response.vue')['default']
GraphqlSidebar: typeof import('./components/graphql/Sidebar.vue')['default']
GraphqlSubscriptionLog: typeof import('./components/graphql/SubscriptionLog.vue')['default']
GraphqlTabHead: typeof import('./components/graphql/TabHead.vue')['default']
GraphqlType: typeof import('./components/graphql/Type.vue')['default']
GraphqlTypeLink: typeof import('./components/graphql/TypeLink.vue')['default']
GraphqlVariable: typeof import('./components/graphql/Variable.vue')['default']
History: typeof import('./components/history/index.vue')['default']
HistoryGraphqlCard: typeof import('./components/history/graphql/Card.vue')['default']
HistoryRestCard: typeof import('./components/history/rest/Card.vue')['default']
HoppButtonPrimary: typeof import('@hoppscotch/ui')['HoppButtonPrimary']
HoppButtonSecondary: typeof import('@hoppscotch/ui')['HoppButtonSecondary']
HoppSmartAnchor: typeof import('@hoppscotch/ui')['HoppSmartAnchor']
HoppSmartCheckbox: typeof import('@hoppscotch/ui')['HoppSmartCheckbox']
HoppSmartConfirmModal: typeof import('@hoppscotch/ui')['HoppSmartConfirmModal']
HoppSmartExpand: typeof import('@hoppscotch/ui')['HoppSmartExpand']
HoppSmartFileChip: typeof import('@hoppscotch/ui')['HoppSmartFileChip']
HoppSmartInput: typeof import('@hoppscotch/ui')['HoppSmartInput']
HoppSmartIntersection: typeof import('@hoppscotch/ui')['HoppSmartIntersection']
HoppSmartItem: typeof import('@hoppscotch/ui')['HoppSmartItem']
HoppSmartLink: typeof import('@hoppscotch/ui')['HoppSmartLink']
HoppSmartModal: typeof import('@hoppscotch/ui')['HoppSmartModal']
HoppSmartPicture: typeof import('@hoppscotch/ui')['HoppSmartPicture']
HoppSmartPlaceholder: typeof import('@hoppscotch/ui')['HoppSmartPlaceholder']
HoppSmartProgressRing: typeof import('@hoppscotch/ui')['HoppSmartProgressRing']
HoppSmartRadio: typeof import('@hoppscotch/ui')['HoppSmartRadio']
HoppSmartRadioGroup: typeof import('@hoppscotch/ui')['HoppSmartRadioGroup']
HoppSmartSelectWrapper: typeof import('@hoppscotch/ui')['HoppSmartSelectWrapper']
HoppSmartSlideOver: typeof import('@hoppscotch/ui')['HoppSmartSlideOver']
HoppSmartSpinner: typeof import('@hoppscotch/ui')['HoppSmartSpinner']
HoppSmartTab: typeof import('@hoppscotch/ui')['HoppSmartTab']
HoppSmartTabs: typeof import('@hoppscotch/ui')['HoppSmartTabs']
HoppSmartToggle: typeof import('@hoppscotch/ui')['HoppSmartToggle']
HoppSmartTree: typeof import('@hoppscotch/ui')['HoppSmartTree']
HoppSmartWindow: typeof import('@hoppscotch/ui')['HoppSmartWindow']
HoppSmartWindows: typeof import('@hoppscotch/ui')['HoppSmartWindows']
HttpAuthorization: typeof import('./components/http/Authorization.vue')['default']
HttpAuthorizationApiKey: typeof import('./components/http/authorization/ApiKey.vue')['default']
HttpAuthorizationBasic: typeof import('./components/http/authorization/Basic.vue')['default']
HttpBody: typeof import('./components/http/Body.vue')['default']
HttpBodyParameters: typeof import('./components/http/BodyParameters.vue')['default']
HttpCodegenModal: typeof import('./components/http/CodegenModal.vue')['default']
HttpHeaders: typeof import('./components/http/Headers.vue')['default']
HttpImportCurl: typeof import('./components/http/ImportCurl.vue')['default']
HttpOAuth2Authorization: typeof import('./components/http/OAuth2Authorization.vue')['default']
HttpParameters: typeof import('./components/http/Parameters.vue')['default']
HttpPreRequestScript: typeof import('./components/http/PreRequestScript.vue')['default']
HttpRawBody: typeof import('./components/http/RawBody.vue')['default']
HttpReqChangeConfirmModal: typeof import('./components/http/ReqChangeConfirmModal.vue')['default']
HttpRequest: typeof import('./components/http/Request.vue')['default']
HttpRequestOptions: typeof import('./components/http/RequestOptions.vue')['default']
HttpRequestTab: typeof import('./components/http/RequestTab.vue')['default']
HttpRequestVariables: typeof import('./components/http/RequestVariables.vue')['default']
HttpResponse: typeof import('./components/http/Response.vue')['default']
HttpResponseMeta: typeof import('./components/http/ResponseMeta.vue')['default']
HttpSidebar: typeof import('./components/http/Sidebar.vue')['default']
HttpTabHead: typeof import('./components/http/TabHead.vue')['default']
HttpTestResult: typeof import('./components/http/TestResult.vue')['default']
HttpTestResultEntry: typeof import('./components/http/TestResultEntry.vue')['default']
HttpTestResultEnv: typeof import('./components/http/TestResultEnv.vue')['default']
HttpTestResultReport: typeof import('./components/http/TestResultReport.vue')['default']
HttpTests: typeof import('./components/http/Tests.vue')['default']
HttpURLEncodedParams: typeof import('./components/http/URLEncodedParams.vue')['default']
IconLucideActivity: typeof import('~icons/lucide/activity')['default']
IconLucideAlertTriangle: typeof import('~icons/lucide/alert-triangle')['default']
IconLucideArrowLeft: typeof import('~icons/lucide/arrow-left')['default']
IconLucideArrowUpRight: typeof import('~icons/lucide/arrow-up-right')['default']
IconLucideBrush: (typeof import("~icons/lucide/brush"))["default"]
IconLucideCheckCircle: (typeof import("~icons/lucide/check-circle"))["default"]
IconLucideChevronRight: (typeof import("~icons/lucide/chevron-right"))["default"]
IconLucideGlobe: (typeof import("~icons/lucide/globe"))["default"]
IconLucideHelpCircle: (typeof import("~icons/lucide/help-circle"))["default"]
IconLucideInbox: (typeof import("~icons/lucide/inbox"))["default"]
IconLucideInfo: (typeof import("~icons/lucide/info"))["default"]
IconLucideLayers: (typeof import("~icons/lucide/layers"))["default"]
IconLucideListEnd: (typeof import("~icons/lucide/list-end"))["default"]
IconLucideMinus: (typeof import("~icons/lucide/minus"))["default"]
IconLucideCheckCircle: typeof import('~icons/lucide/check-circle')['default']
IconLucideChevronRight: typeof import('~icons/lucide/chevron-right')['default']
IconLucideGlobe: typeof import('~icons/lucide/globe')['default']
IconLucideHelpCircle: typeof import('~icons/lucide/help-circle')['default']
IconLucideInbox: typeof import('~icons/lucide/inbox')['default']
IconLucideInfo: typeof import('~icons/lucide/info')['default']
IconLucideLayers: typeof import('~icons/lucide/layers')['default']
IconLucideListEnd: typeof import('~icons/lucide/list-end')['default']
IconLucideMinus: typeof import('~icons/lucide/minus')['default']
IconLucideRss: (typeof import("~icons/lucide/rss"))["default"]
IconLucideSearch: (typeof import("~icons/lucide/search"))["default"]
IconLucideUsers: (typeof import("~icons/lucide/users"))["default"]
IconLucideX: (typeof import("~icons/lucide/x"))["default"]
ImportExportBase: (typeof import("./components/importExport/Base.vue"))["default"]
ImportExportImportExportList: (typeof import("./components/importExport/ImportExportList.vue"))["default"]
ImportExportImportExportSourcesList: (typeof import("./components/importExport/ImportExportSourcesList.vue"))["default"]
ImportExportImportExportStepsFileImport: (typeof import("./components/importExport/ImportExportSteps/FileImport.vue"))["default"]
ImportExportImportExportStepsMyCollectionImport: (typeof import("./components/importExport/ImportExportSteps/MyCollectionImport.vue"))["default"]
ImportExportImportExportStepsUrlImport: (typeof import("./components/importExport/ImportExportSteps/UrlImport.vue"))["default"]
InterceptorsErrorPlaceholder: (typeof import("./components/interceptors/ErrorPlaceholder.vue"))["default"]
InterceptorsExtensionSubtitle: (typeof import("./components/interceptors/ExtensionSubtitle.vue"))["default"]
LensesHeadersRenderer: (typeof import("./components/lenses/HeadersRenderer.vue"))["default"]
LensesHeadersRendererEntry: (typeof import("./components/lenses/HeadersRendererEntry.vue"))["default"]
LensesRenderersAudioLensRenderer: (typeof import("./components/lenses/renderers/AudioLensRenderer.vue"))["default"]
LensesRenderersHTMLLensRenderer: (typeof import("./components/lenses/renderers/HTMLLensRenderer.vue"))["default"]
LensesRenderersImageLensRenderer: (typeof import("./components/lenses/renderers/ImageLensRenderer.vue"))["default"]
LensesRenderersJSONLensRenderer: (typeof import("./components/lenses/renderers/JSONLensRenderer.vue"))["default"]
LensesRenderersPDFLensRenderer: (typeof import("./components/lenses/renderers/PDFLensRenderer.vue"))["default"]
LensesRenderersRawLensRenderer: (typeof import("./components/lenses/renderers/RawLensRenderer.vue"))["default"]
LensesRenderersVideoLensRenderer: (typeof import("./components/lenses/renderers/VideoLensRenderer.vue"))["default"]
LensesRenderersXMLLensRenderer: (typeof import("./components/lenses/renderers/XMLLensRenderer.vue"))["default"]
LensesResponseBodyRenderer: (typeof import("./components/lenses/ResponseBodyRenderer.vue"))["default"]
ProfileUserDelete: (typeof import("./components/profile/UserDelete.vue"))["default"]
RealtimeCommunication: (typeof import("./components/realtime/Communication.vue"))["default"]
RealtimeConnectionConfig: (typeof import("./components/realtime/ConnectionConfig.vue"))["default"]
RealtimeLog: (typeof import("./components/realtime/Log.vue"))["default"]
RealtimeLogEntry: (typeof import("./components/realtime/LogEntry.vue"))["default"]
RealtimeSubscription: (typeof import("./components/realtime/Subscription.vue"))["default"]
SettingsExtension: (typeof import("./components/settings/Extension.vue"))["default"]
SettingsProxy: (typeof import("./components/settings/Proxy.vue"))["default"]
Share: (typeof import("./components/share/index.vue"))["default"]
ShareCreateModal: (typeof import("./components/share/CreateModal.vue"))["default"]
ShareCustomizeModal: (typeof import("./components/share/CustomizeModal.vue"))["default"]
ShareModal: (typeof import("./components/share/Modal.vue"))["default"]
ShareRequest: (typeof import("./components/share/Request.vue"))["default"]
ShareTemplatesButton: (typeof import("./components/share/templates/Button.vue"))["default"]
ShareTemplatesEmbeds: (typeof import("./components/share/templates/Embeds.vue"))["default"]
ShareTemplatesLink: (typeof import("./components/share/templates/Link.vue"))["default"]
SmartAccentModePicker: (typeof import("./components/smart/AccentModePicker.vue"))["default"]
SmartChangeLanguage: (typeof import("./components/smart/ChangeLanguage.vue"))["default"]
SmartColorModePicker: (typeof import("./components/smart/ColorModePicker.vue"))["default"]
SmartEnvInput: (typeof import("./components/smart/EnvInput.vue"))["default"]
TabPrimary: (typeof import("./components/tab/Primary.vue"))["default"]
TabSecondary: (typeof import("./components/tab/Secondary.vue"))["default"]
Teams: (typeof import("./components/teams/index.vue"))["default"]
TeamsAdd: (typeof import("./components/teams/Add.vue"))["default"]
TeamsEdit: (typeof import("./components/teams/Edit.vue"))["default"]
TeamsInvite: (typeof import("./components/teams/Invite.vue"))["default"]
TeamsMemberStack: (typeof import("./components/teams/MemberStack.vue"))["default"]
TeamsModal: (typeof import("./components/teams/Modal.vue"))["default"]
TeamsTeam: (typeof import("./components/teams/Team.vue"))["default"]
Tippy: (typeof import("vue-tippy"))["Tippy"]
WorkspaceCurrent: (typeof import("./components/workspace/Current.vue"))["default"]
WorkspaceSelector: (typeof import("./components/workspace/Selector.vue"))["default"]
IconLucideSearch: typeof import('~icons/lucide/search')['default']
IconLucideUsers: typeof import('~icons/lucide/users')['default']
IconLucideX: typeof import('~icons/lucide/x')['default']
ImportExportBase: typeof import('./components/importExport/Base.vue')['default']
ImportExportImportExportList: typeof import('./components/importExport/ImportExportList.vue')['default']
ImportExportImportExportSourcesList: typeof import('./components/importExport/ImportExportSourcesList.vue')['default']
ImportExportImportExportStepsFileImport: typeof import('./components/importExport/ImportExportSteps/FileImport.vue')['default']
ImportExportImportExportStepsMyCollectionImport: typeof import('./components/importExport/ImportExportSteps/MyCollectionImport.vue')['default']
ImportExportImportExportStepsUrlImport: typeof import('./components/importExport/ImportExportSteps/UrlImport.vue')['default']
InterceptorsErrorPlaceholder: typeof import('./components/interceptors/ErrorPlaceholder.vue')['default']
InterceptorsExtensionSubtitle: typeof import('./components/interceptors/ExtensionSubtitle.vue')['default']
LensesHeadersRenderer: typeof import('./components/lenses/HeadersRenderer.vue')['default']
LensesHeadersRendererEntry: typeof import('./components/lenses/HeadersRendererEntry.vue')['default']
LensesRenderersAudioLensRenderer: typeof import('./components/lenses/renderers/AudioLensRenderer.vue')['default']
LensesRenderersHTMLLensRenderer: typeof import('./components/lenses/renderers/HTMLLensRenderer.vue')['default']
LensesRenderersImageLensRenderer: typeof import('./components/lenses/renderers/ImageLensRenderer.vue')['default']
LensesRenderersJSONLensRenderer: typeof import('./components/lenses/renderers/JSONLensRenderer.vue')['default']
LensesRenderersPDFLensRenderer: typeof import('./components/lenses/renderers/PDFLensRenderer.vue')['default']
LensesRenderersRawLensRenderer: typeof import('./components/lenses/renderers/RawLensRenderer.vue')['default']
LensesRenderersVideoLensRenderer: typeof import('./components/lenses/renderers/VideoLensRenderer.vue')['default']
LensesRenderersXMLLensRenderer: typeof import('./components/lenses/renderers/XMLLensRenderer.vue')['default']
LensesResponseBodyRenderer: typeof import('./components/lenses/ResponseBodyRenderer.vue')['default']
ProfileUserDelete: typeof import('./components/profile/UserDelete.vue')['default']
RealtimeCommunication: typeof import('./components/realtime/Communication.vue')['default']
RealtimeConnectionConfig: typeof import('./components/realtime/ConnectionConfig.vue')['default']
RealtimeLog: typeof import('./components/realtime/Log.vue')['default']
RealtimeLogEntry: typeof import('./components/realtime/LogEntry.vue')['default']
RealtimeSubscription: typeof import('./components/realtime/Subscription.vue')['default']
SettingsExtension: typeof import('./components/settings/Extension.vue')['default']
SettingsProxy: typeof import('./components/settings/Proxy.vue')['default']
Share: typeof import('./components/share/index.vue')['default']
ShareCreateModal: typeof import('./components/share/CreateModal.vue')['default']
ShareCustomizeModal: typeof import('./components/share/CustomizeModal.vue')['default']
ShareModal: typeof import('./components/share/Modal.vue')['default']
ShareRequest: typeof import('./components/share/Request.vue')['default']
ShareTemplatesButton: typeof import('./components/share/templates/Button.vue')['default']
ShareTemplatesEmbeds: typeof import('./components/share/templates/Embeds.vue')['default']
ShareTemplatesLink: typeof import('./components/share/templates/Link.vue')['default']
SmartAccentModePicker: typeof import('./components/smart/AccentModePicker.vue')['default']
SmartChangeLanguage: typeof import('./components/smart/ChangeLanguage.vue')['default']
SmartColorModePicker: typeof import('./components/smart/ColorModePicker.vue')['default']
SmartEnvInput: typeof import('./components/smart/EnvInput.vue')['default']
TabPrimary: typeof import('./components/tab/Primary.vue')['default']
TabSecondary: typeof import('./components/tab/Secondary.vue')['default']
Teams: typeof import('./components/teams/index.vue')['default']
TeamsAdd: typeof import('./components/teams/Add.vue')['default']
TeamsEdit: typeof import('./components/teams/Edit.vue')['default']
TeamsInvite: typeof import('./components/teams/Invite.vue')['default']
TeamsMemberStack: typeof import('./components/teams/MemberStack.vue')['default']
TeamsModal: typeof import('./components/teams/Modal.vue')['default']
TeamsTeam: typeof import('./components/teams/Team.vue')['default']
Tippy: typeof import('vue-tippy')['Tippy']
WorkspaceCurrent: typeof import('./components/workspace/Current.vue')['default']
WorkspaceSelector: typeof import('./components/workspace/Selector.vue')['default']
}
}

View File

@@ -21,19 +21,7 @@
</div>
</div>
<div class="col-span-1 flex items-center justify-between space-x-2">
<button
class="flex h-full flex-1 cursor-text items-center justify-between self-stretch rounded border border-dividerDark bg-primaryDark px-2 text-secondaryLight transition hover:border-dividerDark hover:bg-primaryLight hover:text-secondary focus-visible:border-dividerDark focus-visible:bg-primaryLight focus-visible:text-secondary"
@click="invokeAction('modals.search.toggle', undefined, 'mouseclick')"
>
<span class="inline-flex flex-1 items-center">
<icon-lucide-search class="svg-icons mr-2" />
{{ t("app.search") }}
</span>
<span class="flex space-x-1">
<kbd class="shortcut-key">{{ getPlatformSpecialKey() }}</kbd>
<kbd class="shortcut-key">K</kbd>
</span>
</button>
<AppSpotlightSearch />
</div>
<div class="col-span-2 flex items-center justify-between space-x-2">
<div class="flex">
@@ -251,7 +239,6 @@ import { breakpointsTailwind, useBreakpoints, useNetwork } from "@vueuse/core"
import { computed, reactive, ref, watch } from "vue"
import { useToast } from "~/composables/toast"
import { GetMyTeamsQuery, TeamMemberRole } from "~/helpers/backend/graphql"
import { getPlatformSpecialKey } from "~/helpers/platformutils"
import { platform } from "~/platform"
import IconDownload from "~icons/lucide/download"
import IconLifeBuoy from "~icons/lucide/life-buoy"

View File

@@ -0,0 +1,135 @@
<template>
<div
class="border-animation relative p-[1px] rounded flex-1 self-stretch overflow-hidden flex items-center justify-center"
:class="{
'before:top-1/2 before:left-1/2 before:-translate-x-1/2 before:-translate-y-1/2 before:aspect-square before:w-full before:absolute before:bg-':
!HAS_OPENED_SPOTLIGHT,
}"
aria-hidden="true"
>
<button
class="relative flex flex-1 cursor-text items-center justify-between self-stretch rounded bg-primaryDark px-2 text-secondaryLight transition hover:border-dividerDark hover:bg-primaryLight hover:text-secondary focus-visible:border-dividerDark focus-visible:bg-primaryLight focus-visible:text-secondary overflow-hidden"
@click="
() => {
invokeAction('modals.search.toggle', undefined, 'mouseclick')
!HAS_OPENED_SPOTLIGHT && toggleSetting('HAS_OPENED_SPOTLIGHT')
}
"
>
<span class="inline-flex flex-1 items-center">
<icon-lucide-search class="svg-icons mr-2" />
<span v-if="!HAS_OPENED_SPOTLIGHT" class="flex flex-1">
{{ t("spotlight.phrases.try") }}
<TransitionGroup tag="div" name="list" class="ml-1 relative">
<span
v-for="(phrase, index) in phraseToShow"
:key="phrase.text"
:data-index="index"
class="truncate"
>
"{{ t(phrase.text) }}"
</span>
</TransitionGroup>
</span>
<template v-else>
{{ t("app.search") }}
</template>
</span>
<span class="flex space-x-1">
<kbd class="shortcut-key">{{ getPlatformSpecialKey() }}</kbd>
<kbd class="shortcut-key">K</kbd>
</span>
</button>
</div>
</template>
<script lang="ts" setup>
import { watch, computed, ref } from "vue"
import { useI18n } from "~/composables/i18n"
import { useSetting } from "~/composables/settings"
import { invokeAction } from "~/helpers/actions"
import { getPlatformSpecialKey } from "~/helpers/platformutils"
import { toggleSetting } from "~/newstore/settings"
const t = useI18n()
const HAS_OPENED_SPOTLIGHT = useSetting("HAS_OPENED_SPOTLIGHT")
const phrases = ref([
{ text: "spotlight.phrases.import_collections", show: true },
{ text: "spotlight.phrases.create_environment", show: false },
{ text: "spotlight.phrases.create_workspace", show: false },
{ text: "spotlight.phrases.share_request", show: false },
])
let intervalId: ReturnType<typeof setTimeout> | null = null
//cycle through the phrases
const showNextPhrase = () => {
let i = 0
intervalId = setInterval(() => {
phrases.value[i].show = false
i++
if (i >= phrases.value.length) {
i = 0
}
phrases.value[i].show = true
}, 3000)
}
const stopPhraseInterval = () => {
if (intervalId) clearInterval(intervalId)
}
const phraseToShow = computed(() => {
return phrases.value.filter((phrase) => phrase.show)
})
watch(
HAS_OPENED_SPOTLIGHT,
() => {
!HAS_OPENED_SPOTLIGHT.value ? showNextPhrase() : stopPhraseInterval()
},
{
immediate: true,
}
)
</script>
<style>
/* Transition Classes */
.list-enter-active {
transition: all 1s ease;
}
.list-leave-active {
transition: all 0.4s ease;
}
.list-enter-from,
.list-leave-to {
opacity: 0;
transform: translateY(-30px);
}
.list-leave-active {
position: absolute;
}
/* Conic gradient */
.border-animation::before {
background: conic-gradient(
transparent 270deg,
var(--accent-color),
transparent
);
animation: rotate 4s linear infinite;
}
@keyframes rotate {
from {
transform: translate(-50%, -50%) scale(1.4) rotate(0turn);
}
to {
transform: translate(-50%, -50%) scale(1.4) rotate(1turn);
}
}
</style>

View File

@@ -86,6 +86,8 @@ import { platform } from "~/platform"
import { useService } from "dioc/vue"
import { RESTTabService } from "~/services/tab/rest"
import { GQLTabService } from "~/services/tab/graphql"
import { getDefaultRESTRequest } from "~/helpers/rest/default"
import { getDefaultGQLRequest } from "~/helpers/graphql/default"
const t = useI18n()
const toast = useToast()
@@ -221,6 +223,15 @@ const saveRequestAs = async () => {
requestUpdated.name = requestName.value
if (props.mode === "rest") {
;(requestUpdated as HoppRESTRequest).endpoint =
(requestUpdated as HoppRESTRequest).endpoint ||
getDefaultRESTRequest().endpoint
} else {
;(requestUpdated as HoppGQLRequest).url =
(requestUpdated as HoppGQLRequest).url || getDefaultGQLRequest().url
}
if (picked.value.pickedType === "my-collection") {
if (!isHoppRESTRequest(requestUpdated))
throw new Error("requestUpdated is not a REST Request")

View File

@@ -192,6 +192,8 @@ import { PersistenceService } from "~/services/persistence"
import { PersistedOAuthConfig } from "~/services/oauth/oauth.service"
import { GQLOptionTabs } from "~/components/graphql/RequestOptions.vue"
import { EditingProperties } from "../Properties.vue"
import { defineActionHandler } from "~/helpers/actions"
import { getDefaultGQLRequest } from "~/helpers/graphql/default"
const t = useI18n()
const toast = useToast()
@@ -379,32 +381,26 @@ const editCollection = (
displayModalEdit(true)
}
const onAddRequest = ({
name,
path,
index,
}: {
name: string
path: string
index: number
}) => {
const onAddRequest = ({ name, path }: { name: string; path: string }) => {
const newRequest = {
...tabs.currentActiveTab.value.document.request,
name,
url:
tabs.currentActiveTab.value.document.request.url ||
getDefaultGQLRequest().url,
}
saveGraphqlRequestAs(path, newRequest)
const insertionIndex = saveGraphqlRequestAs(path, newRequest)
const { auth, headers } = cascadeParentCollectionForHeaderAuth(
path,
"graphql"
)
tabs.createNewTab({
saveContext: {
originLocation: "user-collection",
folderPath: path,
requestIndex: index,
requestIndex: insertionIndex,
},
request: newRequest,
isDirty: false,
@@ -676,4 +672,11 @@ const resetSelectedData = () => {
editingRequest.value = null
editingRequestIndex.value = null
}
defineActionHandler("collection.new", () => {
displayModalAdd(true)
})
defineActionHandler("modals.collection.import", () => {
displayModalImportExport(true)
})
</script>

View File

@@ -254,6 +254,7 @@ import { PersistenceService } from "~/services/persistence"
import { PersistedOAuthConfig } from "~/services/oauth/oauth.service"
import { RESTOptionTabs } from "../http/RequestOptions.vue"
import { EditingProperties } from "./Properties.vue"
import { getDefaultRESTRequest } from "~/helpers/rest/default"
const t = useI18n()
const toast = useToast()
@@ -790,6 +791,9 @@ const onAddRequest = (requestName: string) => {
const newRequest = {
...cloneDeep(tabs.currentActiveTab.value.document.request),
name: requestName,
endpoint:
tabs.currentActiveTab.value.document.request.endpoint ||
getDefaultRESTRequest().endpoint,
}
const path = editingFolderPath.value
@@ -2351,4 +2355,7 @@ const getErrorMessage = (err: GQLError<string>) => {
defineActionHandler("collection.new", () => {
displayModalAdd(true)
})
defineActionHandler("modals.collection.import", () => {
displayModalImportExport(true)
})
</script>

View File

@@ -595,7 +595,7 @@ const getComputedAuthHeaders = (
} else if (request.auth.authType === "api-key") {
const { key, addTo } = request.auth
if (addTo === "Headers" && key) {
if (addTo === "HEADERS" && key) {
headers.push({
active: true,
key,

View File

@@ -3,16 +3,13 @@
class="sticky top-0 z-10 flex flex-shrink-0 space-x-2 overflow-x-auto bg-primary p-4"
>
<div class="inline-flex flex-1 space-x-2">
<input
id="url"
<SmartEnvInput
v-model="url"
type="url"
autocomplete="off"
spellcheck="false"
class="w-full rounded border border-divider bg-primaryLight px-4 py-2 text-secondaryDark"
:placeholder="`${t('request.url')}`"
:disabled="connected"
@keyup.enter="onConnectClick"
:placeholder="getDefaultGQLRequest().url"
:placeholder-hover-string="t('request.graphql_placeholder')"
:readonly="connected"
class="rounded border border-divider bg-primaryLight"
@enter="onConnectClick"
/>
<HoppButtonPrimary
id="get"
@@ -72,6 +69,7 @@ import { InterceptorService } from "~/services/interceptor.service"
import { useService } from "dioc/vue"
import { defineActionHandler } from "~/helpers/actions"
import { GQLTabService } from "~/services/tab/graphql"
import { getDefaultGQLRequest } from "~/helpers/graphql/default"
const t = useI18n()
const tabs = useService(GQLTabService)
@@ -98,7 +96,10 @@ const onConnectClick = () => {
}
const gqlConnect = () => {
connect(url.value, tabs.currentActiveTab.value?.document.request.headers)
connect(
url.value || getDefaultGQLRequest().url,
tabs.currentActiveTab.value?.document.request.headers
)
platform.analytics?.logEvent({
type: "HOPP_REQUEST_RUN",
@@ -118,7 +119,9 @@ watch(
tabs.currentActiveTab,
(newVal) => {
if (newVal) {
lastTwoUrls.value.push(newVal.document.request.url)
lastTwoUrls.value.push(
newVal.document.request.url ?? getDefaultGQLRequest().url
)
if (lastTwoUrls.value.length > 2) {
lastTwoUrls.value.shift()
}

View File

@@ -76,6 +76,7 @@ import { InterceptorService } from "~/services/interceptor.service"
import { editGraphqlRequest } from "~/newstore/collections"
import { GQLTabService } from "~/services/tab/graphql"
import { HoppInheritedProperty } from "~/helpers/types/HoppInheritedProperties"
import { getDefaultGQLRequest } from "~/helpers/graphql/default"
const VALID_GQL_OPERATIONS = [
"query",
@@ -119,7 +120,9 @@ const request = useVModel(props, "modelValue", emit)
const url = computedWithControl(
() => tabs.currentActiveTab.value,
() => tabs.currentActiveTab.value.document.request.url
() =>
tabs.currentActiveTab.value.document.request.url ||
getDefaultGQLRequest().url
)
const activeGQLHeadersCount = computed(
@@ -247,10 +250,16 @@ const saveRequest = () => {
tabs.currentActiveTab.value.document.saveContext.originLocation ===
"user-collection"
) {
const finalRequest = {
...tabs.currentActiveTab.value.document.request,
url:
tabs.currentActiveTab.value.document.request.url ||
getDefaultGQLRequest().url,
}
editGraphqlRequest(
tabs.currentActiveTab.value.document.saveContext.folderPath,
tabs.currentActiveTab.value.document.saveContext.requestIndex,
tabs.currentActiveTab.value.document.request
finalRequest
)
tabs.currentActiveTab.value.document.isDirty = false

View File

@@ -54,9 +54,10 @@
>
<SmartEnvInput
v-model="tab.document.request.endpoint"
:placeholder="`${t('request.url')}`"
:placeholder="getDefaultRESTRequest().endpoint"
:auto-complete-source="userHistories"
:auto-complete-env="true"
:placeholder-hover-string="t('request.http_placeholder')"
:inspection-results="tabResults"
@paste="onPasteUrl($event)"
@enter="newSendRequest"
@@ -331,12 +332,12 @@ const tabs = useService(RESTTabService)
const workspaceService = useService(WorkspaceService)
const newSendRequest = async () => {
if (newEndpoint.value === "" || /^\s+$/.test(newEndpoint.value)) {
if (/^\s+$/.test(newEndpoint.value)) {
toast.error(`${t("empty.endpoint")}`)
return
}
ensureMethodInEndpoint()
if (newEndpoint.value) ensureMethodInEndpoint()
loading.value = true
@@ -348,7 +349,20 @@ const newSendRequest = async () => {
workspaceType: workspaceService.currentWorkspace.value.type,
})
const [cancel, streamPromise] = runRESTRequest$(tab)
const finalTab = ref({
...tab.value,
document: {
...tab.value.document,
request: {
...tab.value.document.request,
endpoint:
tab.value.document.request.endpoint ||
getDefaultRESTRequest().endpoint,
},
},
})
const [cancel, streamPromise] = runRESTRequest$(finalTab)
const streamResult = await streamPromise
requestCancelFunc.value = cancel
@@ -472,8 +486,13 @@ const fetchingShareLink = ref(false)
const shareRequest = () => {
if (currentUser.value) {
const finalRequest = {
...tab.value.document.request,
endpoint:
tab.value.document.request.endpoint || getDefaultRESTRequest().endpoint,
}
invokeAction("share.request", {
request: tab.value.document.request,
request: finalRequest,
})
} else {
invokeAction("modals.login.toggle")
@@ -513,11 +532,17 @@ const saveRequest = () => {
showSaveRequestModal.value = true
return
}
if (saveCtx.originLocation === "user-collection") {
const req = tab.value.document.request
const req = tab.value.document.request
const finalRequest = {
...req,
endpoint: req.endpoint.trim() || getDefaultRESTRequest().endpoint,
}
if (saveCtx.originLocation === "user-collection") {
try {
editRESTRequest(saveCtx.folderPath, saveCtx.requestIndex, req)
editRESTRequest(saveCtx.folderPath, saveCtx.requestIndex, finalRequest)
tab.value.document.isDirty = false
@@ -534,8 +559,6 @@ const saveRequest = () => {
saveRequest()
}
} else if (saveCtx.originLocation === "team-collection") {
const req = tab.value.document.request
// TODO: handle error case (NOTE: overwriteRequestTeams is async)
try {
platform.analytics?.logEvent({
@@ -549,7 +572,7 @@ const saveRequest = () => {
requestID: saveCtx.requestID,
data: {
title: req.name,
request: JSON.stringify(req),
request: JSON.stringify(finalRequest),
},
})().then((result) => {
if (E.isLeft(result)) {

View File

@@ -28,7 +28,13 @@
>
<HoppSmartSelectWrapper>
<HoppButtonSecondary
:label="auth.addTo || t('state.none')"
:label="
auth.addTo
? auth.addTo === 'HEADERS'
? t('authorization.pass_by_headers_label')
: t('authorization.pass_by_query_params_label')
: t('state.none')
"
class="ml-2 rounded-none pr-8"
/>
</HoppSmartSelectWrapper>
@@ -40,23 +46,23 @@
@keyup.escape="hide()"
>
<HoppSmartItem
:icon="auth.addTo === 'Headers' ? IconCircleDot : IconCircle"
:active="auth.addTo === 'Headers'"
:label="'Headers'"
:icon="auth.addTo === 'HEADERS' ? IconCircleDot : IconCircle"
:active="auth.addTo === 'HEADERS'"
:label="t('authorization.pass_by_headers_label')"
@click="
() => {
auth.addTo = 'Headers'
auth.addTo = 'HEADERS'
hide()
}
"
/>
<HoppSmartItem
:icon="auth.addTo === 'Query params' ? IconCircleDot : IconCircle"
:active="auth.addTo === 'Query params'"
:label="'Query params'"
:icon="auth.addTo === 'QUERY_PARAMS' ? IconCircleDot : IconCircle"
:active="auth.addTo === 'QUERY_PARAMS'"
:label="t('authorization.pass_by_query_params_label')"
@click="
() => {
auth.addTo = 'Query params'
auth.addTo = 'QUERY_PARAMS'
hide()
}
"

View File

@@ -134,6 +134,7 @@ import * as E from "fp-ts/Either"
import { RESTTabService } from "~/services/tab/rest"
import { useService } from "dioc/vue"
import { watch } from "vue"
import { getDefaultRESTRequest } from "~/helpers/rest/default"
const t = useI18n()
const colorMode = useColorMode()
@@ -511,7 +512,10 @@ const openRequestInNewTab = (request: HoppRESTRequest) => {
}
defineActionHandler("share.request", ({ request }) => {
requestToShare.value = request
requestToShare.value = {
...request,
endpoint: request.endpoint || getDefaultRESTRequest().endpoint,
}
displayShareRequestModal(true)
})
</script>

View File

@@ -73,7 +73,12 @@ import {
keymap,
tooltips,
} from "@codemirror/view"
import { EditorSelection, EditorState, Extension } from "@codemirror/state"
import {
Compartment,
EditorSelection,
EditorState,
Extension,
} from "@codemirror/state"
import { clone } from "lodash-es"
import { history, historyKeymap } from "@codemirror/commands"
import { inputTheme } from "~/helpers/editor/themes/baseTheme"
@@ -109,6 +114,7 @@ const props = withDefaults(
contextMenuEnabled?: boolean
secret?: boolean
autoCompleteEnv?: boolean
placeholderHoverString: string
}>(),
{
modelValue: "",
@@ -124,6 +130,7 @@ const props = withDefaults(
contextMenuEnabled: true,
secret: false,
autoCompleteEnvSource: false,
placeholderHoverString: "",
}
)
@@ -137,6 +144,8 @@ const emit = defineEmits<{
(e: "click", ev: any): void
}>()
const placeholderString = ref(props.placeholder)
const cachedValue = ref(props.modelValue)
const view = ref<EditorView>()
@@ -441,6 +450,9 @@ function handleTextSelection() {
}
}
const placeholderCompt = new Compartment()
const readOnlyCompt = new Compartment()
// Debounce to prevent double click from selecting the word
const debouncedTextSelection = (time: number) =>
useDebounceFn(() => {
@@ -475,6 +487,7 @@ const getExtensions = (readonly: boolean): Extension => {
}),
EditorState.changeFilter.of(() => !readonly),
inputTheme,
readOnlyCompt.of(EditorState.readOnly.of(readonly)),
readonly
? EditorView.theme({
".cm-content": {
@@ -490,7 +503,8 @@ const getExtensions = (readonly: boolean): Extension => {
position: "absolute",
}),
props.environmentHighlights ? envTooltipPlugin : [],
placeholderExt(props.placeholder),
placeholderCompt.of(placeholderExt(props.placeholder)),
EditorView.domEventHandlers({
paste(ev) {
clipboardEv = ev
@@ -505,6 +519,27 @@ const getExtensions = (readonly: boolean): Extension => {
debouncedTextSelection(30)()
}
},
mouseenter() {
//change placeholder to hover string if provided
if (props.placeholderHoverString && !props.readonly) {
placeholderString.value = props.placeholderHoverString
view.value?.dispatch({
effects: placeholderCompt.reconfigure(
placeholderExt(props.placeholderHoverString)
),
})
}
},
mouseleave() {
//change placeholder back to original string
if (props.placeholderHoverString && !props.readonly) {
view.value?.dispatch({
effects: placeholderCompt.reconfigure(
placeholderExt(props.placeholder)
),
})
}
},
}),
props.autoCompleteEnv
? autocompletion({
@@ -568,6 +603,38 @@ const getExtensions = (readonly: boolean): Extension => {
return extensions
}
watch(
() => props.readonly,
(readonly) => {
if (readonly) {
view.value!.dispatch({
effects: [
readOnlyCompt.reconfigure([
EditorState.readOnly.of(readonly),
EditorView.theme({
".cm-content": {
caretColor: "var(--secondary-dark-color)",
color: "var(--secondary-dark-color)",
backgroundColor: "var(--divider-color)",
opacity: 0.25,
},
}),
]),
],
})
} else {
view.value!.dispatch({
effects: [
readOnlyCompt.reconfigure([
EditorState.readOnly.of(readonly),
EditorView.theme({}),
]),
],
})
}
}
)
const triggerTextSelection = () => {
nextTick(() => {
view.value?.focus()

View File

@@ -36,6 +36,7 @@ export type HoppAction =
| "collection.new" // Create root collection
| "flyouts.chat.open" // Shows the keybinds flyout
| "flyouts.keybinds.toggle" // Shows the keybinds flyout
| "modals.collection.import" // Shows the collection import modal
| "modals.search.toggle" // Shows the search modal
| "modals.support.toggle" // Shows the support modal
| "modals.share.toggle" // Shows the share modal

View File

@@ -281,9 +281,9 @@ export const runGQLOperation = async (options: RunQueryOptions) => {
}
} else if (auth.authType === "api-key") {
const { key, value, addTo } = auth
if (addTo === "Headers") {
if (addTo === "HEADERS") {
finalHeaders[key] = value
} else if (addTo === "Query params") {
} else if (addTo === "QUERY_PARAMS") {
params[key] = value
}
}

View File

@@ -260,7 +260,7 @@ const resolveOpenAPIV3SecurityObj = (
return {
authType: "api-key",
authActive: true,
addTo: "Headers",
addTo: "HEADERS",
key: scheme.name,
value: "",
}
@@ -268,7 +268,7 @@ const resolveOpenAPIV3SecurityObj = (
return {
authType: "api-key",
authActive: true,
addTo: "Query params",
addTo: "QUERY_PARAMS",
key: scheme.in,
value: "",
}
@@ -430,7 +430,7 @@ const resolveOpenAPIV2SecurityScheme = (
// V2 only supports in: header and in: query
return {
authType: "api-key",
addTo: scheme.in === "header" ? "Headers" : "Query params",
addTo: scheme.in === "header" ? "HEADERS" : "QUERY_PARAMS",
authActive: true,
key: scheme.name,
value: "",

View File

@@ -150,8 +150,8 @@ const getHoppReqAuth = (item: Item): HoppRESTAuth => {
),
addTo:
(getVariableValue(auth.apikey, "in") ?? "query") === "query"
? "Query params"
: "Headers",
? "QUERY_PARAMS"
: "HEADERS",
}
} else if (auth.type === "bearer") {
return {

View File

@@ -132,30 +132,23 @@ function generateKeybindingString(ev: KeyboardEvent): ShortcutKey | null {
function getPressedKey(ev: KeyboardEvent): Key | null {
// Sometimes the property code is not available on the KeyboardEvent object
const val = (ev.code ?? "").toLowerCase()
const key = (ev.key ?? "").toLowerCase()
// Check arrow keys
if (val === "arrowup") return "up"
else if (val === "arrowdown") return "down"
else if (val === "arrowleft") return "left"
else if (val === "arrowright") return "right"
if (key.startsWith("arrow")) {
return key.slice(5) as Key
}
// Check letter keys
const isLetter = val.startsWith("key")
if (isLetter) return val.substring(3) as Key
const isLetter = key.length === 1 && key >= "a" && key <= "z"
if (isLetter) return key as Key
// Check if number keys
const isDigit = val.startsWith("digit")
if (isDigit) return val.substring(5) as Key
const isDigit = key.length === 1 && key >= "0" && key <= "9"
if (isDigit) return key as Key
// Check if slash
if (val === "slash") return "/"
// Check if period
if (val === "period") return "."
// Check if enter
if (val === "enter") return "enter"
// Check if slash, period or enter
if (key === "/" || key === "." || key === "enter") return key
// If no other cases match, this is not a valid key
return null

View File

@@ -96,7 +96,7 @@ export const getComputedAuthHeaders = (
})
} else if (request.auth.authType === "api-key") {
const { key, addTo } = request.auth
if (addTo === "Headers" && key) {
if (addTo === "HEADERS" && key) {
headers.push({
active: true,
key: parseTemplateString(key, envVars),

View File

@@ -73,7 +73,7 @@ import { useI18n } from "~/composables/i18n"
import { useToast } from "~/composables/toast"
import { InvocationTriggers, defineActionHandler } from "~/helpers/actions"
import { hookKeybindingsListener } from "~/helpers/keybindings"
import { applySetting } from "~/newstore/settings"
import { applySetting, toggleSetting } from "~/newstore/settings"
import { platform } from "~/platform"
import { HoppSpotlightSessionEventData } from "~/platform/analytics"
import { PersistenceService } from "~/services/persistence"
@@ -97,6 +97,8 @@ const t = useI18n()
const persistenceService = useService(PersistenceService)
const spotlightService = useService(SpotlightService)
const HAS_OPENED_SPOTLIGHT = useSetting("HAS_OPENED_SPOTLIGHT")
onBeforeMount(() => {
if (!mdAndLarger.value) {
rightSidebar.value = false
@@ -160,6 +162,7 @@ defineActionHandler("modals.search.toggle", (_, trigger) => {
})
showSearch.value = !showSearch.value
!HAS_OPENED_SPOTLIGHT.value && toggleSetting("HAS_OPENED_SPOTLIGHT")
})
defineActionHandler("modals.support.toggle", () => {

View File

@@ -1393,6 +1393,14 @@ export function editGraphqlRequest(
}
export function saveGraphqlRequestAs(path: string, request: HoppGQLRequest) {
// For calculating the insertion request index
const targetLocation = navigateToFolderWithIndexPath(
graphqlCollectionStore.value.state,
path.split("/").map((x) => parseInt(x))
)
const insertionIndex = targetLocation!.requests.length
graphqlCollectionStore.dispatch({
dispatcher: "saveRequestAs",
payload: {
@@ -1400,6 +1408,8 @@ export function saveGraphqlRequestAs(path: string, request: HoppGQLRequest) {
request,
},
})
return insertionIndex
}
export function removeGraphqlRequest(

View File

@@ -65,6 +65,8 @@ export type SettingsDef = {
SIDEBAR: boolean
SIDEBAR_ON_LEFT: boolean
COLUMN_LAYOUT: boolean
HAS_OPENED_SPOTLIGHT: boolean
}
export const getDefaultSettings = (): SettingsDef => ({
@@ -109,6 +111,8 @@ export const getDefaultSettings = (): SettingsDef => ({
SIDEBAR: true,
SIDEBAR_ON_LEFT: false,
COLUMN_LAYOUT: true,
HAS_OPENED_SPOTLIGHT: false,
})
type ApplySettingPayload = {

View File

@@ -112,7 +112,10 @@ const activeTabs = tabs.getActiveTabs()
const addNewTab = () => {
const tab = tabs.createNewTab({
request: getDefaultGQLRequest(),
request: {
...getDefaultGQLRequest(),
url: "",
},
isDirty: false,
})

View File

@@ -178,7 +178,10 @@ const onTabUpdate = (tab: HoppTab<HoppRESTDocument>) => {
const addNewTab = () => {
const tab = tabs.createNewTab({
request: getDefaultRESTRequest(),
request: {
...getDefaultRESTRequest(),
endpoint: "",
},
isDirty: false,
})
@@ -295,8 +298,14 @@ const shareTabRequest = (tabID: string) => {
const tab = tabs.getTabRef(tabID)
if (tab.value) {
if (currentUser.value) {
const finalRequest = {
...tab.value.document.request,
endpoint:
tab.value.document.request.endpoint ||
getDefaultRESTRequest().endpoint,
}
invokeAction("share.request", {
request: tab.value.document.request,
request: finalRequest,
})
} else {
invokeAction("modals.login.toggle")

View File

@@ -25,7 +25,7 @@ export const REST_COLLECTIONS_MOCK: HoppCollection[] = [
folders: [],
requests: [
{
v: "3",
v: "4",
endpoint: "https://echo.hoppscotch.io",
name: "Echo test",
params: [],
@@ -50,7 +50,7 @@ export const GQL_COLLECTIONS_MOCK: HoppCollection[] = [
folders: [],
requests: [
{
v: 3,
v: 4,
name: "Echo test",
url: "https://echo.hoppscotch.io/graphql",
headers: [],
@@ -138,7 +138,7 @@ export const REST_HISTORY_MOCK: RESTHistoryEntry[] = [
preRequestScript: "",
testScript: "",
requestVariables: [],
v: "3",
v: "4",
},
responseMeta: { duration: 807, statusCode: 200 },
star: false,
@@ -150,7 +150,7 @@ export const GQL_HISTORY_MOCK: GQLHistoryEntry[] = [
{
v: 1,
request: {
v: 3,
v: 4,
name: "Untitled",
url: "https://echo.hoppscotch.io/graphql",
query: "query Request { url }",
@@ -171,7 +171,7 @@ export const GQL_TAB_STATE_MOCK: PersistableTabState<HoppGQLDocument> = {
tabID: "5edbe8d4-65c9-4381-9354-5f1bf05d8ccc",
doc: {
request: {
v: 3,
v: 4,
name: "Untitled",
url: "https://echo.hoppscotch.io/graphql",
headers: [],
@@ -194,7 +194,7 @@ export const REST_TAB_STATE_MOCK: PersistableTabState<HoppRESTDocument> = {
tabID: "e6e8d800-caa8-44a2-a6a6-b4765a3167aa",
doc: {
request: {
v: "3",
v: "4",
endpoint: "https://echo.hoppscotch.io",
name: "Echo test",
params: [],

View File

@@ -66,6 +66,8 @@ const SettingsDefSchema = z.object({
cookie: z.boolean().catch(true),
})
),
HAS_OPENED_SPOTLIGHT: z.optional(z.boolean()),
})
// Common properties shared across REST & GQL collections

View File

@@ -15,6 +15,7 @@ import {
restCollectionStore,
} from "~/newstore/collections"
import IconFolder from "~icons/lucide/folder"
import IconImport from "~icons/lucide/folder-down"
import RESTRequestSpotlightEntry from "~/components/app/spotlight/entry/RESTRequest.vue"
import GQLRequestSpotlightEntry from "~/components/app/spotlight/entry/GQLRequest.vue"
import {
@@ -151,6 +152,10 @@ export class CollectionsSpotlightSearcherService
id: `create-collection`,
name: this.t("collection.new"),
})
minisearch.add({
id: "import-collection",
name: this.t("collection.import"),
})
}
if (pageCategory === "rest") {
@@ -168,6 +173,11 @@ export class CollectionsSpotlightSearcherService
text: this.t("collection.new"),
}
const importCollectionText: SpotlightResultTextType<any> = {
type: "text",
text: this.t("collection.import_collection"),
}
scopeHandle.run(() => {
const isPersonalWorkspace = computed(
() => this.workspaceService.currentWorkspace.value.type === "personal"
@@ -183,44 +193,37 @@ export class CollectionsSpotlightSearcherService
results.value = []
return
}
if (pageCategory === "rest") {
const searchResults = minisearch.search(query).slice(0, 10)
results.value = searchResults.map((result) => ({
id: result.id,
text:
result.id === "create-collection"
? newCollectionText
: {
type: "custom",
component: markRaw(RESTRequestSpotlightEntry),
componentProps: {
folderPath: result.id.split("rest-")[1],
},
},
icon: markRaw(IconFolder),
score: result.score,
}))
} else if (pageCategory === "graphql") {
const searchResults = minisearch.search(query).slice(0, 10)
results.value = searchResults.map((result) => ({
id: result.id,
text:
result.id === "create-collection"
? newCollectionText
: {
type: "custom",
component: markRaw(GQLRequestSpotlightEntry),
componentProps: {
folderPath: result.id.split("gql-")[1],
},
},
icon: markRaw(IconFolder),
score: result.score,
}))
const getResultText = (id: string): SpotlightResultTextType<any> => {
if (id === "create-collection") return newCollectionText
else if (id === "import-collection") return importCollectionText
return {
type: "custom",
component: markRaw(
pageCategory === "rest"
? RESTRequestSpotlightEntry
: GQLRequestSpotlightEntry
),
componentProps: {
folderPath: id.split(
pageCategory === "rest" ? "rest-" : "gql-"
)[1],
},
}
}
const getResultIcon = (id: string) => {
if (id === "import-collection") return markRaw(IconImport)
return markRaw(IconFolder)
}
const searchResults = minisearch.search(query).slice(0, 10)
results.value = searchResults.map((result) => ({
id: result.id,
text: getResultText(result.id),
icon: getResultIcon(result.id),
score: result.score,
}))
})
})
@@ -288,6 +291,9 @@ export class CollectionsSpotlightSearcherService
public onResultSelect(result: SpotlightSearcherResult): void {
if (result.id === "create-collection") return invokeAction("collection.new")
if (result.id === "import-collection")
return invokeAction(`modals.collection.import`)
const [type, path] = result.id.split("-")
if (type === "rest") {

View File

@@ -1,20 +1,24 @@
import { Service } from "dioc"
import {
SpotlightResultTextType,
SpotlightSearcher,
SpotlightSearcherResult,
SpotlightSearcherSessionState,
SpotlightService,
} from ".."
import { getI18n } from "~/modules/i18n"
import { Ref, computed, effectScope, markRaw, watch } from "vue"
import { Ref, computed, effectScope, markRaw, ref, watch } from "vue"
import { TeamSearchService } from "~/helpers/teams/TeamsSearch.service"
import { cloneDeep, debounce } from "lodash-es"
import IconFolder from "~icons/lucide/folder"
import IconImport from "~icons/lucide/folder-down"
import { WorkspaceService } from "~/services/workspace.service"
import RESTTeamRequestEntry from "~/components/app/spotlight/entry/RESTTeamRequestEntry.vue"
import { RESTTabService } from "~/services/tab/rest"
import { HoppInheritedProperty } from "~/helpers/types/HoppInheritedProperties"
import { HoppRESTRequest } from "@hoppscotch/data"
import MiniSearch from "minisearch"
import { invokeAction } from "~/helpers/actions"
export class TeamsSpotlightSearcherService
extends Service
@@ -41,9 +45,79 @@ export class TeamsSpotlightSearcherService
this.spotlight.registerSearcher(this)
}
private getCurrentPageCategory() {
// TODO: Better logic for this ?
try {
const url = new URL(window.location.href)
if (url.pathname.startsWith("/graphql")) {
return "graphql"
} else if (url.pathname === "/") {
return "rest"
}
return "other"
} catch (e) {
return "other"
}
}
createSearchSession(
query: Readonly<Ref<string>>
): [Ref<SpotlightSearcherSessionState>, () => void] {
const pageCategory = this.getCurrentPageCategory()
// Only show the searcher on the REST page
if (pageCategory !== "rest") {
return [computed(() => ({ loading: false, results: [] })), () => {}]
}
const results = ref<SpotlightSearcherResult[]>([])
const minisearch = new MiniSearch({
fields: ["name"],
storeFields: ["name"],
searchOptions: {
prefix: true,
fuzzy: true,
boost: {
name: 2,
},
weights: {
fuzzy: 0.2,
prefix: 0.8,
},
},
})
minisearch.add({
id: `create-collection`,
name: this.t("collection.new"),
})
minisearch.add({
id: "import-collection",
name: this.t("collection.import"),
})
const newCollectionText: SpotlightResultTextType<any> = {
type: "text",
text: this.t("collection.new"),
}
const importCollectionText: SpotlightResultTextType<any> = {
type: "text",
text: this.t("collection.import_collection"),
}
const getResultText = (id: string): SpotlightResultTextType<any> => {
if (id === "create-collection") return newCollectionText
return importCollectionText
}
const getResultIcon = (id: string) => {
if (id === "import-collection") return markRaw(IconImport)
return markRaw(IconFolder)
}
const isTeamWorkspace = computed(
() => this.workspaceService.currentWorkspace.value.type === "team"
)
@@ -59,6 +133,13 @@ export class TeamsSpotlightSearcherService
if (this.workspaceService.currentWorkspace.value.type === "team") {
const teamID = this.workspaceService.currentWorkspace.value.teamID
debouncedSearch(query, teamID)?.catch(() => {})
const searchResults = minisearch.search(query).slice(0, 10)
results.value = searchResults.map((result) => ({
id: result.id,
text: getResultText(result.id),
icon: getResultIcon(result.id),
score: result.score,
}))
}
},
{
@@ -91,36 +172,47 @@ export class TeamsSpotlightSearcherService
}
const resultObj = computed<SpotlightSearcherSessionState>(() => {
return isTeamWorkspace.value
? {
loading: this.teamsSearch.teamsSearchResultsLoading.value,
results:
this.teamsSearch.teamsSearchResultsFormattedForSpotlight.value.map(
(result) => ({
id: result.request.id,
icon: markRaw(IconFolder),
score: 1, // make a better scoring system for this
text: {
type: "custom",
component: markRaw(RESTTeamRequestEntry),
componentProps: {
collectionTitles: result.collectionTitles,
request: result.request,
},
},
})
),
}
: {
loading: false,
results: [],
}
})
if (isTeamWorkspace.value) {
const teamsSearchResults =
this.teamsSearch.teamsSearchResultsFormattedForSpotlight.value
const minisearchResults = results.value
const mergedResults = [
...teamsSearchResults.map((result) => ({
id: result.request.id,
icon: markRaw(IconFolder),
score: 1, // make a better scoring system for this
text: {
type: "custom",
component: markRaw(RESTTeamRequestEntry),
componentProps: {
collectionTitles: result.collectionTitles,
request: result.request,
},
},
})),
...minisearchResults,
] as SpotlightSearcherResult[]
return {
loading: this.teamsSearch.teamsSearchResultsLoading.value,
results: mergedResults,
}
}
return {
loading: false,
results: [],
}
})
return [resultObj, onSessionEnd]
}
onResultSelect(result: SpotlightSearcherResult): void {
if (result.id === "create-collection") return invokeAction("collection.new")
if (result.id === "import-collection")
return invokeAction(`modals.collection.import`)
let inheritedProperties: HoppInheritedProperty | undefined = undefined
const selectedRequest = this.teamsSearch.searchResultsRequests[result.id]

View File

@@ -13,7 +13,10 @@ export class GQLTabService extends TabService<HoppGQLDocument> {
this.tabMap.set("test", {
id: "test",
document: {
request: getDefaultGQLRequest(),
request: {
...getDefaultGQLRequest(),
url: "",
},
isDirty: false,
optionTabPreference: "query",
},

View File

@@ -13,7 +13,10 @@ export class RESTTabService extends TabService<HoppRESTDocument> {
this.tabMap.set("test", {
id: "test",
document: {
request: getDefaultRESTRequest(),
request: {
...getDefaultRESTRequest(),
endpoint: "",
},
isDirty: false,
optionTabPreference: "params",
},

View File

@@ -3,31 +3,33 @@ import { z } from "zod"
import V1_VERSION from "./v/1"
import V2_VERSION from "./v/2"
import V3_VERSION from "./v/3"
import V4_VERSION from "./v/4"
export { GQLHeader } from "./v/1"
export {
HoppGQLAuthAPIKey,
HoppGQLAuthBasic,
HoppGQLAuthBearer,
HoppGQLAuthNone,
HoppGQLAuthInherit,
} from "./v/2"
export { HoppGQLAuth } from "./v/3"
export { HoppGQLAuth } from "./v/4"
export { HoppGQLAuthOAuth2 } from "./v/3"
export { HoppGQLAuthAPIKey } from "./v/4"
export const GQL_REQ_SCHEMA_VERSION = 3
export const GQL_REQ_SCHEMA_VERSION = 4
const versionedObject = z.object({
v: z.number(),
})
export const HoppGQLRequest = createVersionedEntity({
latestVersion: 3,
latestVersion: 4,
versionMap: {
1: V1_VERSION,
2: V2_VERSION,
3: V3_VERSION,
4: V4_VERSION,
},
getVersion(x) {
const result = versionedObject.safeParse(x)

View File

@@ -0,0 +1,71 @@
import { z } from "zod"
import { defineVersion } from "verzod"
import { HoppRESTAuthOAuth2 } from "../../rest"
import {
HoppGQLAuthAPIKey as HoppGQLAuthAPIKeyOld,
HoppGQLAuthBasic,
HoppGQLAuthBearer,
HoppGQLAuthInherit,
HoppGQLAuthNone,
} from "./2"
import { V3_SCHEMA } from "./3"
export { HoppRESTAuthOAuth2 as HoppGQLAuthOAuth2 } from "../../rest"
export const HoppGQLAuthAPIKey = HoppGQLAuthAPIKeyOld.extend({
addTo: z.enum(["HEADERS", "QUERY_PARAMS"]).catch("HEADERS"),
})
export type HoppGqlAuthOAuth2 = z.infer<typeof HoppRESTAuthOAuth2>
export const HoppGQLAuth = z
.discriminatedUnion("authType", [
HoppGQLAuthNone,
HoppGQLAuthInherit,
HoppGQLAuthBasic,
HoppGQLAuthBearer,
HoppGQLAuthAPIKey,
HoppRESTAuthOAuth2, // both rest and gql have the same auth type for oauth2
])
.and(
z.object({
authActive: z.boolean(),
})
)
export type HoppGQLAuth = z.infer<typeof HoppGQLAuth>
export const V4_SCHEMA = V3_SCHEMA.extend({
v: z.literal(4),
auth: HoppGQLAuth,
})
export default defineVersion({
initial: false,
schema: V4_SCHEMA,
up(old: z.infer<typeof V3_SCHEMA>) {
if (old.auth.authType === "api-key") {
return {
...old,
v: 4 as const,
auth: {
...old.auth,
addTo:
old.auth.addTo === "Query params"
? ("QUERY_PARAMS" as const)
: ("HEADERS" as const),
},
}
}
return {
...old,
v: 4 as const,
auth: {
...old.auth,
},
}
},
})

View File

@@ -5,12 +5,13 @@ import V0_VERSION from "./v/0"
import V1_VERSION from "./v/1"
import V2_VERSION from "./v/2"
import V3_VERSION from "./v/3"
import V4_VERSION from "./v/4"
import { createVersionedEntity, InferredEntity } from "verzod"
import { lodashIsEqualEq, mapThenEq, undefinedEq } from "../utils/eq"
import { HoppRESTReqBody, HoppRESTHeaders, HoppRESTParams } from "./v/1"
import { HoppRESTAuth } from "./v/3"
import { HoppRESTAuth } from "./v/4"
import { HoppRESTRequestVariables } from "./v/2"
import { z } from "zod"
@@ -19,7 +20,6 @@ export * from "./content-types"
export {
FormDataKeyValue,
HoppRESTReqBodyFormData,
HoppRESTAuthAPIKey,
HoppRESTAuthBasic,
HoppRESTAuthInherit,
HoppRESTAuthBearer,
@@ -29,7 +29,6 @@ export {
} from "./v/1"
export {
HoppRESTAuth,
HoppRESTAuthOAuth2,
AuthCodeGrantTypeParams,
ClientCredentialsGrantTypeParams,
@@ -37,6 +36,8 @@ export {
PasswordGrantTypeParams,
} from "./v/3"
export { HoppRESTAuth, HoppRESTAuthAPIKey } from "./v/4"
export { HoppRESTRequestVariables } from "./v/2"
const versionedObject = z.object({
@@ -45,12 +46,13 @@ const versionedObject = z.object({
})
export const HoppRESTRequest = createVersionedEntity({
latestVersion: 3,
latestVersion: 4,
versionMap: {
0: V0_VERSION,
1: V1_VERSION,
2: V2_VERSION,
3: V3_VERSION,
4: V4_VERSION,
},
getVersion(data) {
// For V1 onwards we have the v string storing the number
@@ -92,7 +94,7 @@ const HoppRESTRequestEq = Eq.struct<HoppRESTRequest>({
),
})
export const RESTReqSchemaVersion = "3"
export const RESTReqSchemaVersion = "4"
export type HoppRESTParam = HoppRESTRequest["params"][number]
export type HoppRESTHeader = HoppRESTRequest["headers"][number]
@@ -187,7 +189,7 @@ export function makeRESTRequest(
export function getDefaultRESTRequest(): HoppRESTRequest {
return {
v: "3",
v: "4",
endpoint: "https://echo.hoppscotch.io",
name: "Untitled",
params: [],

View File

@@ -0,0 +1,69 @@
import { z } from "zod"
import { defineVersion } from "verzod"
import { HoppRESTAuthOAuth2, V3_SCHEMA } from "./3"
import {
HoppRESTAuthAPIKey as HoppRESTAuthAPIKeyOld,
HoppRESTAuthBasic,
HoppRESTAuthBearer,
HoppRESTAuthInherit,
HoppRESTAuthNone,
} from "./1"
// in this new version, we update the old 'Headers' and 'Query params' to be more consistent with OAuth addTo values
// also in the previous version addTo was a string, which prevented some bugs from being caught by the type system
// this version uses an enum, so we get the values as literals in the type system
export const HoppRESTAuthAPIKey = HoppRESTAuthAPIKeyOld.extend({
addTo: z.enum(["HEADERS", "QUERY_PARAMS"]).catch("HEADERS"),
})
export type HoppRESTAuthAPIKey = z.infer<typeof HoppRESTAuthAPIKey>
export const HoppRESTAuth = z
.discriminatedUnion("authType", [
HoppRESTAuthNone,
HoppRESTAuthInherit,
HoppRESTAuthBasic,
HoppRESTAuthBearer,
HoppRESTAuthOAuth2,
HoppRESTAuthAPIKey,
])
.and(
z.object({
authActive: z.boolean(),
})
)
export type HoppRESTAuth = z.infer<typeof HoppRESTAuth>
export const V4_SCHEMA = V3_SCHEMA.extend({
v: z.literal("4"),
auth: HoppRESTAuth,
})
export default defineVersion({
schema: V4_SCHEMA,
initial: false,
up(old: z.infer<typeof V3_SCHEMA>) {
if (old.auth.authType === "api-key") {
return {
...old,
v: "4" as const,
auth: {
...old.auth,
addTo:
old.auth.addTo === "Query params"
? ("QUERY_PARAMS" as const)
: ("HEADERS" as const),
},
}
}
return {
...old,
auth: {
...old.auth,
},
v: "4" as const,
}
},
})

View File

@@ -1,10 +0,0 @@
export default {
preset: "ts-jest",
testEnvironment: "jsdom",
collectCoverage: true,
setupFilesAfterEnv: ["./jest.setup.ts"],
moduleNameMapper: {
"~/(.*)": "<rootDir>/src/$1",
"^lodash-es$": "lodash",
},
}

View File

@@ -1 +0,0 @@
require("@relmify/jest-fp-ts")

View File

@@ -1,2 +1,2 @@
export { default } from "./dist/node.d.ts"
export * from "./dist/node.d.ts"
export { default } from "./dist/node/index.d.ts"
export * from "./dist/node/index.d.ts"

View File

@@ -30,7 +30,7 @@
"scripts": {
"lint": "eslint --ext .ts,.js --ignore-path .gitignore .",
"lintfix": "eslint --fix --ext .ts,.js --ignore-path .gitignore .",
"test": "pnpm exec jest",
"test": "vitest run",
"build": "vite build && tsc --emitDeclarationOnly",
"clean": "pnpm tsc --build --clean",
"postinstall": "pnpm run build",
@@ -69,10 +69,18 @@
"eslint-config-prettier": "8.6.0",
"eslint-plugin-prettier": "4.2.1",
"io-ts": "2.2.16",
"jest": "27.5.1",
"prettier": "2.8.4",
"ts-jest": "27.1.5",
"typescript": "4.9.5",
"vite": "5.0.5"
"vite": "5.0.5",
"vitest": "0.34.6"
},
"peerDependencies": {
"isolated-vm": "4.7.2"
},
"peerDependenciesMeta": {
"isolated-vm": {
"optional": true
}
}
}

View File

@@ -0,0 +1,15 @@
// Vitest doesn't work without globals
// Ref: https://github.com/relmify/jest-fp-ts/issues/11
import decodeMatchers from "@relmify/jest-fp-ts/dist/decodeMatchers"
import eitherMatchers from "@relmify/jest-fp-ts/dist/eitherMatchers"
import optionMatchers from "@relmify/jest-fp-ts/dist/optionMatchers"
import theseMatchers from "@relmify/jest-fp-ts/dist/theseMatchers"
import eitherOrTheseMatchers from "@relmify/jest-fp-ts/dist/eitherOrTheseMatchers"
import { expect } from "vitest"
expect.extend(decodeMatchers.matchers)
expect.extend(eitherMatchers.matchers)
expect.extend(optionMatchers.matchers)
expect.extend(theseMatchers.matchers)
expect.extend(eitherOrTheseMatchers.matchers)

View File

@@ -1,8 +1,9 @@
import * as TE from "fp-ts/TaskEither"
import { pipe } from "fp-ts/function"
import { runPreRequestScript } from "~/pre-request/node-vm"
import { runTestScript } from "~/test-runner/node-vm"
import { describe, expect, test } from "vitest"
import { runPreRequestScript, runTestScript } from "~/node"
import { TestResponse, TestResult } from "~/types"
describe("Base64 helper functions", () => {

View File

@@ -1,8 +1,9 @@
import "@relmify/jest-fp-ts"
import * as TE from "fp-ts/TaskEither"
import { pipe } from "fp-ts/function"
import { runTestScript } from "~/test-runner/node-vm"
import { describe, expect, test } from "vitest"
import { runTestScript } from "~/node"
import { TestResponse, TestResult } from "~/types"
const fakeResponse: TestResponse = {

View File

@@ -1,8 +1,9 @@
import "@relmify/jest-fp-ts"
import * as TE from "fp-ts/TaskEither"
import { pipe } from "fp-ts/function"
import { runTestScript } from "~/test-runner/node-vm"
import { describe, expect, test } from "vitest"
import { runTestScript } from "~/node"
import { TestResponse, TestResult } from "~/types"
const fakeResponse: TestResponse = {

View File

@@ -1,7 +1,9 @@
import * as TE from "fp-ts/TaskEither"
import { pipe } from "fp-ts/function"
import { runTestScript } from "~/test-runner/node-vm"
import { describe, expect, test } from "vitest"
import { runTestScript } from "~/node"
import { TestResponse, TestResult } from "~/types"
const fakeResponse: TestResponse = {

View File

@@ -1,7 +1,9 @@
import * as TE from "fp-ts/TaskEither"
import { pipe } from "fp-ts/function"
import { runTestScript } from "~/test-runner/node-vm"
import { describe, expect, test } from "vitest"
import { runTestScript } from "~/node"
import { TestResponse, TestResult } from "~/types"
const fakeResponse: TestResponse = {

View File

@@ -1,7 +1,9 @@
import * as TE from "fp-ts/TaskEither"
import { pipe } from "fp-ts/function"
import { runTestScript } from "~/test-runner/node-vm"
import { describe, expect, test } from "vitest"
import { runTestScript } from "~/node"
import { TestResponse, TestResult } from "~/types"
const fakeResponse: TestResponse = {

View File

@@ -1,8 +1,9 @@
import "@relmify/jest-fp-ts"
import * as TE from "fp-ts/TaskEither"
import { pipe } from "fp-ts/function"
import { runTestScript } from "~/test-runner/node-vm"
import { describe, expect, test } from "vitest"
import { runTestScript } from "~/node"
import { TestResponse } from "~/types"
const fakeResponse: TestResponse = {
@@ -23,7 +24,7 @@ describe("toBe", () => {
return expect(
func(
`
pw.expect(2).toBe(2)
pw.expect(2).toBe(2)
`,
fakeResponse
)()

View File

@@ -1,8 +1,9 @@
import "@relmify/jest-fp-ts"
import * as TE from "fp-ts/TaskEither"
import { pipe } from "fp-ts/function"
import { runTestScript } from "~/test-runner/node-vm"
import { describe, expect, test } from "vitest"
import { runTestScript } from "~/node"
import { TestResponse } from "~/types"
const fakeResponse: TestResponse = {

View File

@@ -1,7 +1,9 @@
import * as TE from "fp-ts/TaskEither"
import { pipe } from "fp-ts/function"
import { runTestScript } from "~/test-runner/node-vm"
import { describe, expect, test } from "vitest"
import { runTestScript } from "~/node"
import { TestResponse } from "~/types"
const fakeResponse: TestResponse = {

View File

@@ -1,7 +1,9 @@
import * as TE from "fp-ts/TaskEither"
import { pipe } from "fp-ts/function"
import { runTestScript } from "~/test-runner/node-vm"
import { describe, expect, test } from "vitest"
import { runTestScript } from "~/node"
import { TestResponse } from "~/types"
const fakeResponse: TestResponse = {

View File

@@ -1,7 +1,9 @@
import * as TE from "fp-ts/TaskEither"
import { pipe } from "fp-ts/function"
import { runTestScript } from "~/test-runner/node-vm"
import { describe, expect, test } from "vitest"
import { runTestScript } from "~/node"
import { TestResponse } from "~/types"
const fakeResponse: TestResponse = {

View File

@@ -1,6 +1,6 @@
import "@relmify/jest-fp-ts"
import { describe, expect, test } from "vitest"
import { runPreRequestScript } from "~/pre-request/node-vm"
import { runPreRequestScript } from "~/node"
describe("runPreRequestScript", () => {
test("returns the updated environment properly", () => {

View File

@@ -1,4 +1,6 @@
import { preventCyclicObjects } from "~/utils"
import { preventCyclicObjects } from "~/shared-utils"
import { describe, expect, test } from "vitest"
describe("preventCyclicObjects", () => {
test("succeeds with a simple object", () => {

View File

@@ -1,7 +1,9 @@
import * as TE from "fp-ts/TaskEither"
import { pipe } from "fp-ts/function"
import { runTestScript } from "~/test-runner/node-vm"
import { describe, expect, test } from "vitest"
import { runTestScript } from "~/node"
import { TestResponse } from "~/types"
const fakeResponse: TestResponse = {

View File

@@ -1,2 +0,0 @@
export * from "./pre-request/node-vm"
export * from "./test-runner/node-vm"

View File

@@ -0,0 +1,2 @@
export { runPreRequestScript } from "./pre-request"
export { runTestScript } from "./test-runner"

View File

@@ -0,0 +1,91 @@
import { pipe } from "fp-ts/function"
import * as TE from "fp-ts/lib/TaskEither"
import { createRequire } from "module"
import type ivmT from "isolated-vm"
import { TestResult } from "~/types"
import { getPreRequestScriptMethods } from "~/shared-utils"
import { getSerializedAPIMethods } from "./utils"
const nodeRequire = createRequire(import.meta.url)
const ivm = nodeRequire("isolated-vm")
export const runPreRequestScript = (
preRequestScript: string,
envs: TestResult["envs"]
): TE.TaskEither<string, TestResult["envs"]> =>
pipe(
TE.tryCatch(
async () => {
const isolate: ivmT.Isolate = new ivm.Isolate()
const context = await isolate.createContext()
return { isolate, context }
},
(reason) => `Context initialization failed: ${reason}`
),
TE.chain(({ isolate, context }) =>
pipe(
TE.tryCatch(
async () => {
const jail = context.global
const { pw, updatedEnvs } = getPreRequestScriptMethods(envs)
const serializedAPIMethods = getSerializedAPIMethods(pw)
jail.setSync("serializedAPIMethods", serializedAPIMethods, {
copy: true,
})
jail.setSync("atob", atob)
jail.setSync("btoa", btoa)
// Methods in the isolate context can't be invoked straightaway
const finalScript = `
const pw = new Proxy(serializedAPIMethods, {
get: (pwObjTarget, pwObjProp) => {
const topLevelEntry = pwObjTarget[pwObjProp]
// "pw.env" set of API methods
if (topLevelEntry && typeof topLevelEntry === "object") {
return new Proxy(topLevelEntry, {
get: (subTarget, subProp) => {
const subLevelProperty = subTarget[subProp]
if (subLevelProperty && subLevelProperty.typeof === "function") {
return (...args) => subLevelProperty.applySync(null, args)
}
},
})
}
}
})
${preRequestScript}
`
// Create a script and compile it
const script = await isolate.compileScript(finalScript)
// Run the pre-request script in the provided context
await script.run(context)
return updatedEnvs
},
(reason) => reason
),
TE.fold(
(error) => TE.left(`Script execution failed: ${error}`),
(result) =>
pipe(
TE.tryCatch(
async () => {
await isolate.dispose()
return result
},
(disposeError) => `Isolate disposal failed: ${disposeError}`
)
)
)
)
)
)

View File

@@ -0,0 +1,217 @@
import * as E from "fp-ts/Either"
import * as TE from "fp-ts/TaskEither"
import { pipe } from "fp-ts/function"
import { createRequire } from "module"
import type ivmT from "isolated-vm"
import { TestResponse, TestResult } from "~/types"
import {
getTestRunnerScriptMethods,
preventCyclicObjects,
} from "~/shared-utils"
import { getSerializedAPIMethods } from "./utils"
const nodeRequire = createRequire(import.meta.url)
const ivm = nodeRequire("isolated-vm")
export const runTestScript = (
testScript: string,
envs: TestResult["envs"],
response: TestResponse
): TE.TaskEither<string, TestResult> =>
pipe(
TE.tryCatch(
async () => {
const isolate: ivmT.Isolate = new ivm.Isolate()
const context = await isolate.createContext()
return { isolate, context }
},
(reason) => `Context initialization failed: ${reason}`
),
TE.chain(({ isolate, context }) =>
pipe(
TE.tryCatch(
async () =>
executeScriptInContext(
testScript,
envs,
response,
isolate,
context
),
(reason) => `Script execution failed: ${reason}`
),
TE.chain((result) =>
TE.tryCatch(
async () => {
await isolate.dispose()
return result
},
(disposeReason) => `Isolate disposal failed: ${disposeReason}`
)
)
)
)
)
const executeScriptInContext = (
testScript: string,
envs: TestResult["envs"],
response: TestResponse,
isolate: ivmT.Isolate,
context: ivmT.Context
): Promise<TestResult> => {
return new Promise((resolve, reject) => {
// Parse response object
const responseObjHandle = preventCyclicObjects(response)
if (E.isLeft(responseObjHandle)) {
return reject(`Response parsing failed: ${responseObjHandle.left}`)
}
const jail = context.global
const { pw, testRunStack, updatedEnvs } = getTestRunnerScriptMethods(envs)
const serializedAPIMethods = getSerializedAPIMethods({
...pw,
response: responseObjHandle.right,
})
jail.setSync("serializedAPIMethods", serializedAPIMethods, { copy: true })
jail.setSync("atob", atob)
jail.setSync("btoa", btoa)
jail.setSync("ivm", ivm)
// Methods in the isolate context can't be invoked straightaway
const finalScript = `
const pw = new Proxy(serializedAPIMethods, {
get: (pwObj, pwObjProp) => {
// pw.expect(), pw.env, etc.
const topLevelEntry = pwObj[pwObjProp]
// If the entry exists and is a function
// pw.expect(), pw.test(), etc.
if (topLevelEntry && topLevelEntry.typeof === "function") {
// pw.test() just involves invoking the function via "applySync()"
if (pwObjProp === "test") {
return (...args) => topLevelEntry.applySync(null, args)
}
// pw.expect() returns an object with matcher methods
return (...args) => {
// Invoke "pw.expect()" and get access to the object with matcher methods
const expectFnResult = topLevelEntry.applySync(
null,
args.map((expectVal) => {
if (typeof expectVal === "object") {
if (expectVal === null) {
return null
}
// Only arrays and objects stringified here should be parsed from the "pw.expect()" method definition
// The usecase is that any JSON string supplied should be preserved
// An extra "isStringifiedWithinIsolate" prop is added to indicate it has to be parsed
if (Array.isArray(expectVal)) {
return JSON.stringify({
arr: expectVal,
isStringifiedWithinIsolate: true,
})
}
return JSON.stringify({
...expectVal,
isStringifiedWithinIsolate: true,
})
}
return expectVal
})
)
// Matcher methods that can be chained with "pw.expect()"
// pw.expect().toBe(), etc
if (expectFnResult.typeof === "object") {
// Access the getter that points to the negated matcher methods via "{ accessors: true }"
const matcherMethods = {
not: expectFnResult.getSync("not", { accessors: true }),
}
// Serialize matcher methods for use in the isolate context
const matcherMethodNames = [
"toBe",
"toBeLevel2xx",
"toBeLevel3xx",
"toBeLevel4xx",
"toBeLevel5xx",
"toBeType",
"toHaveLength",
"toInclude",
]
matcherMethodNames.forEach((methodName) => {
matcherMethods[methodName] = expectFnResult.getSync(methodName)
})
return new Proxy(matcherMethods, {
get: (matcherMethodTarget, matcherMethodProp) => {
// pw.expect().not.toBe(), etc
const matcherMethodEntry = matcherMethodTarget[matcherMethodProp]
if (matcherMethodProp === "not") {
return new Proxy(matcherMethodEntry, {
get: (negatedObjTarget, negatedObjprop) => {
// Return the negated matcher method defn that is invoked from the test script
const negatedMatcherMethodDefn = negatedObjTarget.getSync(negatedObjprop)
return negatedMatcherMethodDefn
},
})
}
// Return the matcher method defn that is invoked from the test script
return matcherMethodEntry
},
})
}
}
}
// "pw.env" set of API methods
if (typeof topLevelEntry === "object" && pwObjProp !== "response") {
return new Proxy(topLevelEntry, {
get: (subTarget, subProp) => {
const subLevelProperty = subTarget[subProp]
if (
subLevelProperty &&
subLevelProperty.typeof === "function"
) {
return (...args) => subLevelProperty.applySync(null, args)
}
},
})
}
return topLevelEntry
},
})
${testScript}
`
// Create a script and compile it
const script = isolate.compileScript(finalScript)
// Run the test script in the provided context
script
.then((script) => script.run(context))
.then(() => {
resolve({
tests: testRunStack,
envs: updatedEnvs,
})
})
.catch((error: Error) => {
reject(error)
})
})
}

View File

@@ -0,0 +1,23 @@
import { createRequire } from "module"
const nodeRequire = createRequire(import.meta.url)
const ivm = nodeRequire("isolated-vm")
// Helper function to recursively wrap methods in `ivm.Reference`
export const getSerializedAPIMethods = (
namespaceObj: Record<string, unknown>
): Record<string, unknown> => {
const result: Record<string, unknown> = {}
for (const [key, value] of Object.entries(namespaceObj)) {
if (typeof value === "object" && value !== null && !Array.isArray(value)) {
result[key] = getSerializedAPIMethods(value as Record<string, unknown>)
} else if (typeof value === "function") {
result[key] = new ivm.Reference(value)
} else {
result[key] = value
}
}
return result
}

View File

@@ -1,38 +0,0 @@
import { pipe } from "fp-ts/function"
import * as TE from "fp-ts/lib/TaskEither"
import { createContext, runInContext } from "vm"
import { TestResult } from "~/types"
import { getPreRequestScriptMethods } from "~/utils"
export const runPreRequestScript = (
preRequestScript: string,
envs: TestResult["envs"]
): TE.TaskEither<string, TestResult["envs"]> =>
pipe(
TE.tryCatch(
async () => {
return createContext()
},
(reason) => `Context initialization failed: ${reason}`
),
TE.chain((context) =>
TE.tryCatch(
() =>
new Promise((resolve) => {
const { pw, updatedEnvs } = getPreRequestScriptMethods(envs)
// Expose pw to the context
context.pw = pw
context.atob = atob
context.btoa = btoa
// Run the pre-request script in the provided context
runInContext(preRequestScript, context)
resolve(updatedEnvs)
}),
(reason) => `Script execution failed: ${reason}`
)
)
)

View File

@@ -182,6 +182,36 @@ const getSharedMethods = (envs: TestResult["envs"]) => {
}
}
const getResolvedExpectValue = (expectVal: any) => {
if (typeof expectVal !== "string") {
return expectVal
}
try {
const parsedExpectVal = JSON.parse(expectVal)
// Supplying non-primitive values is not permitted in the `isStringifiedWithinIsolate` property indicates that the object was stringified before executing the script from the isolate context
// This is done to ensure a JSON string supplied as the "expectVal" is not parsed and preserved as is
if (typeof parsedExpectVal === "object") {
if (parsedExpectVal.isStringifiedWithinIsolate !== true) {
return expectVal
}
// For an array, the contents are stored in the `arr` property
if (Array.isArray(parsedExpectVal.arr)) {
return parsedExpectVal.arr
}
delete parsedExpectVal.isStringifiedWithinIsolate
return parsedExpectVal
}
return expectVal
} catch (_) {
return expectVal
}
}
export function preventCyclicObjects(
obj: Record<string, any>
): E.Left<string> | E.Right<Record<string, any>> {
@@ -215,15 +245,18 @@ export const createExpectation = (
) => {
const result: Record<string, unknown> = {}
// Non-primitive values supplied are stringified in the isolate context
const resolvedExpectVal = getResolvedExpectValue(expectVal)
const toBeFn = (expectedVal: any) => {
let assertion = expectVal === expectedVal
let assertion = resolvedExpectVal === expectedVal
if (negated) {
assertion = !assertion
}
const status = assertion ? "pass" : "fail"
const message = `Expected '${expectVal}' to${
const message = `Expected '${resolvedExpectVal}' to${
negated ? " not" : ""
} be '${expectedVal}'`
@@ -240,7 +273,7 @@ export const createExpectation = (
rangeStart: number,
rangeEnd: number
) => {
const parsedExpectVal = parseInt(expectVal)
const parsedExpectVal = parseInt(resolvedExpectVal)
if (!Number.isNaN(parsedExpectVal)) {
let assertion =
@@ -260,7 +293,7 @@ export const createExpectation = (
message,
})
} else {
const message = `Expected ${level}-level status but could not parse value '${expectVal}'`
const message = `Expected ${level}-level status but could not parse value '${resolvedExpectVal}'`
currTestStack[currTestStack.length - 1].expectResults.push({
status: "error",
message,
@@ -288,14 +321,14 @@ export const createExpectation = (
"function",
].includes(expectedType)
) {
let assertion = typeof expectVal === expectedType
let assertion = typeof resolvedExpectVal === expectedType
if (negated) {
assertion = !assertion
}
const status = assertion ? "pass" : "fail"
const message = `Expected '${expectVal}' to${
const message = `Expected '${resolvedExpectVal}' to${
negated ? " not" : ""
} be type '${expectedType}'`
@@ -316,7 +349,12 @@ export const createExpectation = (
}
const toHaveLengthFn = (expectedLength: any) => {
if (!(Array.isArray(expectVal) || typeof expectVal === "string")) {
if (
!(
Array.isArray(resolvedExpectVal) ||
typeof resolvedExpectVal === "string"
)
) {
const message =
"Expected toHaveLength to be called for an array or string"
currTestStack[currTestStack.length - 1].expectResults.push({
@@ -328,7 +366,7 @@ export const createExpectation = (
}
if (typeof expectedLength === "number" && !Number.isNaN(expectedLength)) {
let assertion = expectVal.length === expectedLength
let assertion = resolvedExpectVal.length === expectedLength
if (negated) {
assertion = !assertion
@@ -355,7 +393,12 @@ export const createExpectation = (
}
const toIncludeFn = (needle: any) => {
if (!(Array.isArray(expectVal) || typeof expectVal === "string")) {
if (
!(
Array.isArray(resolvedExpectVal) ||
typeof resolvedExpectVal === "string"
)
) {
const message = "Expected toInclude to be called for an array or string"
currTestStack[currTestStack.length - 1].expectResults.push({
status: "error",
@@ -382,13 +425,13 @@ export const createExpectation = (
return undefined
}
let assertion = expectVal.includes(needle)
let assertion = resolvedExpectVal.includes(needle)
if (negated) {
assertion = !assertion
}
const expectValPretty = JSON.stringify(expectVal)
const expectValPretty = JSON.stringify(resolvedExpectVal)
const needlePretty = JSON.stringify(needle)
const status = assertion ? "pass" : "fail"
const message = `Expected ${expectValPretty} to${

View File

@@ -1,57 +0,0 @@
import * as E from "fp-ts/Either"
import * as TE from "fp-ts/TaskEither"
import { pipe } from "fp-ts/function"
import { createContext, runInContext } from "vm"
import { TestResponse, TestResult } from "~/types"
import { getTestRunnerScriptMethods, preventCyclicObjects } from "~/utils"
export const runTestScript = (
testScript: string,
envs: TestResult["envs"],
response: TestResponse
): TE.TaskEither<string, TestResult> =>
pipe(
TE.tryCatch(
async () => {
return createContext()
},
(reason) => `Context initialization failed: ${reason}`
),
TE.chain((context) =>
TE.tryCatch(
() => executeScriptInContext(testScript, envs, response, context),
(reason) => `Script execution failed: ${reason}`
)
)
)
const executeScriptInContext = (
testScript: string,
envs: TestResult["envs"],
response: TestResponse,
context: any
): Promise<TestResult> => {
return new Promise((resolve, reject) => {
// Parse response object
const responseObjHandle = preventCyclicObjects(response)
if (E.isLeft(responseObjHandle)) {
return reject(`Response parsing failed: ${responseObjHandle.left}`)
}
const { pw, testRunStack, updatedEnvs } = getTestRunnerScriptMethods(envs)
// Expose pw to the context
context.pw = { ...pw, response: responseObjHandle.right }
context.atob = atob
context.btoa = btoa
// Run the test script in the provided context
runInContext(testScript, context)
resolve({
tests: testRunStack,
envs: updatedEnvs,
})
})
}

View File

@@ -1,2 +0,0 @@
export * from "./pre-request/web-worker"
export * from "./test-runner/web-worker"

View File

@@ -0,0 +1,2 @@
export { runPreRequestScript } from "./pre-request"
export { runTestScript } from "./test-runner"

View File

@@ -1,7 +1,7 @@
import * as TE from "fp-ts/TaskEither"
import { TestResult } from "~/types"
import { getPreRequestScriptMethods } from "~/utils"
import { getPreRequestScriptMethods } from "~/shared-utils"
const executeScriptInContext = (
preRequestScript: string,

View File

@@ -2,7 +2,10 @@ import * as E from "fp-ts/Either"
import * as TE from "fp-ts/TaskEither"
import { SandboxTestResult, TestResponse, TestResult } from "~/types"
import { getTestRunnerScriptMethods, preventCyclicObjects } from "~/utils"
import {
getTestRunnerScriptMethods,
preventCyclicObjects,
} from "~/shared-utils"
const executeScriptInContext = (
testScript: string,

View File

@@ -7,16 +7,20 @@ export default defineConfig({
emptyOutDir: true,
lib: {
entry: {
web: "./src/web.ts",
node: "./src/node.ts",
web: "./src/web/index.ts",
node: "./src/node/index.ts",
},
name: "js-sandbox",
formats: ["es", "cjs"],
},
rollupOptions: {
external: ["vm"],
external: ["module"],
},
},
test: {
environment: "node",
setupFiles: ["./setupFiles.ts"],
},
resolve: {
alias: {
"~": resolve(__dirname, "./src"),

View File

@@ -1,2 +1,2 @@
export { default } from "./dist/web.d.ts"
export * from "./dist/web.d.ts"
export { default } from "./dist/web/index.d.ts"
export * from "./dist/web/index.d.ts"

View File

@@ -1260,7 +1260,7 @@ dependencies = [
[[package]]
name = "hoppscotch-desktop"
version = "24.3.0"
version = "24.3.1"
dependencies = [
"cocoa 0.25.0",
"hex_color",
@@ -1274,6 +1274,7 @@ dependencies = [
"tauri-plugin-window-state",
"url",
"windows 0.52.0",
"winver",
]
[[package]]
@@ -4415,6 +4416,15 @@ dependencies = [
"windows-sys 0.48.0",
]
[[package]]
name = "winver"
version = "1.0.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "9e0e7162b9e282fd75a0a832cce93994bdb21208d848a418cd05a5fdd9b9ab33"
dependencies = [
"windows 0.48.0",
]
[[package]]
name = "wry"
version = "0.24.6"

View File

@@ -41,6 +41,7 @@ windows = { version = "0.52.0", features = [
"Win32_Foundation",
"Win32_UI_Controls",
] }
winver = "1"
[features]
# this feature is used for production builds or when `devPath` points to the filesystem

View File

@@ -16,6 +16,8 @@ use windows::Win32::Graphics::Dwm::DwmSetWindowAttribute;
use windows::Win32::Foundation::HWND;
use windows::Win32::Graphics::Dwm::{DWMWA_USE_IMMERSIVE_DARK_MODE};
use winver::WindowsVersion;
fn hex_color_to_colorref(color: HexColor) -> COLORREF {
// TODO: Remove this unsafe, This operation doesn't need to be unsafe!
unsafe {
@@ -42,18 +44,25 @@ fn update_bg_color(hwnd: &HWND, bg_color: HexColor) {
ptr::addr_of!(use_dark_mode) as *const c_void,
size_of::<BOOL>().try_into().unwrap()
).unwrap();
}
DwmSetWindowAttribute(
HWND(hwnd.0),
DWMWA_CAPTION_COLOR,
ptr::addr_of!(final_color) as *const c_void,
size_of::<COLORREF>().try_into().unwrap()
).unwrap();
let version = WindowsVersion::detect().unwrap();
if version >= WindowsVersion::new(10, 0, 22000) {
unsafe {
DwmSetWindowAttribute(
HWND(hwnd.0),
DWMWA_CAPTION_COLOR,
ptr::addr_of!(final_color) as *const c_void,
size_of::<COLORREF>().try_into().unwrap()
).unwrap();
}
}
let flags = WTNCA_NODRAWCAPTION | WTNCA_NODRAWICON;
let mask = WTNCA_NODRAWCAPTION | WTNCA_NODRAWICON | WTNCA_NOSYSMENU | WTNCA_NOMIRRORHELP;
let options = WinThemeAttribute { flag: flags, mask };
let flags = WTNCA_NODRAWCAPTION | WTNCA_NODRAWICON;
let mask = WTNCA_NODRAWCAPTION | WTNCA_NODRAWICON | WTNCA_NOSYSMENU | WTNCA_NOMIRRORHELP;
let options = WinThemeAttribute { flag: flags, mask };
unsafe {
SetWindowThemeAttribute(
HWND(hwnd.0),
WTA_NONCLIENT,

View File

@@ -8,7 +8,7 @@
},
"package": {
"productName": "Hoppscotch",
"version": "24.3.0"
"version": "24.3.1"
},
"tauri": {
"allowlist": {

Some files were not shown because too many files have changed in this diff Show More