Compare commits

...

62 Commits

Author SHA1 Message Date
nivedin
850954efdf chore: minoir ui and code refactor 2024-05-29 14:09:19 +05:30
nivedin
bf65654126 chore: update settings page interceptor section UI 2024-05-29 14:09:19 +05:30
nivedin
27ea170d85 refactor: update interceptor section in settings page 2024-05-29 14:09:19 +05:30
nivedin
6cf0c427a7 fix: persist interceptor state for loged out user 2024-05-29 14:09:19 +05:30
Mir Arif Hasan
b7a3ae231b HSB-431 fix: email case sensitive in email provider (#4042)
* feat: code level email insensitivity added

* test: fix broken test case

* chore: updated testcase for findUserByEmail in user module

---------

Co-authored-by: Balu Babu <balub997@gmail.com>
2024-05-15 12:37:35 +05:30
Nivedin
f8ac6dfeb1 chore: add workspace switcher login A/B testing flow (#4053)
Co-authored-by: Andrew Bastin <andrewbastin.k@gmail.com>
2024-05-10 16:35:42 +05:30
Andrew Bastin
7d2d335b37 chore: revert back default interceptor for sh app to browser 2024-05-10 16:13:51 +05:30
Andrew Bastin
76875db865 chore: bump selfhost-desktop lockfile version 2024-05-10 15:04:16 +05:30
Balu Babu
96e2d87b57 feat: update node version to node20-apline3.19 (#4040) 2024-05-10 14:24:34 +05:30
islamzeki
be353d9f72 chore(i18n): update tr.json (#4039) 2024-05-10 14:16:04 +05:30
Nivedin
38bc2c12c3 refactor: set default interceptor to "proxy" (#4051) 2024-05-10 14:01:43 +05:30
Stéfany Larissa
97644fa508 chore: update pt-br translations (#4003) 2024-05-10 13:56:58 +05:30
Dmitry
eb3446ae23 locale: update ru locale (#4005)
Co-authored-by: Dmitry Mukovkin <d.mukovkin@cft.ru>
Co-authored-by: Liyas Thomas <liyascthomas@gmail.com>
Co-authored-by: Joel Jacob Stephen <70131076+JoelJacobStephen@users.noreply.github.com>
2024-05-10 13:51:25 +05:30
Dmitry
6c29961d09 Added new button to save request modal (#3976)
Co-authored-by: Dmitry Mukovkin <d.mukovkin@cft.ru>
Co-authored-by: Nivedin <53208152+nivedin@users.noreply.github.com>
2024-05-09 20:52:22 +05:30
Nivedin
ef1117d8cc refactor: switch workspace after creation (#4015)
Co-authored-by: amk-dev <akash.k.mohan98@gmail.com>
2024-05-09 20:28:24 +05:30
Balu Babu
5c4b651aee feat: healthcheck for external services (#4048)
* chore: installed terminus package and scaffolded the health module

* feat: added healthcheck for database with prisma

* chore: renamed label from prisma to database in health check route

* chore: reverted target of hopp-old-backend service to prod
2024-05-07 20:05:17 +05:30
Andrew Bastin
391e5a20f5 chore: bump versions to 2024.3.3 2024-05-06 22:57:33 +05:30
Andrew Bastin
4b8f3bd8da fix: code generate modal erroring out (#4033) 2024-05-06 22:44:13 +05:30
Joel Jacob Stephen
94248076e6 refactor(sh-admin): improved handling of server configurations in admin dashboard (#3971)
Co-authored-by: jamesgeorge007 <jamesgeorge998001@gmail.com>
2024-05-06 21:50:31 +05:30
Nivedin
eecc3db4e9 chore: update placeholder text (#4023) 2024-04-30 16:49:32 +05:30
Andrew Bastin
426e7594f4 fix: tab systems erroring out due to out of sync tabOrdering and tabMap 2024-04-29 20:32:27 +05:30
Andrew Bastin
934dc473f0 chore: bump version to 2024.3.2 2024-04-29 19:21:48 +05:30
Andrew Bastin
be57255bf7 refactor: update to dioc v3 (#4009)
Co-authored-by: jamesgeorge007 <jamesgeorge998001@gmail.com>
2024-04-29 19:06:18 +05:30
Balu Babu
f89561da54 fix: resolved mailer module email issue (#4000) 2024-04-29 12:05:07 +05:30
Joel Jacob Stephen
c2c4e620c2 fix(common): rest and graphql pages not being rendered when french is selected as app language (#4004)
Co-authored-by: nivedin <nivedinp@gmail.com>
2024-04-25 18:01:46 +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
jamesgeorge007
787aab650f fix: redirect to the users list view on successful deletion from the profile view
SH Admin dashboard
2024-03-28 21:20:48 +05:30
Anwarul Islam
1f7a8edb14 fix: lint errors removed by using satisfies or as for type (#3934)
Co-authored-by: jamesgeorge007 <jamesgeorge998001@gmail.com>
Co-authored-by: amk-dev <akash.k.mohan98@gmail.com>
2024-03-28 20:28:48 +05:30
James George
81f1e05a6c chore(sh-admin): alert the user while deleting users who are team owners (#3937)
Co-authored-by: amk-dev <akash.k.mohan98@gmail.com>
Co-authored-by: nivedin <nivedinp@gmail.com>
2024-03-28 20:17:24 +05:30
James George
0a71783eaa fix(common): ensure requests are translated to the latest version during import and search actions (#3931) 2024-03-25 17:09:54 +05:30
Nivedin
c326f54f7e feat: added mutation and function to platform for updating user profile name (#3929)
Co-authored-by: amk-dev <akash.k.mohan98@gmail.com>
Co-authored-by: jamesgeorge007 <jamesgeorge998001@gmail.com>
2024-03-25 14:41:25 +05:30
Dmitry
1113c79e20 fix: can't import curl command with data-urlencode (#3905)
Co-authored-by: Anwarul Islam <anwaarulislaam@gmail.com>
Co-authored-by: jamesgeorge007 <jamesgeorge998001@gmail.com>
2024-03-25 13:12:48 +05:30
Akash K
6fd30f9aca chore: spotlight improvements for team requests search (#3930)
Co-authored-by: jamesgeorge007 <jamesgeorge998001@gmail.com>
2024-03-25 12:41:18 +05:30
Anwarul Islam
2c5b0dcd1b feat: focus codemirror view when ImportCurl component launched (#3911)
Co-authored-by: jamesgeorge007 <jamesgeorge998001@gmail.com>
2024-03-22 18:21:16 +05:30
Anwarul Islam
6f4455ba03 fix: request failing on change content type to multipart-formdata (#3922)
Co-authored-by: jamesgeorge007 <jamesgeorge998001@gmail.com>
2024-03-22 18:11:16 +05:30
Nivedin
ba8c4480d9 fix: workspace list section bugs (#3925)
Co-authored-by: jamesgeorge007 <jamesgeorge998001@gmail.com>
2024-03-22 18:02:28 +05:30
Joel Jacob Stephen
380397cc55 refactor(sh-admin): improvements to pending invites page in dashboard (#3926) 2024-03-22 17:54:40 +05:30
Akash K
d19807b212 chore: split axios request options into platform (#3927)
Co-authored-by: jamesgeorge007 <jamesgeorge998001@gmail.com>
2024-03-22 13:42:41 +05:30
Balu Babu
c89c2a5f5c feat: added new mutation to update username in hopp app (#3924)
* feat: added new mutation to update username in hopp app

* feat: display name length validation added

---------

Co-authored-by: mirarifhasan <arif.ishan05@gmail.com>
2024-03-22 12:07:38 +05:30
Anwarul Islam
256553b9bb chore: update copy for header inspection (#3907)
Co-authored-by: James George <jamesgeorge998001@gmail.com>
2024-03-21 22:15:23 +05:30
Akash K
89d9951f3b fix: fix typo in team search url (#3923)
fix: fix typo in team search endpoint url

Co-authored-by: James George <jamesgeorge998001@gmail.com>
2024-03-21 16:44:58 +05:30
Balu Babu
dd65ad3103 chore: added input validation to search query (#3921) 2024-03-21 16:13:11 +05:30
Balu Babu
018ed3db26 refactor: AIO healthcheck bash script (#3920)
* chore: added logic to make script with with subpath

* chore: removed variable from failed echo message
2024-03-21 16:12:13 +05:30
Andrew Bastin
a9cd6c0c01 chore: update internal deployment docker compose config 2024-03-21 02:38:55 +05:30
James George
e53382666a fix(common): prevent exception with ShortcodeListAdapter initialization (#3917) 2024-03-20 20:29:04 +05:30
James George
7621ff2961 feat: add extended support for versioned entities in the CLI (#3912) 2024-03-20 20:13:22 +05:30
Akash K
fc20b76080 fix: direct import from url failing (#3918) 2024-03-20 20:06:51 +05:30
Nivedin
146c73d7b6 feat: github enterprise SSO provider addition (#3914) 2024-03-20 20:01:56 +05:30
217 changed files with 19916 additions and 15324 deletions

View File

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

View File

@@ -1,40 +1,14 @@
# Docker Compose config used for internal test and QA deployments # THIS IS NOT TO BE USED FOR PERSONAL DEPLOYMENTS!
# This just spins up the AIO container along with an attached DB to the standard HTTP ports with subpath access mode # Internal Docker Compose Image used for internal testing deployments
# TODO: Add Healthcheck for the AIO container
version: "3.7" version: "3.7"
services: services:
# The service that spins up all 3 services at once in one container
hoppscotch-aio:
container_name: hoppscotch-aio
restart: unless-stopped
build:
dockerfile: prod.Dockerfile
context: .
target: aio
environment:
- DATABASE_URL=postgresql://postgres:testpass@hoppscotch-db:5432/hoppscotch
- ENABLE_SUBPATH_BASED_ACCESS=true
depends_on:
hoppscotch-db:
condition: service_healthy
ports:
- "3080:80"
# The preset DB service, you can delete/comment the below lines if
# you are using an external postgres instance
# This will be exposed at port 5432
hoppscotch-db: hoppscotch-db:
image: postgres:15 image: postgres:15
ports:
- "5432:5432"
user: postgres user: postgres
environment: environment:
# The default user defined by the docker image
POSTGRES_USER: postgres POSTGRES_USER: postgres
# NOTE: Please UPDATE THIS PASSWORD!
POSTGRES_PASSWORD: testpass POSTGRES_PASSWORD: testpass
POSTGRES_DB: hoppscotch POSTGRES_DB: hoppscotch
healthcheck: healthcheck:
@@ -46,3 +20,29 @@ services:
interval: 5s interval: 5s
timeout: 5s timeout: 5s
retries: 10 retries: 10
hoppscotch-aio:
container_name: hoppscotch-aio
build:
dockerfile: prod.Dockerfile
context: .
target: aio
environment:
- DATABASE_URL=postgresql://postgres:testpass@hoppscotch-db:5432/hoppscotch
- ENABLE_SUBPATH_BASED_ACCESS=true
env_file:
- ./.env
depends_on:
hoppscotch-db:
condition: service_healthy
command: ["sh", "-c", "pnpm exec prisma migrate deploy && node /usr/src/app/aio_run.mjs"]
healthcheck:
test:
- CMD
- curl
- '-f'
- 'http://localhost:80'
interval: 2s
timeout: 10s
retries: 30

View File

@@ -9,6 +9,10 @@ curlCheck() {
fi fi
} }
curlCheck "http://localhost:3000" if [ "$ENABLE_SUBPATH_BASED_ACCESS" = "true" ]; then
curlCheck "http://localhost:3100" curlCheck "http://localhost:80/backend/ping"
curlCheck "http://localhost:3170/ping" else
curlCheck "http://localhost:3000"
curlCheck "http://localhost:3100"
curlCheck "http://localhost:3170/ping"
fi

View File

@@ -38,7 +38,7 @@
}, },
"packageExtensions": { "packageExtensions": {
"httpsnippet@3.0.1": { "httpsnippet@3.0.1": {
"peerDependencies": { "dependencies": {
"ajv": "6.12.3" "ajv": "6.12.3"
} }
} }

View File

@@ -1,4 +1,4 @@
FROM node:18.8.0 AS builder FROM node:20.12.2 AS builder
WORKDIR /usr/src/app WORKDIR /usr/src/app

View File

@@ -3,9 +3,7 @@
"collection": "@nestjs/schematics", "collection": "@nestjs/schematics",
"sourceRoot": "src", "sourceRoot": "src",
"compilerOptions": { "compilerOptions": {
"assets": [ "assets": [{ "include": "mailer/templates/**/*", "outDir": "dist" }],
"**/*.hbs"
],
"watchAssets": true "watchAssets": true
} }
} }

View File

@@ -1,6 +1,6 @@
{ {
"name": "hoppscotch-backend", "name": "hoppscotch-backend",
"version": "2024.3.0", "version": "2024.3.3",
"description": "", "description": "",
"author": "", "author": "",
"private": true, "private": true,
@@ -35,6 +35,7 @@
"@nestjs/passport": "10.0.2", "@nestjs/passport": "10.0.2",
"@nestjs/platform-express": "10.2.7", "@nestjs/platform-express": "10.2.7",
"@nestjs/schedule": "4.0.1", "@nestjs/schedule": "4.0.1",
"@nestjs/terminus": "10.2.3",
"@nestjs/throttler": "5.0.1", "@nestjs/throttler": "5.0.1",
"@prisma/client": "5.8.1", "@prisma/client": "5.8.1",
"argon2": "0.30.3", "argon2": "0.30.3",
@@ -121,4 +122,4 @@
"^src/(.*)$": "<rootDir>/$1" "^src/(.*)$": "<rootDir>/$1"
} }
} }
} }

View File

@@ -121,6 +121,7 @@ describe('AdminService', () => {
NOT: { NOT: {
inviteeEmail: { inviteeEmail: {
in: [dbAdminUsers[0].email], in: [dbAdminUsers[0].email],
mode: 'insensitive',
}, },
}, },
}, },
@@ -229,7 +230,10 @@ describe('AdminService', () => {
expect(mockPrisma.invitedUsers.deleteMany).toHaveBeenCalledWith({ expect(mockPrisma.invitedUsers.deleteMany).toHaveBeenCalledWith({
where: { where: {
inviteeEmail: { in: [invitedUsers[0].inviteeEmail] }, inviteeEmail: {
in: [invitedUsers[0].inviteeEmail],
mode: 'insensitive',
},
}, },
}); });
expect(result).toEqualRight(true); expect(result).toEqualRight(true);

View File

@@ -89,12 +89,17 @@ export class AdminService {
adminEmail: string, adminEmail: string,
inviteeEmail: string, inviteeEmail: string,
) { ) {
if (inviteeEmail == adminEmail) return E.left(DUPLICATE_EMAIL); if (inviteeEmail.toLowerCase() == adminEmail.toLowerCase()) {
return E.left(DUPLICATE_EMAIL);
}
if (!validateEmail(inviteeEmail)) return E.left(INVALID_EMAIL); if (!validateEmail(inviteeEmail)) return E.left(INVALID_EMAIL);
const alreadyInvitedUser = await this.prisma.invitedUsers.findFirst({ const alreadyInvitedUser = await this.prisma.invitedUsers.findFirst({
where: { where: {
inviteeEmail: inviteeEmail, inviteeEmail: {
equals: inviteeEmail,
mode: 'insensitive',
},
}, },
}); });
if (alreadyInvitedUser != null) return E.left(USER_ALREADY_INVITED); if (alreadyInvitedUser != null) return E.left(USER_ALREADY_INVITED);
@@ -159,7 +164,7 @@ export class AdminService {
try { try {
await this.prisma.invitedUsers.deleteMany({ await this.prisma.invitedUsers.deleteMany({
where: { where: {
inviteeEmail: { in: inviteeEmails }, inviteeEmail: { in: inviteeEmails, mode: 'insensitive' },
}, },
}); });
return E.right(true); return E.right(true);
@@ -189,6 +194,7 @@ export class AdminService {
NOT: { NOT: {
inviteeEmail: { inviteeEmail: {
in: userEmailObjs.map((user) => user.email), in: userEmailObjs.map((user) => user.email),
mode: 'insensitive',
}, },
}, },
}, },

View File

@@ -26,6 +26,7 @@ import { loadInfraConfiguration } from './infra-config/helper';
import { MailerModule } from './mailer/mailer.module'; import { MailerModule } from './mailer/mailer.module';
import { PosthogModule } from './posthog/posthog.module'; import { PosthogModule } from './posthog/posthog.module';
import { ScheduleModule } from '@nestjs/schedule'; import { ScheduleModule } from '@nestjs/schedule';
import { HealthModule } from './health/health.module';
@Module({ @Module({
imports: [ imports: [
@@ -100,6 +101,7 @@ import { ScheduleModule } from '@nestjs/schedule';
InfraConfigModule, InfraConfigModule,
PosthogModule, PosthogModule,
ScheduleModule.forRoot(), ScheduleModule.forRoot(),
HealthModule,
], ],
providers: [GQLComplexityPlugin], providers: [GQLComplexityPlugin],
controllers: [AppController], controllers: [AppController],

View File

@@ -84,6 +84,12 @@ export const USER_ALREADY_INVITED = 'admin/user_already_invited' as const;
*/ */
export const USER_UPDATE_FAILED = 'user/update_failed' as const; export const USER_UPDATE_FAILED = 'user/update_failed' as const;
/**
* User display name validation failure
* (UserService)
*/
export const USER_SHORT_DISPLAY_NAME = 'user/short_display_name' as const;
/** /**
* User deletion failure * User deletion failure
* (UserService) * (UserService)
@@ -750,3 +756,8 @@ export const DATABASE_TABLE_NOT_EXIST =
* (InfraConfigService) * (InfraConfigService)
*/ */
export const POSTHOG_CLIENT_NOT_INITIALIZED = 'posthog/client_not_initialized'; export const POSTHOG_CLIENT_NOT_INITIALIZED = 'posthog/client_not_initialized';
/**
* Inputs supplied are invalid
*/
export const INVALID_PARAMS = 'invalid_parameters' as const;

View File

@@ -0,0 +1,24 @@
import { Controller, Get } from '@nestjs/common';
import {
HealthCheck,
HealthCheckService,
PrismaHealthIndicator,
} from '@nestjs/terminus';
import { PrismaService } from 'src/prisma/prisma.service';
@Controller('health')
export class HealthController {
constructor(
private health: HealthCheckService,
private prismaHealth: PrismaHealthIndicator,
private prisma: PrismaService,
) {}
@Get()
@HealthCheck()
check() {
return this.health.check([
async () => this.prismaHealth.pingCheck('database', this.prisma),
]);
}
}

View File

@@ -0,0 +1,10 @@
import { Module } from '@nestjs/common';
import { HealthController } from './health.controller';
import { PrismaModule } from 'src/prisma/prisma.module';
import { TerminusModule } from '@nestjs/terminus';
@Module({
imports: [PrismaModule, TerminusModule],
controllers: [HealthController],
})
export class HealthModule {}

View File

@@ -156,6 +156,25 @@ export async function getDefaultInfraConfigs(): Promise<
return infraConfigDefaultObjs; 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 * Verify if 'infra_config' table is loaded with all entries
* @returns boolean * @returns boolean
@@ -163,12 +182,7 @@ export async function getDefaultInfraConfigs(): Promise<
export async function isInfraConfigTablePopulated(): Promise<boolean> { export async function isInfraConfigTablePopulated(): Promise<boolean> {
const prisma = new PrismaService(); const prisma = new PrismaService();
try { try {
const dbInfraConfigs = await prisma.infraConfig.findMany(); const propsRemainingToInsert = await getMissingInfraConfigEntries();
const infraConfigDefaultObjs = await getDefaultInfraConfigs();
const propsRemainingToInsert = infraConfigDefaultObjs.filter(
(p) => !dbInfraConfigs.find((e) => e.name === p.name),
);
if (propsRemainingToInsert.length > 0) { if (propsRemainingToInsert.length > 0) {
console.log( console.log(

View File

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

View File

@@ -299,7 +299,10 @@ export class ShortcodeService implements UserDataHandler, OnModuleInit {
where: userEmail where: userEmail
? { ? {
User: { User: {
email: userEmail, email: {
equals: userEmail,
mode: 'insensitive',
},
}, },
} }
: undefined, : undefined,

View File

@@ -1,4 +1,11 @@
import { Controller, Get, Param, Query, UseGuards } from '@nestjs/common'; import {
Controller,
Get,
HttpStatus,
Param,
Query,
UseGuards,
} from '@nestjs/common';
import { TeamCollectionService } from './team-collection.service'; import { TeamCollectionService } from './team-collection.service';
import * as E from 'fp-ts/Either'; import * as E from 'fp-ts/Either';
import { ThrottlerBehindProxyGuard } from 'src/guards/throttler-behind-proxy.guard'; import { ThrottlerBehindProxyGuard } from 'src/guards/throttler-behind-proxy.guard';
@@ -7,6 +14,8 @@ import { RequiresTeamRole } from 'src/team/decorators/requires-team-role.decorat
import { TeamMemberRole } from '@prisma/client'; import { TeamMemberRole } from '@prisma/client';
import { RESTTeamMemberGuard } from 'src/team/guards/rest-team-member.guard'; import { RESTTeamMemberGuard } from 'src/team/guards/rest-team-member.guard';
import { throwHTTPErr } from 'src/utils'; import { throwHTTPErr } from 'src/utils';
import { RESTError } from 'src/types/RESTError';
import { INVALID_PARAMS } from 'src/errors';
@UseGuards(ThrottlerBehindProxyGuard) @UseGuards(ThrottlerBehindProxyGuard)
@Controller({ path: 'team-collection', version: '1' }) @Controller({ path: 'team-collection', version: '1' })
@@ -26,8 +35,15 @@ export class TeamCollectionController {
@Query('take') take: string, @Query('take') take: string,
@Query('skip') skip: string, @Query('skip') skip: string,
) { ) {
if (!teamID || !searchQuery) {
return <RESTError>{
message: INVALID_PARAMS,
statusCode: HttpStatus.BAD_REQUEST,
};
}
const res = await this.teamCollectionService.searchByTitle( const res = await this.teamCollectionService.searchByTitle(
searchQuery, searchQuery.trim(),
teamID, teamID,
parseInt(take), parseInt(take),
parseInt(skip), parseInt(skip),

View File

@@ -75,12 +75,13 @@ export class TeamInvitationService {
if (!isEmailValid) return E.left(INVALID_EMAIL); if (!isEmailValid) return E.left(INVALID_EMAIL);
try { try {
const teamInvite = await this.prisma.teamInvitation.findUniqueOrThrow({ const teamInvite = await this.prisma.teamInvitation.findFirstOrThrow({
where: { where: {
teamID_inviteeEmail: { inviteeEmail: {
inviteeEmail: inviteeEmail, equals: inviteeEmail,
teamID: teamID, mode: 'insensitive',
}, },
teamID,
}, },
}); });

View File

@@ -58,6 +58,29 @@ export class UserResolver {
if (E.isLeft(updatedUser)) throwErr(updatedUser.left); if (E.isLeft(updatedUser)) throwErr(updatedUser.left);
return updatedUser.right; return updatedUser.right;
} }
@Mutation(() => User, {
description: 'Update a users display name',
})
@UseGuards(GqlAuthGuard)
async updateDisplayName(
@GqlUser() user: AuthUser,
@Args({
name: 'updatedDisplayName',
description: 'New name of user',
type: () => String,
})
updatedDisplayName: string,
) {
const updatedUser = await this.userService.updateUserDisplayName(
user.uid,
updatedDisplayName,
);
if (E.isLeft(updatedUser)) throwErr(updatedUser.left);
return updatedUser.right;
}
@Mutation(() => Boolean, { @Mutation(() => Boolean, {
description: 'Delete an user account', description: 'Delete an user account',
}) })

View File

@@ -1,4 +1,9 @@
import { JSON_INVALID, USERS_NOT_FOUND, USER_NOT_FOUND } from 'src/errors'; import {
JSON_INVALID,
USERS_NOT_FOUND,
USER_NOT_FOUND,
USER_SHORT_DISPLAY_NAME,
} from 'src/errors';
import { mockDeep, mockReset } from 'jest-mock-extended'; import { mockDeep, mockReset } from 'jest-mock-extended';
import { PrismaService } from 'src/prisma/prisma.service'; import { PrismaService } from 'src/prisma/prisma.service';
import { AuthUser } from 'src/types/AuthUser'; import { AuthUser } from 'src/types/AuthUser';
@@ -144,7 +149,7 @@ beforeEach(() => {
describe('UserService', () => { describe('UserService', () => {
describe('findUserByEmail', () => { describe('findUserByEmail', () => {
test('should successfully return a valid user given a valid email', async () => { test('should successfully return a valid user given a valid email', async () => {
mockPrisma.user.findUniqueOrThrow.mockResolvedValueOnce(user); mockPrisma.user.findFirst.mockResolvedValueOnce(user);
const result = await userService.findUserByEmail( const result = await userService.findUserByEmail(
'dwight@dundermifflin.com', 'dwight@dundermifflin.com',
@@ -153,7 +158,7 @@ describe('UserService', () => {
}); });
test('should return a null user given a invalid email', async () => { test('should return a null user given a invalid email', async () => {
mockPrisma.user.findUniqueOrThrow.mockRejectedValueOnce('NotFoundError'); mockPrisma.user.findFirst.mockResolvedValueOnce(null);
const result = await userService.findUserByEmail('jim@dundermifflin.com'); const result = await userService.findUserByEmail('jim@dundermifflin.com');
expect(result).resolves.toBeNone; expect(result).resolves.toBeNone;
@@ -480,6 +485,14 @@ describe('UserService', () => {
); );
expect(result).toEqualLeft(USER_NOT_FOUND); expect(result).toEqualLeft(USER_NOT_FOUND);
}); });
test('should resolve left and error when short display name is passed', async () => {
const newDisplayName = '';
const result = await userService.updateUserDisplayName(
user.uid,
newDisplayName,
);
expect(result).toEqualLeft(USER_SHORT_DISPLAY_NAME);
});
}); });
describe('fetchAllUsers', () => { describe('fetchAllUsers', () => {

View File

@@ -8,7 +8,11 @@ import * as T from 'fp-ts/Task';
import * as A from 'fp-ts/Array'; import * as A from 'fp-ts/Array';
import { pipe, constVoid } from 'fp-ts/function'; import { pipe, constVoid } from 'fp-ts/function';
import { AuthUser } from 'src/types/AuthUser'; import { AuthUser } from 'src/types/AuthUser';
import { USERS_NOT_FOUND, USER_NOT_FOUND } from 'src/errors'; import {
USERS_NOT_FOUND,
USER_NOT_FOUND,
USER_SHORT_DISPLAY_NAME,
} from 'src/errors';
import { SessionType, User } from './user.model'; import { SessionType, User } from './user.model';
import { USER_UPDATE_FAILED } from 'src/errors'; import { USER_UPDATE_FAILED } from 'src/errors';
import { PubSubService } from 'src/pubsub/pubsub.service'; import { PubSubService } from 'src/pubsub/pubsub.service';
@@ -58,16 +62,16 @@ export class UserService {
* @returns Option of found User * @returns Option of found User
*/ */
async findUserByEmail(email: string): Promise<O.None | O.Some<AuthUser>> { async findUserByEmail(email: string): Promise<O.None | O.Some<AuthUser>> {
try { const user = await this.prisma.user.findFirst({
const user = await this.prisma.user.findUniqueOrThrow({ where: {
where: { email: {
email: email, equals: email,
mode: 'insensitive',
}, },
}); },
return O.some(user); });
} catch (error) { if (!user) return O.none;
return O.none; return O.some(user);
}
} }
/** /**
@@ -291,6 +295,10 @@ export class UserService {
* @returns a Either of User or error * @returns a Either of User or error
*/ */
async updateUserDisplayName(userUID: string, displayName: string) { async updateUserDisplayName(userUID: string, displayName: string) {
if (!displayName || displayName.length === 0) {
return E.left(USER_SHORT_DISPLAY_NAME);
}
try { try {
const dbUpdatedUser = await this.prisma.user.update({ const dbUpdatedUser = await this.prisma.user.update({
where: { uid: userUID }, where: { uid: userUID },

View File

@@ -52,11 +52,34 @@ hopp [options or commands] arguments
Taking the above example, `pw.env.get("ENV1")` will return `"value1"` Taking the above example, `pw.env.get("ENV1")` will return `"value1"`
## Install ## 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:
``` - Once the dependencies are installed, install [@hoppscotch/cli](https://www.npmjs.com/package/@hoppscotch/cli) from npm by running:
npm i -g @hoppscotch/cli ```
``` npm i -g @hoppscotch/cli
```
## **Developing:** ## **Developing:**

View File

@@ -1,6 +1,31 @@
#!/usr/bin/env node #!/usr/bin/env node
// * The entry point of the CLI // * The entry point of the CLI
// @ts-check
import { cli } from "../dist/index.js"; 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", "name": "@hoppscotch/cli",
"version": "0.7.0", "version": "0.8.0",
"description": "A CLI to run Hoppscotch test scripts in CI environments.", "description": "A CLI to run Hoppscotch test scripts in CI environments.",
"homepage": "https://hoppscotch.io", "homepage": "https://hoppscotch.io",
"type": "module", "type": "module",
@@ -44,6 +44,7 @@
"axios": "1.6.7", "axios": "1.6.7",
"chalk": "5.3.0", "chalk": "5.3.0",
"commander": "11.1.0", "commander": "11.1.0",
"isolated-vm": "4.7.2",
"lodash-es": "4.17.21", "lodash-es": "4.17.21",
"qs": "6.11.2", "qs": "6.11.2",
"verzod": "0.2.2", "verzod": "0.2.2",
@@ -65,4 +66,4 @@
"tsup": "8.0.2", "tsup": "8.0.2",
"typescript": "5.3.3" "typescript": "5.3.3"
} }
} }

View File

@@ -20,7 +20,7 @@ describe("Test `hopp test <file>` command:", () => {
const out = getErrorCode(stderr); const out = getErrorCode(stderr);
expect(out).toBe<HoppErrorCode>("INVALID_ARGUMENT"); expect(out).toBe<HoppErrorCode>("INVALID_ARGUMENT");
}); });
}) });
describe("Supplied collection export file validations", () => { describe("Supplied collection export file validations", () => {
test("Errors with the code `FILE_NOT_FOUND` if the supplied collection export file doesn't exist", async () => { test("Errors with the code `FILE_NOT_FOUND` if the supplied collection export file doesn't exist", async () => {
@@ -66,6 +66,43 @@ describe("Test `hopp test <file>` command:", () => {
}); });
}); });
describe("Versioned entities", () => {
describe("Collections & Requests", () => {
const testFixtures = [
{ fileName: "coll-v1-req-v0.json", collVersion: 1, reqVersion: 0 },
{ fileName: "coll-v1-req-v1.json", collVersion: 1, reqVersion: 1 },
{ fileName: "coll-v2-req-v2.json", collVersion: 2, reqVersion: 2 },
{ fileName: "coll-v2-req-v3.json", collVersion: 2, reqVersion: 3 },
];
testFixtures.forEach(({ collVersion, fileName, reqVersion }) => {
test(`Successfully processes a supplied collection export file where the collection is based on the "v${collVersion}" schema and the request following the "v${reqVersion}" schema`, async () => {
const args = `test ${getTestJsonFilePath(fileName, "collection")}`;
const { error } = await runCLI(args);
expect(error).toBeNull();
});
});
});
describe("Environments", () => {
const testFixtures = [
{ fileName: "env-v0.json", version: 0 },
{ fileName: "env-v1.json", version: 1 },
];
testFixtures.forEach(({ fileName, version }) => {
test(`Successfully processes the supplied collection and environment export files where the environment is based on the "v${version}" schema`, async () => {
const ENV_PATH = getTestJsonFilePath(fileName, "environment");
const args = `test ${getTestJsonFilePath("sample-coll.json", "collection")} --env ${ENV_PATH}`;
const { error } = await runCLI(args);
expect(error).toBeNull();
});
});
});
});
test("Successfully processes a supplied collection export file of the expected format", async () => { test("Successfully processes a supplied collection export file of the expected format", async () => {
const args = `test ${getTestJsonFilePath("passes-coll.json", "collection")}`; const args = `test ${getTestJsonFilePath("passes-coll.json", "collection")}`;
const { error } = await runCLI(args); const { error } = await runCLI(args);
@@ -75,7 +112,8 @@ describe("Test `hopp test <file>` command:", () => {
test("Successfully inherits headers and authorization set at the root collection", async () => { test("Successfully inherits headers and authorization set at the root collection", async () => {
const args = `test ${getTestJsonFilePath( const args = `test ${getTestJsonFilePath(
"collection-level-headers-auth-coll.json", "collection" "collection-level-headers-auth-coll.json",
"collection"
)}`; )}`;
const { error } = await runCLI(args); const { error } = await runCLI(args);
@@ -84,7 +122,8 @@ describe("Test `hopp test <file>` command:", () => {
test("Persists environment variables set in the pre-request script for consumption in the test script", async () => { test("Persists environment variables set in the pre-request script for consumption in the test script", async () => {
const args = `test ${getTestJsonFilePath( const args = `test ${getTestJsonFilePath(
"pre-req-script-env-var-persistence-coll.json", "collection" "pre-req-script-env-var-persistence-coll.json",
"collection"
)}`; )}`;
const { error } = await runCLI(args); const { error } = await runCLI(args);
@@ -106,7 +145,8 @@ describe("Test `hopp test <file> --env <file>` command:", () => {
test("Errors with the code `INVALID_FILE_TYPE` if the supplied environment export file doesn't end with the `.json` extension", async () => { test("Errors with the code `INVALID_FILE_TYPE` if the supplied environment export file doesn't end with the `.json` extension", async () => {
const args = `${VALID_TEST_ARGS} --env ${getTestJsonFilePath( const args = `${VALID_TEST_ARGS} --env ${getTestJsonFilePath(
"notjson-coll.txt", "collection" "notjson-coll.txt",
"collection"
)}`; )}`;
const { stderr } = await runCLI(args); const { stderr } = await runCLI(args);
@@ -123,7 +163,10 @@ describe("Test `hopp test <file> --env <file>` command:", () => {
}); });
test("Errors with the code `MALFORMED_ENV_FILE` on supplying a malformed environment export file", async () => { test("Errors with the code `MALFORMED_ENV_FILE` on supplying a malformed environment export file", async () => {
const ENV_PATH = getTestJsonFilePath("malformed-envs.json", "environment"); const ENV_PATH = getTestJsonFilePath(
"malformed-envs.json",
"environment"
);
const args = `${VALID_TEST_ARGS} --env ${ENV_PATH}`; const args = `${VALID_TEST_ARGS} --env ${ENV_PATH}`;
const { stderr } = await runCLI(args); const { stderr } = await runCLI(args);
@@ -142,7 +185,10 @@ describe("Test `hopp test <file> --env <file>` command:", () => {
}); });
test("Successfully resolves values from the supplied environment export file", async () => { test("Successfully resolves values from the supplied environment export file", async () => {
const TESTS_PATH = getTestJsonFilePath("env-flag-tests-coll.json", "collection"); const TESTS_PATH = getTestJsonFilePath(
"env-flag-tests-coll.json",
"collection"
);
const ENV_PATH = getTestJsonFilePath("env-flag-envs.json", "environment"); const ENV_PATH = getTestJsonFilePath("env-flag-envs.json", "environment");
const args = `test ${TESTS_PATH} --env ${ENV_PATH}`; const args = `test ${TESTS_PATH} --env ${ENV_PATH}`;
@@ -151,8 +197,14 @@ describe("Test `hopp test <file> --env <file>` command:", () => {
}); });
test("Successfully resolves environment variables referenced in the request body", async () => { test("Successfully resolves environment variables referenced in the request body", async () => {
const COLL_PATH = getTestJsonFilePath("req-body-env-vars-coll.json", "collection"); const COLL_PATH = getTestJsonFilePath(
const ENVS_PATH = getTestJsonFilePath("req-body-env-vars-envs.json", "environment"); "req-body-env-vars-coll.json",
"collection"
);
const ENVS_PATH = getTestJsonFilePath(
"req-body-env-vars-envs.json",
"environment"
);
const args = `test ${COLL_PATH} --env ${ENVS_PATH}`; const args = `test ${COLL_PATH} --env ${ENVS_PATH}`;
const { error } = await runCLI(args); const { error } = await runCLI(args);
@@ -160,7 +212,10 @@ describe("Test `hopp test <file> --env <file>` command:", () => {
}); });
test("Works with shorth `-e` flag", async () => { test("Works with shorth `-e` flag", async () => {
const TESTS_PATH = getTestJsonFilePath("env-flag-tests-coll.json", "collection"); const TESTS_PATH = getTestJsonFilePath(
"env-flag-tests-coll.json",
"collection"
);
const ENV_PATH = getTestJsonFilePath("env-flag-envs.json", "environment"); const ENV_PATH = getTestJsonFilePath("env-flag-envs.json", "environment");
const args = `test ${TESTS_PATH} -e ${ENV_PATH}`; const args = `test ${TESTS_PATH} -e ${ENV_PATH}`;
@@ -169,7 +224,7 @@ describe("Test `hopp test <file> --env <file>` command:", () => {
}); });
describe("Secret environment variables", () => { describe("Secret environment variables", () => {
jest.setTimeout(10000); jest.setTimeout(100000);
// Reads secret environment values from system environment // 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 () => { test("Successfully picks the values for secret environment variables from `process.env` and persists the variables set from the pre-request script", async () => {
@@ -183,7 +238,10 @@ describe("Test `hopp test <file> --env <file>` command:", () => {
secretHeaderValue: "secret-header-value", secretHeaderValue: "secret-header-value",
}; };
const COLL_PATH = getTestJsonFilePath("secret-envs-coll.json", "collection"); const COLL_PATH = getTestJsonFilePath(
"secret-envs-coll.json",
"collection"
);
const ENVS_PATH = getTestJsonFilePath("secret-envs.json", "environment"); const ENVS_PATH = getTestJsonFilePath("secret-envs.json", "environment");
const args = `test ${COLL_PATH} --env ${ENVS_PATH}`; const args = `test ${COLL_PATH} --env ${ENVS_PATH}`;
@@ -197,8 +255,14 @@ describe("Test `hopp test <file> --env <file>` command:", () => {
// Prefers values specified in the environment export file over values set in the system environment // Prefers values specified in the environment export file over values set in the system environment
test("Successfully picks the values for secret environment variables set directly in the environment export file and persists the environment variables set from the pre-request script", async () => { test("Successfully picks the values for secret environment variables set directly in the environment export file and persists the environment variables set from the pre-request script", async () => {
const COLL_PATH = getTestJsonFilePath("secret-envs-coll.json", "collection"); const COLL_PATH = getTestJsonFilePath(
const ENVS_PATH = getTestJsonFilePath("secret-supplied-values-envs.json", "environment"); "secret-envs-coll.json",
"collection"
);
const ENVS_PATH = getTestJsonFilePath(
"secret-supplied-values-envs.json",
"environment"
);
const args = `test ${COLL_PATH} --env ${ENVS_PATH}`; const args = `test ${COLL_PATH} --env ${ENVS_PATH}`;
const { error, stdout } = await runCLI(args); const { error, stdout } = await runCLI(args);
@@ -212,9 +276,13 @@ describe("Test `hopp test <file> --env <file>` command:", () => {
// Values set from the scripting context takes the highest precedence // Values set from the scripting context takes the highest precedence
test("Setting values for secret environment variables from the pre-request script overrides values set at the supplied environment export file", async () => { test("Setting values for secret environment variables from the pre-request script overrides values set at the supplied environment export file", async () => {
const COLL_PATH = getTestJsonFilePath( const COLL_PATH = getTestJsonFilePath(
"secret-envs-persistence-coll.json", "collection" "secret-envs-persistence-coll.json",
"collection"
);
const ENVS_PATH = getTestJsonFilePath(
"secret-supplied-values-envs.json",
"environment"
); );
const ENVS_PATH = getTestJsonFilePath("secret-supplied-values-envs.json", "environment");
const args = `test ${COLL_PATH} --env ${ENVS_PATH}`; const args = `test ${COLL_PATH} --env ${ENVS_PATH}`;
const { error, stdout } = await runCLI(args); const { error, stdout } = await runCLI(args);
@@ -227,10 +295,12 @@ describe("Test `hopp test <file> --env <file>` command:", () => {
test("Persists secret environment variable values set from the pre-request script for consumption in the request and post-request script context", async () => { test("Persists secret environment variable values set from the pre-request script for consumption in the request and post-request script context", async () => {
const COLL_PATH = getTestJsonFilePath( const COLL_PATH = getTestJsonFilePath(
"secret-envs-persistence-scripting-coll.json", "collection" "secret-envs-persistence-scripting-coll.json",
"collection"
); );
const ENVS_PATH = getTestJsonFilePath( const ENVS_PATH = getTestJsonFilePath(
"secret-envs-persistence-scripting-envs.json", "environment" "secret-envs-persistence-scripting-envs.json",
"environment"
); );
const args = `test ${COLL_PATH} --env ${ENVS_PATH}`; const args = `test ${COLL_PATH} --env ${ENVS_PATH}`;

View File

@@ -1,84 +0,0 @@
import { isRESTCollection } from "../../../utils/checks";
describe("isRESTCollection", () => {
test("Undefined collection value.", () => {
expect(isRESTCollection(undefined)).toBeFalsy();
});
test("Invalid id value.", () => {
expect(
isRESTCollection({
v: 1,
name: "test",
id: 1,
})
).toBeFalsy();
});
test("Invalid requests value.", () => {
expect(
isRESTCollection({
v: 1,
name: "test",
id: "1",
requests: null,
})
).toBeFalsy();
});
test("Invalid folders value.", () => {
expect(
isRESTCollection({
v: 1,
name: "test",
id: "1",
requests: [],
folders: undefined,
})
).toBeFalsy();
});
test("Invalid RESTCollection(s) in folders.", () => {
expect(
isRESTCollection({
v: 1,
name: "test",
id: "1",
requests: [],
folders: [
{
v: 1,
name: "test1",
id: "2",
requests: undefined,
folders: [],
},
],
})
).toBeFalsy();
});
test("Invalid HoppRESTRequest(s) in requests.", () => {
expect(
isRESTCollection({
v: 1,
name: "test",
id: "1",
requests: [{}],
folders: [],
})
).toBeFalsy();
});
test("Valid RESTCollection.", () => {
expect(
isRESTCollection({
v: 1,
name: "test",
id: "1",
requests: [],
folders: [],
})
).toBeTruthy();
});
});

View File

@@ -0,0 +1,55 @@
{
"v": 1,
"name": "coll-v1",
"folders": [
{
"v": 1,
"name": "coll-v1-child",
"folders": [],
"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-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

@@ -0,0 +1,97 @@
{
"v": 1,
"name": "coll-v1",
"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://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",
"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"
}
}
]
}

View File

@@ -0,0 +1,109 @@
{
"v": 2,
"name": "coll-v2",
"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://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",
"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": []
}

View File

@@ -0,0 +1,109 @@
{
"v": 2,
"name": "coll-v2",
"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://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",
"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": []
}

View File

@@ -1,18 +1,18 @@
[ [
{ {
"v": 1, "v": 2,
"name": "CollectionA", "name": "CollectionA",
"folders": [ "folders": [
{ {
"v": 1, "v": 2,
"name": "FolderA", "name": "FolderA",
"folders": [ "folders": [
{ {
"v": 1, "v": 2,
"name": "FolderB", "name": "FolderB",
"folders": [ "folders": [
{ {
"v": 1, "v": 2,
"name": "FolderC", "name": "FolderC",
"folders": [], "folders": [],
"requests": [ "requests": [
@@ -75,7 +75,7 @@
"auth": { "auth": {
"authType": "api-key", "authType": "api-key",
"authActive": true, "authActive": true,
"addTo": "Headers", "addTo": "HEADERS",
"key": "key", "key": "key",
"value": "test-key" "value": "test-key"
}, },
@@ -153,11 +153,11 @@
} }
}, },
{ {
"v": 1, "v": 2,
"name": "CollectionB", "name": "CollectionB",
"folders": [ "folders": [
{ {
"v": 1, "v": 2,
"name": "FolderA", "name": "FolderA",
"folders": [], "folders": [],
"requests": [ "requests": [

View File

@@ -0,0 +1,26 @@
{
"v": 1,
"name": "tests",
"folders": [],
"requests": [
{
"v": "2",
"endpoint": "<<baseURL>>",
"name": "",
"params": [],
"headers": [],
"method": "GET",
"auth": {
"authType": "none",
"authActive": true
},
"preRequestScript": "",
"testScript": "// Check status code is 200\npw.test(\"Status code is 200\", ()=> {\n pw.expect(pw.response.status).toBe(200);\n});\n\n// Check JSON response property\npw.test(\"Check JSON response property\", ()=> {\n pw.expect(pw.response.body.method).toBe(\"GET\");\n pw.expect(pw.response.body.headers).toBeType(\"object\");\n});",
"body": {
"contentType": null,
"body": null
},
"requestVariables": []
}
]
}

View File

@@ -5,8 +5,14 @@
"requests": [ "requests": [
{ {
"v": "3", "v": "3",
"auth": { "authType": "none", "authActive": true }, "auth": {
"body": { "body": null, "contentType": null }, "authType": "none",
"authActive": true
},
"body": {
"body": null,
"contentType": null
},
"name": "test-secret-headers", "name": "test-secret-headers",
"method": "GET", "method": "GET",
"params": [], "params": [],
@@ -18,13 +24,16 @@
} }
], ],
"requestVariables": [], "requestVariables": [],
"endpoint": "<<baseURL>>/headers", "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})", "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)" "preRequestScript": "const secretHeaderValueFromPreReqScript = pw.env.get(\"secretHeaderValue\")\npw.env.set(\"secretHeaderValueFromPreReqScript\", secretHeaderValueFromPreReqScript)"
}, },
{ {
"v": "3", "v": "3",
"auth": { "authType": "none", "authActive": true }, "auth": {
"authType": "none",
"authActive": true
},
"body": { "body": {
"body": "{\n \"secretBodyKey\": \"<<secretBodyValue>>\"\n}", "body": "{\n \"secretBodyKey\": \"<<secretBodyValue>>\"\n}",
"contentType": "application/json" "contentType": "application/json"
@@ -34,14 +43,20 @@
"params": [], "params": [],
"headers": [], "headers": [],
"requestVariables": [], "requestVariables": [],
"endpoint": "<<baseURL>>/post", "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(pw.response.body.json.secretBodyKey).toBe(secretBodyValue)\n }\n\n pw.expect(pw.env.get(\"secretBodyValueFromPreReqScript\")).toBe(\"secret-body-value\")\n})", "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)" "preRequestScript": "const secretBodyValueFromPreReqScript = pw.env.get(\"secretBodyValue\")\npw.env.set(\"secretBodyValueFromPreReqScript\", secretBodyValueFromPreReqScript)"
}, },
{ {
"v": "3", "v": "3",
"auth": { "authType": "none", "authActive": true }, "auth": {
"body": { "body": null, "contentType": null }, "authType": "none",
"authActive": true
},
"body": {
"body": null,
"contentType": null
},
"name": "test-secret-query-params", "name": "test-secret-query-params",
"method": "GET", "method": "GET",
"params": [ "params": [
@@ -53,7 +68,7 @@
], ],
"headers": [], "headers": [],
"requestVariables": [], "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})", "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)" "preRequestScript": "const secretQueryParamValueFromPreReqScript = pw.env.get(\"secretQueryParamValue\")\npw.env.set(\"secretQueryParamValueFromPreReqScript\", secretQueryParamValueFromPreReqScript)"
}, },
@@ -65,14 +80,17 @@
"username": "<<secretBasicAuthUsername>>", "username": "<<secretBasicAuthUsername>>",
"authActive": true "authActive": true
}, },
"body": { "body": null, "contentType": null }, "body": {
"body": null,
"contentType": null
},
"name": "test-secret-basic-auth", "name": "test-secret-basic-auth",
"method": "GET", "method": "GET",
"params": [], "params": [],
"headers": [], "headers": [],
"requestVariables": [], "requestVariables": [],
"endpoint": "<<baseURL>>/basic-auth/<<secretBasicAuthUsername>>/<<secretBasicAuthPassword>>", "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 if (secretBasicAuthUsername && secretBasicAuthPassword) {\n const { authenticated, user } = pw.response.body\n pw.expect(authenticated).toBe(true)\n pw.expect(user).toBe(secretBasicAuthUsername)\n }\n});", "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": "" "preRequestScript": ""
}, },
{ {
@@ -84,30 +102,42 @@
"username": "testuser", "username": "testuser",
"authActive": true "authActive": true
}, },
"body": { "body": null, "contentType": null }, "body": {
"body": null,
"contentType": null
},
"name": "test-secret-bearer-auth", "name": "test-secret-bearer-auth",
"method": "GET", "method": "GET",
"params": [], "params": [],
"headers": [], "headers": [],
"requestVariables": [], "requestVariables": [],
"endpoint": "<<baseURL>>/bearer", "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 if (secretBearerToken) { \n pw.expect(pw.response.body.token).toBe(secretBearerToken)\n pw.expect(preReqSecretBearerToken).toBe(\"test-token\")\n }\n});", "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)" "preRequestScript": "const secretBearerToken = pw.env.get(\"secretBearerToken\")\npw.env.set(\"preReqSecretBearerToken\", secretBearerToken)"
}, },
{ {
"v": "3", "v": "3",
"auth": { "authType": "none", "authActive": true }, "auth": {
"body": { "body": null, "contentType": null }, "authType": "none",
"authActive": true
},
"body": {
"body": null,
"contentType": null
},
"name": "test-secret-fallback", "name": "test-secret-fallback",
"method": "GET", "method": "GET",
"params": [], "params": [],
"headers": [], "headers": [],
"requestVariables": [], "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})", "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": "" "preRequestScript": ""
} }
], ],
"auth": { "authType": "inherit", "authActive": false }, "auth": {
"authType": "inherit",
"authActive": false
},
"headers": [] "headers": []
} }

View File

@@ -1,6 +1,6 @@
{ {
"v": 2, "v": 2,
"name": "secret-envs-setters-coll", "name": "secret-envs-persistence-coll",
"folders": [], "folders": [],
"requests": [ "requests": [
{ {
@@ -24,8 +24,8 @@
"active": true "active": true
} }
], ],
"endpoint": "<<baseURL>>/headers", "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})", "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)" "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 "active": true
} }
], ],
"endpoint": "<<baseURL>>/headers", "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})", "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)" "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": [], "params": [],
"requestVariables": [], "requestVariables": [],
"headers": [], "headers": [],
"endpoint": "<<baseURL>>/post", "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(pw.response.body.json.secretBodyKey).toBe(secretBodyValue)\n }\n\n pw.expect(pw.env.get(\"secretBodyValueFromPreReqScript\")).toBe(\"secret-body-value\")\n})", "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)" "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": [], "requestVariables": [],
"headers": [], "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})", "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)" "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": [], "params": [],
"requestVariables": [], "requestVariables": [],
"headers": [], "headers": [],
"endpoint": "<<baseURL>>/basic-auth/<<secretBasicAuthUsername>>/<<secretBasicAuthPassword>>", "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 if (secretBasicAuthUsername && secretBasicAuthPassword) {\n const { authenticated, user } = pw.response.body\n pw.expect(authenticated).toBe(true)\n pw.expect(user).toBe(secretBasicAuthUsername)\n }\n});", "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}" "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": [], "params": [],
"requestVariables": [], "requestVariables": [],
"headers": [], "headers": [],
"endpoint": "<<baseURL>>/bearer", "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 if (secretBearerToken) { \n pw.expect(pw.response.body.token).toBe(secretBearerToken)\n pw.expect(preReqSecretBearerToken).toBe(\"test-token\")\n }\n});", "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)" "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 "authActive": false
}, },
"headers": [] "headers": []
} }

View File

@@ -5,7 +5,7 @@
"requests": [ "requests": [
{ {
"v": "3", "v": "3",
"endpoint": "https://httpbin.org/post", "endpoint": "https://echo.hoppscotch.io/post",
"name": "req", "name": "req",
"params": [], "params": [],
"headers": [ "headers": [
@@ -18,7 +18,7 @@
"method": "POST", "method": "POST",
"auth": { "authType": "none", "authActive": true }, "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\")", "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": { "body": {
"contentType": "application/json", "contentType": "application/json",
"body": "{\n \"key\": \"<<customBodyValue>>\"\n}" "body": "{\n \"key\": \"<<customBodyValue>>\"\n}"

View File

@@ -0,0 +1,9 @@
{
"name": "env-v0",
"variables": [
{
"key": "baseURL",
"value": "https://echo.hoppscotch.io"
}
]
}

View File

@@ -0,0 +1,10 @@
{
"name": "env-v0",
"variables": [
{
"key": "baseURL",
"value": "https://echo.hoppscotch.io",
"secret": false
}
]
}

View File

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

View File

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

View File

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

View File

@@ -1,6 +1,7 @@
import chalk from "chalk"; import chalk from "chalk";
import { Command } from "commander"; import { Command } from "commander";
import * as E from "fp-ts/Either"; import * as E from "fp-ts/Either";
import { version } from "../package.json"; import { version } from "../package.json";
import { test } from "./commands/test"; import { test } from "./commands/test";
import { handleError } from "./handlers/error"; 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" "https://docs.hoppscotch.io/documentation/clients/cli"
)}`; )}`;
const program = new Command() const program = new Command();
program program
.name("hopp") .name("hopp")

View File

@@ -47,7 +47,7 @@ export async function parseEnvsData(path: string) {
if (HoppEnvKeyPairResult.success) { if (HoppEnvKeyPairResult.success) {
for (const [key, value] of Object.entries(HoppEnvKeyPairResult.data)) { for (const [key, value] of Object.entries(HoppEnvKeyPairResult.data)) {
envPairs.push({ key, value }); envPairs.push({ key, value, secret: false });
} }
} else if (HoppEnvExportObjectResult.type === "ok") { } else if (HoppEnvExportObjectResult.type === "ok") {
envPairs.push(...HoppEnvExportObjectResult.value.variables); envPairs.push(...HoppEnvExportObjectResult.value.variables);

View File

@@ -1,5 +1,3 @@
import { HoppCollection, isHoppRESTRequest } from "@hoppscotch/data";
import * as A from "fp-ts/Array";
import { CommanderError } from "commander"; import { CommanderError } from "commander";
import { HoppCLIError, HoppErrnoException } from "../types/errors"; import { HoppCLIError, HoppErrnoException } from "../types/errors";
@@ -14,48 +12,6 @@ export const hasProperty = <P extends PropertyKey>(
prop: P prop: P
): target is Record<P, unknown> => prop in target; ): target is Record<P, unknown> => prop in target;
/**
* Typeguard to check valid Hoppscotch REST Collection.
* @param param The object to be checked.
* @returns True, if unknown parameter is valid Hoppscotch REST Collection;
* False, otherwise.
*/
export const isRESTCollection = (param: unknown): param is HoppCollection => {
if (!!param && typeof param === "object") {
if (!hasProperty(param, "v") || typeof param.v !== "number") {
return false;
}
if (!hasProperty(param, "name") || typeof param.name !== "string") {
return false;
}
if (hasProperty(param, "id") && typeof param.id !== "string") {
return false;
}
if (!hasProperty(param, "requests") || !Array.isArray(param.requests)) {
return false;
} else {
// Checks each requests array to be valid HoppRESTRequest.
const checkRequests = A.every(isHoppRESTRequest)(param.requests);
if (!checkRequests) {
return false;
}
}
if (!hasProperty(param, "folders") || !Array.isArray(param.folders)) {
return false;
} else {
// Checks each folder to be valid REST collection.
const checkFolders = A.every(isRESTCollection)(param.folders);
if (!checkFolders) {
return false;
}
}
return true;
}
return false;
};
/** /**
* Checks if given error data is of type HoppCLIError, based on existence * Checks if given error data is of type HoppCLIError, based on existence
* of code property. * of code property.

View File

@@ -1,8 +1,46 @@
import { HoppCollection, HoppRESTRequest } from "@hoppscotch/data";
import fs from "fs/promises"; import fs from "fs/promises";
import { FormDataEntry } from "../types/request"; import { entityReference } from "verzod";
import { z } from "zod";
import { error } from "../types/errors"; import { error } from "../types/errors";
import { isRESTCollection, isHoppErrnoException } from "./checks"; import { FormDataEntry } from "../types/request";
import { HoppCollection } from "@hoppscotch/data"; 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. * Parses array of FormDataEntry to FormData.
@@ -67,7 +105,11 @@ export async function parseCollectionData(
? contents ? contents
: [contents]; : [contents];
if (maybeArrayOfCollections.some((x) => !isRESTCollection(x))) { const collectionSchemaParsedResult = z
.array(entityReference(HoppCollection))
.safeParse(maybeArrayOfCollections);
if (!collectionSchemaParsedResult.success) {
throw error({ throw error({
code: "MALFORMED_COLLECTION", code: "MALFORMED_COLLECTION",
path, path,
@@ -75,5 +117,5 @@ export async function parseCollectionData(
}); });
} }
return maybeArrayOfCollections as HoppCollection[]; return getValidRequests(collectionSchemaParsedResult.data, path);
} }

View File

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

View File

@@ -10,6 +10,9 @@ module.exports = {
parserOptions: { parserOptions: {
sourceType: "module", sourceType: "module",
requireConfigFile: false, requireConfigFile: false,
ecmaFeatures: {
jsx: false,
},
}, },
extends: [ extends: [
"@vue/typescript/recommended", "@vue/typescript/recommended",

View File

@@ -27,6 +27,7 @@
"hide_secret": "Hide secret", "hide_secret": "Hide secret",
"label": "Label", "label": "Label",
"learn_more": "Learn more", "learn_more": "Learn more",
"download_here": "Download here",
"less": "Less", "less": "Less",
"more": "More", "more": "More",
"new": "New", "new": "New",
@@ -103,8 +104,10 @@
"auth": { "auth": {
"account_exists": "Account exists with different credential - Login to link both accounts", "account_exists": "Account exists with different credential - Login to link both accounts",
"all_sign_in_options": "All sign in options", "all_sign_in_options": "All sign in options",
"continue_with_auth_provider": "Continue with {provider}",
"continue_with_email": "Continue with Email", "continue_with_email": "Continue with Email",
"continue_with_github": "Continue with GitHub", "continue_with_github": "Continue with GitHub",
"continue_with_github_enterprise": "Continue with GitHub Enterprise",
"continue_with_google": "Continue with Google", "continue_with_google": "Continue with Google",
"continue_with_microsoft": "Continue with Microsoft", "continue_with_microsoft": "Continue with Microsoft",
"email": "Email", "email": "Email",
@@ -159,6 +162,8 @@
"label_client_credentials": "Client Credentials" "label_client_credentials": "Client Credentials"
}, },
"pass_key_by": "Pass by", "pass_key_by": "Pass by",
"pass_by_query_params_label": "Query Parameters",
"pass_by_headers_label": "Headers",
"password": "Password", "password": "Password",
"save_to_inherit": "Please save this request in any collection to inherit the authorization", "save_to_inherit": "Please save this request in any collection to inherit the authorization",
"token": "Token", "token": "Token",
@@ -170,6 +175,7 @@
"different_parent": "Cannot reorder collection with different parent", "different_parent": "Cannot reorder collection with different parent",
"edit": "Edit Collection", "edit": "Edit Collection",
"import_or_create": "Import or create a collection", "import_or_create": "Import or create a collection",
"import_collection":"Import Collection",
"invalid_name": "Please provide a name for the collection", "invalid_name": "Please provide a name for the collection",
"invalid_root_move": "Collection already in the root", "invalid_root_move": "Collection already in the root",
"moved": "Moved Successfully", "moved": "Moved Successfully",
@@ -312,6 +318,7 @@
"danger_zone": "Danger zone", "danger_zone": "Danger zone",
"delete_account": "Your account is currently an owner in these workspaces:", "delete_account": "Your account is currently an owner in these workspaces:",
"delete_account_description": "You must either remove yourself, transfer ownership, or delete these workspaces before you can delete your account.", "delete_account_description": "You must either remove yourself, transfer ownership, or delete these workspaces before you can delete your account.",
"empty_profile_name": "Profile name cannot be empty",
"empty_req_name": "Empty Request Name", "empty_req_name": "Empty Request Name",
"f12_details": "(F12 for details)", "f12_details": "(F12 for details)",
"gql_prettify_invalid_query": "Couldn't prettify an invalid query, solve query syntax errors and try again", "gql_prettify_invalid_query": "Couldn't prettify an invalid query, solve query syntax errors and try again",
@@ -331,6 +338,7 @@
"page_not_found": "This page could not be found", "page_not_found": "This page could not be found",
"please_install_extension": "Please install the extension and add origin to the extension.", "please_install_extension": "Please install the extension and add origin to the extension.",
"proxy_error": "Proxy error", "proxy_error": "Proxy error",
"same_profile_name": "Updated profile name is same as the current profile name",
"script_fail": "Could not execute pre-request script", "script_fail": "Could not execute pre-request script",
"something_went_wrong": "Something went wrong", "something_went_wrong": "Something went wrong",
"test_script_fail": "Could not execute post-request script", "test_script_fail": "Could not execute post-request script",
@@ -366,7 +374,8 @@
"mutations": "Mutations", "mutations": "Mutations",
"schema": "Schema", "schema": "Schema",
"subscriptions": "Subscriptions", "subscriptions": "Subscriptions",
"switch_connection": "Switch connection" "switch_connection": "Switch connection",
"url_placeholder": "Enter a GraphQL endpoint URL"
}, },
"graphql_collections": { "graphql_collections": {
"title": "GraphQL Collections" "title": "GraphQL Collections"
@@ -446,7 +455,7 @@
"not_found": "Environment variable “{environment}” not found." "not_found": "Environment variable “{environment}” not found."
}, },
"header": { "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": "The browser doesn't allow Hoppscotch to set Cookie Headers. Please use Authorization Headers instead. However, our Hoppscotch Desktop App is live now and supports Cookies."
}, },
"response": { "response": {
"401_error": "Please check your authentication credentials.", "401_error": "Please check your authentication credentials.",
@@ -590,6 +599,7 @@
"title": "Request", "title": "Request",
"type": "Request type", "type": "Request type",
"url": "URL", "url": "URL",
"url_placeholder": "Enter a URL or paste a cURL command",
"variables": "Variables", "variables": "Variables",
"view_my_links": "View my links" "view_my_links": "View my links"
}, },
@@ -842,6 +852,13 @@
"new": "Create new workspace", "new": "Create new workspace",
"switch_to_personal": "Switch to your personal workspace", "switch_to_personal": "Switch to your personal workspace",
"title": "Workspaces" "title": "Workspaces"
},
"phrases":{
"try": "Try",
"import_collections": "Import collections",
"create_environment": "Create environment",
"create_workspace": "Create workspace",
"share_request": "Share request"
} }
}, },
"sse": { "sse": {

View File

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

View File

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

View File

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

File diff suppressed because it is too large Load Diff

View File

@@ -11,12 +11,13 @@
"connect": "Подключиться", "connect": "Подключиться",
"connecting": "Соединение...", "connecting": "Соединение...",
"copy": "Скопировать", "copy": "Скопировать",
"create": "Create", "create": "Создать",
"delete": "Удалить", "delete": "Удалить",
"disconnect": "Отключиться", "disconnect": "Отключиться",
"dismiss": "Скрыть", "dismiss": "Скрыть",
"dont_save": "Не сохранять", "dont_save": "Не сохранять",
"download_file": "Скачать файл", "download_file": "Скачать файл",
"download_here": "Download here",
"drag_to_reorder": "Перетягивайте для сортировки", "drag_to_reorder": "Перетягивайте для сортировки",
"duplicate": "Дублировать", "duplicate": "Дублировать",
"edit": "Редактировать", "edit": "Редактировать",
@@ -24,6 +25,7 @@
"go_back": "Вернуться", "go_back": "Вернуться",
"go_forward": "Вперёд", "go_forward": "Вперёд",
"group_by": "Сгруппировать по", "group_by": "Сгруппировать по",
"hide_secret": "Hide secret",
"label": "Название", "label": "Название",
"learn_more": "Узнать больше", "learn_more": "Узнать больше",
"less": "Меньше", "less": "Меньше",
@@ -33,7 +35,7 @@
"open_workspace": "Открыть пространство", "open_workspace": "Открыть пространство",
"paste": "Вставить", "paste": "Вставить",
"prettify": "Форматировать", "prettify": "Форматировать",
"properties": "Properties", "properties": "Параметры",
"remove": "Удалить", "remove": "Удалить",
"rename": "Переименовать", "rename": "Переименовать",
"restore": "Восстановить", "restore": "Восстановить",
@@ -42,13 +44,14 @@
"scroll_to_top": "Вверх", "scroll_to_top": "Вверх",
"search": "Поиск", "search": "Поиск",
"send": "Отправить", "send": "Отправить",
"share": "Share", "share": "Поделиться",
"show_secret": "Show secret",
"start": "Начать", "start": "Начать",
"starting": "Запускаю", "starting": "Запускаю",
"stop": "Стоп", "stop": "Стоп",
"to_close": "что бы закрыть", "to_close": "закрыть",
"to_navigate": "для навигации", "to_navigate": "для навигации",
"to_select": "выборать", "to_select": "выбрать",
"turn_off": "Выключить", "turn_off": "Выключить",
"turn_on": "Включить", "turn_on": "Включить",
"undo": "Отменить", "undo": "Отменить",
@@ -66,12 +69,12 @@
"copy_interface_type": "Copy interface type", "copy_interface_type": "Copy interface type",
"copy_user_id": "Копировать токен пользователя", "copy_user_id": "Копировать токен пользователя",
"developer_option": "Настройки разработчика", "developer_option": "Настройки разработчика",
"developer_option_description": "Инструмент разработчика помогает обслуживить и развивить Hoppscotch", "developer_option_description": "Инструмент разработчика помогает обслуживать и развивать Hoppscotch",
"discord": "Discord", "discord": "Discord",
"documentation": "Документация", "documentation": "Документация",
"github": "GitHub", "github": "GitHub",
"help": "Справка, отзывы и документация", "help": "Справка, отзывы и документация",
"home": "Дом", "home": "На главную",
"invite": "Пригласить", "invite": "Пригласить",
"invite_description": "В Hoppscotch мы разработали простой и интуитивно понятный интерфейс для создания и управления вашими API. Hoppscotch - это инструмент, который помогает создавать, тестировать, документировать и делиться своими API.", "invite_description": "В Hoppscotch мы разработали простой и интуитивно понятный интерфейс для создания и управления вашими API. Hoppscotch - это инструмент, который помогает создавать, тестировать, документировать и делиться своими API.",
"invite_your_friends": "Пригласить своих друзей", "invite_your_friends": "Пригласить своих друзей",
@@ -85,7 +88,7 @@
"reload": "Перезагрузить", "reload": "Перезагрузить",
"search": "Поиск", "search": "Поиск",
"share": "Поделиться", "share": "Поделиться",
"shortcuts": "Ярлыки", "shortcuts": "Горячие клавиши",
"social_description": "Подписывайся на наши соц. сети и оставайся всегда в курсе последних новостей, обновлений и релизов.", "social_description": "Подписывайся на наши соц. сети и оставайся всегда в курсе последних новостей, обновлений и релизов.",
"social_links": "Социальные сети", "social_links": "Социальные сети",
"spotlight": "Прожектор", "spotlight": "Прожектор",
@@ -96,17 +99,19 @@
"type_a_command_search": "Введите команду или выполните поиск…", "type_a_command_search": "Введите команду или выполните поиск…",
"we_use_cookies": "Мы используем куки", "we_use_cookies": "Мы используем куки",
"whats_new": "Что нового?", "whats_new": "Что нового?",
"wiki": "Вики" "wiki": "Узнать больше"
}, },
"auth": { "auth": {
"account_exists": "Учетная запись существует с разными учетными данными - войдите, чтобы связать обе учетные записи", "account_exists": "Учетная запись существует с разными учетными данными - войдите, чтобы связать обе учетные записи",
"all_sign_in_options": "Все варианты входа", "all_sign_in_options": "Все варианты входа",
"continue_with_auth_provider": "Continue with {provider}",
"continue_with_email": "Продолжить с электронной почтой", "continue_with_email": "Продолжить с электронной почтой",
"continue_with_github": "Продолжить с GitHub", "continue_with_github": "Продолжить с GitHub",
"continue_with_github_enterprise": "Continue with GitHub Enterprise",
"continue_with_google": "Продолжить с Google", "continue_with_google": "Продолжить с Google",
"continue_with_microsoft": "Продолжить с Microsoft", "continue_with_microsoft": "Продолжить с Microsoft",
"email": "Электронное письмо", "email": "Электронное письмо",
"logged_out": "Вышли из", "logged_out": "Успешно вышли. Будем скучать!",
"login": "Авторизоваться", "login": "Авторизоваться",
"login_success": "Успешный вход в систему", "login_success": "Успешный вход в систему",
"login_to_hoppscotch": "Войти в Hoppscotch", "login_to_hoppscotch": "Войти в Hoppscotch",
@@ -121,7 +126,7 @@
"generate_token": "Сгенерировать токен", "generate_token": "Сгенерировать токен",
"graphql_headers": "Authorization Headers are sent as part of the payload to connection_init", "graphql_headers": "Authorization Headers are sent as part of the payload to connection_init",
"include_in_url": "Добавить в URL", "include_in_url": "Добавить в URL",
"inherited_from": "Inherited {auth} from parent collection {collection} ", "inherited_from": "Унаследован тип аутентификации {auth} из родительской коллекции {collection}",
"learn": "Узнать больше", "learn": "Узнать больше",
"oauth": { "oauth": {
"redirect_auth_server_returned_error": "Auth Server returned an error state", "redirect_auth_server_returned_error": "Auth Server returned an error state",
@@ -135,11 +140,32 @@
"redirect_no_token_endpoint": "No Token Endpoint 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_oauth_redirect": "Something went wrong during OAuth Redirect",
"something_went_wrong_on_token_generation": "Something went wrong on token generation", "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" "token_generation_oidc_discovery_failed": "Failure on token generation: OpenID Connect Discovery Failed",
"grant_type": "Grant Type",
"grant_type_auth_code": "Authorization Code",
"token_fetched_successfully": "Token fetched successfully",
"token_fetch_failed": "Failed to fetch token",
"validation_failed": "Validation Failed, please check the form fields",
"label_authorization_endpoint": "Authorization Endpoint",
"label_client_id": "Client ID",
"label_client_secret": "Client Secret",
"label_code_challenge": "Code Challenge",
"label_code_challenge_method": "Code Challenge Method",
"label_code_verifier": "Code Verifier",
"label_scopes": "Scopes",
"label_token_endpoint": "Token Endpoint",
"label_use_pkce": "Use PKCE",
"label_implicit": "Implicit",
"label_password": "Password",
"label_username": "Username",
"label_auth_code": "Authorization Code",
"label_client_credentials": "Client Credentials"
}, },
"pass_by_headers_label": "Headers",
"pass_by_query_params_label": "Query Parameters",
"pass_key_by": "Pass by", "pass_key_by": "Pass by",
"password": "Пароль", "password": "Пароль",
"save_to_inherit": "Please save this request in any collection to inherit the authorization", "save_to_inherit": "Чтобы унаследовать аутентификации, нужно сохранить запрос в коллекции",
"token": "Токен", "token": "Токен",
"type": "Метод авторизации", "type": "Метод авторизации",
"username": "Имя пользователя" "username": "Имя пользователя"
@@ -149,6 +175,7 @@
"different_parent": "Нельзя сортировать коллекцию с разной родительской коллекцией", "different_parent": "Нельзя сортировать коллекцию с разной родительской коллекцией",
"edit": "Редактировать коллекцию", "edit": "Редактировать коллекцию",
"import_or_create": "Вы можете импортировать существующую или создать новую коллекцию", "import_or_create": "Вы можете импортировать существующую или создать новую коллекцию",
"import_collection": "Импортировать коллекцию",
"invalid_name": "Укажите допустимое название коллекции", "invalid_name": "Укажите допустимое название коллекции",
"invalid_root_move": "Коллекция уже в корне", "invalid_root_move": "Коллекция уже в корне",
"moved": "Перемещено успешно", "moved": "Перемещено успешно",
@@ -157,38 +184,36 @@
"name_length_insufficient": "Имя коллекции должно иметь 3 или более символов", "name_length_insufficient": "Имя коллекции должно иметь 3 или более символов",
"new": "Создать коллекцию", "new": "Создать коллекцию",
"order_changed": "Порядок коллекции обновлён", "order_changed": "Порядок коллекции обновлён",
"properties": "Collection Properties", "properties": "Параметры коллекции",
"properties_updated": "Collection Properties Updated", "properties_updated": "Параметры коллекции обновлены",
"renamed": "Коллекция переименована", "renamed": "Коллекция переименована",
"request_in_use": "Запрос обрабатывается", "request_in_use": "Запрос обрабатывается",
"save_as": "Сохранить как", "save_as": "Сохранить как",
"save_to_collection": "Сохранить в коллекцию", "save_to_collection": "Сохранить в коллекцию",
"select": "Выбрать коллекцию", "select": "Выбрать коллекцию",
"select_location": "Выберите местоположение", "select_location": "Выберите местоположение"
"select_team": "Выберите команду",
"team_collections": "Коллекции команд"
}, },
"confirm": { "confirm": {
"close_unsaved_tab": "Вы уверены что хотите закрыть эту вкладку?", "close_unsaved_tab": "Вы уверены, что хотите закрыть эту вкладку?",
"close_unsaved_tabs": "Вы уверены что хотите закрыть все эти вкладки? Несохранённые данные {count} вкладок будут утеряны.", "close_unsaved_tabs": "Вы уверены, что хотите закрыть все эти вкладки? Несохранённые данные {count} вкладок будут утеряны.",
"exit_team": "Вы точно хотите покинуть эту команду?", "exit_team": "Вы точно хотите покинуть эту команду?",
"logout": "Вы действительно хотите выйти?", "logout": "Вы действительно хотите выйти?",
"remove_collection": "Вы уверены, что хотите навсегда удалить эту коллекцию?", "remove_collection": "Вы уверены, что хотите навсегда удалить эту коллекцию?",
"remove_environment": "Вы действительно хотите удалить эту среду без возможности восстановления?", "remove_environment": "Вы действительно хотите удалить это окружение без возможности восстановления?",
"remove_folder": "Вы уверены, что хотите навсегда удалить эту папку?", "remove_folder": "Вы уверены, что хотите навсегда удалить эту папку?",
"remove_history": "Вы уверены, что хотите навсегда удалить всю историю?", "remove_history": "Вы уверены, что хотите навсегда удалить всю историю?",
"remove_request": "Вы уверены, что хотите навсегда удалить этот запрос?", "remove_request": "Вы уверены, что хотите навсегда удалить этот запрос?",
"remove_shared_request": "Are you sure you want to permanently delete this shared request?", "remove_shared_request": "Вы уверены, что хотите навсегда удалить этот запрос?",
"remove_team": "Вы уверены, что хотите удалить эту команду?", "remove_team": "Вы уверены, что хотите удалить эту команду?",
"remove_telemetry": "Вы действительно хотите отказаться от телеметрии?", "remove_telemetry": "Вы действительно хотите отказаться от телеметрии?",
"request_change": "Вы уверены что хотите сбросить текущий запрос, все не сохранённые данные будт утеряны?", "request_change": "Вы уверены, что хотите сбросить текущий запрос, все не сохранённые данные будт утеряны?",
"save_unsaved_tab": "Вы хотите сохранить изменения в этой вкладке?", "save_unsaved_tab": "Вы хотите сохранить изменения в этой вкладке?",
"sync": "Вы уверены, что хотите синхронизировать это рабочее пространство?" "sync": "Вы уверены, что хотите синхронизировать это рабочее пространство?"
}, },
"context_menu": { "context_menu": {
"add_parameters": "Add to parameters", "add_parameters": "Добавить в список параметров",
"open_request_in_new_tab": "Open request in new tab", "open_request_in_new_tab": "Открыть запрос в новом окне",
"set_environment_variable": "Set as variable" "set_environment_variable": "Добавить значение в переменную"
}, },
"cookies": { "cookies": {
"modal": { "modal": {
@@ -227,24 +252,25 @@
"collections": "Коллекции пустые", "collections": "Коллекции пустые",
"documentation": "Подключите GraphQL endpoint, чтобы увидеть документацию.", "documentation": "Подключите GraphQL endpoint, чтобы увидеть документацию.",
"endpoint": "Endpoint не может быть пустым", "endpoint": "Endpoint не может быть пустым",
"environments": "Окружения пусты", "environments": "Переменных окружения нет",
"folder": "Папка пуста", "folder": "Папка пуста",
"headers": "У этого запроса нет заголовков", "headers": "У этого запроса нет заголовков",
"history": "История пуста", "history": "История пуста",
"invites": "Вы еще никого не приглашали", "invites": "Вы еще никого не приглашали",
"members": "В этой команде еще нет участников", "members": "В этой команде еще нет участников",
"parameters": "Этот запрос не имеет параметров", "parameters": "Этот запрос не содержит параметров",
"pending_invites": "Пока что нет ожидающих заявок на вступление в команду", "pending_invites": "Пока что нет ожидающих заявок на вступление в команду",
"profile": "Войдите, чтобы просмотреть свой профиль", "profile": "Войдите, чтобы просмотреть свой профиль",
"protocols": "Протоколы пустые", "protocols": "Протоколы пустые",
"request_variables": "Этот запрос не содержит никаких переменных",
"secret_environments": "Секреты хранятся только на этом устройстве и не синхронизируются с сервером",
"schema": "Подключиться к конечной точке GraphQL", "schema": "Подключиться к конечной точке GraphQL",
"shared_requests": "Shared requests are empty", "shared_requests": "Вы еще не делились запросами с другими",
"shared_requests_logout": "Login to view your shared requests or create a new one", "shared_requests_logout": "Нужно войти, чтобы делиться запросами и управлять ими",
"subscription": "Нет подписок", "subscription": "Нет подписок",
"team_name": "Название команды пусто", "team_name": "Название команды пусто",
"teams": "Команды пустые", "teams": "Команды пустые",
"tests": "Для этого запроса нет тестов", "tests": "Для этого запроса нет тестов"
"shortcodes": "Нет коротких ссылок"
}, },
"environment": { "environment": {
"add_to_global": "Добавить в глобальное окружение", "add_to_global": "Добавить в глобальное окружение",
@@ -252,53 +278,57 @@
"create_new": "Создать новое окружение", "create_new": "Создать новое окружение",
"created": "Окружение создано", "created": "Окружение создано",
"deleted": "Окружение удалено", "deleted": "Окружение удалено",
"duplicated": "Environment duplicated", "duplicated": "Окружение продублировано",
"edit": "Редактировать окружение", "edit": "Редактировать окружение",
"empty_variables": "No variables", "empty_variables": "Переменные еще не добавлены",
"global": "Global", "global": "Global",
"global_variables": "Global variables", "global_variables": "Глобальные переменные",
"import_or_create": "Импортировать или создать новое окружение", "import_or_create": "Импортировать или создать новое окружение",
"invalid_name": "Укажите допустимое имя для окружения", "invalid_name": "Укажите допустимое имя для окружения",
"list": "Переменные окружения", "list": "Переменные окружения",
"my_environments": "Мои окружения", "my_environments": "Мои окружения",
"name": "Name", "name": "Имя",
"nested_overflow": "максимальный уровень вложения переменных окружения - 10", "nested_overflow": "максимальный уровень вложения переменных окружения - 10",
"new": "Новая среда", "new": "Новая среда",
"no_active_environment": "Нет активных окружений", "no_active_environment": "Нет активных окружений",
"no_environment": "Нет окружения", "no_environment": "Нет окружения",
"no_environment_description": "Не выбрано окружение, выберите что делать с переменными.", "no_environment_description": "Не выбрано окружение, выберите что делать с переменными.",
"quick_peek": "Environment Quick Peek", "quick_peek": "Быстрый просмотр переменных",
"replace_with_variable": "Replace with variable", "replace_with_variable": "Replace with variable",
"scope": "Scope", "scope": "Scope",
"secrets": "Секретные переменные",
"secret_value": "Секретное значение",
"select": "Выберите среду", "select": "Выберите среду",
"set": "Set environment", "set": "Выбрать окружение",
"set_as_environment": "Set as environment", "set_as_environment": "Поместить значение в переменную",
"team_environments": "Окружения команды", "team_environments": "Окружения команды",
"title": "Окружения", "title": "Окружения",
"updated": "Окружение обновлено", "updated": "Окружение обновлено",
"value": "Value", "value": "Значение",
"variable": "Variable", "variable": "Переменная",
"variables": "Переменные",
"variable_list": "Список переменных" "variable_list": "Список переменных"
}, },
"error": { "error": {
"authproviders_load_error": "Unable to load auth providers", "authproviders_load_error": "Unable to load auth providers",
"browser_support_sse": "Похоже, в этом браузере нет поддержки событий, отправленных сервером.", "browser_support_sse": "Похоже, в этом браузере нет поддержки событий, отправленных сервером.",
"check_console_details": "Подробности смотрите в журнале консоли.", "check_console_details": "Подробности смотрите в журнале консоли.",
"check_how_to_add_origin": "Инструкция как добавить origin в настройки расширения", "check_how_to_add_origin": "Инструкция как это сделать",
"curl_invalid_format": "cURL неправильно отформатирован", "curl_invalid_format": "cURL неправильно отформатирован",
"danger_zone": "Опасная зона", "danger_zone": "Опасная зона",
"delete_account": "Вы являетесь владельцем этой команды:", "delete_account": "Вы являетесь владельцем этой команды:",
"delete_account_description": "Прежде чем удалить аккаунт вам необходимо либо назначить владельцом другого пользователя, либо удалить команды в которых вы являетесь владельцем.", "delete_account_description": "Прежде чем удалить аккаунт вам необходимо либо назначить владельцом другого пользователя, либо удалить команды в которых вы являетесь владельцем.",
"empty_profile_name": "Имя пользователя не может быть пустым",
"empty_req_name": "Пустое имя запроса", "empty_req_name": "Пустое имя запроса",
"f12_details": "(F12 для подробностей)", "f12_details": "(F12 для подробностей)",
"gql_prettify_invalid_query": "Не удалось определить недопустимый запрос, устранить синтаксические ошибки запроса и повторить попытку.", "gql_prettify_invalid_query": "Не удалось отформатировать, т.к. в запросе есть синтаксические ошибки. Устраните их и повторите попытку.",
"incomplete_config_urls": "Не заполнены URL конфигурации", "incomplete_config_urls": "Не заполнены URL конфигурации",
"incorrect_email": "Не корректный Email", "incorrect_email": "Не корректный Email",
"invalid_link": "Не корректная ссылка", "invalid_link": "Не корректная ссылка",
"invalid_link_description": "Ссылка, по которой вы перешли, - недействительна, либо срок ее действия истек.", "invalid_link_description": "Ссылка, по которой вы перешли, - недействительна, либо срок ее действия истек.",
"invalid_embed_link": "The embed does not exist or is invalid.", "invalid_embed_link": "The embed does not exist or is invalid.",
"json_parsing_failed": "Не корректный JSON", "json_parsing_failed": "Не корректный JSON",
"json_prettify_invalid_body": "Не удалось определить недопустимое тело, устранить синтаксические ошибки json и повторить попытку.", "json_prettify_invalid_body": "Не удалось определить формат строки, устраните синтаксические ошибки и повторите попытку.",
"network_error": "Похоже, возникла проблема с соединением. Попробуйте еще раз.", "network_error": "Похоже, возникла проблема с соединением. Попробуйте еще раз.",
"network_fail": "Не удалось отправить запрос", "network_fail": "Не удалось отправить запрос",
"no_collections_to_export": "Нечего экспортировать. Для начала нужно создать коллекцию.", "no_collections_to_export": "Нечего экспортировать. Для начала нужно создать коллекцию.",
@@ -306,8 +336,10 @@
"no_environments_to_export": "Нечего экспортировать. Для начала нужно создать переменные окружения.", "no_environments_to_export": "Нечего экспортировать. Для начала нужно создать переменные окружения.",
"no_results_found": "Совпадения не найдены", "no_results_found": "Совпадения не найдены",
"page_not_found": "Эта страница не найдена", "page_not_found": "Эта страница не найдена",
"please_install_extension": "Нужно установить специальное расширение и добавить этот домен как новый origin в настройках расширения.", "please_install_extension": "Ничего страшного. Просто нужно установить специальное расширение в браузере.",
"proxy_error": "Proxy error", "proxy_error": "Proxy error",
"reading_files": "Произошла ошибка при чтении файла или нескольких файлов",
"same_profile_name": "Задано имя пользователя такое же как и было",
"script_fail": "Не удалось выполнить сценарий предварительного запроса", "script_fail": "Не удалось выполнить сценарий предварительного запроса",
"something_went_wrong": "Что-то пошло не так", "something_went_wrong": "Что-то пошло не так",
"test_script_fail": "Не удалось выполнить тестирование запроса" "test_script_fail": "Не удалось выполнить тестирование запроса"
@@ -315,13 +347,12 @@
"export": { "export": {
"as_json": "Экспорт как JSON", "as_json": "Экспорт как JSON",
"create_secret_gist": "Создать секретный Gist", "create_secret_gist": "Создать секретный Gist",
"create_secret_gist_tooltip_text": "Export as secret Gist", "create_secret_gist_tooltip_text": "Экспортировать как секретный Gist",
"failed": "Something went wrong while exporting", "failed": "Произошла ошибка во время экспорта",
"secret_gist_success": "Successfully exported as secret Gist", "secret_gist_success": "Успешно экспортировано как секретный Gist",
"require_github": "Войдите через GitHub, чтобы создать секретную суть", "require_github": "Войдите через GitHub, чтобы создать секретную суть",
"title": "Экспорт", "title": "Экспорт",
"success": "Successfully exported", "success": "Успешно экспортировано"
"gist_created": "Gist создан"
}, },
"filter": { "filter": {
"all": "Все", "all": "Все",
@@ -346,7 +377,7 @@
"switch_connection": "Изменить соединение" "switch_connection": "Изменить соединение"
}, },
"graphql_collections": { "graphql_collections": {
"title": "GraphQL Collections" "title": "Коллекции GraphQL"
}, },
"group": { "group": {
"time": "Время", "time": "Время",
@@ -359,8 +390,8 @@
}, },
"helpers": { "helpers": {
"authorization": "Заголовок авторизации будет автоматически сгенерирован при отправке запроса.", "authorization": "Заголовок авторизации будет автоматически сгенерирован при отправке запроса.",
"collection_properties_authorization": " This authorization will be set for every request in this collection.", "collection_properties_authorization": "Этот заголовок авторизации будет подставляться при каждом запросе в этой коллекции.",
"collection_properties_header": "This header will be set for every request in this collection.", "collection_properties_header": "Этот заголовок будет подставляться при каждом запросе в этой коллекции.",
"generate_documentation_first": "Сначала создайте документацию", "generate_documentation_first": "Сначала создайте документацию",
"network_fail": "Невозможно достичь конечной точки API. Проверьте подключение к сети и попробуйте еще раз.", "network_fail": "Невозможно достичь конечной точки API. Проверьте подключение к сети и попробуйте еще раз.",
"offline": "Кажется, вы не в сети. Данные в этой рабочей области могут быть устаревшими.", "offline": "Кажется, вы не в сети. Данные в этой рабочей области могут быть устаревшими.",
@@ -380,10 +411,12 @@
"import": { "import": {
"collections": "Импортировать коллекции", "collections": "Импортировать коллекции",
"curl": "Импортировать из cURL", "curl": "Импортировать из cURL",
"environments_from_gist": "Import From Gist", "environments_from_gist": "Импортировать из Gist",
"environments_from_gist_description": "Import Hoppscotch Environments From Gist", "environments_from_gist_description": "Импортировать переменные окружения Hoppscotch из Gist",
"failed": "Ошибка импорта", "failed": "Ошибка импорта",
"from_file": "Import from File", "file_size_limit_exceeded_warning_multiple_files": "Выбранные файлы превышают рекомендованный лимит в 10MB. Были импортированы только первые {files}",
"file_size_limit_exceeded_warning_single_file": "Размер выбранного в данный момент файла превышает рекомендуемый лимит в 10 МБ. Пожалуйста, выберите другой файл.",
"from_file": "Импортировать из одного или нескольких файлов",
"from_gist": "Импорт из Gist", "from_gist": "Импорт из Gist",
"from_gist_description": "Импортировать через Gist URL", "from_gist_description": "Импортировать через Gist URL",
"from_insomnia": "Импортировать с Insomnia", "from_insomnia": "Импортировать с Insomnia",
@@ -398,9 +431,9 @@
"from_postman_description": "Импортировать из коллекции Postman", "from_postman_description": "Импортировать из коллекции Postman",
"from_url": "Импортировать из URL", "from_url": "Импортировать из URL",
"gist_url": "Введите URL-адрес Gist", "gist_url": "Введите URL-адрес Gist",
"gql_collections_from_gist_description": "Import GraphQL Collections From Gist", "gql_collections_from_gist_description": "Импортировать GraphQL коллекцию из Gist",
"hoppscotch_environment": "Hoppscotch Environment", "hoppscotch_environment": "Hoppscotch Environment",
"hoppscotch_environment_description": "Import Hoppscotch Environment JSON file", "hoppscotch_environment_description": "Импортировать окружение Hoppscotch из JSON файла",
"import_from_url_invalid_fetch": "Не удалить получить данные по этому URL", "import_from_url_invalid_fetch": "Не удалить получить данные по этому URL",
"import_from_url_invalid_file_format": "Ошибка при импорте коллекций", "import_from_url_invalid_file_format": "Ошибка при импорте коллекций",
"import_from_url_invalid_type": "Неподдерживаемый тип. Поддерживаемые типы: 'hoppscotch', 'openapi', 'postman', 'insomnia'", "import_from_url_invalid_type": "Неподдерживаемый тип. Поддерживаемые типы: 'hoppscotch', 'openapi', 'postman', 'insomnia'",
@@ -409,16 +442,19 @@
"json_description": "Импортировать из коллекции Hoppscotch", "json_description": "Импортировать из коллекции Hoppscotch",
"postman_environment": "Postman Environment", "postman_environment": "Postman Environment",
"postman_environment_description": "Import Postman Environment from a JSON file", "postman_environment_description": "Import Postman Environment from a JSON file",
"success": "Успешно импортировано",
"title": "Импортировать" "title": "Импортировать"
}, },
"inspections": { "inspections": {
"description": "Inspect possible errors", "description": "Показать возможные ошибки",
"environment": { "environment": {
"add_environment": "Add to Environment", "add_environment": "Добавить переменную",
"not_found": "Environment variable “{environment}” not found." "add_environment_value": "Заполнить значение",
"empty_value": "Значение переменной окружения '{variable}' пустое",
"not_found": "Переменная окружения “{environment}” не задана."
}, },
"header": { "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": "Из-за ограничений безопасности в веб версии нельзя задать Cookie параметры. Пожалуйста, используйте Hoppscotch Desktop приложение или используйте заголовок Authorization вместо этого."
}, },
"response": { "response": {
"401_error": "Please check your authentication credentials.", "401_error": "Please check your authentication credentials.",
@@ -427,12 +463,12 @@
"default_error": "Please check your request.", "default_error": "Please check your request.",
"network_error": "Please check your network connection." "network_error": "Please check your network connection."
}, },
"title": "Inspector", "title": "Помощник",
"url": { "url": {
"extension_not_installed": "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.", "extension_unknown_origin": "Убедитесь, что текущий домен добавлен в список доверенных ресурсов в расширении браузера",
"extention_enable_action": "Enable Browser Extension", "extention_enable_action": "Подключить расширение",
"extention_not_enabled": "Extension not enabled." "extention_not_enabled": "Расширение в браузере не подключено."
} }
}, },
"layout": { "layout": {
@@ -445,11 +481,11 @@
"modal": { "modal": {
"close_unsaved_tab": "У вас есть не сохранённые изменения", "close_unsaved_tab": "У вас есть не сохранённые изменения",
"collections": "Коллекции", "collections": "Коллекции",
"confirm": "Подтверждать", "confirm": "Подтвердите действие",
"customize_request": "Customize Request", "customize_request": "Customize Request",
"edit_request": "Изменить запрос", "edit_request": "Изменить запрос",
"import_export": "Импорт Экспорт", "import_export": "Импорт Экспорт",
"share_request": "Share Request" "share_request": "Поделиться запросом"
}, },
"mqtt": { "mqtt": {
"already_subscribed": "Вы уже подписаны на этот топик", "already_subscribed": "Вы уже подписаны на этот топик",
@@ -485,7 +521,7 @@
"doc": "Документы", "doc": "Документы",
"graphql": "GraphQL", "graphql": "GraphQL",
"profile": "Профиль", "profile": "Профиль",
"realtime": "В реальном времени", "realtime": "Realtime",
"rest": "REST", "rest": "REST",
"settings": "Настройки" "settings": "Настройки"
}, },
@@ -507,8 +543,8 @@
"roles": "Роли", "roles": "Роли",
"roles_description": "Роли позволяют настраивать доступ конкретным людям к публичным коллекциям.", "roles_description": "Роли позволяют настраивать доступ конкретным людям к публичным коллекциям.",
"updated": "Профиль обновлен", "updated": "Профиль обновлен",
"viewer": "Зритель", "viewer": "Читатель",
"viewer_description": "Зрительно могут только просматривать и использовать запросы." "viewer_description": "Могут только просматривать и использовать запросы."
}, },
"remove": { "remove": {
"star": "Удалить звезду" "star": "Удалить звезду"
@@ -530,11 +566,11 @@
"enter_curl": "Введите сюда команду cURL", "enter_curl": "Введите сюда команду cURL",
"generate_code": "Сгенерировать код", "generate_code": "Сгенерировать код",
"generated_code": "Сгенерированный код", "generated_code": "Сгенерированный код",
"go_to_authorization_tab": "Go to Authorization", "go_to_authorization_tab": "Перейти на вкладку авторизации",
"go_to_body_tab": "Go to Body tab", "go_to_body_tab": "Перейти на вкладку тела запроса",
"header_list": "Список заголовков", "header_list": "Список заголовков",
"invalid_name": "Укажите имя для запроса", "invalid_name": "Укажите имя для запроса",
"method": "Методика", "method": "Метод",
"moved": "Запрос перемещён", "moved": "Запрос перемещён",
"name": "Имя запроса", "name": "Имя запроса",
"new": "Новый запрос", "new": "Новый запрос",
@@ -548,22 +584,22 @@
"payload": "Полезная нагрузка", "payload": "Полезная нагрузка",
"query": "Запрос", "query": "Запрос",
"raw_body": "Необработанное тело запроса", "raw_body": "Необработанное тело запроса",
"rename": "Переименость запрос", "rename": "Переименовать запрос",
"renamed": "Запрос переименован", "renamed": "Запрос переименован",
"request_variables": "Переменные запроса",
"run": "Запустить", "run": "Запустить",
"save": "Сохранить", "save": "Сохранить",
"save_as": "Сохранить как", "save_as": "Сохранить как",
"saved": "Запрос сохранен", "saved": "Запрос сохранен",
"share": "Делиться", "share": "Поделиться",
"share_description": "Поделиться Hoppscotch с друзьями", "share_description": "Поделиться Hoppscotch с друзьями",
"share_request": "Share Request", "share_request": "Поделиться запросом",
"stop": "Stop", "stop": "Стоп",
"title": "Запрос", "title": "Запрос",
"type": "Тип запроса", "type": "Тип запроса",
"url": "URL", "url": "URL",
"variables": "Переменные", "variables": "Переменные",
"view_my_links": "Посмотреть мои ссылки", "view_my_links": "Посмотреть мои ссылки"
"copy_link": "Копировать ссылку"
}, },
"response": { "response": {
"audio": "Аудио", "audio": "Аудио",
@@ -586,7 +622,7 @@
}, },
"settings": { "settings": {
"accent_color": "Основной цвет", "accent_color": "Основной цвет",
"account": "Счет", "account": "Аккаунт",
"account_deleted": "Ваш аккаунт был удалён", "account_deleted": "Ваш аккаунт был удалён",
"account_description": "Настройте параметры своей учетной записи.", "account_description": "Настройте параметры своей учетной записи.",
"account_email_description": "Ваш основной адрес электронной почты.", "account_email_description": "Ваш основной адрес электронной почты.",
@@ -639,29 +675,29 @@
"verify_email": "Подтвердить Email" "verify_email": "Подтвердить Email"
}, },
"shared_requests": { "shared_requests": {
"button": "Button", "button": "Кнопка",
"button_info": "Create a 'Run in Hoppscotch' button for your website, blog or a README.", "button_info": "Создать кнопку 'Run in Hoppscotch' на свой сайт, блог или README.",
"copy_html": "Copy HTML", "copy_html": "Копировать HTML код",
"copy_link": "Copy Link", "copy_link": "Копировать ссылку",
"copy_markdown": "Copy Markdown", "copy_markdown": "Копировать Markdown",
"creating_widget": "Creating widget", "creating_widget": "Создание виджет",
"customize": "Customize", "customize": "Настроить",
"deleted": "Shared request deleted", "deleted": "Запрос удален",
"description": "Select a widget, you can change and customize this later", "description": "Выберите вид как вы поделитесь запросом, позже вы сможете дополнительно его настроить",
"embed": "Embed", "embed": "Встраиваемое окно",
"embed_info": "Add a mini 'Hoppscotch API Playground' to your website, blog or documentation.", "embed_info": "Добавьте небольшую площадку 'Hoppscotch API Playground' на свой веб-сайт, блог или документацию.",
"link": "Link", "link": "Ссылка",
"link_info": "Create a shareable link to share with anyone on the internet with view access.", "link_info": "Создайте общедоступную ссылку, которой можно поделиться с любым пользователем, имеющим доступ к просмотру.",
"modified": "Shared request modified", "modified": "Запрос изменен",
"not_found": "Shared request not found", "not_found": "Такой ссылке не нашлось",
"open_new_tab": "Open in new tab", "open_new_tab": "Открыть в новом окне",
"preview": "Preview", "preview": "Preview",
"run_in_hoppscotch": "Run in Hoppscotch", "run_in_hoppscotch": "Run in Hoppscotch",
"theme": { "theme": {
"dark": "Dark", "dark": "Темная",
"light": "Light", "light": "Светлая",
"system": "System", "system": "Системная",
"title": "Theme" "title": "Тема"
} }
}, },
"shortcut": { "shortcut": {
@@ -669,7 +705,7 @@
"close_current_menu": "Закрыть текущее меню", "close_current_menu": "Закрыть текущее меню",
"command_menu": "Меню поиска и команд", "command_menu": "Меню поиска и команд",
"help_menu": "Меню помощи", "help_menu": "Меню помощи",
"show_all": "Горячие клавиши", "show_all": "Список горячих клавиш",
"title": "Общий" "title": "Общий"
}, },
"miscellaneous": { "miscellaneous": {
@@ -696,20 +732,19 @@
"get_method": "Выберите метод GET", "get_method": "Выберите метод GET",
"head_method": "Выберите метод HEAD", "head_method": "Выберите метод HEAD",
"import_curl": "Импортировать из cURL", "import_curl": "Импортировать из cURL",
"method": "Методика", "method": "Метод",
"next_method": "Выберите следующий метод", "next_method": "Выберите следующий метод",
"post_method": "Выберите метод POST", "post_method": "Выберите метод POST",
"previous_method": "Выбрать предыдущий метод", "previous_method": "Выбрать предыдущий метод",
"put_method": "Выберите метод PUT", "put_method": "Выберите метод PUT",
"rename": "Переименовать запрос", "rename": "Переименовать запрос",
"reset_request": "Сбросить запрос", "reset_request": "Сбросить запрос",
"save_request": "Сохарнить запрос", "save_request": "Сохранить запрос",
"save_to_collections": "Сохранить в коллекции", "save_to_collections": "Сохранить в коллекции",
"send_request": "Послать запрос", "send_request": "Послать запрос",
"share_request": "Share Request", "share_request": "Поделиться запросом",
"show_code": "Generate code snippet", "show_code": "Сгенерировать фрагмент кода из запроса",
"title": "Запрос", "title": "Запрос"
"copy_request_link": "Копировать ссылку на запрос"
}, },
"response": { "response": {
"copy": "Копировать запрос в буфер обмена", "copy": "Копировать запрос в буфер обмена",
@@ -717,11 +752,11 @@
"title": "Запрос" "title": "Запрос"
}, },
"theme": { "theme": {
"black": "Черный режим", "black": "Переключить на черный режим",
"dark": "Тёмный режим", "dark": "Переключить на тёмный режим",
"light": "Светлый режим", "light": "Переключить на светлый режим",
"system": "Определяется системой", "system": "Переключить на тему, исходя из настроек системы",
"title": "Тема" "title": "Внешний вид"
} }
}, },
"show": { "show": {
@@ -730,6 +765,11 @@
"more": "Показать больше", "more": "Показать больше",
"sidebar": "Показать боковую панель" "sidebar": "Показать боковую панель"
}, },
"site_protection": {
"error_fetching_site_protection_status": "Something Went Wrong While Fetching Site Protection Status",
"login_to_continue": "Login to continue",
"login_to_continue_description": "You need to be logged in to access this Hoppscotch Enterprise Instance."
},
"socketio": { "socketio": {
"communication": "Коммуникация", "communication": "Коммуникация",
"connection_not_authorized": "Это SocketIO соединение не использует какую-либо авторизацию.", "connection_not_authorized": "Это SocketIO соединение не использует какую-либо авторизацию.",
@@ -739,82 +779,89 @@
"url": "URL" "url": "URL"
}, },
"spotlight": { "spotlight": {
"change_language": "Change Language", "change_language": "Изменить язык",
"environments": { "environments": {
"delete": "Delete current environment", "delete": "Удалить текущее окружение",
"duplicate": "Duplicate current environment", "duplicate": "Дублировать текущее окружение",
"duplicate_global": "Duplicate global environment", "duplicate_global": "Дублировать глобальное окружение",
"edit": "Edit current environment", "edit": "Редактировать текущее окружение",
"edit_global": "Edit global environment", "edit_global": "Редактировать глобальное окружение",
"new": "Create new environment", "new": "Создать новое окружение",
"new_variable": "Create a new environment variable", "new_variable": "Создать новую переменную окружения",
"title": "Environments" "title": "Окружение"
}, },
"general": { "general": {
"chat": "Chat with support", "chat": "Чат с поддержкой",
"help_menu": "Help and support", "help_menu": "Помощь",
"open_docs": "Read Documentation", "open_docs": "Почитать документацию",
"open_github": "Open GitHub repository", "open_github": "Открыть GitHub репозиторий",
"open_keybindings": "Keyboard shortcuts", "open_keybindings": "Горячие клавиши",
"social": "Social", "social": "Соц. сети",
"title": "General" "title": "Общее"
}, },
"graphql": { "graphql": {
"connect": "Connect to server", "connect": "Подключиться к серверу",
"disconnect": "Disconnect from server" "disconnect": "Отключиться от сервера"
}, },
"miscellaneous": { "miscellaneous": {
"invite": "Invite your friends to Hoppscotch", "invite": "Пригласить друзей в Hoppscotch",
"title": "Miscellaneous" "title": "Другое"
},
"phrases": {
"create_environment": "Создать окружение",
"create_workspace": "Создать пространство",
"import_collections": "Импортировать коллекцию",
"share_request": "Поделиться запросом",
"try": "Попробовать"
}, },
"request": { "request": {
"save_as_new": "Save as new request", "save_as_new": "Сохранить как новый запрос",
"select_method": "Select method", "select_method": "Выбрать метод",
"switch_to": "Switch to", "switch_to": "Переключиться",
"tab_authorization": "Authorization tab", "tab_authorization": "На вкладку авторизации",
"tab_body": "Body tab", "tab_body": "На вкладку тела запроса",
"tab_headers": "Headers tab", "tab_headers": "На вкладку заголовков",
"tab_parameters": "Parameters tab", "tab_parameters": "На вкладку параметров",
"tab_pre_request_script": "Pre-request script tab", "tab_pre_request_script": "На вкладку пред-скрипта запроса",
"tab_query": "Query tab", "tab_query": "На вкладку запроса",
"tab_tests": "Tests tab", "tab_tests": "На вкладку тестов",
"tab_variables": "Variables tab" "tab_variables": "На вкладку переменных запроса"
}, },
"response": { "response": {
"copy": "Copy response", "copy": "Копировать содержимое ответа",
"download": "Download response as file", "download": "Сказать содержимое ответа как файл",
"title": "Response" "title": "Ответ запроса"
}, },
"section": { "section": {
"interceptor": "Interceptor", "interceptor": "Перехватчик",
"interface": "Interface", "interface": "Интерфейс",
"theme": "Theme", "theme": "Внешний вид",
"user": "User" "user": "Пользователь"
}, },
"settings": { "settings": {
"change_interceptor": "Change Interceptor", "change_interceptor": "Изменить перехватчик",
"change_language": "Change Language", "change_language": "Изменить язык",
"theme": { "theme": {
"black": "Black", "black": "Черная",
"dark": "Dark", "dark": "Темная",
"light": "Light", "light": "Светлая",
"system": "System preference" "system": "Как задано в системе"
} }
}, },
"tab": { "tab": {
"close_current": "Close current tab", "close_current": "Закрыть текущую вкладку",
"close_others": "Close all other tabs", "close_others": "Закрыть все вкладки",
"duplicate": "Duplicate current tab", "duplicate": "Продублировать текущую вкладку",
"new_tab": "Open a new tab", "new_tab": "Открыть в новой вкладке",
"title": "Tabs" "title": "Вкладки"
}, },
"workspace": { "workspace": {
"delete": "Delete current team", "delete": "Удалить текущую команду",
"edit": "Edit current team", "edit": "Редактировать текущую команду",
"invite": "Invite people to team", "invite": "Пригласить людей в команду",
"new": "Create new team", "new": "Создать новую команду",
"switch_to_personal": "Switch to your personal workspace", "switch_to_personal": "Переключить на персональное пространство",
"title": "Teams" "title": "Команды"
} }
}, },
"sse": { "sse": {
@@ -824,7 +871,7 @@
}, },
"state": { "state": {
"bulk_mode": "Множественное редактирование", "bulk_mode": "Множественное редактирование",
"bulk_mode_placeholder": "Каждый параметр должен начинаться с новой строки\nКлючи и значения разедляются двоеточием\nИспользуйте # для комментария", "bulk_mode_placeholder": "Каждый параметр должен начинаться с новой строки\nКлючи и значения разделяются двоеточием\nИспользуйте # для комментария",
"cleared": "Очищено", "cleared": "Очищено",
"connected": "Связаны", "connected": "Связаны",
"connected_to": "Подключено к {name}", "connected_to": "Подключено к {name}",
@@ -843,20 +890,20 @@
"download_failed": "Download failed", "download_failed": "Download failed",
"download_started": "Скачивание началось", "download_started": "Скачивание началось",
"enabled": "Включено", "enabled": "Включено",
"file_imported": "Файл импортирован", "file_imported": "Файл успешно импортирован",
"finished_in": "Завершено через {duration} мс", "finished_in": "Завершено через {duration} мс",
"hide": "Hide", "hide": "Скрыть",
"history_deleted": "История удалена", "history_deleted": "История удалена",
"linewrap": "Обернуть линии", "linewrap": "Обернуть линии",
"loading": "Загрузка...", "loading": "Загрузка...",
"message_received": "Сообщение: {message} получено по топику: {topic}", "message_received": "Сообщение: {message} получено по топику: {topic}",
"mqtt_subscription_failed": "Что-то пошло не так, при попытке подписаться на топик: {topic}", "mqtt_subscription_failed": "Что-то пошло не так, при попытке подписаться на топик: {topic}",
"none": "Никто", "none": "Не задан",
"nothing_found": "Ничего не найдено для", "nothing_found": "Ничего не найдено для",
"published_error": "Что-то пошло не так при попытке опубликовать сообщение в топик {topic}: {message}", "published_error": "Что-то пошло не так при попытке опубликовать сообщение в топик {topic}: {message}",
"published_message": "Опубликовано сообщение: {message} в топик: {topic}", "published_message": "Опубликовано сообщение: {message} в топик: {topic}",
"reconnection_error": "Не удалось переподключиться", "reconnection_error": "Не удалось переподключиться",
"show": "Show", "show": "Показать",
"subscribed_failed": "Не удалось подписаться на топик: {topic}", "subscribed_failed": "Не удалось подписаться на топик: {topic}",
"subscribed_success": "Успешно подписался на топик: {topic}", "subscribed_success": "Успешно подписался на топик: {topic}",
"unsubscribed_failed": "Не удалось отписаться от топика: {topic}", "unsubscribed_failed": "Не удалось отписаться от топика: {topic}",
@@ -871,7 +918,6 @@
"forum": "Задавайте вопросы и получайте ответы", "forum": "Задавайте вопросы и получайте ответы",
"github": "Подпишитесь на нас на Github", "github": "Подпишитесь на нас на Github",
"shortcuts": "Просматривайте приложение быстрее", "shortcuts": "Просматривайте приложение быстрее",
"team": "Свяжитесь с командой",
"title": "Служба поддержки", "title": "Служба поддержки",
"twitter": "Следуйте за нами на Twitter" "twitter": "Следуйте за нами на Twitter"
}, },
@@ -882,7 +928,7 @@
"close_others": "Закрыть остальные вкладки", "close_others": "Закрыть остальные вкладки",
"collections": "Коллекции", "collections": "Коллекции",
"documentation": "Документация", "documentation": "Документация",
"duplicate": "Duplicate Tab", "duplicate": "Дублировать вкладку",
"environments": "Окружения", "environments": "Окружения",
"headers": "Заголовки", "headers": "Заголовки",
"history": "История", "history": "История",
@@ -892,7 +938,8 @@
"queries": "Запросы", "queries": "Запросы",
"query": "Запрос", "query": "Запрос",
"schema": "Схема", "schema": "Схема",
"shared_requests": "Shared Requests", "shared_requests": "Запросы в общем доступе",
"share_tab_request": "Поделиться запросом",
"socketio": "Socket.IO", "socketio": "Socket.IO",
"sse": "SSE", "sse": "SSE",
"tests": "Тесты", "tests": "Тесты",
@@ -921,7 +968,6 @@
"invite_tooltip": "Пригласить людей в Ваше рабочее пространство", "invite_tooltip": "Пригласить людей в Ваше рабочее пространство",
"invited_to_team": "{owner} приглашает Вас присоединиться к команде {team}", "invited_to_team": "{owner} приглашает Вас присоединиться к команде {team}",
"join": "Приглашение принято", "join": "Приглашение принято",
"join_beta": "Присоединяйтесь к бета-программе, чтобы получить доступ к командам.",
"join_team": "Присоединиться к {team}", "join_team": "Присоединиться к {team}",
"joined_team": "Вы присоединились к команде {team}", "joined_team": "Вы присоединились к команде {team}",
"joined_team_description": "Теперь Вы участник этой команды", "joined_team_description": "Теперь Вы участник этой команды",
@@ -950,6 +996,7 @@
"permissions": "Разрешения", "permissions": "Разрешения",
"same_target_destination": "Таже цель и конечная точка", "same_target_destination": "Таже цель и конечная точка",
"saved": "Команда сохранена", "saved": "Команда сохранена",
"search_title": "Team Requests",
"select_a_team": "Выбрать команду", "select_a_team": "Выбрать команду",
"success_invites": "Принятые приглашения", "success_invites": "Принятые приглашения",
"title": "Команды", "title": "Команды",
@@ -981,16 +1028,8 @@
"workspace": { "workspace": {
"change": "Изменить пространство", "change": "Изменить пространство",
"personal": "Моё пространство", "personal": "Моё пространство",
"other_workspaces": "Пространства",
"team": "Пространство команды", "team": "Пространство команды",
"title": "Рабочие пространства" "title": "Рабочие пространства"
},
"shortcodes": {
"actions": "Действия",
"created_on": "Создано",
"deleted": "Удалёна",
"method": "Метод",
"not_found": "Короткая ссылка не найдена",
"short_code": "Короткая ссылка",
"url": "URL"
} }
} }

View File

@@ -68,9 +68,9 @@
"developer_option": "Developer options", "developer_option": "Developer options",
"developer_option_description": "Developer tools which helps in development and maintenance of Hoppscotch.", "developer_option_description": "Developer tools which helps in development and maintenance of Hoppscotch.",
"discord": "Discord", "discord": "Discord",
"documentation": "Dökümanlar", "documentation": "Dokümanlar",
"github": "GitHub", "github": "GitHub",
"help": "Yardım, geri bildirim ve dökümanlar", "help": "Yardım, geri bildirim ve dokümanlar",
"home": "Ana sayfa", "home": "Ana sayfa",
"invite": "Davet et", "invite": "Davet et",
"invite_description": "Hoppscotch'ta API'lerinizi oluşturmak ve yönetmek için basit ve sezgisel bir arayüz tasarladık. Hoppscotch, API'lerinizi oluşturmanıza, test etmenize, belgelemenize ve paylaşmanıza yardımcı olan bir araçtır.", "invite_description": "Hoppscotch'ta API'lerinizi oluşturmak ve yönetmek için basit ve sezgisel bir arayüz tasarladık. Hoppscotch, API'lerinizi oluşturmanıza, test etmenize, belgelemenize ve paylaşmanıza yardımcı olan bir araçtır.",
@@ -225,7 +225,7 @@
"body": "Bu isteğin bir gövdesi yok", "body": "Bu isteğin bir gövdesi yok",
"collection": "Koleksiyon boş", "collection": "Koleksiyon boş",
"collections": "Koleksiyonlar boş", "collections": "Koleksiyonlar boş",
"documentation": "Dökümanları görmek için GraphQL uç noktasını bağlayın", "documentation": "Dokümanları görmek için GraphQL uç noktasını bağlayın",
"endpoint": "Uç nokta boş olamaz", "endpoint": "Uç nokta boş olamaz",
"environments": "Ortamlar boş", "environments": "Ortamlar boş",
"folder": "Klasör boş", "folder": "Klasör boş",
@@ -735,7 +735,7 @@
"url": "Bağlantı" "url": "Bağlantı"
}, },
"spotlight": { "spotlight": {
"change_language": "Change Language", "change_language": "Dil Değiştir",
"environments": { "environments": {
"delete": "Delete current environment", "delete": "Delete current environment",
"duplicate": "Duplicate current environment", "duplicate": "Duplicate current environment",
@@ -744,7 +744,7 @@
"edit_global": "Edit global environment", "edit_global": "Edit global environment",
"new": "Create new environment", "new": "Create new environment",
"new_variable": "Create a new environment variable", "new_variable": "Create a new environment variable",
"title": "Environments" "title": "Ortamlar"
}, },
"general": { "general": {
"chat": "Chat with support", "chat": "Chat with support",

View File

@@ -1,7 +1,7 @@
{ {
"name": "@hoppscotch/common", "name": "@hoppscotch/common",
"private": true, "private": true,
"version": "2024.3.0", "version": "2024.3.3",
"scripts": { "scripts": {
"dev": "pnpm exec npm-run-all -p -l dev:*", "dev": "pnpm exec npm-run-all -p -l dev:*",
"test": "vitest --run", "test": "vitest --run",
@@ -50,7 +50,7 @@
"axios": "1.6.2", "axios": "1.6.2",
"buffer": "6.0.3", "buffer": "6.0.3",
"cookie-es": "1.0.0", "cookie-es": "1.0.0",
"dioc": "1.0.1", "dioc": "3.0.1",
"esprima": "4.0.1", "esprima": "4.0.1",
"events": "3.3.0", "events": "3.3.0",
"fp-ts": "2.16.1", "fp-ts": "2.16.1",
@@ -127,8 +127,8 @@
"@types/splitpanes": "2.2.6", "@types/splitpanes": "2.2.6",
"@types/uuid": "9.0.7", "@types/uuid": "9.0.7",
"@types/yargs-parser": "21.0.3", "@types/yargs-parser": "21.0.3",
"@typescript-eslint/eslint-plugin": "6.13.2", "@typescript-eslint/eslint-plugin": "7.3.1",
"@typescript-eslint/parser": "6.13.2", "@typescript-eslint/parser": "7.3.1",
"@vitejs/plugin-vue": "4.5.1", "@vitejs/plugin-vue": "4.5.1",
"@vue/compiler-sfc": "3.3.10", "@vue/compiler-sfc": "3.3.10",
"@vue/eslint-config-typescript": "12.0.0", "@vue/eslint-config-typescript": "12.0.0",
@@ -136,9 +136,9 @@
"autoprefixer": "10.4.16", "autoprefixer": "10.4.16",
"cross-env": "7.0.3", "cross-env": "7.0.3",
"dotenv": "16.3.1", "dotenv": "16.3.1",
"eslint": "8.55.0", "eslint": "8.57.0",
"eslint-plugin-prettier": "5.0.1", "eslint-plugin-prettier": "5.1.3",
"eslint-plugin-vue": "9.19.2", "eslint-plugin-vue": "9.24.0",
"glob": "10.3.10", "glob": "10.3.10",
"npm-run-all": "4.1.5", "npm-run-all": "4.1.5",
"openapi-types": "12.1.3", "openapi-types": "12.1.3",
@@ -164,4 +164,4 @@
"vitest": "0.34.6", "vitest": "0.34.6",
"vue-tsc": "1.8.24" "vue-tsc": "1.8.24"
} }
} }

View File

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

View File

@@ -21,19 +21,7 @@
</div> </div>
</div> </div>
<div class="col-span-1 flex items-center justify-between space-x-2"> <div class="col-span-1 flex items-center justify-between space-x-2">
<button <AppSpotlightSearch />
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>
</div> </div>
<div class="col-span-2 flex items-center justify-between space-x-2"> <div class="col-span-2 flex items-center justify-between space-x-2">
<div class="flex"> <div class="flex">
@@ -55,12 +43,19 @@
@click="invokeAction('modals.support.toggle')" @click="invokeAction('modals.support.toggle')"
/> />
</div> </div>
<div class="flex"> <div
class="flex"
:class="{
'flex-row-reverse gap-2':
workspaceSelectorFlagEnabled && !currentUser,
}"
>
<div <div
v-if="currentUser === null" v-if="currentUser === null"
class="inline-flex items-center space-x-2" class="inline-flex items-center space-x-2"
> >
<HoppButtonSecondary <HoppButtonSecondary
v-if="!workspaceSelectorFlagEnabled"
:icon="IconUploadCloud" :icon="IconUploadCloud"
:label="t('header.save_workspace')" :label="t('header.save_workspace')"
class="!focus-visible:text-emerald-600 !hover:text-emerald-600 hidden h-8 border border-emerald-600/25 bg-emerald-500/10 !text-emerald-500 hover:border-emerald-600/20 hover:bg-emerald-600/20 focus-visible:border-emerald-600/20 focus-visible:bg-emerald-600/20 md:flex" class="!focus-visible:text-emerald-600 !hover:text-emerald-600 hidden h-8 border border-emerald-600/25 bg-emerald-500/10 !text-emerald-500 hover:border-emerald-600/20 hover:bg-emerald-600/20 focus-visible:border-emerald-600/20 focus-visible:bg-emerald-600/20 md:flex"
@@ -72,18 +67,22 @@
@click="invokeAction('modals.login.toggle')" @click="invokeAction('modals.login.toggle')"
/> />
</div> </div>
<div v-else class="inline-flex items-center space-x-2"> <TeamsMemberStack
<TeamsMemberStack v-else-if="
v-if=" currentUser !== null &&
workspace.type === 'team' && workspace.type === 'team' &&
selectedTeam && selectedTeam &&
selectedTeam.teamMembers.length > 1 selectedTeam.teamMembers.length > 1
" "
:team-members="selectedTeam.teamMembers" :team-members="selectedTeam.teamMembers"
show-count show-count
class="mx-2" class="mx-2"
@handle-click="handleTeamEdit()" @handle-click="handleTeamEdit()"
/> />
<div
v-if="workspaceSelectorFlagEnabled || currentUser"
class="inline-flex items-center space-x-2"
>
<div <div
class="flex h-8 divide-x divide-emerald-600/25 rounded border border-emerald-600/25 bg-emerald-500/10 focus-within:divide-emerald-600/20 focus-within:border-emerald-600/20 focus-within:bg-emerald-600/20 hover:divide-emerald-600/20 hover:border-emerald-600/20 hover:bg-emerald-600/20" class="flex h-8 divide-x divide-emerald-600/25 rounded border border-emerald-600/25 bg-emerald-500/10 focus-within:divide-emerald-600/20 focus-within:border-emerald-600/20 focus-within:bg-emerald-600/20 hover:divide-emerald-600/20 hover:border-emerald-600/20 hover:bg-emerald-600/20"
> >
@@ -96,6 +95,7 @@
/> />
<HoppButtonSecondary <HoppButtonSecondary
v-if=" v-if="
currentUser &&
workspace.type === 'team' && workspace.type === 'team' &&
selectedTeam && selectedTeam &&
selectedTeam?.myRole === 'OWNER' selectedTeam?.myRole === 'OWNER'
@@ -136,7 +136,7 @@
</div> </div>
</template> </template>
</tippy> </tippy>
<span class="px-2"> <span v-if="currentUser" class="px-2">
<tippy <tippy
interactive interactive
trigger="click" trigger="click"
@@ -251,7 +251,6 @@ import { breakpointsTailwind, useBreakpoints, useNetwork } from "@vueuse/core"
import { computed, reactive, ref, watch } from "vue" import { computed, reactive, ref, watch } from "vue"
import { useToast } from "~/composables/toast" import { useToast } from "~/composables/toast"
import { GetMyTeamsQuery, TeamMemberRole } from "~/helpers/backend/graphql" import { GetMyTeamsQuery, TeamMemberRole } from "~/helpers/backend/graphql"
import { getPlatformSpecialKey } from "~/helpers/platformutils"
import { platform } from "~/platform" import { platform } from "~/platform"
import IconDownload from "~icons/lucide/download" import IconDownload from "~icons/lucide/download"
import IconLifeBuoy from "~icons/lucide/life-buoy" import IconLifeBuoy from "~icons/lucide/life-buoy"
@@ -272,6 +271,13 @@ import {
const t = useI18n() const t = useI18n()
const toast = useToast() const toast = useToast()
/**
* Feature flag to enable the workspace selector login conversion
*/
const workspaceSelectorFlagEnabled = computed(
() => !!platform.platformFeatureFlags.workspaceSwitcherLogin?.value
)
/** /**
* Once the PWA code is initialized, this holds a method * Once the PWA code is initialized, this holds a method
* that can be called to show the user the installation * that can be called to show the user the installation
@@ -393,6 +399,8 @@ const inviteTeam = (team: { name: string }, teamID: string) => {
// Show the workspace selected team invite modal if the user is an owner of the team else show the default invite modal // Show the workspace selected team invite modal if the user is an owner of the team else show the default invite modal
const handleInvite = () => { const handleInvite = () => {
if (!currentUser.value) return invokeAction("modals.login.toggle")
if ( if (
workspace.value.type === "team" && workspace.value.type === "team" &&
workspace.value.teamID && workspace.value.teamID &&

View File

@@ -1,6 +1,6 @@
<template> <template>
<div class="flex flex-col space-y-2"> <div class="flex flex-col space-y-2">
<div class="flex flex-col px-4 pt-2"> <div v-if="isTooltipComponent" class="flex flex-col px-4 pt-2">
<h2 class="inline-flex pb-1 font-semibold text-secondaryDark"> <h2 class="inline-flex pb-1 font-semibold text-secondaryDark">
{{ t("settings.interceptor") }} {{ t("settings.interceptor") }}
</h2> </h2>
@@ -19,6 +19,9 @@
:value="interceptor.interceptorID" :value="interceptor.interceptorID"
:label="unref(interceptor.name(t))" :label="unref(interceptor.name(t))"
:selected="interceptorSelection === interceptor.interceptorID" :selected="interceptorSelection === interceptor.interceptorID"
:class="{
'!px-0 hover:bg-transparent': !isTooltipComponent,
}"
@change="interceptorSelection = interceptor.interceptorID" @change="interceptorSelection = interceptor.interceptorID"
/> />
@@ -39,6 +42,15 @@ import { InterceptorService } from "~/services/interceptor.service"
const t = useI18n() const t = useI18n()
withDefaults(
defineProps<{
isTooltipComponent?: boolean
}>(),
{
isTooltipComponent: true,
}
)
const interceptorService = useService(InterceptorService) const interceptorService = useService(InterceptorService)
const interceptorSelection = const interceptorSelection =

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

@@ -32,7 +32,6 @@ import { useI18n } from "~/composables/i18n"
import { useToast } from "~/composables/toast" import { useToast } from "~/composables/toast"
import { appendRESTCollections, restCollections$ } from "~/newstore/collections" import { appendRESTCollections, restCollections$ } from "~/newstore/collections"
import MyCollectionImport from "~/components/importExport/ImportExportSteps/MyCollectionImport.vue" import MyCollectionImport from "~/components/importExport/ImportExportSteps/MyCollectionImport.vue"
import { GetMyTeamsQuery } from "~/helpers/backend/graphql"
import IconFolderPlus from "~icons/lucide/folder-plus" import IconFolderPlus from "~icons/lucide/folder-plus"
import IconOpenAPI from "~icons/lucide/file" import IconOpenAPI from "~icons/lucide/file"
@@ -55,16 +54,15 @@ import { teamCollectionsExporter } from "~/helpers/import-export/export/teamColl
import { GistSource } from "~/helpers/import-export/import/import-sources/GistSource" import { GistSource } from "~/helpers/import-export/import/import-sources/GistSource"
import { ImporterOrExporter } from "~/components/importExport/types" import { ImporterOrExporter } from "~/components/importExport/types"
import { TeamWorkspace } from "~/services/workspace.service"
const t = useI18n() const t = useI18n()
const toast = useToast() const toast = useToast()
type SelectedTeam = GetMyTeamsQuery["myTeams"][number] | undefined
type CollectionType = type CollectionType =
| { | {
type: "team-collections" type: "team-collections"
selectedTeam: SelectedTeam selectedTeam: TeamWorkspace
} }
| { type: "my-collections" } | { type: "my-collections" }
@@ -433,7 +431,7 @@ const HoppTeamCollectionsExporter: ImporterOrExporter = {
props.collectionsType.selectedTeam props.collectionsType.selectedTeam
) { ) {
const res = await teamCollectionsExporter( const res = await teamCollectionsExporter(
props.collectionsType.selectedTeam.id props.collectionsType.selectedTeam.teamID
) )
if (E.isRight(res)) { if (E.isRight(res)) {
@@ -569,8 +567,8 @@ const hasTeamWriteAccess = computed(() => {
} }
return ( return (
collectionsType.selectedTeam.myRole === "EDITOR" || collectionsType.selectedTeam.role === "EDITOR" ||
collectionsType.selectedTeam.myRole === "OWNER" collectionsType.selectedTeam.role === "OWNER"
) )
}) })
@@ -578,17 +576,17 @@ const selectedTeamID = computed(() => {
const { collectionsType } = props const { collectionsType } = props
return collectionsType.type === "team-collections" return collectionsType.type === "team-collections"
? collectionsType.selectedTeam?.id ? collectionsType.selectedTeam?.teamID
: undefined : undefined
}) })
const getCollectionJSON = async () => { const getCollectionJSON = async () => {
if ( if (
props.collectionsType.type === "team-collections" && props.collectionsType.type === "team-collections" &&
props.collectionsType.selectedTeam?.id props.collectionsType.selectedTeam?.teamID
) { ) {
const res = await getTeamCollectionJSON( const res = await getTeamCollectionJSON(
props.collectionsType.selectedTeam?.id props.collectionsType.selectedTeam?.teamID
) )
return E.isRight(res) return E.isRight(res)

View File

@@ -65,14 +65,20 @@
<script setup lang="ts"> <script setup lang="ts">
import { useI18n } from "@composables/i18n" import { useI18n } from "@composables/i18n"
import { HoppCollection, HoppRESTAuth, HoppRESTHeaders } from "@hoppscotch/data" import {
import { clone } from "lodash-es" GQLHeader,
import { HoppInheritedProperty } from "~/helpers/types/HoppInheritedProperties" HoppCollection,
import { PersistenceService } from "~/services/persistence" HoppGQLAuth,
HoppRESTAuth,
HoppRESTHeaders,
} from "@hoppscotch/data"
import { useVModel } from "@vueuse/core"
import { useService } from "dioc/vue" import { useService } from "dioc/vue"
import { clone } from "lodash-es"
import { ref, watch } from "vue" import { ref, watch } from "vue"
import { useVModel } from "@vueuse/core" import { HoppInheritedProperty } from "~/helpers/types/HoppInheritedProperties"
import { PersistenceService } from "~/services/persistence"
const persistenceService = useService(PersistenceService) const persistenceService = useService(PersistenceService)
const t = useI18n() const t = useI18n()
@@ -84,6 +90,9 @@ export type EditingProperties = {
inheritedProperties?: HoppInheritedProperty inheritedProperties?: HoppInheritedProperty
} }
type HoppCollectionAuth = HoppRESTAuth | HoppGQLAuth
type HoppCollectionHeaders = HoppRESTHeaders | GQLHeader[]
const props = withDefaults( const props = withDefaults(
defineProps<{ defineProps<{
show: boolean show: boolean
@@ -109,8 +118,8 @@ const emit = defineEmits<{
}>() }>()
const editableCollection = ref<{ const editableCollection = ref<{
headers: HoppRESTHeaders headers: HoppCollectionHeaders
auth: HoppRESTAuth auth: HoppCollectionAuth
}>({ }>({
headers: [], headers: [],
auth: { auth: {
@@ -122,15 +131,16 @@ const editableCollection = ref<{
watch( watch(
editableCollection, editableCollection,
(updatedEditableCollection) => { (updatedEditableCollection) => {
if (props.show) { if (props.show && props.editingProperties) {
const unsavedCollectionProperties: EditingProperties = {
collection: updatedEditableCollection,
isRootCollection: props.editingProperties?.isRootCollection ?? false,
path: props.editingProperties?.path,
inheritedProperties: props.editingProperties?.inheritedProperties,
}
persistenceService.setLocalConfig( persistenceService.setLocalConfig(
"unsaved_collection_properties", "unsaved_collection_properties",
JSON.stringify(<EditingProperties>{ JSON.stringify(unsavedCollectionProperties)
collection: updatedEditableCollection,
isRootCollection: props.editingProperties?.isRootCollection,
path: props.editingProperties?.path,
inheritedProperties: props.editingProperties?.inheritedProperties,
})
) )
} }
}, },
@@ -146,10 +156,10 @@ watch(
(show) => { (show) => {
if (show && props.editingProperties?.collection) { if (show && props.editingProperties?.collection) {
editableCollection.value.auth = clone( editableCollection.value.auth = clone(
props.editingProperties.collection.auth as HoppRESTAuth props.editingProperties.collection.auth as HoppCollectionAuth
) )
editableCollection.value.headers = clone( editableCollection.value.headers = clone(
props.editingProperties.collection.headers as HoppRESTHeaders props.editingProperties.collection.headers as HoppCollectionHeaders
) )
} else { } else {
editableCollection.value = { editableCollection.value = {

View File

@@ -56,23 +56,25 @@
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import { computed, nextTick, reactive, ref, watch } from "vue" import { useI18n } from "@composables/i18n"
import { cloneDeep } from "lodash-es" import { useToast } from "@composables/toast"
import { import {
HoppGQLRequest, HoppGQLRequest,
HoppRESTRequest, HoppRESTRequest,
isHoppRESTRequest, isHoppRESTRequest,
} from "@hoppscotch/data" } from "@hoppscotch/data"
import { pipe } from "fp-ts/function" import { computedWithControl } from "@vueuse/core"
import { useService } from "dioc/vue"
import * as TE from "fp-ts/TaskEither" import * as TE from "fp-ts/TaskEither"
import { GetMyTeamsQuery } from "~/helpers/backend/graphql" import { pipe } from "fp-ts/function"
import { cloneDeep } from "lodash-es"
import { computed, nextTick, reactive, ref, watch } from "vue"
import { GQLError } from "~/helpers/backend/GQLClient"
import { import {
createRequestInCollection, createRequestInCollection,
updateTeamRequest, updateTeamRequest,
} from "~/helpers/backend/mutations/TeamRequest" } from "~/helpers/backend/mutations/TeamRequest"
import { Picked } from "~/helpers/types/HoppPicked" import { Picked } from "~/helpers/types/HoppPicked"
import { useI18n } from "@composables/i18n"
import { useToast } from "@composables/toast"
import { import {
cascadeParentCollectionForHeaderAuth, cascadeParentCollectionForHeaderAuth,
editGraphqlRequest, editGraphqlRequest,
@@ -80,12 +82,10 @@ import {
saveGraphqlRequestAs, saveGraphqlRequestAs,
saveRESTRequestAs, saveRESTRequestAs,
} from "~/newstore/collections" } from "~/newstore/collections"
import { GQLError } from "~/helpers/backend/GQLClient"
import { computedWithControl } from "@vueuse/core"
import { platform } from "~/platform" import { platform } from "~/platform"
import { useService } from "dioc/vue"
import { RESTTabService } from "~/services/tab/rest"
import { GQLTabService } from "~/services/tab/graphql" import { GQLTabService } from "~/services/tab/graphql"
import { RESTTabService } from "~/services/tab/rest"
import { TeamWorkspace } from "~/services/workspace.service"
const t = useI18n() const t = useI18n()
const toast = useToast() const toast = useToast()
@@ -93,12 +93,10 @@ const toast = useToast()
const RESTTabs = useService(RESTTabService) const RESTTabs = useService(RESTTabService)
const GQLTabs = useService(GQLTabService) const GQLTabs = useService(GQLTabService)
type SelectedTeam = GetMyTeamsQuery["myTeams"][number] | undefined
type CollectionType = type CollectionType =
| { | {
type: "team-collections" type: "team-collections"
selectedTeam: SelectedTeam selectedTeam: TeamWorkspace
} }
| { type: "my-collections"; selectedTeam: undefined } | { type: "my-collections"; selectedTeam: undefined }
@@ -192,7 +190,7 @@ watch(
} }
) )
const updateTeam = (newTeam: SelectedTeam) => { const updateTeam = (newTeam: TeamWorkspace) => {
collectionsType.value.selectedTeam = newTeam collectionsType.value.selectedTeam = newTeam
} }
@@ -493,7 +491,7 @@ const updateTeamCollectionOrFolder = (
const data = { const data = {
title: requestUpdated.name, title: requestUpdated.name,
request: JSON.stringify(requestUpdated), request: JSON.stringify(requestUpdated),
teamID: collectionsType.value.selectedTeam.id, teamID: collectionsType.value.selectedTeam.teamID,
} }
pipe( pipe(
createRequestInCollection(collectionID, data), createRequestInCollection(collectionID, data),

View File

@@ -387,7 +387,6 @@ import IconPlus from "~icons/lucide/plus"
import IconHelpCircle from "~icons/lucide/help-circle" import IconHelpCircle from "~icons/lucide/help-circle"
import IconImport from "~icons/lucide/folder-down" import IconImport from "~icons/lucide/folder-down"
import { computed, PropType, Ref, toRef } from "vue" import { computed, PropType, Ref, toRef } from "vue"
import { GetMyTeamsQuery } from "~/helpers/backend/graphql"
import { useI18n } from "@composables/i18n" import { useI18n } from "@composables/i18n"
import { useColorMode } from "@composables/theming" import { useColorMode } from "@composables/theming"
import { TeamCollection } from "~/helpers/teams/TeamCollection" import { TeamCollection } from "~/helpers/teams/TeamCollection"
@@ -400,17 +399,16 @@ import * as O from "fp-ts/Option"
import { Picked } from "~/helpers/types/HoppPicked.js" import { Picked } from "~/helpers/types/HoppPicked.js"
import { RESTTabService } from "~/services/tab/rest" import { RESTTabService } from "~/services/tab/rest"
import { useService } from "dioc/vue" import { useService } from "dioc/vue"
import { TeamWorkspace } from "~/services/workspace.service"
const t = useI18n() const t = useI18n()
const colorMode = useColorMode() const colorMode = useColorMode()
const tabs = useService(RESTTabService) const tabs = useService(RESTTabService)
type SelectedTeam = GetMyTeamsQuery["myTeams"][number] | undefined
type CollectionType = type CollectionType =
| { | {
type: "team-collections" type: "team-collections"
selectedTeam: SelectedTeam selectedTeam: TeamWorkspace
} }
| { type: "my-collections"; selectedTeam: undefined } | { type: "my-collections"; selectedTeam: undefined }
@@ -614,7 +612,7 @@ const hasNoTeamAccess = computed(
() => () =>
props.collectionsType.type === "team-collections" && props.collectionsType.type === "team-collections" &&
(props.collectionsType.selectedTeam === undefined || (props.collectionsType.selectedTeam === undefined ||
props.collectionsType.selectedTeam.myRole === "VIEWER") props.collectionsType.selectedTeam.role === "VIEWER")
) )
const isSelected = ({ const isSelected = ({

View File

@@ -180,7 +180,6 @@ import { GQLTabService } from "~/services/tab/graphql"
import { computed } from "vue" import { computed } from "vue"
import { import {
HoppCollection, HoppCollection,
HoppGQLAuth,
HoppGQLRequest, HoppGQLRequest,
makeGQLRequest, makeGQLRequest,
} from "@hoppscotch/data" } from "@hoppscotch/data"
@@ -193,6 +192,7 @@ import { PersistenceService } from "~/services/persistence"
import { PersistedOAuthConfig } from "~/services/oauth/oauth.service" import { PersistedOAuthConfig } from "~/services/oauth/oauth.service"
import { GQLOptionTabs } from "~/components/graphql/RequestOptions.vue" import { GQLOptionTabs } from "~/components/graphql/RequestOptions.vue"
import { EditingProperties } from "../Properties.vue" import { EditingProperties } from "../Properties.vue"
import { defineActionHandler } from "~/helpers/actions"
const t = useI18n() const t = useI18n()
const toast = useToast() const toast = useToast()
@@ -200,7 +200,7 @@ const toast = useToast()
defineProps<{ defineProps<{
// Whether to activate the ability to pick items (activates 'select' events) // Whether to activate the ability to pick items (activates 'select' events)
saveRequest: boolean saveRequest: boolean
picked: Picked picked: Picked | null
}>() }>()
const collections = useReadonlyStream(graphqlCollections$, [], "deep") const collections = useReadonlyStream(graphqlCollections$, [], "deep")
@@ -226,7 +226,7 @@ const editingRequest = ref<HoppGQLRequest | null>(null)
const editingRequestIndex = ref<number | null>(null) const editingRequestIndex = ref<number | null>(null)
const editingProperties = ref<{ const editingProperties = ref<{
collection: HoppCollection | null collection: Partial<HoppCollection> | null
isRootCollection: boolean isRootCollection: boolean
path: string path: string
inheritedProperties?: HoppInheritedProperty inheritedProperties?: HoppInheritedProperty
@@ -265,8 +265,9 @@ onMounted(() => {
) )
if (unsavedCollectionPropertiesString) { if (unsavedCollectionPropertiesString) {
const unsavedCollectionProperties: EditingProperties<"GraphQL"> = const unsavedCollectionProperties: EditingProperties = JSON.parse(
JSON.parse(unsavedCollectionPropertiesString) unsavedCollectionPropertiesString
)
const auth = unsavedCollectionProperties.collection?.auth const auth = unsavedCollectionProperties.collection?.auth
@@ -610,7 +611,7 @@ const editProperties = ({
if (collectionIndex === null || collection === null) return if (collectionIndex === null || collection === null) return
const parentIndex = collectionIndex.split("/").slice(0, -1).join("/") // remove last folder to get parent folder const parentIndex = collectionIndex.split("/").slice(0, -1).join("/") // remove last folder to get parent folder
let inheritedProperties = {} let inheritedProperties = undefined
if (parentIndex) { if (parentIndex) {
const { auth, headers } = cascadeParentCollectionForHeaderAuth( const { auth, headers } = cascadeParentCollectionForHeaderAuth(
@@ -621,7 +622,7 @@ const editProperties = ({
inheritedProperties = { inheritedProperties = {
auth, auth,
headers, headers,
} as HoppInheritedProperty }
} }
editingProperties.value = { editingProperties.value = {
@@ -635,11 +636,15 @@ const editProperties = ({
} }
const setCollectionProperties = (newCollection: { const setCollectionProperties = (newCollection: {
collection: HoppCollection collection: Partial<HoppCollection> | null
path: string path: string
isRootCollection: boolean isRootCollection: boolean
}) => { }) => {
const { collection, path, isRootCollection } = newCollection const { collection, path, isRootCollection } = newCollection
if (!collection) {
return
}
if (isRootCollection) { if (isRootCollection) {
editGraphqlCollection(parseInt(path), collection) editGraphqlCollection(parseInt(path), collection)
} else { } else {
@@ -672,4 +677,11 @@ const resetSelectedData = () => {
editingRequest.value = null editingRequest.value = null
editingRequestIndex.value = null editingRequestIndex.value = null
} }
defineActionHandler("collection.new", () => {
displayModalAdd(true)
})
defineActionHandler("modals.collection.import", () => {
displayModalImportExport(true)
})
</script> </script>

View File

@@ -178,7 +178,6 @@ import { useI18n } from "@composables/i18n"
import { Picked } from "~/helpers/types/HoppPicked" import { Picked } from "~/helpers/types/HoppPicked"
import { useReadonlyStream } from "~/composables/stream" import { useReadonlyStream } from "~/composables/stream"
import { useLocalState } from "~/newstore/localstate" import { useLocalState } from "~/newstore/localstate"
import { GetMyTeamsQuery } from "~/helpers/backend/graphql"
import { pipe } from "fp-ts/function" import { pipe } from "fp-ts/function"
import * as TE from "fp-ts/TaskEither" import * as TE from "fp-ts/TaskEither"
import { import {
@@ -245,7 +244,7 @@ import {
} from "~/helpers/collection/collection" } from "~/helpers/collection/collection"
import { currentReorderingStatus$ } from "~/newstore/reordering" import { currentReorderingStatus$ } from "~/newstore/reordering"
import { defineActionHandler, invokeAction } from "~/helpers/actions" import { defineActionHandler, invokeAction } from "~/helpers/actions"
import { WorkspaceService } from "~/services/workspace.service" import { TeamWorkspace, WorkspaceService } from "~/services/workspace.service"
import { useService } from "dioc/vue" import { useService } from "dioc/vue"
import { RESTTabService } from "~/services/tab/rest" import { RESTTabService } from "~/services/tab/rest"
import { HoppInheritedProperty } from "~/helpers/types/HoppInheritedProperties" import { HoppInheritedProperty } from "~/helpers/types/HoppInheritedProperties"
@@ -274,16 +273,14 @@ const props = defineProps({
const emit = defineEmits<{ const emit = defineEmits<{
(event: "select", payload: Picked | null): void (event: "select", payload: Picked | null): void
(event: "update-team", team: SelectedTeam): void (event: "update-team", team: TeamWorkspace): void
(event: "update-collection-type", type: CollectionType["type"]): void (event: "update-collection-type", type: CollectionType["type"]): void
}>() }>()
type SelectedTeam = GetMyTeamsQuery["myTeams"][number] | undefined
type CollectionType = type CollectionType =
| { | {
type: "team-collections" type: "team-collections"
selectedTeam: SelectedTeam selectedTeam: TeamWorkspace
} }
| { type: "my-collections"; selectedTeam: undefined } | { type: "my-collections"; selectedTeam: undefined }
@@ -330,9 +327,7 @@ const requestMoveLoading = ref<string[]>([])
// TeamList-Adapter // TeamList-Adapter
const workspaceService = useService(WorkspaceService) const workspaceService = useService(WorkspaceService)
const teamListAdapter = workspaceService.acquireTeamListAdapter(null) const teamListAdapter = workspaceService.acquireTeamListAdapter(null)
const myTeams = useReadonlyStream(teamListAdapter.teamList$, null)
const REMEMBERED_TEAM_ID = useLocalState("REMEMBERED_TEAM_ID") const REMEMBERED_TEAM_ID = useLocalState("REMEMBERED_TEAM_ID")
const teamListFetched = ref(false)
// Team Collection Adapter // Team Collection Adapter
const teamCollectionAdapter = new TeamCollectionAdapter(null) const teamCollectionAdapter = new TeamCollectionAdapter(null)
@@ -378,10 +373,10 @@ watch(
filterTexts, filterTexts,
(newFilterText) => { (newFilterText) => {
if (collectionsType.value.type === "team-collections") { if (collectionsType.value.type === "team-collections") {
const selectedTeamID = collectionsType.value.selectedTeam?.id const selectedTeamID = collectionsType.value.selectedTeam?.teamID
selectedTeamID && selectedTeamID &&
debouncedSearch(newFilterText, selectedTeamID)?.catch((_) => {}) debouncedSearch(newFilterText, selectedTeamID)?.catch(() => {})
} }
}, },
{ {
@@ -414,14 +409,11 @@ onMounted(() => {
) )
if (unsavedCollectionPropertiesString) { if (unsavedCollectionPropertiesString) {
const unsavedCollectionProperties: EditingProperties<"REST"> = JSON.parse( const unsavedCollectionProperties: EditingProperties = JSON.parse(
unsavedCollectionPropertiesString unsavedCollectionPropertiesString
) )
// casting because the type `EditingProperties["collection"]["auth"] and the usage in Properties.vue is different. there it's casted as an any. const auth = unsavedCollectionProperties.collection?.auth
// FUTURE-TODO: look into this
// @ts-expect-error because of the above reason
const auth = unsavedCollectionProperties.collection?.auth as HoppRESTAuth
if (auth?.authType === "oauth-2") { if (auth?.authType === "oauth-2") {
const grantTypeInfo = auth.grantTypeInfo const grantTypeInfo = auth.grantTypeInfo
@@ -438,28 +430,6 @@ onMounted(() => {
} }
}) })
watch(
() => myTeams.value,
(newTeams) => {
if (newTeams && !teamListFetched.value) {
teamListFetched.value = true
if (REMEMBERED_TEAM_ID.value && currentUser.value) {
const team = newTeams.find((t) => t.id === REMEMBERED_TEAM_ID.value)
if (team) updateSelectedTeam(team)
}
}
}
)
watch(
() => collectionsType.value.selectedTeam,
(newTeam) => {
if (newTeam) {
teamCollectionAdapter.changeTeamID(newTeam.id)
}
}
)
const switchToMyCollections = () => { const switchToMyCollections = () => {
collectionsType.value.type = "my-collections" collectionsType.value.type = "my-collections"
collectionsType.value.selectedTeam = undefined collectionsType.value.selectedTeam = undefined
@@ -491,11 +461,12 @@ const expandTeamCollection = (collectionID: string) => {
teamCollectionAdapter.expandCollection(collectionID) teamCollectionAdapter.expandCollection(collectionID)
} }
const updateSelectedTeam = (team: SelectedTeam) => { const updateSelectedTeam = (team: TeamWorkspace) => {
if (team) { if (team) {
collectionsType.value.type = "team-collections" collectionsType.value.type = "team-collections"
teamCollectionAdapter.changeTeamID(team.teamID)
collectionsType.value.selectedTeam = team collectionsType.value.selectedTeam = team
REMEMBERED_TEAM_ID.value = team.id REMEMBERED_TEAM_ID.value = team.teamID
emit("update-team", team) emit("update-team", team)
emit("update-collection-type", "team-collections") emit("update-collection-type", "team-collections")
} }
@@ -504,23 +475,14 @@ const updateSelectedTeam = (team: SelectedTeam) => {
const workspace = workspaceService.currentWorkspace const workspace = workspaceService.currentWorkspace
// Used to switch collection type and team when user switch workspace in the global workspace switcher // Used to switch collection type and team when user switch workspace in the global workspace switcher
// Check if there is a teamID in the workspace, if yes, switch to team collections and select the team
// If there is no teamID, switch to my collections
watch( watch(
() => { workspace,
const space = workspace.value (newWorkspace) => {
return space.type === "personal" ? undefined : space.teamID if (newWorkspace.type === "personal") {
}, switchToMyCollections()
(teamID) => { } else if (newWorkspace.type === "team") {
if (teamID) { updateSelectedTeam(newWorkspace)
const team = myTeams.value?.find((t) => t.id === teamID)
if (team) {
updateSelectedTeam(team)
}
return
} }
return switchToMyCollections()
}, },
{ {
immediate: true, immediate: true,
@@ -548,7 +510,7 @@ const hasTeamWriteAccess = computed(() => {
return false return false
} }
const role = collectionsType.value.selectedTeam?.myRole const role = collectionsType.value.selectedTeam?.role
return role === "OWNER" || role === "EDITOR" return role === "OWNER" || role === "EDITOR"
}) })
@@ -763,7 +725,7 @@ const addNewRootCollection = (name: string) => {
}) })
pipe( pipe(
createNewRootCollection(name, collectionsType.value.selectedTeam.id), createNewRootCollection(name, collectionsType.value.selectedTeam.teamID),
TE.match( TE.match(
(err: GQLError<string>) => { (err: GQLError<string>) => {
toast.error(`${getErrorMessage(err)}`) toast.error(`${getErrorMessage(err)}`)
@@ -834,7 +796,7 @@ const onAddRequest = (requestName: string) => {
const data = { const data = {
request: JSON.stringify(newRequest), request: JSON.stringify(newRequest),
teamID: collectionsType.value.selectedTeam.id, teamID: collectionsType.value.selectedTeam.teamID,
title: requestName, title: requestName,
} }
@@ -1161,7 +1123,7 @@ const duplicateRequest = (payload: {
const data = { const data = {
request: JSON.stringify(newRequest), request: JSON.stringify(newRequest),
teamID: collectionsType.value.selectedTeam.id, teamID: collectionsType.value.selectedTeam.teamID,
title: `${request.name} - ${t("action.duplicate")}`, title: `${request.name} - ${t("action.duplicate")}`,
} }
@@ -2354,4 +2316,7 @@ const getErrorMessage = (err: GQLError<string>) => {
defineActionHandler("collection.new", () => { defineActionHandler("collection.new", () => {
displayModalAdd(true) displayModalAdd(true)
}) })
defineActionHandler("modals.collection.import", () => {
displayModalImportExport(true)
})
</script> </script>

View File

@@ -364,6 +364,7 @@ const switchToTeamWorkspace = (team: GetMyTeamsQuery["myTeams"][number]) => {
teamID: team.id, teamID: team.id,
teamName: team.name, teamName: team.name,
type: "team", type: "team",
role: team.myRole,
}) })
} }
watch( watch(

View File

@@ -46,41 +46,38 @@
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import { computed, ref, watch } from "vue"
import { isEqual } from "lodash-es"
import { platform } from "~/platform"
import { GetMyTeamsQuery } from "~/helpers/backend/graphql"
import { useReadonlyStream, useStream } from "@composables/stream" import { useReadonlyStream, useStream } from "@composables/stream"
import { Environment } from "@hoppscotch/data"
import { useService } from "dioc/vue"
import * as TE from "fp-ts/TaskEither"
import { pipe } from "fp-ts/function"
import { isEqual } from "lodash-es"
import { computed, ref, watch } from "vue"
import { useI18n } from "~/composables/i18n" import { useI18n } from "~/composables/i18n"
import { useToast } from "~/composables/toast"
import { defineActionHandler } from "~/helpers/actions"
import { GQLError } from "~/helpers/backend/GQLClient"
import { deleteTeamEnvironment } from "~/helpers/backend/mutations/TeamEnvironment"
import TeamEnvironmentAdapter from "~/helpers/teams/TeamEnvironmentAdapter"
import { import {
deleteEnvironment,
getSelectedEnvironmentIndex, getSelectedEnvironmentIndex,
globalEnv$, globalEnv$,
selectedEnvironmentIndex$, selectedEnvironmentIndex$,
setSelectedEnvironmentIndex, setSelectedEnvironmentIndex,
} from "~/newstore/environments" } from "~/newstore/environments"
import TeamEnvironmentAdapter from "~/helpers/teams/TeamEnvironmentAdapter"
import { defineActionHandler } from "~/helpers/actions"
import { useLocalState } from "~/newstore/localstate" import { useLocalState } from "~/newstore/localstate"
import { pipe } from "fp-ts/function" import { platform } from "~/platform"
import * as TE from "fp-ts/TaskEither" import { TeamWorkspace, WorkspaceService } from "~/services/workspace.service"
import { GQLError } from "~/helpers/backend/GQLClient"
import { deleteEnvironment } from "~/newstore/environments"
import { deleteTeamEnvironment } from "~/helpers/backend/mutations/TeamEnvironment"
import { useToast } from "~/composables/toast"
import { WorkspaceService } from "~/services/workspace.service"
import { useService } from "dioc/vue"
import { Environment } from "@hoppscotch/data"
const t = useI18n() const t = useI18n()
const toast = useToast() const toast = useToast()
type EnvironmentType = "my-environments" | "team-environments" type EnvironmentType = "my-environments" | "team-environments"
type SelectedTeam = GetMyTeamsQuery["myTeams"][number] | undefined
type EnvironmentsChooseType = { type EnvironmentsChooseType = {
type: EnvironmentType type: EnvironmentType
selectedTeam: SelectedTeam selectedTeam: TeamWorkspace | undefined
} }
const environmentType = ref<EnvironmentsChooseType>({ const environmentType = ref<EnvironmentsChooseType>({
@@ -102,11 +99,7 @@ const currentUser = useReadonlyStream(
platform.auth.getCurrentUser() platform.auth.getCurrentUser()
) )
// TeamList-Adapter
const workspaceService = useService(WorkspaceService) const workspaceService = useService(WorkspaceService)
const teamListAdapter = workspaceService.acquireTeamListAdapter(null)
const myTeams = useReadonlyStream(teamListAdapter.teamList$, null)
const teamListFetched = ref(false)
const REMEMBERED_TEAM_ID = useLocalState("REMEMBERED_TEAM_ID") const REMEMBERED_TEAM_ID = useLocalState("REMEMBERED_TEAM_ID")
const adapter = new TeamEnvironmentAdapter(undefined) const adapter = new TeamEnvironmentAdapter(undefined)
@@ -118,29 +111,17 @@ const loading = computed(
() => adapterLoading.value && teamEnvironmentList.value.length === 0 () => adapterLoading.value && teamEnvironmentList.value.length === 0
) )
watch(
() => myTeams.value,
(newTeams) => {
if (newTeams && !teamListFetched.value) {
teamListFetched.value = true
if (REMEMBERED_TEAM_ID.value && currentUser.value) {
const team = newTeams.find((t) => t.id === REMEMBERED_TEAM_ID.value)
if (team) updateSelectedTeam(team)
}
}
}
)
const switchToMyEnvironments = () => { const switchToMyEnvironments = () => {
environmentType.value.selectedTeam = undefined environmentType.value.selectedTeam = undefined
updateEnvironmentType("my-environments") updateEnvironmentType("my-environments")
adapter.changeTeamID(undefined) adapter.changeTeamID(undefined)
} }
const updateSelectedTeam = (newSelectedTeam: SelectedTeam | undefined) => { const updateSelectedTeam = (newSelectedTeam: TeamWorkspace | undefined) => {
if (newSelectedTeam) { if (newSelectedTeam) {
adapter.changeTeamID(newSelectedTeam.teamID)
environmentType.value.selectedTeam = newSelectedTeam environmentType.value.selectedTeam = newSelectedTeam
REMEMBERED_TEAM_ID.value = newSelectedTeam.id REMEMBERED_TEAM_ID.value = newSelectedTeam.teamID
updateEnvironmentType("team-environments") updateEnvironmentType("team-environments")
} }
} }
@@ -148,15 +129,6 @@ const updateEnvironmentType = (newEnvironmentType: EnvironmentType) => {
environmentType.value.type = newEnvironmentType environmentType.value.type = newEnvironmentType
} }
watch(
() => environmentType.value.selectedTeam,
(newTeam) => {
if (newTeam) {
adapter.changeTeamID(newTeam.id)
}
}
)
const workspace = workspaceService.currentWorkspace const workspace = workspaceService.currentWorkspace
// Switch to my environments if workspace is personal and to team environments if workspace is team // Switch to my environments if workspace is personal and to team environments if workspace is team
@@ -170,8 +142,7 @@ watch(workspace, (newWorkspace) => {
}) })
} }
} else if (newWorkspace.type === "team") { } else if (newWorkspace.type === "team") {
const team = myTeams.value?.find((t) => t.id === newWorkspace.teamID) updateSelectedTeam(newWorkspace)
updateSelectedTeam(team)
} }
}) })

View File

@@ -54,9 +54,7 @@
:key="tab.id" :key="tab.id"
:label="tab.label" :label="tab.label"
> >
<div <div class="divide-y divide-dividerLight">
class="divide-y divide-dividerLight rounded border border-divider"
>
<HoppSmartPlaceholder <HoppSmartPlaceholder
v-if="tab.variables.length === 0" v-if="tab.variables.length === 0"
:src="`/images/states/${colorMode.value}/blockchain.svg`" :src="`/images/states/${colorMode.value}/blockchain.svg`"

View File

@@ -56,9 +56,7 @@
:key="tab.id" :key="tab.id"
:label="tab.label" :label="tab.label"
> >
<div <div class="divide-y divide-dividerLight">
class="divide-y divide-dividerLight rounded border border-divider"
>
<HoppSmartPlaceholder <HoppSmartPlaceholder
v-if="tab.variables.length === 0" v-if="tab.variables.length === 0"
:src="`/images/states/${colorMode.value}/blockchain.svg`" :src="`/images/states/${colorMode.value}/blockchain.svg`"

View File

@@ -4,7 +4,7 @@
class="sticky top-upperPrimaryStickyFold z-10 flex flex-1 flex-shrink-0 justify-between overflow-x-auto border-b border-dividerLight bg-primary" class="sticky top-upperPrimaryStickyFold z-10 flex flex-1 flex-shrink-0 justify-between overflow-x-auto border-b border-dividerLight bg-primary"
> >
<HoppButtonSecondary <HoppButtonSecondary
v-if="team === undefined || team.myRole === 'VIEWER'" v-if="team === undefined || team.role === 'VIEWER'"
v-tippy="{ theme: 'tooltip' }" v-tippy="{ theme: 'tooltip' }"
disabled disabled
class="!rounded-none" class="!rounded-none"
@@ -28,7 +28,7 @@
:icon="IconHelpCircle" :icon="IconHelpCircle"
/> />
<HoppButtonSecondary <HoppButtonSecondary
v-if="team !== undefined && team.myRole === 'VIEWER'" v-if="team !== undefined && team.role === 'VIEWER'"
v-tippy="{ theme: 'tooltip' }" v-tippy="{ theme: 'tooltip' }"
disabled disabled
:icon="IconImport" :icon="IconImport"
@@ -84,7 +84,7 @@
)" )"
:key="`environment-${index}`" :key="`environment-${index}`"
:environment="environment" :environment="environment"
:is-viewer="team?.myRole === 'VIEWER'" :is-viewer="team?.role === 'VIEWER'"
@edit-environment="editEnvironment(environment)" @edit-environment="editEnvironment(environment)"
/> />
</div> </div>
@@ -103,16 +103,16 @@
:show="showModalDetails" :show="showModalDetails"
:action="action" :action="action"
:editing-environment="editingEnvironment" :editing-environment="editingEnvironment"
:editing-team-id="team?.id" :editing-team-id="team?.teamID"
:editing-variable-name="editingVariableName" :editing-variable-name="editingVariableName"
:is-secret-option-selected="secretOptionSelected" :is-secret-option-selected="secretOptionSelected"
:is-viewer="team?.myRole === 'VIEWER'" :is-viewer="team?.role === 'VIEWER'"
@hide-modal="displayModalEdit(false)" @hide-modal="displayModalEdit(false)"
/> />
<EnvironmentsImportExport <EnvironmentsImportExport
v-if="showModalImportExport" v-if="showModalImportExport"
:team-environments="teamEnvironments" :team-environments="teamEnvironments"
:team-id="team?.id" :team-id="team?.teamID"
environment-type="TEAM_ENV" environment-type="TEAM_ENV"
@hide-modal="displayModalImportExport(false)" @hide-modal="displayModalImportExport(false)"
/> />
@@ -129,16 +129,14 @@ import IconPlus from "~icons/lucide/plus"
import IconHelpCircle from "~icons/lucide/help-circle" import IconHelpCircle from "~icons/lucide/help-circle"
import IconImport from "~icons/lucide/folder-down" import IconImport from "~icons/lucide/folder-down"
import { defineActionHandler } from "~/helpers/actions" import { defineActionHandler } from "~/helpers/actions"
import { GetMyTeamsQuery } from "~/helpers/backend/graphql" import { TeamWorkspace } from "~/services/workspace.service"
const t = useI18n() const t = useI18n()
const colorMode = useColorMode() const colorMode = useColorMode()
type SelectedTeam = GetMyTeamsQuery["myTeams"][number] | undefined
const props = defineProps<{ const props = defineProps<{
team: SelectedTeam team: TeamWorkspace | undefined
teamEnvironments: TeamEnvironment[] teamEnvironments: TeamEnvironment[]
adapterError: GQLError<string> | null adapterError: GQLError<string> | null
loading: boolean loading: boolean
@@ -151,7 +149,7 @@ const editingEnvironment = ref<TeamEnvironment | null>(null)
const editingVariableName = ref("") const editingVariableName = ref("")
const secretOptionSelected = ref(false) const secretOptionSelected = ref(false)
const isTeamViewer = computed(() => props.team?.myRole === "VIEWER") const isTeamViewer = computed(() => props.team?.role === "VIEWER")
const displayModalAdd = (shouldDisplay: boolean) => { const displayModalAdd = (shouldDisplay: boolean) => {
action.value = "new" action.value = "new"

View File

@@ -23,10 +23,10 @@
@click="provider.action" @click="provider.action"
/> />
<hr v-if="additonalLoginItems.length > 0" /> <hr v-if="additionalLoginItems.length > 0" />
<HoppSmartItem <HoppSmartItem
v-for="loginItem in additonalLoginItems" v-for="loginItem in additionalLoginItems"
:key="loginItem.id" :key="loginItem.id"
:icon="loginItem.icon" :icon="loginItem.icon"
:label="loginItem.text(t)" :label="loginItem.text(t)"
@@ -170,7 +170,7 @@ type AuthProviderItem = {
} }
let allowedAuthProviders: AuthProviderItem[] = [] let allowedAuthProviders: AuthProviderItem[] = []
let additonalLoginItems: LoginItemDef[] = [] const additionalLoginItems: LoginItemDef[] = []
const doAdditionalLoginItemClickAction = async (item: LoginItemDef) => { const doAdditionalLoginItemClickAction = async (item: LoginItemDef) => {
await item.onClick() await item.onClick()
@@ -199,10 +199,33 @@ onMounted(async () => {
allowedAuthProviders = enabledAuthProviders allowedAuthProviders = enabledAuthProviders
// setup the additional login items // setup the additional login items
additonalLoginItems = platform.auth.additionalLoginItems?.forEach((item) => {
platform.auth.additionalLoginItems?.filter((item) => if (res.right.includes(item.id)) {
res.right.includes(item.id) additionalLoginItems.push(item)
) ?? [] }
// since the BE send the OIDC auth providers as OIDC:providerName,
// we need to split the string and use the providerName as the text
if (item.id === "OIDC") {
res.right
.filter((provider) => provider.startsWith("OIDC"))
.forEach((provider) => {
const OIDCName = provider.split(":")[1]
const loginItemText = OIDCName
? () =>
t("auth.continue_with_auth_provider", {
provider: OIDCName,
})
: item.text
const OIDCLoginItem = {
...item,
text: loginItemText,
}
additionalLoginItems.push(OIDCLoginItem)
})
}
})
isLoadingAllowedAuthProviders.value = false isLoadingAllowedAuthProviders.value = false
}) })
@@ -311,6 +334,14 @@ const authProvidersAvailable: AuthProviderItem[] = [
action: signInWithGithub, action: signInWithGithub,
isLoading: signingInWithGitHub, isLoading: signingInWithGitHub,
}, },
// the authprovider either send github or github:enterprise and both are handled by the same route
{
id: "GITHUB:ENTERPRISE",
icon: IconGithub,
label: t("auth.continue_with_github_enterprise"),
action: signInWithGithub,
isLoading: signingInWithGitHub,
},
{ {
id: "GOOGLE", id: "GOOGLE",
icon: IconGoogle, icon: IconGoogle,

View File

@@ -299,7 +299,7 @@ const selectOAuth2AuthType = () => {
? existingGrantTypeInfo ? existingGrantTypeInfo
: defaultGrantTypeInfo : defaultGrantTypeInfo
auth.value = <HoppGQLAuth>{ auth.value = {
...auth.value, ...auth.value,
authType: "oauth-2", authType: "oauth-2",
addTo: "HEADERS", addTo: "HEADERS",

View File

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

View File

@@ -10,7 +10,7 @@
autocomplete="off" autocomplete="off"
spellcheck="false" spellcheck="false"
class="w-full rounded border border-divider bg-primaryLight px-4 py-2 text-secondaryDark" class="w-full rounded border border-divider bg-primaryLight px-4 py-2 text-secondaryDark"
:placeholder="`${t('request.url')}`" :placeholder="`${t('graphql.url_placeholder')}`"
:disabled="connected" :disabled="connected"
@keyup.enter="onConnectClick" @keyup.enter="onConnectClick"
/> />

View File

@@ -307,7 +307,7 @@ const selectOAuth2AuthType = () => {
? existingGrantTypeInfo ? existingGrantTypeInfo
: defaultGrantTypeInfo : defaultGrantTypeInfo
auth.value = <HoppRESTAuth>{ auth.value = {
...auth.value, ...auth.value,
authType: "oauth-2", authType: "oauth-2",
addTo: "HEADERS", addTo: "HEADERS",

View File

@@ -98,6 +98,7 @@ import { RESTTabService } from "~/services/tab/rest"
import { useService } from "dioc/vue" import { useService } from "dioc/vue"
import { useNestedSetting } from "~/composables/settings" import { useNestedSetting } from "~/composables/settings"
import { toggleNestedSetting } from "~/newstore/settings" import { toggleNestedSetting } from "~/newstore/settings"
import { EditorView } from "@codemirror/view"
const t = useI18n() const t = useI18n()
@@ -124,6 +125,7 @@ useCodemirror(
linter: null, linter: null,
completer: null, completer: null,
environmentHighlights: false, environmentHighlights: false,
onInit: (view: EditorView) => view.focus(),
}) })
) )

View File

@@ -913,14 +913,16 @@ const generateOAuthToken = async () => {
if ( if (
grantTypesInvolvingRedirect.includes(auth.value.grantTypeInfo.grantType) grantTypesInvolvingRedirect.includes(auth.value.grantTypeInfo.grantType)
) { ) {
const authConfig: PersistedOAuthConfig = {
source: props.source,
context: props.isCollectionProperty
? { type: "collection-properties", metadata: {} }
: { type: "request-tab", metadata: {} },
grant_type: auth.value.grantTypeInfo.grantType,
}
persistenceService.setLocalConfig( persistenceService.setLocalConfig(
"oauth_temp_config", "oauth_temp_config",
JSON.stringify(<PersistedOAuthConfig>{ JSON.stringify(authConfig)
source: props.source,
context: props.isCollectionProperty
? { type: "collection-properties", metadata: {} }
: { type: "request-tab" },
})
) )
} }

View File

@@ -54,7 +54,7 @@
> >
<SmartEnvInput <SmartEnvInput
v-model="tab.document.request.endpoint" v-model="tab.document.request.endpoint"
:placeholder="`${t('request.url')}`" :placeholder="`${t('request.url_placeholder')}`"
:auto-complete-source="userHistories" :auto-complete-source="userHistories"
:auto-complete-env="true" :auto-complete-env="true"
:inspection-results="tabResults" :inspection-results="tabResults"

View File

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

View File

@@ -36,16 +36,6 @@
/> />
</span> </span>
</div> </div>
<div class="space-y-4 py-4">
<div class="flex items-center">
<HoppSmartToggle
:on="extensionEnabled"
@change="extensionEnabled = !extensionEnabled"
>
{{ t("settings.extensions_use_toggle") }}
</HoppSmartToggle>
</div>
</div>
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
@@ -55,34 +45,12 @@ import IconCheckCircle from "~icons/lucide/check-circle"
import { useI18n } from "@composables/i18n" import { useI18n } from "@composables/i18n"
import { ExtensionInterceptorService } from "~/platform/std/interceptors/extension" import { ExtensionInterceptorService } from "~/platform/std/interceptors/extension"
import { useService } from "dioc/vue" import { useService } from "dioc/vue"
import { computed } from "vue"
import { InterceptorService } from "~/services/interceptor.service"
import { platform } from "~/platform"
const t = useI18n() const t = useI18n()
const interceptorService = useService(InterceptorService)
const extensionService = useService(ExtensionInterceptorService) const extensionService = useService(ExtensionInterceptorService)
const extensionVersion = extensionService.extensionVersion const extensionVersion = extensionService.extensionVersion
const hasChromeExtInstalled = extensionService.chromeExtensionInstalled const hasChromeExtInstalled = extensionService.chromeExtensionInstalled
const hasFirefoxExtInstalled = extensionService.firefoxExtensionInstalled const hasFirefoxExtInstalled = extensionService.firefoxExtensionInstalled
const extensionEnabled = computed({
get() {
return (
interceptorService.currentInterceptorID.value ===
extensionService.interceptorID
)
},
set(active) {
if (active) {
interceptorService.currentInterceptorID.value =
extensionService.interceptorID
} else {
interceptorService.currentInterceptorID.value =
platform.interceptors.default
}
},
})
</script> </script>

View File

@@ -8,16 +8,6 @@
:label="t('app.proxy_privacy_policy')" :label="t('app.proxy_privacy_policy')"
/>. />.
</div> </div>
<div class="space-y-4 py-4">
<div class="flex items-center">
<HoppSmartToggle
:on="proxyEnabled"
@change="proxyEnabled = !proxyEnabled"
>
{{ t("settings.proxy_use_toggle") }}
</HoppSmartToggle>
</div>
</div>
<div class="flex items-center space-x-2 py-4"> <div class="flex items-center space-x-2 py-4">
<HoppSmartInput <HoppSmartInput
v-model="PROXY_URL" v-model="PROXY_URL"
@@ -50,7 +40,6 @@ import { computed } from "vue"
import { useService } from "dioc/vue" import { useService } from "dioc/vue"
import { InterceptorService } from "~/services/interceptor.service" import { InterceptorService } from "~/services/interceptor.service"
import { proxyInterceptor } from "~/platform/std/interceptors/proxy" import { proxyInterceptor } from "~/platform/std/interceptors/proxy"
import { platform } from "~/platform"
const t = useI18n() const t = useI18n()
const toast = useToast() const toast = useToast()
@@ -59,23 +48,11 @@ const interceptorService = useService(InterceptorService)
const PROXY_URL = useSetting("PROXY_URL") const PROXY_URL = useSetting("PROXY_URL")
const proxyEnabled = computed({ const proxyEnabled = computed(
get() { () =>
return ( interceptorService.currentInterceptorID.value ===
interceptorService.currentInterceptorID.value === proxyInterceptor.interceptorID
proxyInterceptor.interceptorID )
)
},
set(active) {
if (active) {
interceptorService.currentInterceptorID.value =
proxyInterceptor.interceptorID
} else {
interceptorService.currentInterceptorID.value =
platform.interceptors.default
}
},
})
const clearIcon = refAutoReset<typeof IconRotateCCW | typeof IconCheck>( const clearIcon = refAutoReset<typeof IconRotateCCW | typeof IconCheck>(
IconRotateCCW, IconRotateCCW,

View File

@@ -273,6 +273,10 @@ const loading = computed(
) )
onLoggedIn(() => { onLoggedIn(() => {
if (adapter.isInitialized()) {
return
}
try { try {
// wait for a bit to let the auth token to be set // wait for a bit to let the auth token to be set
// because in some race conditions, the token is not set this fixes that // because in some race conditions, the token is not set this fixes that

View File

@@ -37,13 +37,17 @@ import { TeamNameCodec } from "~/helpers/backend/types/TeamName"
import { useI18n } from "@composables/i18n" import { useI18n } from "@composables/i18n"
import { useToast } from "@composables/toast" import { useToast } from "@composables/toast"
import { platform } from "~/platform" import { platform } from "~/platform"
import { useService } from "dioc/vue"
import { WorkspaceService } from "~/services/workspace.service"
import { useLocalState } from "~/newstore/localstate"
const t = useI18n() const t = useI18n()
const toast = useToast() const toast = useToast()
defineProps<{ const props = defineProps<{
show: boolean show: boolean
switchWorkspaceAfterCreation?: boolean
}>() }>()
const emit = defineEmits<{ const emit = defineEmits<{
@@ -52,8 +56,12 @@ const emit = defineEmits<{
const editingName = ref<string | null>(null) const editingName = ref<string | null>(null)
const REMEMBERED_TEAM_ID = useLocalState("REMEMBERED_TEAM_ID")
const isLoading = ref(false) const isLoading = ref(false)
const workspaceService = useService(WorkspaceService)
const addNewTeam = async () => { const addNewTeam = async () => {
isLoading.value = true isLoading.value = true
await pipe( await pipe(
@@ -76,8 +84,19 @@ const addNewTeam = async () => {
// Handle GQL errors (use err obj) // Handle GQL errors (use err obj)
} }
}, },
() => { (team) => {
toast.success(`${t("team.new_created")}`) toast.success(`${t("team.new_created")}`)
if (props.switchWorkspaceAfterCreation) {
REMEMBERED_TEAM_ID.value = team.id
workspaceService.changeWorkspace({
teamID: team.id,
teamName: team.name,
type: "team",
role: team.myRole,
})
}
hideModal() hideModal()
} }
) )

View File

@@ -20,7 +20,7 @@
: '' : ''
" "
> >
<div class="p-4"> <div class="p-4 truncate">
<label <label
class="font-semibold text-secondaryDark" class="font-semibold text-secondaryDark"
:class="{ 'cursor-pointer': compact && team.myRole === 'OWNER' }" :class="{ 'cursor-pointer': compact && team.myRole === 'OWNER' }"
@@ -131,6 +131,7 @@
<HoppSmartConfirmModal <HoppSmartConfirmModal
:show="confirmRemove" :show="confirmRemove"
:title="t('confirm.remove_team')" :title="t('confirm.remove_team')"
:loading-state="loading"
@hide-modal="confirmRemove = false" @hide-modal="confirmRemove = false"
@resolve="deleteTeam()" @resolve="deleteTeam()"
/> />
@@ -161,6 +162,8 @@ import IconMoreVertical from "~icons/lucide/more-vertical"
import IconUserX from "~icons/lucide/user-x" import IconUserX from "~icons/lucide/user-x"
import IconUserPlus from "~icons/lucide/user-plus" import IconUserPlus from "~icons/lucide/user-plus"
import IconTrash2 from "~icons/lucide/trash-2" import IconTrash2 from "~icons/lucide/trash-2"
import { useService } from "dioc/vue"
import { WorkspaceService } from "~/services/workspace.service"
const t = useI18n() const t = useI18n()
@@ -173,6 +176,7 @@ const props = defineProps<{
const emit = defineEmits<{ const emit = defineEmits<{
(e: "edit-team"): void (e: "edit-team"): void
(e: "invite-team"): void (e: "invite-team"): void
(e: "refetch-teams"): void
}>() }>()
const toast = useToast() const toast = useToast()
@@ -180,7 +184,12 @@ const toast = useToast()
const confirmRemove = ref(false) const confirmRemove = ref(false)
const confirmExit = ref(false) const confirmExit = ref(false)
const loading = ref(false)
const workspaceService = useService(WorkspaceService)
const deleteTeam = () => { const deleteTeam = () => {
loading.value = true
pipe( pipe(
backendDeleteTeam(props.teamID), backendDeleteTeam(props.teamID),
TE.match( TE.match(
@@ -188,9 +197,25 @@ const deleteTeam = () => {
// TODO: Better errors ? We know the possible errors now // TODO: Better errors ? We know the possible errors now
toast.error(`${t("error.something_went_wrong")}`) toast.error(`${t("error.something_went_wrong")}`)
console.error(err) console.error(err)
loading.value = false
confirmRemove.value = false
}, },
() => { () => {
toast.success(`${t("team.deleted")}`) toast.success(`${t("team.deleted")}`)
loading.value = false
emit("refetch-teams")
const currentWorkspace = workspaceService.currentWorkspace.value
// If the current workspace is the deleted workspace, change the workspace to personal
if (
currentWorkspace.type === "team" &&
currentWorkspace.teamID === props.teamID
) {
workspaceService.changeWorkspace({ type: "personal" })
}
confirmRemove.value = false
} }
) )
)() // Tasks (and TEs) are lazy, so call the function returned )() // Tasks (and TEs) are lazy, so call the function returned

View File

@@ -4,6 +4,7 @@
<HoppButtonSecondary <HoppButtonSecondary
:label="`${t('team.create_new')}`" :label="`${t('team.create_new')}`"
outline outline
:icon="IconPlus"
@click="displayModalAdd(true)" @click="displayModalAdd(true)"
/> />
<div v-if="loading" class="flex flex-col items-center justify-center"> <div v-if="loading" class="flex flex-col items-center justify-center">
@@ -16,13 +17,6 @@
:alt="`${t('empty.teams')}`" :alt="`${t('empty.teams')}`"
:text="`${t('empty.teams')}`" :text="`${t('empty.teams')}`"
> >
<template #body>
<HoppButtonSecondary
:label="`${t('team.create_new')}`"
filled
@click="displayModalAdd(true)"
/>
</template>
</HoppSmartPlaceholder> </HoppSmartPlaceholder>
<div <div
v-else-if="!loading" v-else-if="!loading"
@@ -39,6 +33,7 @@
:compact="modal" :compact="modal"
@edit-team="editTeam(team, team.id)" @edit-team="editTeam(team, team.id)"
@invite-team="inviteTeam(team, team.id)" @invite-team="inviteTeam(team, team.id)"
@refetch-teams="refetchTeams"
/> />
</div> </div>
<div v-if="!loading && adapterError" class="flex flex-col items-center"> <div v-if="!loading && adapterError" class="flex flex-col items-center">
@@ -76,6 +71,7 @@ import { useReadonlyStream } from "@composables/stream"
import { useColorMode } from "@composables/theming" import { useColorMode } from "@composables/theming"
import { WorkspaceService } from "~/services/workspace.service" import { WorkspaceService } from "~/services/workspace.service"
import { useService } from "dioc/vue" import { useService } from "dioc/vue"
import IconPlus from "~icons/lucide/plus"
const t = useI18n() const t = useI18n()

View File

@@ -59,14 +59,18 @@
/> />
</div> </div>
<div <div
v-if="!loading && teamListAdapterError" v-else-if="teamListAdapterError"
class="flex flex-col items-center py-4" class="flex flex-col items-center py-4"
> >
<icon-lucide-help-circle class="svg-icons mb-4" /> <icon-lucide-help-circle class="svg-icons mb-4" />
{{ t("error.something_went_wrong") }} {{ t("error.something_went_wrong") }}
</div> </div>
</div> </div>
<TeamsAdd :show="showModalAdd" @hide-modal="displayModalAdd(false)" /> <TeamsAdd
:show="showModalAdd"
:switch-workspace-after-creation="true"
@hide-modal="displayModalAdd(false)"
/>
</div> </div>
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
@@ -81,7 +85,7 @@ import { useColorMode } from "@composables/theming"
import { GetMyTeamsQuery } from "~/helpers/backend/graphql" import { GetMyTeamsQuery } from "~/helpers/backend/graphql"
import IconDone from "~icons/lucide/check" import IconDone from "~icons/lucide/check"
import { useLocalState } from "~/newstore/localstate" import { useLocalState } from "~/newstore/localstate"
import { defineActionHandler } from "~/helpers/actions" import { defineActionHandler, invokeAction } from "~/helpers/actions"
import { WorkspaceService } from "~/services/workspace.service" import { WorkspaceService } from "~/services/workspace.service"
import { useService } from "dioc/vue" import { useService } from "dioc/vue"
import { useElementVisibility, useIntervalFn } from "@vueuse/core" import { useElementVisibility, useIntervalFn } from "@vueuse/core"
@@ -154,6 +158,7 @@ const switchToTeamWorkspace = (team: GetMyTeamsQuery["myTeams"][number]) => {
teamID: team.id, teamID: team.id,
teamName: team.name, teamName: team.name,
type: "team", type: "team",
role: team.myRole,
}) })
} }
@@ -169,11 +174,14 @@ watch(
(user) => { (user) => {
if (!user) { if (!user) {
switchToPersonalWorkspace() switchToPersonalWorkspace()
teamListadapter.dispose()
} }
} }
) )
const displayModalAdd = (shouldDisplay: boolean) => { const displayModalAdd = (shouldDisplay: boolean) => {
if (!currentUser.value) return invokeAction("modals.login.toggle")
showModalAdd.value = shouldDisplay showModalAdd.value = shouldDisplay
teamListadapter.fetchList() teamListadapter.fetchList()
} }

View File

@@ -68,6 +68,9 @@ type CodeMirrorOptions = {
// callback on editor update // callback on editor update
onUpdate?: (view: ViewUpdate) => void onUpdate?: (view: ViewUpdate) => void
// callback on view initialization
onInit?: (view: EditorView) => void
} }
const hoppCompleterExt = (completer: Completer): Extension => { const hoppCompleterExt = (completer: Completer): Extension => {
@@ -208,7 +211,9 @@ export function useCodemirror(
el: Ref<any | null>, el: Ref<any | null>,
value: Ref<string | undefined>, value: Ref<string | undefined>,
options: CodeMirrorOptions options: CodeMirrorOptions
): { cursor: Ref<{ line: number; ch: number }> } { ): {
cursor: Ref<{ line: number; ch: number }>
} {
const { subscribeToStream } = useStreamSubscriber() const { subscribeToStream } = useStreamSubscriber()
// Set default value for contextMenuEnabled if not provided // Set default value for contextMenuEnabled if not provided
@@ -383,6 +388,8 @@ export function useCodemirror(
extensions, extensions,
}), }),
}) })
options.onInit?.(view.value)
} }
onMounted(() => { onMounted(() => {

View File

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

View File

@@ -868,6 +868,38 @@ const samples = [
requestVariables: [], requestVariables: [],
}), }),
}, },
{
command: `curl --location 'https://api.example.net/id/1164/requests' \
--header 'Accept: application/vnd.test-data.v2.1+json' \
--header 'Content-Type: application/x-www-form-urlencoded' \
--data-urlencode 'data={"type":"test","typeId":"101"}' \
--data-urlencode 'data2={"type":"test2","typeId":"123"}'`,
response: makeRESTRequest({
method: "POST",
name: "Untitled",
endpoint: "https://api.example.net/id/1164/requests",
auth: {
authType: "inherit",
authActive: true,
},
body: {
contentType: "application/x-www-form-urlencoded",
body: `data: {"type":"test","typeId":"101"}
data2: {"type":"test2","typeId":"123"}`,
},
params: [],
headers: [
{
active: true,
key: "Accept",
value: "application/vnd.test-data.v2.1+json",
},
],
preRequestScript: "",
testScript: "",
requestVariables: [],
}),
},
] ]
describe("Parse curl command to Hopp REST Request", () => { describe("Parse curl command to Hopp REST Request", () => {

View File

@@ -33,7 +33,27 @@ export const parseCurlCommand = (curlCommand: string) => {
// const compressed = !!parsedArguments.compressed // const compressed = !!parsedArguments.compressed
curlCommand = preProcessCurlCommand(curlCommand) curlCommand = preProcessCurlCommand(curlCommand)
const parsedArguments = parser(curlCommand)
const args: parser.Arguments = parser(curlCommand)
const parsedArguments = pipe(
args,
O.fromPredicate(
(args) =>
objHasProperty("dataUrlencode", "string")(args) ||
objHasProperty("dataUrlencode", "object")(args)
),
O.map((args) => {
const urlEncodedData: string[] = Array.isArray(args.dataUrlencode)
? args.dataUrlencode
: [args.dataUrlencode]
const data = A.map(decodeURIComponent)(urlEncodedData)
return { ...args, d: data }
}),
O.getOrElse(() => args)
)
const headerObject = getHeaders(parsedArguments) const headerObject = getHeaders(parsedArguments)
const { headers } = headerObject const { headers } = headerObject

View File

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

View File

@@ -1,15 +1,19 @@
import { pipe, flow } from "fp-ts/function" import {
import * as TE from "fp-ts/TaskEither" HoppCollection,
HoppRESTRequest,
getDefaultGQLRequest,
getDefaultRESTRequest,
translateToNewRESTCollection,
} from "@hoppscotch/data"
import * as A from "fp-ts/Array"
import * as O from "fp-ts/Option" import * as O from "fp-ts/Option"
import * as RA from "fp-ts/ReadonlyArray" import * as RA from "fp-ts/ReadonlyArray"
import * as A from "fp-ts/Array" import * as TE from "fp-ts/TaskEither"
import { translateToNewRESTCollection, HoppCollection } from "@hoppscotch/data" import { flow, pipe } from "fp-ts/function"
import { IMPORTER_INVALID_FILE_FORMAT } from "." import { HoppGQLRequest, translateToNewGQLCollection } from "@hoppscotch/data"
import { safeParseJSON } from "~/helpers/functional/json" import { safeParseJSON } from "~/helpers/functional/json"
import { translateToNewGQLCollection } from "@hoppscotch/data" import { IMPORTER_INVALID_FILE_FORMAT } from "."
import { entityReference } from "verzod"
import { z } from "zod"
export const hoppRESTImporter = (content: string[]) => export const hoppRESTImporter = (content: string[]) =>
pipe( pipe(
@@ -32,8 +36,24 @@ export const hoppRESTImporter = (content: string[]) =>
* else translate it into one. * else translate it into one.
*/ */
const validateCollection = (collection: unknown) => { const validateCollection = (collection: unknown) => {
const result = entityReference(HoppCollection).safeParse(collection) const collectionSchemaParsedResult = HoppCollection.safeParse(collection)
if (result.success) return O.some(result.data)
if (collectionSchemaParsedResult.type === "ok") {
const requests = collectionSchemaParsedResult.value.requests.map(
(request) => {
const requestSchemaParsedResult = HoppRESTRequest.safeParse(request)
return requestSchemaParsedResult.type === "ok"
? requestSchemaParsedResult.value
: getDefaultRESTRequest()
}
)
return O.some({
...collectionSchemaParsedResult.value,
requests,
})
}
return O.some(translateToNewRESTCollection(collection)) return O.some(translateToNewRESTCollection(collection))
} }
@@ -64,9 +84,24 @@ export const hoppGQLImporter = (content: string) =>
* @returns the collection if it is valid, else a translated version of the collection * @returns the collection if it is valid, else a translated version of the collection
*/ */
export const validateGQLCollection = (collection: unknown) => { export const validateGQLCollection = (collection: unknown) => {
const result = z.array(entityReference(HoppCollection)).safeParse(collection) const collectionSchemaParsedResult = HoppCollection.safeParse(collection)
if (result.success) return O.some(result.data) if (collectionSchemaParsedResult.type === "ok") {
const requests = collectionSchemaParsedResult.value.requests.map(
(request) => {
const requestSchemaParsedResult = HoppGQLRequest.safeParse(request)
return requestSchemaParsedResult.type === "ok"
? requestSchemaParsedResult.value
: getDefaultGQLRequest()
}
)
return O.some({
...collectionSchemaParsedResult.value,
requests,
})
}
return O.some(translateToNewGQLCollection(collection)) return O.some(translateToNewGQLCollection(collection))
} }

View File

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

View File

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

View File

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

View File

@@ -50,6 +50,7 @@ export default class TeamListAdapter {
} }
public dispose() { public dispose() {
this.teamList$.next([])
this.isDispose = true this.isDispose = true
clearTimeout(this.timeoutHandle as any) clearTimeout(this.timeoutHandle as any)
this.timeoutHandle = null this.timeoutHandle = null

View File

@@ -1,4 +1,14 @@
import { ref } from "vue" import {
HoppRESTAuth,
HoppRESTHeader,
HoppRESTRequest,
getDefaultRESTRequest,
} from "@hoppscotch/data"
import axios from "axios"
import { Service } from "dioc"
import * as E from "fp-ts/Either"
import { Ref, ref } from "vue"
import { runGQLQuery } from "../backend/GQLClient" import { runGQLQuery } from "../backend/GQLClient"
import { import {
GetCollectionChildrenDocument, GetCollectionChildrenDocument,
@@ -7,14 +17,10 @@ import {
GetSingleRequestDocument, GetSingleRequestDocument,
} from "../backend/graphql" } from "../backend/graphql"
import { TeamCollection } from "./TeamCollection" import { TeamCollection } from "./TeamCollection"
import { HoppRESTAuth, HoppRESTHeader } from "@hoppscotch/data"
import * as E from "fp-ts/Either" import { platform } from "~/platform"
import { HoppInheritedProperty } from "../types/HoppInheritedProperties" import { HoppInheritedProperty } from "../types/HoppInheritedProperties"
import { TeamRequest } from "./TeamRequest" import { TeamRequest } from "./TeamRequest"
import { Service } from "dioc"
import axios from "axios"
import { Ref } from "vue"
type CollectionSearchMeta = { type CollectionSearchMeta = {
isSearchResult?: boolean isSearchResult?: boolean
@@ -149,12 +155,21 @@ function convertToTeamTree(
if (isAlreadyInserted) return if (isAlreadyInserted) return
if (parentCollection) { if (parentCollection) {
const requestSchemaParsedResult = HoppRESTRequest.safeParse(
request.request
)
const effectiveRequest =
requestSchemaParsedResult.type === "ok"
? requestSchemaParsedResult.value
: getDefaultRESTRequest()
parentCollection.requests = parentCollection.requests || [] parentCollection.requests = parentCollection.requests || []
parentCollection.requests.push({ parentCollection.requests.push({
id: request.id, id: request.id,
collectionID: request.collectionID, collectionID: request.collectionID,
title: request.title, title: request.title,
request: request.request, request: effectiveRequest,
}) })
} }
}) })
@@ -186,7 +201,7 @@ export class TeamSearchService extends Service {
expandingCollections: Ref<string[]> = ref([]) expandingCollections: Ref<string[]> = ref([])
expandedCollections: Ref<string[]> = ref([]) expandedCollections: Ref<string[]> = ref([])
// FUTURE-TODO: ideally this should return the search results / formatted results instead of directly manipulating the result set // TODO: ideally this should return the search results / formatted results instead of directly manipulating the result set
// eg: do the spotlight formatting in the spotlight searcher and not here // eg: do the spotlight formatting in the spotlight searcher and not here
searchTeams = async (query: string, teamID: string) => { searchTeams = async (query: string, teamID: string) => {
if (!query.length) { if (!query.length) {
@@ -199,16 +214,16 @@ export class TeamSearchService extends Service {
this.searchResultsRequests = {} this.searchResultsRequests = {}
this.expandedCollections.value = [] this.expandedCollections.value = []
const axiosPlatformConfig = platform.auth.axiosPlatformConfig?.() ?? {}
try { try {
const searchResponse = await axios.get( const searchResponse = await axios.get(
`${ `${
this.endpoint this.endpoint
}/team-collection/search/${teamID}?searchQuery=${encodeURIComponent( }/team-collection/search/${teamID}?searchQuery=${encodeURIComponent(
query query
)}}`, )}`,
{ axiosPlatformConfig
withCredentials: true,
}
) )
if (searchResponse.status !== 200) { if (searchResponse.status !== 200) {

View File

@@ -96,7 +96,7 @@ export const getComputedAuthHeaders = (
}) })
} else if (request.auth.authType === "api-key") { } else if (request.auth.authType === "api-key") {
const { key, addTo } = request.auth const { key, addTo } = request.auth
if (addTo === "Headers" && key) { if (addTo === "HEADERS" && key) {
headers.push({ headers.push({
active: true, active: true,
key: parseTemplateString(key, envVars), key: parseTemplateString(key, envVars),
@@ -274,7 +274,7 @@ function getFinalBodyFromRequest(
if (request.body.contentType === "application/x-www-form-urlencoded") { if (request.body.contentType === "application/x-www-form-urlencoded") {
const parsedBodyRecord = pipe( const parsedBodyRecord = pipe(
request.body.body, request.body.body ?? "",
parseRawKeyValueEntriesE, parseRawKeyValueEntriesE,
E.map( E.map(
flow( flow(
@@ -311,7 +311,7 @@ function getFinalBodyFromRequest(
if (request.body.contentType === "multipart/form-data") { if (request.body.contentType === "multipart/form-data") {
return pipe( return pipe(
request.body.body, request.body.body ?? [],
A.filter((x) => (x.key !== "" || x.isFile) && x.active), // Remove empty keys A.filter((x) => (x.key !== "" || x.isFile) && x.active), // Remove empty keys
// Sort files down // Sort files down

View File

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

View File

@@ -1,5 +1,5 @@
import { HoppModule } from "." import { HoppModule } from "."
import { Container, Service } from "dioc" import { Container, ServiceClassInstance } from "dioc"
import { diocPlugin } from "dioc/vue" import { diocPlugin } from "dioc/vue"
import { DebugService } from "~/services/debug.service" import { DebugService } from "~/services/debug.service"
import { platform } from "~/platform" import { platform } from "~/platform"
@@ -22,7 +22,7 @@ if (import.meta.env.DEV) {
* services. Please use `useService` if within components or try to convert your * services. Please use `useService` if within components or try to convert your
* legacy subsystem into a service if possible. * legacy subsystem into a service if possible.
*/ */
export function getService<T extends typeof Service<any> & { ID: string }>( export function getService<T extends ServiceClassInstance<any>>(
service: T service: T
): InstanceType<T> { ): InstanceType<T> {
return serviceContainer.bind(service) return serviceContainer.bind(service)
@@ -30,11 +30,10 @@ export function getService<T extends typeof Service<any> & { ID: string }>(
export default <HoppModule>{ export default <HoppModule>{
onVueAppInit(app) { onVueAppInit(app) {
// TODO: look into this
// @ts-expect-error Something weird with Vue versions
app.use(diocPlugin, { app.use(diocPlugin, {
container: serviceContainer, container: serviceContainer,
}) })
for (const service of platform.addedServices ?? []) { for (const service of platform.addedServices ?? []) {
serviceContainer.bind(service) serviceContainer.bind(service)
} }

View File

@@ -1,9 +1,8 @@
import { pluck, distinctUntilChanged } from "rxjs/operators"
import { cloneDeep, defaultsDeep, has } from "lodash-es" import { cloneDeep, defaultsDeep, has } from "lodash-es"
import { Observable } from "rxjs" import { Observable } from "rxjs"
import { distinctUntilChanged, pluck } from "rxjs/operators"
import DispatchingStore, { defineDispatchers } from "./DispatchingStore"
import type { KeysMatching } from "~/types/ts-utils" import type { KeysMatching } from "~/types/ts-utils"
import DispatchingStore, { defineDispatchers } from "./DispatchingStore"
export const HoppBgColors = ["system", "light", "dark", "black"] as const export const HoppBgColors = ["system", "light", "dark", "black"] as const
@@ -65,6 +64,8 @@ export type SettingsDef = {
SIDEBAR: boolean SIDEBAR: boolean
SIDEBAR_ON_LEFT: boolean SIDEBAR_ON_LEFT: boolean
COLUMN_LAYOUT: boolean COLUMN_LAYOUT: boolean
HAS_OPENED_SPOTLIGHT: boolean
} }
export const getDefaultSettings = (): SettingsDef => ({ export const getDefaultSettings = (): SettingsDef => ({
@@ -91,7 +92,8 @@ export const getDefaultSettings = (): SettingsDef => ({
cookie: true, cookie: true,
}, },
CURRENT_INTERCEPTOR_ID: "browser", // TODO: Allow the platform definition to take this place // Set empty because interceptor module will set the default value
CURRENT_INTERCEPTOR_ID: "",
// TODO: Interceptor related settings should move under the interceptor systems // TODO: Interceptor related settings should move under the interceptor systems
PROXY_URL: "https://proxy.hoppscotch.io/", PROXY_URL: "https://proxy.hoppscotch.io/",
@@ -109,6 +111,8 @@ export const getDefaultSettings = (): SettingsDef => ({
SIDEBAR: true, SIDEBAR: true,
SIDEBAR_ON_LEFT: false, SIDEBAR_ON_LEFT: false,
COLUMN_LAYOUT: true, COLUMN_LAYOUT: true,
HAS_OPENED_SPOTLIGHT: false,
}) })
type ApplySettingPayload = { type ApplySettingPayload = {

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