gemercheung 2 年之前
当前提交
5a18f03643

+ 14 - 0
.env.development

@@ -0,0 +1,14 @@
+SOCKET_NAME='socket-test'
+SOCKET_PORT=6666
+SOCKET_PATH="/ws-sync"
+REDIS_HOST=127.0.0.1
+REDIS_PORT=6379
+REDIS_PASSWORD=redis9394
+REDIS_DB=0
+REDIS_ADAPTER_DB=8
+REDIS_PREFIX=chat
+WATCH_USER=4dage
+WATCH_PASSWORD=4dage168.
+TEMP_ROOM_VALID=3600
+DOTENV_SILENT=true
+DOTENV_FLOW_SILENT=true

+ 14 - 0
.env.production

@@ -0,0 +1,14 @@
+SOCKET_NAME='socket-test'
+SOCKET_PORT=6666
+SOCKET_PATH="/ws-sync"
+REDIS_HOST=127.0.0.1
+REDIS_PORT=6379
+REDIS_PASSWORD=redis9394
+REDIS_DB=0
+REDIS_ADAPTER_DB=8
+REDIS_PREFIX=chat
+WATCH_USER=4dage
+WATCH_PASSWORD=4dage168.
+TEMP_ROOM_VALID=3600
+DOTENV_SILENT=true
+DOTENV_FLOW_SILENT=true

+ 25 - 0
.eslintrc.js

@@ -0,0 +1,25 @@
+module.exports = {
+  parser: '@typescript-eslint/parser',
+  parserOptions: {
+    project: 'tsconfig.json',
+    tsconfigRootDir: __dirname,
+    sourceType: 'module',
+  },
+  plugins: ['@typescript-eslint/eslint-plugin'],
+  extends: [
+    'plugin:@typescript-eslint/recommended',
+    'plugin:prettier/recommended',
+  ],
+  root: true,
+  env: {
+    node: true,
+    jest: true,
+  },
+  ignorePatterns: ['.eslintrc.js'],
+  rules: {
+    '@typescript-eslint/interface-name-prefix': 'off',
+    '@typescript-eslint/explicit-function-return-type': 'off',
+    '@typescript-eslint/explicit-module-boundary-types': 'off',
+    '@typescript-eslint/no-explicit-any': 'off',
+  },
+};

+ 35 - 0
.gitignore

@@ -0,0 +1,35 @@
+# compiled output
+/dist
+/node_modules
+
+# Logs
+logs
+*.log
+npm-debug.log*
+pnpm-debug.log*
+yarn-debug.log*
+yarn-error.log*
+lerna-debug.log*
+
+# OS
+.DS_Store
+
+# Tests
+/coverage
+/.nyc_output
+
+# IDEs and editors
+/.idea
+.project
+.classpath
+.c9/
+*.launch
+.settings/
+*.sublime-workspace
+
+# IDE - VSCode
+.vscode/*
+!.vscode/settings.json
+!.vscode/tasks.json
+!.vscode/launch.json
+!.vscode/extensions.json

+ 4 - 0
.prettierrc

@@ -0,0 +1,4 @@
+{
+  "singleQuote": true,
+  "trailingComma": "all"
+}

+ 73 - 0
README.md

@@ -0,0 +1,73 @@
+<p align="center">
+  <a href="http://nestjs.com/" target="blank"><img src="https://nestjs.com/img/logo-small.svg" width="200" alt="Nest Logo" /></a>
+</p>
+
+[circleci-image]: https://img.shields.io/circleci/build/github/nestjs/nest/master?token=abc123def456
+[circleci-url]: https://circleci.com/gh/nestjs/nest
+
+  <p align="center">A progressive <a href="http://nodejs.org" target="_blank">Node.js</a> framework for building efficient and scalable server-side applications.</p>
+    <p align="center">
+<a href="https://www.npmjs.com/~nestjscore" target="_blank"><img src="https://img.shields.io/npm/v/@nestjs/core.svg" alt="NPM Version" /></a>
+<a href="https://www.npmjs.com/~nestjscore" target="_blank"><img src="https://img.shields.io/npm/l/@nestjs/core.svg" alt="Package License" /></a>
+<a href="https://www.npmjs.com/~nestjscore" target="_blank"><img src="https://img.shields.io/npm/dm/@nestjs/common.svg" alt="NPM Downloads" /></a>
+<a href="https://circleci.com/gh/nestjs/nest" target="_blank"><img src="https://img.shields.io/circleci/build/github/nestjs/nest/master" alt="CircleCI" /></a>
+<a href="https://coveralls.io/github/nestjs/nest?branch=master" target="_blank"><img src="https://coveralls.io/repos/github/nestjs/nest/badge.svg?branch=master#9" alt="Coverage" /></a>
+<a href="https://discord.gg/G7Qnnhy" target="_blank"><img src="https://img.shields.io/badge/discord-online-brightgreen.svg" alt="Discord"/></a>
+<a href="https://opencollective.com/nest#backer" target="_blank"><img src="https://opencollective.com/nest/backers/badge.svg" alt="Backers on Open Collective" /></a>
+<a href="https://opencollective.com/nest#sponsor" target="_blank"><img src="https://opencollective.com/nest/sponsors/badge.svg" alt="Sponsors on Open Collective" /></a>
+  <a href="https://paypal.me/kamilmysliwiec" target="_blank"><img src="https://img.shields.io/badge/Donate-PayPal-ff3f59.svg"/></a>
+    <a href="https://opencollective.com/nest#sponsor"  target="_blank"><img src="https://img.shields.io/badge/Support%20us-Open%20Collective-41B883.svg" alt="Support us"></a>
+  <a href="https://twitter.com/nestframework" target="_blank"><img src="https://img.shields.io/twitter/follow/nestframework.svg?style=social&label=Follow"></a>
+</p>
+  <!--[![Backers on Open Collective](https://opencollective.com/nest/backers/badge.svg)](https://opencollective.com/nest#backer)
+  [![Sponsors on Open Collective](https://opencollective.com/nest/sponsors/badge.svg)](https://opencollective.com/nest#sponsor)-->
+
+## Description
+
+[Nest](https://github.com/nestjs/nest) framework TypeScript starter repository.
+
+## Installation
+
+```bash
+$ pnpm install
+```
+
+## Running the app
+
+```bash
+# development
+$ pnpm run start
+
+# watch mode
+$ pnpm run start:dev
+
+# production mode
+$ pnpm run start:prod
+```
+
+## Test
+
+```bash
+# unit tests
+$ pnpm run test
+
+# e2e tests
+$ pnpm run test:e2e
+
+# test coverage
+$ pnpm run test:cov
+```
+
+## Support
+
+Nest is an MIT-licensed open source project. It can grow thanks to the sponsors and support by the amazing backers. If you'd like to join them, please [read more here](https://docs.nestjs.com/support).
+
+## Stay in touch
+
+- Author - [Kamil Myśliwiec](https://kamilmysliwiec.com)
+- Website - [https://nestjs.com](https://nestjs.com/)
+- Twitter - [@nestframework](https://twitter.com/nestframework)
+
+## License
+
+Nest is [MIT licensed](LICENSE).

+ 15 - 0
log/.46f0af219249542fdc264b9b0460d72fee271bbf-audit.json

@@ -0,0 +1,15 @@
+{
+    "keep": {
+        "days": true,
+        "amount": 30
+    },
+    "auditLog": "D:\\codes\\daikan\\socket-server-v4\\log\\.46f0af219249542fdc264b9b0460d72fee271bbf-audit.json",
+    "files": [
+        {
+            "date": 1676369333465,
+            "name": "D:\\codes\\daikan\\socket-server-v4\\log\\2023-02-14.log",
+            "hash": "2ab3113bffdc301303be9f0789c1993a5dd376f54ab83789f1cafe128dbac5a8"
+        }
+    ],
+    "hashType": "sha256"
+}

+ 8 - 0
nest-cli.json

@@ -0,0 +1,8 @@
+{
+  "$schema": "https://json.schemastore.org/nest-cli",
+  "collection": "@nestjs/schematics",
+  "sourceRoot": "src",
+  "compilerOptions": {
+    "deleteOutDir": true
+  }
+}

+ 87 - 0
package.json

@@ -0,0 +1,87 @@
+{
+  "name": "socket-server-v4",
+  "version": "0.0.1",
+  "description": "",
+  "author": "",
+  "private": true,
+  "license": "UNLICENSED",
+  "scripts": {
+    "build": "nest build",
+    "format": "prettier --write \"src/**/*.ts\" \"test/**/*.ts\"",
+    "start": "nest start",
+    "start:dev": "nest start --watch",
+    "start:debug": "nest start --debug --watch",
+    "start:prod": "node dist/main",
+    "lint": "eslint \"{src,apps,libs,test}/**/*.ts\" --fix",
+    "test": "jest",
+    "test:watch": "jest --watch",
+    "test:cov": "jest --coverage",
+    "test:debug": "node --inspect-brk -r tsconfig-paths/register -r ts-node/register node_modules/.bin/jest --runInBand",
+    "test:e2e": "jest --config ./test/jest-e2e.json"
+  },
+  "dependencies": {
+    "@liaoliaots/nestjs-redis": "^9.0.5",
+    "@nestjs/common": "^9.0.0",
+    "@nestjs/config": "^2.3.1",
+    "@nestjs/core": "^9.0.0",
+    "@nestjs/platform-express": "^9.0.0",
+    "@nestjs/platform-socket.io": "^9.3.8",
+    "@nestjs/throttler": "^4.0.0",
+    "@nestjs/websockets": "^9.3.8",
+    "@socket.io/admin-ui": "^0.5.1",
+    "@socket.io/redis-adapter": "^8.1.0",
+    "bcrypt": "^5.1.0",
+    "chalk": "^4",
+    "ioredis": "^5.3.1",
+    "logform": "^2.5.1",
+    "nest-winston": "^1.8.0",
+    "nestjs-throttler-storage-redis": "^0.3.0",
+    "redis": "^4.6.4",
+    "reflect-metadata": "^0.1.13",
+    "rxjs": "^7.2.0",
+    "socket.io": "^4.6.0",
+    "socket.io-msgpack-parser": "^3.0.2",
+    "winston": "^3.8.2",
+    "winston-daily-rotate-file": "^4.7.1"
+  },
+  "devDependencies": {
+    "@nestjs/cli": "^9.0.0",
+    "@nestjs/schematics": "^9.0.0",
+    "@nestjs/testing": "^9.0.0",
+    "@types/express": "^4.17.13",
+    "@types/jest": "29.2.4",
+    "@types/node": "18.11.18",
+    "@types/supertest": "^2.0.11",
+    "@typescript-eslint/eslint-plugin": "^5.0.0",
+    "@typescript-eslint/parser": "^5.0.0",
+    "eslint": "^8.0.1",
+    "eslint-config-prettier": "^8.3.0",
+    "eslint-plugin-prettier": "^4.0.0",
+    "jest": "29.3.1",
+    "prettier": "^2.3.2",
+    "source-map-support": "^0.5.20",
+    "supertest": "^6.1.3",
+    "ts-jest": "29.0.3",
+    "ts-loader": "^9.2.3",
+    "ts-node": "^10.0.0",
+    "tsconfig-paths": "4.1.1",
+    "typescript": "^4.7.4"
+  },
+  "jest": {
+    "moduleFileExtensions": [
+      "js",
+      "json",
+      "ts"
+    ],
+    "rootDir": "src",
+    "testRegex": ".*\\.spec\\.ts$",
+    "transform": {
+      "^.+\\.(t|j)s$": "ts-jest"
+    },
+    "collectCoverageFrom": [
+      "**/*.(t|j)s"
+    ],
+    "coverageDirectory": "../coverage",
+    "testEnvironment": "node"
+  }
+}

文件差异内容过多而无法显示
+ 5689 - 0
pnpm-lock.yaml


+ 22 - 0
src/app.controller.spec.ts

@@ -0,0 +1,22 @@
+import { Test, TestingModule } from '@nestjs/testing';
+import { AppController } from './app.controller';
+import { AppService } from './app.service';
+
+describe('AppController', () => {
+  let appController: AppController;
+
+  beforeEach(async () => {
+    const app: TestingModule = await Test.createTestingModule({
+      controllers: [AppController],
+      providers: [AppService],
+    }).compile();
+
+    appController = app.get<AppController>(AppController);
+  });
+
+  describe('root', () => {
+    it('should return "Hello World!"', () => {
+      expect(appController.getHello()).toBe('Hello World!');
+    });
+  });
+});

+ 12 - 0
src/app.controller.ts

@@ -0,0 +1,12 @@
+import { Controller, Get } from '@nestjs/common';
+import { AppService } from './app.service';
+
+@Controller()
+export class AppController {
+  constructor(private readonly appService: AppService) {}
+
+  @Get()
+  getHello(): string {
+    return this.appService.getHello();
+  }
+}

+ 37 - 0
src/app.module.ts

@@ -0,0 +1,37 @@
+import { Module } from '@nestjs/common';
+import { AppController } from './app.controller';
+import { AppService } from './app.service';
+import { ChatGateway } from './chat/chat.gateway';
+import { RedisModule, RedisService } from '@liaoliaots/nestjs-redis';
+import { ThrottlerModule } from '@nestjs/throttler';
+import { ThrottlerStorageRedisService } from 'nestjs-throttler-storage-redis';
+import { ConfigModule } from '@nestjs/config';
+
+const config = ConfigModule.forRoot({
+  envFilePath: ['.env.development', '.env.production'],
+});
+
+const url = `redis://:${process.env.REDIS_PASSWORD}@${process.env.REDIS_HOST}:${process.env.REDIS_PORT}/${process.env.REDIS_DB}`;
+
+const redisMo = RedisModule.forRoot({
+  config: {
+    url: url,
+  },
+});
+const storeThor = ThrottlerModule.forRootAsync({
+  useFactory(redisService: RedisService) {
+    const redis = redisService.getClient();
+    return {
+      ttl: 60,
+      limit: 600,
+      storage: new ThrottlerStorageRedisService(redis),
+    };
+  },
+  inject: [RedisService],
+});
+@Module({
+  imports: [config, redisMo, storeThor],
+  controllers: [AppController],
+  providers: [AppService, ChatGateway],
+})
+export class AppModule {}

+ 8 - 0
src/app.service.ts

@@ -0,0 +1,8 @@
+import { Injectable } from '@nestjs/common';
+
+@Injectable()
+export class AppService {
+  getHello(): string {
+    return 'Hello World!';
+  }
+}

+ 18 - 0
src/chat/chat.gateway.spec.ts

@@ -0,0 +1,18 @@
+import { Test, TestingModule } from '@nestjs/testing';
+import { ChatGateway } from './chat.gateway';
+
+describe('ChatGateway', () => {
+  let gateway: ChatGateway;
+
+  beforeEach(async () => {
+    const module: TestingModule = await Test.createTestingModule({
+      providers: [ChatGateway],
+    }).compile();
+
+    gateway = module.get<ChatGateway>(ChatGateway);
+  });
+
+  it('should be defined', () => {
+    expect(gateway).toBeDefined();
+  });
+});

+ 63 - 0
src/chat/chat.gateway.ts

@@ -0,0 +1,63 @@
+import { InjectRedis } from '@liaoliaots/nestjs-redis';
+import {
+  ConnectedSocket,
+  MessageBody,
+  OnGatewayConnection,
+  OnGatewayDisconnect,
+  OnGatewayInit,
+  SubscribeMessage,
+  WebSocketGateway,
+  WebSocketServer,
+} from '@nestjs/websockets';
+import { Server, Socket } from 'socket.io';
+import Redis from 'ioredis';
+import { instrument } from '@socket.io/admin-ui';
+import * as bcrypt from 'bcrypt';
+
+@WebSocketGateway(8889, {
+  transports: ['websocket'],
+  cors: '*',
+  // namespace: "ws",
+  path: '/ws-sync',
+  // parser: require('socket.io-msgpack-parser'),
+})
+export class ChatGateway
+  implements OnGatewayInit, OnGatewayDisconnect, OnGatewayConnection
+{
+  constructor(@InjectRedis() private readonly redis: Redis) {}
+  @WebSocketServer() server: Server;
+  handleConnection(client: any, ...args: any[]) {
+    // console.log('handleConnection', client, args);
+  }
+  handleDisconnect(client: any) {
+    // console.log('handleDisconnect', client);
+  }
+  afterInit(server: any) {
+    instrument(server, {
+      auth: {
+        type: 'basic',
+        username: '4dage',
+        password: bcrypt.hashSync('4dage168.', 10),
+      },
+      namespaceName: '/watch',
+    });
+  }
+
+  @SubscribeMessage('join')
+  async handleMessage(
+    @MessageBody() message: any,
+    @ConnectedSocket() socket: Socket,
+  ): Promise<void> {
+    console.log('join', message);
+    socket.join(message.roomId);
+    const tt = await this.redis.get('test');
+    console.log('redis', tt);
+    this.server.to(message.roomId).emit('action', { type: 'user-join' });
+
+    // socket.emit('join', { test: true });
+    // socket.broadcast
+    //   .to(message.roomId)
+    //   .emit('action',
+    //    { type: 'user-join',{ test: true } });
+  }
+}

+ 59 - 0
src/logger/config.ts

@@ -0,0 +1,59 @@
+import * as clc from 'chalk';
+import { Format as LogFormat } from 'logform';
+import { inspect } from 'util';
+import { NestLikeConsoleFormatOptions } from 'nest-winston/dist/winston.interfaces';
+import { format } from 'winston';
+
+const nestLikeColorScheme = {
+  info: clc.greenBright,
+  error: clc.red,
+  warn: clc.yellow,
+  debug: clc.magentaBright,
+  verbose: clc.cyanBright,
+};
+
+/**
+ * NestLikeConsoleFormat
+ * {@link https://github.com/gremo/nest-winston/blob/master/winston.utilities.ts}
+ * @param appName AppName
+ * @param options Options
+ */
+export function nestLikeConsoleFormat(
+  appName = 'NestWinston',
+  options?: NestLikeConsoleFormatOptions,
+): LogFormat {
+  return format.printf(
+    ({ context, level, timestamp, message, ms, ...meta }) => {
+      if (typeof timestamp !== 'undefined')
+        // Only format the timestamp to a locale representation if it's ISO 8601 format. Any format
+        // That is not a valid date string will throw, just ignore it (it will be printed as-is).
+        try {
+          if (timestamp === new Date(timestamp).toISOString())
+            // eslint-disable-next-line no-param-reassign
+            timestamp = new Date(timestamp).toLocaleString();
+        } catch (error) {
+          // eslint-disable-next-line no-empty
+        }
+
+      const color =
+        nestLikeColorScheme[level] || ((text: string): string => text);
+
+      const stringifiedMeta = JSON.stringify(meta);
+      const formattedMeta = options?.prettyPrint
+        ? inspect(JSON.parse(stringifiedMeta), { colors: true, depth: null })
+        : stringifiedMeta;
+
+      return (
+        `${color(`[${appName}]`)} ` + // eslint-disable-line prefer-template
+        `${clc.yellow(level.charAt(0).toUpperCase() + level.slice(1))}\t` + // eslint-disable-line prefer-template
+        (typeof timestamp !== 'undefined' ? `${timestamp} ` : '') + // eslint-disable-line prefer-template
+        (typeof context !== 'undefined' // eslint-disable-line prefer-template
+          ? `${clc.yellow('[' + context + ']')} ` // eslint-disable-line prefer-template
+          : '') + // eslint-disable-line prefer-template
+        `${color(message)}` + // eslint-disable-line prefer-template
+        (formattedMeta && formattedMeta !== '{}' ? ` - ${formattedMeta}` : '') + // eslint-disable-line prefer-template
+        (typeof ms !== 'undefined' ? ` ${clc.yellow(ms)}` : '') // eslint-disable-line prefer-template
+      );
+    },
+  );
+}

+ 44 - 0
src/main.ts

@@ -0,0 +1,44 @@
+import { NestFactory } from '@nestjs/core';
+import { AppModule } from './app.module';
+import { RedisIoAdapter } from './redis-io-adapter/redis-io-adapter';
+import { nestLikeConsoleFormat } from './logger/config';
+import { WinstonModule } from 'nest-winston';
+import * as winston from 'winston';
+import 'winston-daily-rotate-file';
+import { join } from 'path';
+
+async function bootstrap() {
+  // const app = await NestFactory.create(AppModule);
+  const app = await NestFactory.create(AppModule, {
+    logger: WinstonModule.createLogger({
+      level: 'debug',
+      transports: [
+        new winston.transports.Console({
+          format: winston.format.combine(
+            winston.format.timestamp({ format: 'DD/MM/YYYY HH:mm:ss' }),
+            winston.format.ms(),
+            nestLikeConsoleFormat('MyApp'),
+          ),
+        }),
+        new winston.transports.DailyRotateFile({
+          format: winston.format.combine(
+            winston.format.timestamp(),
+            winston.format.json(),
+          ),
+          filename: join(__dirname, '..', 'log/%DATE%.log'),
+          datePattern: 'YYYY-MM-DD',
+          zippedArchive: true,
+          maxSize: '20m',
+          maxFiles: '30d',
+          level: 'debug',
+        }),
+      ],
+    }),
+  });
+  const redisIoAdapter = new RedisIoAdapter(app);
+  await redisIoAdapter.connectToRedis();
+  app.useWebSocketAdapter(redisIoAdapter);
+
+  await app.listen(8888);
+}
+bootstrap();

+ 7 - 0
src/redis-io-adapter/redis-io-adapter.spec.ts

@@ -0,0 +1,7 @@
+import { RedisIoAdapter } from './redis-io-adapter';
+
+describe('RedisIoAdapter', () => {
+  it('should be defined', () => {
+    expect(new RedisIoAdapter()).toBeDefined();
+  });
+});

+ 28 - 0
src/redis-io-adapter/redis-io-adapter.ts

@@ -0,0 +1,28 @@
+import { IoAdapter } from '@nestjs/platform-socket.io';
+import { ServerOptions } from 'socket.io';
+import { createAdapter } from '@socket.io/redis-adapter';
+import { createClient } from 'redis';
+
+export class RedisIoAdapter extends IoAdapter {
+  private adapterConstructor: ReturnType<typeof createAdapter>;
+
+  async connectToRedis(): Promise<void> {
+    const url = `redis://:${process.env.REDIS_PASSWORD}@${process.env.REDIS_HOST}:${process.env.REDIS_PORT}/${process.env.REDIS_ADAPTER_DB}`;
+    console.log('url', url);
+    const pubClient = createClient({
+      // url: `redis://:redis9394@127.0.0.1:6379/0`,
+      url: url,
+    });
+    const subClient = pubClient.duplicate();
+
+    await Promise.all([pubClient.connect(), subClient.connect()]);
+
+    this.adapterConstructor = createAdapter(pubClient, subClient);
+  }
+
+  createIOServer(port: number, options?: ServerOptions): any {
+    const server = super.createIOServer(port, options);
+    server.adapter(this.adapterConstructor);
+    return server;
+  }
+}

+ 24 - 0
test/app.e2e-spec.ts

@@ -0,0 +1,24 @@
+import { Test, TestingModule } from '@nestjs/testing';
+import { INestApplication } from '@nestjs/common';
+import * as request from 'supertest';
+import { AppModule } from './../src/app.module';
+
+describe('AppController (e2e)', () => {
+  let app: INestApplication;
+
+  beforeEach(async () => {
+    const moduleFixture: TestingModule = await Test.createTestingModule({
+      imports: [AppModule],
+    }).compile();
+
+    app = moduleFixture.createNestApplication();
+    await app.init();
+  });
+
+  it('/ (GET)', () => {
+    return request(app.getHttpServer())
+      .get('/')
+      .expect(200)
+      .expect('Hello World!');
+  });
+});

+ 9 - 0
test/jest-e2e.json

@@ -0,0 +1,9 @@
+{
+  "moduleFileExtensions": ["js", "json", "ts"],
+  "rootDir": ".",
+  "testEnvironment": "node",
+  "testRegex": ".e2e-spec.ts$",
+  "transform": {
+    "^.+\\.(t|j)s$": "ts-jest"
+  }
+}

+ 4 - 0
tsconfig.build.json

@@ -0,0 +1,4 @@
+{
+  "extends": "./tsconfig.json",
+  "exclude": ["node_modules", "test", "dist", "**/*spec.ts"]
+}

+ 22 - 0
tsconfig.json

@@ -0,0 +1,22 @@
+{
+  "compilerOptions": {
+    "module": "commonjs",
+    "declaration": true,
+    "removeComments": true,
+    "emitDecoratorMetadata": true,
+    "experimentalDecorators": true,
+    "allowSyntheticDefaultImports": true,
+    "moduleResolution": "Node",
+    "target": "es2017",
+    "sourceMap": true,
+    "outDir": "./dist",
+    "baseUrl": "./",
+    "incremental": true,
+    "skipLibCheck": true,
+    "strictNullChecks": false,
+    "noImplicitAny": false,
+    "strictBindCallApply": false,
+    "forceConsistentCasingInFileNames": false,
+    "noFallthroughCasesInSwitch": false
+  }
+}