diff --git a/.github/workflows/dart.yml b/.github/workflows/dart.yml new file mode 100644 index 000000000..3383c717e --- /dev/null +++ b/.github/workflows/dart.yml @@ -0,0 +1,42 @@ +# This workflow uses actions that are not certified by GitHub. +# They are provided by a third-party and are governed by +# separate terms of service, privacy policy, and support +# documentation. + +name: Dart + +on: + push: + branches: [ "main" ] + pull_request: + branches: [ "main" ] + +jobs: + build: + runs-on: ubuntu-latest + + steps: + - uses: actions/checkout@v4 + + # Note: This workflow uses the latest stable version of the Dart SDK. + # You can specify other versions if desired, see documentation here: + # https://github.com/dart-lang/setup-dart/blob/main/README.md + # - uses: dart-lang/setup-dart@v1 + - uses: dart-lang/setup-dart@9a04e6d73cca37bd455e0608d7e5092f881fd603 + + - name: Install dependencies + run: dart pub get + + # Uncomment this step to verify the use of 'dart format' on each commit. + # - name: Verify formatting + # run: dart format --output=none --set-exit-if-changed . + + # Consider passing '--fatal-infos' for slightly stricter analysis. + - name: Analyze project source + run: dart analyze + + # Your project will need to have tests in test/ and a dependency on + # package:test for this step to succeed. Note that Flutter projects will + # want to change this to 'flutter test'. + - name: Run tests + run: dart test diff --git a/.github/workflows/dotnet.yml b/.github/workflows/dotnet.yml new file mode 100644 index 000000000..217f7cbec --- /dev/null +++ b/.github/workflows/dotnet.yml @@ -0,0 +1,28 @@ +# This workflow will build a .NET project +# For more information see: https://docs.github.com/en/actions/automating-builds-and-tests/building-and-testing-net + +name: .NET + +on: + push: + branches: [ "main" ] + pull_request: + branches: [ "main" ] + +jobs: + build: + + runs-on: ubuntu-latest + + steps: + - uses: actions/checkout@v4 + - name: Setup .NET + uses: actions/setup-dotnet@v4 + with: + dotnet-version: 8.0.x + - name: Restore dependencies + run: dotnet restore + - name: Build + run: dotnet build --no-restore + - name: Test + run: dotnet test --no-build --verbosity normal diff --git a/.github/workflows/go.yml b/.github/workflows/go.yml new file mode 100644 index 000000000..0b443f376 --- /dev/null +++ b/.github/workflows/go.yml @@ -0,0 +1,28 @@ +# This workflow will build a golang project +# For more information see: https://docs.github.com/en/actions/automating-builds-and-tests/building-and-testing-go + +name: Go + +on: + push: + branches: [ "main" ] + pull_request: + branches: [ "main" ] + +jobs: + + build: + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v4 + + - name: Set up Go + uses: actions/setup-go@v4 + with: + go-version: '1.20' + + - name: Build + run: go build -v ./... + + - name: Test + run: go test -v ./... diff --git a/.github/workflows/gradle.yml b/.github/workflows/gradle.yml new file mode 100644 index 000000000..27400c2ba --- /dev/null +++ b/.github/workflows/gradle.yml @@ -0,0 +1,67 @@ +# This workflow uses actions that are not certified by GitHub. +# They are provided by a third-party and are governed by +# separate terms of service, privacy policy, and support +# documentation. +# This workflow will build a Java project with Gradle and cache/restore any dependencies to improve the workflow execution time +# For more information see: https://docs.github.com/en/actions/automating-builds-and-tests/building-and-testing-java-with-gradle + +name: Java CI with Gradle + +on: + push: + branches: [ "main" ] + pull_request: + branches: [ "main" ] + +jobs: + build: + + runs-on: ubuntu-latest + permissions: + contents: read + + steps: + - uses: actions/checkout@v4 + - name: Set up JDK 17 + uses: actions/setup-java@v4 + with: + java-version: '17' + distribution: 'temurin' + + # Configure Gradle for optimal use in GiHub Actions, including caching of downloaded dependencies. + # See: https://github.com/gradle/actions/blob/main/setup-gradle/README.md + - name: Setup Gradle + uses: gradle/actions/setup-gradle@417ae3ccd767c252f5661f1ace9f835f9654f2b5 # v3.1.0 + + - name: Build with Gradle Wrapper + run: ./gradlew build + + # NOTE: The Gradle Wrapper is the default and recommended way to run Gradle (https://docs.gradle.org/current/userguide/gradle_wrapper.html). + # If your project does not have the Gradle Wrapper configured, you can use the following configuration to run Gradle with a specified version. + # + # - name: Setup Gradle + # uses: gradle/actions/setup-gradle@417ae3ccd767c252f5661f1ace9f835f9654f2b5 # v3.1.0 + # with: + # gradle-version: '8.5' + # + # - name: Build with Gradle 8.5 + # run: gradle build + + dependency-submission: + + runs-on: ubuntu-latest + permissions: + contents: write + + steps: + - uses: actions/checkout@v4 + - name: Set up JDK 17 + uses: actions/setup-java@v4 + with: + java-version: '17' + distribution: 'temurin' + + # Generates and submits a dependency graph, enabling Dependabot Alerts for all project dependencies. + # See: https://github.com/gradle/actions/blob/main/dependency-submission/README.md + - name: Generate and submit dependency graph + uses: gradle/actions/dependency-submission@417ae3ccd767c252f5661f1ace9f835f9654f2b5 # v3.1.0 diff --git a/.github/workflows/laravel.yml b/.github/workflows/laravel.yml new file mode 100644 index 000000000..fd8c644e6 --- /dev/null +++ b/.github/workflows/laravel.yml @@ -0,0 +1,35 @@ +name: Laravel + +on: + push: + branches: [ "main" ] + pull_request: + branches: [ "main" ] + +jobs: + laravel-tests: + + runs-on: ubuntu-latest + + steps: + - uses: shivammathur/setup-php@15c43e89cdef867065b0213be354c2841860869e + with: + php-version: '8.0' + - uses: actions/checkout@v4 + - name: Copy .env + run: php -r "file_exists('.env') || copy('.env.example', '.env');" + - name: Install Dependencies + run: composer install -q --no-ansi --no-interaction --no-scripts --no-progress --prefer-dist + - name: Generate key + run: php artisan key:generate + - name: Directory Permissions + run: chmod -R 777 storage bootstrap/cache + - name: Create Database + run: | + mkdir -p database + touch database/database.sqlite + - name: Execute tests (Unit and Feature tests) via PHPUnit/Pest + env: + DB_CONNECTION: sqlite + DB_DATABASE: database/database.sqlite + run: php artisan test diff --git a/.github/workflows/node.js.yml b/.github/workflows/node.js.yml new file mode 100644 index 000000000..6d573d77d --- /dev/null +++ b/.github/workflows/node.js.yml @@ -0,0 +1,31 @@ +# This workflow will do a clean installation of node dependencies, cache/restore them, build the source code and run tests across different versions of node +# For more information see: https://docs.github.com/en/actions/automating-builds-and-tests/building-and-testing-nodejs + +name: Node.js CI + +on: + push: + branches: [ "main" ] + pull_request: + branches: [ "main" ] + +jobs: + build: + + runs-on: ubuntu-latest + + strategy: + matrix: + node-version: [14.x, 16.x, 18.x] + # See supported Node.js release schedule at https://nodejs.org/en/about/releases/ + + steps: + - uses: actions/checkout@v4 + - name: Use Node.js ${{ matrix.node-version }} + uses: actions/setup-node@v3 + with: + node-version: ${{ matrix.node-version }} + cache: 'npm' + - run: npm ci + - run: npm run build --if-present + - run: npm test diff --git a/.github/workflows/r.yml b/.github/workflows/r.yml new file mode 100644 index 000000000..ed8aacfa6 --- /dev/null +++ b/.github/workflows/r.yml @@ -0,0 +1,40 @@ +# This workflow uses actions that are not certified by GitHub. +# They are provided by a third-party and are governed by +# separate terms of service, privacy policy, and support +# documentation. +# +# See https://github.com/r-lib/actions/tree/master/examples#readme for +# additional example workflows available for the R community. + +name: R + +on: + push: + branches: [ "main" ] + pull_request: + branches: [ "main" ] + +permissions: + contents: read + +jobs: + build: + runs-on: macos-latest + strategy: + matrix: + r-version: ['3.6.3', '4.1.1'] + + steps: + - uses: actions/checkout@v4 + - name: Set up R ${{ matrix.r-version }} + uses: r-lib/actions/setup-r@f57f1301a053485946083d7a45022b278929a78a + with: + r-version: ${{ matrix.r-version }} + - name: Install dependencies + run: | + install.packages(c("remotes", "rcmdcheck")) + remotes::install_deps(dependencies = TRUE) + shell: Rscript {0} + - name: Check + run: rcmdcheck::rcmdcheck(args = "--no-manual", error_on = "error") + shell: Rscript {0} diff --git a/.github/workflows/ruby.yml b/.github/workflows/ruby.yml new file mode 100644 index 000000000..5d80832d3 --- /dev/null +++ b/.github/workflows/ruby.yml @@ -0,0 +1,38 @@ +# This workflow uses actions that are not certified by GitHub. +# They are provided by a third-party and are governed by +# separate terms of service, privacy policy, and support +# documentation. +# This workflow will download a prebuilt Ruby version, install dependencies and run tests with Rake +# For more information see: https://github.com/marketplace/actions/setup-ruby-jruby-and-truffleruby + +name: Ruby + +on: + push: + branches: [ "main" ] + pull_request: + branches: [ "main" ] + +permissions: + contents: read + +jobs: + test: + + runs-on: ubuntu-latest + strategy: + matrix: + ruby-version: ['2.6', '2.7', '3.0'] + + steps: + - uses: actions/checkout@v4 + - name: Set up Ruby + # To automatically get bug fixes and new Ruby versions for ruby/setup-ruby, + # change this to (see https://github.com/ruby/setup-ruby#versioning): + # uses: ruby/setup-ruby@v1 + uses: ruby/setup-ruby@55283cc23133118229fd3f97f9336ee23a179fcf # v1.146.0 + with: + ruby-version: ${{ matrix.ruby-version }} + bundler-cache: true # runs 'bundle install' and caches installed gems automatically + - name: Run tests + run: bundle exec rake diff --git a/.github/workflows/symfony.yml b/.github/workflows/symfony.yml new file mode 100644 index 000000000..e9b7c26f6 --- /dev/null +++ b/.github/workflows/symfony.yml @@ -0,0 +1,47 @@ +# This workflow uses actions that are not certified by GitHub. +# They are provided by a third-party and are governed by +# separate terms of service, privacy policy, and support +# documentation. + +name: Symfony + +on: + push: + branches: [ "main" ] + pull_request: + branches: [ "main" ] + +permissions: + contents: read + +jobs: + symfony-tests: + runs-on: ubuntu-latest + steps: + # To automatically get bug fixes and new Php versions for shivammathur/setup-php, + # change this to (see https://github.com/shivammathur/setup-php#bookmark-versioning): + # uses: shivammathur/setup-php@v2 + - uses: shivammathur/setup-php@2cb9b829437ee246e9b3cac53555a39208ca6d28 + with: + php-version: '8.0' + - uses: actions/checkout@v4 + - name: Copy .env.test.local + run: php -r "file_exists('.env.test.local') || copy('.env.test', '.env.test.local');" + - name: Cache Composer packages + id: composer-cache + uses: actions/cache@v3 + with: + path: vendor + key: ${{ runner.os }}-php-${{ hashFiles('**/composer.lock') }} + restore-keys: | + ${{ runner.os }}-php- + - name: Install Dependencies + run: composer install -q --no-ansi --no-interaction --no-scripts --no-progress --prefer-dist + - name: Create Database + run: | + mkdir -p data + touch data/database.sqlite + - name: Execute tests (Unit and Feature tests) via PHPUnit + env: + DATABASE_URL: sqlite:///%kernel.project_dir%/data/database.sqlite + run: vendor/bin/phpunit diff --git a/.github/workflows/terraform.yml b/.github/workflows/terraform.yml new file mode 100644 index 000000000..540e8040b --- /dev/null +++ b/.github/workflows/terraform.yml @@ -0,0 +1,93 @@ +# This workflow installs the latest version of Terraform CLI and configures the Terraform CLI configuration file +# with an API token for Terraform Cloud (app.terraform.io). On pull request events, this workflow will run +# `terraform init`, `terraform fmt`, and `terraform plan` (speculative plan via Terraform Cloud). On push events +# to the "main" branch, `terraform apply` will be executed. +# +# Documentation for `hashicorp/setup-terraform` is located here: https://github.com/hashicorp/setup-terraform +# +# To use this workflow, you will need to complete the following setup steps. +# +# 1. Create a `main.tf` file in the root of this repository with the `remote` backend and one or more resources defined. +# Example `main.tf`: +# # The configuration for the `remote` backend. +# terraform { +# backend "remote" { +# # The name of your Terraform Cloud organization. +# organization = "example-organization" +# +# # The name of the Terraform Cloud workspace to store Terraform state files in. +# workspaces { +# name = "example-workspace" +# } +# } +# } +# +# # An example resource that does nothing. +# resource "null_resource" "example" { +# triggers = { +# value = "A example resource that does nothing!" +# } +# } +# +# +# 2. Generate a Terraform Cloud user API token and store it as a GitHub secret (e.g. TF_API_TOKEN) on this repository. +# Documentation: +# - https://www.terraform.io/docs/cloud/users-teams-organizations/api-tokens.html +# - https://help.github.com/en/actions/configuring-and-managing-workflows/creating-and-storing-encrypted-secrets +# +# 3. Reference the GitHub secret in step using the `hashicorp/setup-terraform` GitHub Action. +# Example: +# - name: Setup Terraform +# uses: hashicorp/setup-terraform@v1 +# with: +# cli_config_credentials_token: ${{ secrets.TF_API_TOKEN }} + +name: 'Terraform' + +on: + push: + branches: [ "main" ] + pull_request: + +permissions: + contents: read + +jobs: + terraform: + name: 'Terraform' + runs-on: ubuntu-latest + environment: production + + # Use the Bash shell regardless whether the GitHub Actions runner is ubuntu-latest, macos-latest, or windows-latest + defaults: + run: + shell: bash + + steps: + # Checkout the repository to the GitHub Actions runner + - name: Checkout + uses: actions/checkout@v4 + + # Install the latest version of Terraform CLI and configure the Terraform CLI configuration file with a Terraform Cloud user API token + - name: Setup Terraform + uses: hashicorp/setup-terraform@v1 + with: + cli_config_credentials_token: ${{ secrets.TF_API_TOKEN }} + + # Initialize a new or existing Terraform working directory by creating initial files, loading any remote state, downloading modules, etc. + - name: Terraform Init + run: terraform init + + # Checks that all Terraform configuration files adhere to a canonical format + - name: Terraform Format + run: terraform fmt -check + + # Generates an execution plan for Terraform + - name: Terraform Plan + run: terraform plan -input=false + + # On push to "main", build or change infrastructure according to Terraform configuration files + # Note: It is recommended to set up a required "strict" status check in your repository for "Terraform Cloud". See the documentation on "strict" required status checks for more information: https://help.github.com/en/github/administering-a-repository/types-of-required-status-checks + - name: Terraform Apply + if: github.ref == 'refs/heads/"main"' && github.event_name == 'push' + run: terraform apply -auto-approve -input=false diff --git a/.github/workflows/webpack.yml b/.github/workflows/webpack.yml new file mode 100644 index 000000000..25e64c08e --- /dev/null +++ b/.github/workflows/webpack.yml @@ -0,0 +1,28 @@ +name: NodeJS with Webpack + +on: + push: + branches: [ "main" ] + pull_request: + branches: [ "main" ] + +jobs: + build: + runs-on: ubuntu-latest + + strategy: + matrix: + node-version: [14.x, 16.x, 18.x] + + steps: + - uses: actions/checkout@v4 + + - name: Use Node.js ${{ matrix.node-version }} + uses: actions/setup-node@v3 + with: + node-version: ${{ matrix.node-version }} + + - name: Build + run: | + npm install + npx webpack diff --git a/api/__init__.py b/api/__init__.py index f02c44442..e61ac4911 100644 --- a/api/__init__.py +++ b/api/__init__.py @@ -1,10 +1,12 @@ from flask import Flask -from .endpoints import endpoints + from .authentication import jwt_manager +from .endpoints import endpoints + def create_app(): app = Flask(__name__) - app.config.from_object('config') - app.register_blueprint(endpoints, url_prefix='/api') + app.config.from_object("config") + app.register_blueprint(endpoints, url_prefix="/api") jwt_manager.init_app(app) return app diff --git a/api/endpoints.py b/api/endpoints.py index 8deff8235..6bc4572c6 100644 --- a/api/endpoints.py +++ b/api/endpoints.py @@ -1,24 +1,28 @@ -from flask import Blueprint, request, jsonify +from flask import Blueprint, jsonify, request + from .authentication import jwt_required -from .serializers import UserSchema, TransactionSchema +from .serializers import TransactionSchema, UserSchema + +endpoints = Blueprint("endpoints", __name__) -endpoints = Blueprint('endpoints', __name__) -@endpoints.route('/users', methods=['POST']) +@endpoints.route("/users", methods=["POST"]) def create_user(): user_schema = UserSchema() user = user_schema.load(request.get_json()) # Create user in the database return user_schema.jsonify(user), 201 -@endpoints.route('/users/', methods=['GET']) + +@endpoints.route("/users/", methods=["GET"]) @jwt_required() def get_user(user_id): user = get_user_from_database(user_id) user_schema = UserSchema() return user_schema.jsonify(user) -@endpoints.route('/users/', methods=['PUT']) + +@endpoints.route("/users/", methods=["PUT"]) @jwt_required() def update_user(user_id): user = get_user_from_database(user_id) @@ -27,14 +31,16 @@ def update_user(user_id): # Update user in the database return user_schema.jsonify(user) -@endpoints.route('/users/', methods=['DELETE']) + +@endpoints.route("/users/", methods=["DELETE"]) @jwt_required() def delete_user(user_id): user = get_user_from_database(user_id) # Delete user from the database - return '', 204 + return "", 204 -@endpoints.route('/transactions', methods=['POST']) + +@endpoints.route("/transactions", methods=["POST"]) @jwt_required() def create_transaction(): transaction_schema = TransactionSchema() @@ -42,21 +48,24 @@ def create_transaction(): # Create transaction in the database return transaction_schema.jsonify(transaction), 201 -@endpoints.route('/transactions', methods=['GET']) + +@endpoints.route("/transactions", methods=["GET"]) @jwt_required() def get_transactions(): transactions = get_transactions_from_database() transaction_schema = TransactionSchema(many=True) return transaction_schema.jsonify(transactions) -@endpoints.route('/transactions/', methods=['GET']) + +@endpoints.route("/transactions/", methods=["GET"]) @jwt_required() def get_transaction(transaction_id): transaction = get_transaction_from_database(transaction_id) transaction_schema = TransactionSchema() return transaction_schema.jsonify(transaction) -@endpoints.route('/transactions/', methods=['PUT']) + +@endpoints.route("/transactions/", methods=["PUT"]) @jwt_required() def update_transaction(transaction_id): transaction = get_transaction_from_database(transaction_id) @@ -65,9 +74,10 @@ def update_transaction(transaction_id): # Update transaction in the database return transaction_schema.jsonify(transaction) -@endpoints.route('/transactions/', methods=['DELETE']) + +@endpoints.route("/transactions/", methods=["DELETE"]) @jwt_required() def delete_transaction(transaction_id): transaction = get_transaction_from_database(transaction_id) # Delete transaction from the database - return '', 204 + return "", 204 diff --git a/api/serializers.py b/api/serializers.py index 881af2596..ad0d700fe 100644 --- a/api/serializers.py +++ b/api/serializers.py @@ -1,11 +1,13 @@ from marshmallow import Schema, fields + class UserSchema(Schema): id = fields.Int(dump_only=True) name = fields.Str(required=True) email = fields.Email(required=True) password = fields.Str(required=True, load_only=True) + class TransactionSchema(Schema): id = fields.Int(dump_only=True) user_id = fields.Int(required=True) diff --git a/api/test_endpoints.py b/api/test_endpoints.py index 805c982cc..864c6d238 100644 --- a/api/test_endpoints.py +++ b/api/test_endpoints.py @@ -1,7 +1,10 @@ import unittest + from app import create_app + from api.endpoints import endpoints -from api.serializers import UserSchema, TransactionSchema +from api.serializers import TransactionSchema, UserSchema + class TestEndpoints(unittest.TestCase): def setUp(self): @@ -16,11 +19,11 @@ def tearDown(self): def test_create_user(self): user_data = { - 'name': 'Test User', - 'email': 'test@example.com', - 'password': 'testpassword' + "name": "Test User", + "email": "test@example.com", + "password": "testpassword", } - response = self.client.post('/api/users', json=user_data) + response = self.client.post("/api/users", json=user_data) self.assertEqual(response.status_code, 201) user_schema = UserSchema() user = user_schema.load(response.get_json()) @@ -28,54 +31,51 @@ def test_create_user(self): def test_get_user(self): user_data = { - 'name': 'Test User', - 'email': 'test@example.com', - 'password': 'testpassword' + "name": "Test User", + "email": "test@example.com", + "password": "testpassword", } - self.client.post('/api/users', json=user_data) - response = self.client.get('/api/users/1') + self.client.post("/api/users", json=user_data) + response = self.client.get("/api/users/1") self.assertEqual(response.status_code, 200) user_schema = UserSchema() user = user_schema.load(response.get_json()) - self.assertEqual(user.name, 'Test User') - self.assertEqual(user.email, 'test@example.com') + self.assertEqual(user.name, "Test User") + self.assertEqual(user.email, "test@example.com") def test_update_user(self): user_data = { - 'name': 'Test User', - 'email': 'test@example.com', - 'password': 'testpassword' + "name": "Test User", + "email": "test@example.com", + "password": "testpassword", } - self.client.post('/api/users', json=user_data) - user_data['name'] = 'Updated Test User' - response = self.client.put('/api/users/1', json=user_data) + self.client.post("/api/users", json=user_data) + user_data["name"] = "Updated Test User" + response = self.client.put("/api/users/1", json=user_data) self.assertEqual(response.status_code, 200) user_schema = UserSchema() user = user_schema.load(response.get_json()) - self.assertEqual(user.name, 'Updated Test User') + self.assertEqual(user.name, "Updated Test User") def test_delete_user(self): user_data = { - 'name': 'Test User', - 'email': 'test@example.com', - 'password': 'testpassword' + "name": "Test User", + "email": "test@example.com", + "password": "testpassword", } - self.client.post('/api/users', json=user_data) - response = self.client.delete('/api/users/1') + self.client.post("/api/users", json=user_data) + response = self.client.delete("/api/users/1") self.assertEqual(response.status_code, 204) def test_create_transaction(self): user_data = { - 'name': 'Test User', - 'email': 'test@example.com', - 'password': 'testpassword' - } - self.client.post('/api/users', json=user_data) - transaction_data = { - 'user_id': 1, - 'amount': 100.0 + "name": "Test User", + "email": "test@example.com", + "password": "testpassword", } - response = self.client.post('/api/transactions', json=transaction_data) + self.client.post("/api/users", json=user_data) + transaction_data = {"user_id": 1, "amount": 100.0} + response = self.client.post("/api/transactions", json=transaction_data) self.assertEqual(response.status_code, 201) transaction_schema = TransactionSchema() transaction = transaction_schema.load(response.get_json()) @@ -83,17 +83,14 @@ def test_create_transaction(self): def test_get_transactions(self): user_data = { - 'name': 'Test User', - 'email': 'test@example.com', - 'password':'testpassword' + "name": "Test User", + "email": "test@example.com", + "password": "testpassword", } - self.client.post('/api/users', json=user_data) - transaction_data = { - 'user_id': 1, - 'amount': 100.0 - } - self.client.post('/api/transactions', json=transaction_data) - response = self.client.get('/api/transactions') + self.client.post("/api/users", json=user_data) + transaction_data = {"user_id": 1, "amount": 100.0} + self.client.post("/api/transactions", json=transaction_data) + response = self.client.get("/api/transactions") self.assertEqual(response.status_code, 200) transaction_schema = TransactionSchema(many=True) transactions = transaction_schema.load(response.get_json()) @@ -101,17 +98,14 @@ def test_get_transactions(self): def test_get_transaction(self): user_data = { - 'name': 'Test User', - 'email': 'test@example.com', - 'password': 'testpassword' - } - self.client.post('/api/users', json=user_data) - transaction_data = { - 'user_id': 1, - 'amount': 100.0 + "name": "Test User", + "email": "test@example.com", + "password": "testpassword", } - self.client.post('/api/transactions', json=transaction_data) - response = self.client.get('/api/transactions/1') + self.client.post("/api/users", json=user_data) + transaction_data = {"user_id": 1, "amount": 100.0} + self.client.post("/api/transactions", json=transaction_data) + response = self.client.get("/api/transactions/1") self.assertEqual(response.status_code, 200) transaction_schema = TransactionSchema() transaction = transaction_schema.load(response.get_json()) @@ -119,18 +113,15 @@ def test_get_transaction(self): def test_update_transaction(self): user_data = { - 'name': 'Test User', - 'email': 'test@example.com', - 'password': 'testpassword' + "name": "Test User", + "email": "test@example.com", + "password": "testpassword", } - self.client.post('/api/users', json=user_data) - transaction_data = { - 'user_id': 1, - 'amount': 100.0 - } - self.client.post('/api/transactions', json=transaction_data) - transaction_data['amount'] = 200.0 - response = self.client.put('/api/transactions/1', json=transaction_data) + self.client.post("/api/users", json=user_data) + transaction_data = {"user_id": 1, "amount": 100.0} + self.client.post("/api/transactions", json=transaction_data) + transaction_data["amount"] = 200.0 + response = self.client.put("/api/transactions/1", json=transaction_data) self.assertEqual(response.status_code, 200) transaction_schema = TransactionSchema() transaction = transaction_schema.load(response.get_json()) @@ -138,15 +129,12 @@ def test_update_transaction(self): def test_delete_transaction(self): user_data = { - 'name': 'Test User', - 'email': 'test@example.com', - 'password': 'testpassword' - } - self.client.post('/api/users', json=user_data) - transaction_data = { - 'user_id': 1, - 'amount': 100.0 + "name": "Test User", + "email": "test@example.com", + "password": "testpassword", } - self.client.post('/api/transactions', json=transaction_data) - response = self.client.delete('/api/transactions/1') + self.client.post("/api/users", json=user_data) + transaction_data = {"user_id": 1, "amount": 100.0} + self.client.post("/api/transactions", json=transaction_data) + response = self.client.delete("/api/transactions/1") self.assertEqual(response.status_code, 204) diff --git a/banking/account_management.py b/banking/account_management.py index 1fe44c0e3..4a41344cd 100644 --- a/banking/account_management.py +++ b/banking/account_management.py @@ -3,6 +3,7 @@ import logging from typing import Dict + def create_account(account_type: str, balance: float, customer_id: int) -> Dict: """ Create a new account for a customer. @@ -18,22 +19,23 @@ def create_account(account_type: str, balance: float, customer_id: int) -> Dict: # implementation logging.info(f"Creating account for customer {customer_id}") # Validate input parameters - if account_type not in ['checking', 'savings']: + if account_type not in ["checking", "savings"]: raise ValueError("Invalid account type") if balance < 0: raise ValueError("Balance cannot be negative") # Create account and store it in a database or in-memory data structure account = { - 'id': generate_unique_id(), - 'type': account_type, - 'balance': balance, - 'customer_id': customer_id + "id": generate_unique_id(), + "type": account_type, + "balance": balance, + "customer_id": customer_id, } # Return the account information return account + def update_account(account_id: int, account_type: str, balance: float) -> bool: """ Update an existing account. @@ -49,7 +51,7 @@ def update_account(account_id: int, account_type: str, balance: float) -> bool: # implementation logging.info(f"Updating account {account_id}") # Validate input parameters - if account_type not in ['checking', 'savings']: + if account_type not in ["checking", "savings"]: raise ValueError("Invalid account type") if balance < 0: raise ValueError("Balance cannot be negative") @@ -58,8 +60,8 @@ def update_account(account_id: int, account_type: str, balance: float) -> bool: account = get_account_by_id(account_id) # Update the account information - account['type'] = account_type - account['balance'] = balance + account["type"] = account_type + account["balance"] = balance # Save the updated account back to the database or in-memory data structure save_account(account) @@ -67,6 +69,7 @@ def update_account(account_id: int, account_type: str, balance: float) -> bool: # Return True to indicate success return True + def delete_account(account_id: int) -> bool: """ Delete an account. diff --git a/banking/bank.py b/banking/bank.py index dab15c962..e04c75b68 100644 --- a/banking/bank.py +++ b/banking/bank.py @@ -1,12 +1,25 @@ -from banking.accounts import BankAccount -from banking.transactions import transfer +from banking.accounts import ( + BankAccount, + CheckingAccount, + InsufficientFundsError, + SavingsAccount, +) +from banking.transactions import pay_bill, transfer + -def create_account(account_number): """ - Creates a new bank account with the specified account number. + Creates a new bank account with the specified account type and account number. """ - return BankAccount(account_number) + if account_type == "savings": + account = SavingsAccount(account_number) + elif account_type == "checking": + account = CheckingAccount(account_number) + else: + raise ValueError(f"Invalid account type: {account_type}") + + return account + def get_account(account_number): @@ -18,9 +31,11 @@ def get_account(account_number): # Create new account if it doesn't exist if account_data is None: - account = create_account(account_number) + account_type = None else: - account = BankAccount(account_number, balance=account_data["balance"]) + account_type = account_data["type"] + + account = create_account(account_type, account_number) return account @@ -30,18 +45,37 @@ def save_account(account): Saves the specified bank account to the database or other storage system. """ # Save account data to database or other storage system - save_account_data(account.account_number, account.balance) + save_account_data(account.account_number, account.type, account.balance) + def process_transaction(transaction): """ - Processes the specified transaction by transferring funds between bank accounts. + Processes the specified transaction by transferring funds between bank accounts or paying a bill. """ - from_account = get_account(transaction["from_account"]) - to_account = get_account(transaction["to_account"]) - amount = transaction["amount"] + transaction_type = transaction["type"] + + if transaction_type == "transfer": + from_account_number = transaction["from_account"] + to_account_number = transaction["to_account"] + amount = transaction["amount"] + + from_account = get_account(from_account_number) + to_account = get_account(to_account_number) + + transfer(from_account, to_account, amount) + + save_account(from_account) + save_account(to_account) + + elif transaction_type == "pay_bill": + account_number = transaction["account"] + payee = transaction["payee"] + amount = transaction["amount"] + due_date = transaction["due_date"] + + account = get_account(account_number) - transfer(from_account, to_account, amount) + pay_bill(account, payee, amount, due_date) - save_account(from_account) - save_account(to_account) + save_account(account) diff --git a/blockchain/__init__.py b/blockchain/__init__.py index 2fe5e113f..112888f97 100644 --- a/blockchain/__init__.py +++ b/blockchain/__init__.py @@ -1,6 +1,6 @@ -from .pi_network import PiNetwork -from .ethereum import Ethereum from .bitcoin import Bitcoin from .block import Block from .chain import Blockchain +from .ethereum import Ethereum from .miner import Miner +from .pi_network import PiNetwork diff --git a/blockchain/block.py b/blockchain/block.py index ca6f44246..7945ecbae 100644 --- a/blockchain/block.py +++ b/blockchain/block.py @@ -1,6 +1,7 @@ import hashlib import time + class Block: def __init__(self, index, previous_hash, timestamp, data, hash): self.index = index @@ -12,13 +13,13 @@ def __init__(self, index, previous_hash, timestamp, data, hash): @staticmethod def calculate_hash(index, previous_hash, timestamp, data): value = str(index) + str(previous_hash) + str(timestamp) + str(data) - return hashlib.sha256(value.encode('utf-8')).hexdigest() + return hashlib.sha256(value.encode("utf-8")).hexdigest() def __str__(self): - return 'Block: {index}, Previous Hash: {previous_hash}, Timestamp: {timestamp}, Data: {data}, Hash: {hash}'.format( + return "Block: {index}, Previous Hash: {previous_hash}, Timestamp: {timestamp}, Data: {data}, Hash: {hash}".format( index=self.index, previous_hash=self.previous_hash, timestamp=self.timestamp, data=self.data, - hash=self.hash + hash=self.hash, ) diff --git a/blockchain/chain.py b/blockchain/chain.py index ad5ad80d5..ee2bc218f 100644 --- a/blockchain/chain.py +++ b/blockchain/chain.py @@ -1,23 +1,35 @@ import hashlib + class Blockchain: def __init__(self): self.chain = [self.create_genesis_block()] self.difficulty = 2 def create_genesis_block(self): - return Block(0, '0' * 64, int(time.time()), 'Genesis Block', self.calculate_hash(0, '0' * 64, int(time.time()), 'Genesis Block')) + return Block( + 0, + "0" * 64, + int(time.time()), + "Genesis Block", + self.calculate_hash(0, "0" * 64, int(time.time()), "Genesis Block"), + ) def calculate_hash(self, index, previous_hash, timestamp, data): value = str(index) + str(previous_hash) + str(timestamp) + str(data) - return hashlib.sha256(value.encode('utf-8')).hexdigest() + return hashlib.sha256(value.encode("utf-8")).hexdigest() def is_chain_valid(self, chain): for i in range(1, len(chain)): current_block = chain[i] previous_block = chain[i - 1] - if current_block.hash != self.calculate_hash(current_block.index, current_block.previous_hash, current_block.timestamp, current_block.data): + if current_block.hash != self.calculate_hash( + current_block.index, + current_block.previous_hash, + current_block.timestamp, + current_block.data, + ): return False if current_block.previous_hash != previous_block.hash: @@ -26,8 +38,15 @@ def is_chain_valid(self, chain): return True def add_block(self, data): - new_block = Block(len(self.chain), self.chain[-1].hash, int(time.time()), data, None) - new_block.hash = self.calculate_hash(new_block.index, new_block.previous_hash, new_block.timestamp, new_block.data) + new_block = Block( + len(self.chain), self.chain[-1].hash, int(time.time()), data, None + ) + new_block.hash = self.calculate_hash( + new_block.index, + new_block.previous_hash, + new_block.timestamp, + new_block.data, + ) self.chain.append(new_block) def replace_chain(self, chain): diff --git a/blockchain/miner.py b/blockchain/miner.py index 2a0a17daf..024156d65 100644 --- a/blockchain/miner.py +++ b/blockchain/miner.py @@ -1,6 +1,8 @@ import time + from blockchain.block import Blockchain + class Miner: def __init__(self, blockchain): self.blockchain = blockchain @@ -9,12 +11,12 @@ def mine_block(self, data): while True: if self.blockchain.is_chain_valid(self.blockchain.chain): self.blockchain.add_block(data) - print('Block successfully mined!') - print('Proof of work: {}'.format(self.blockchain.chain[-1].hash)) + print("Block successfully mined!") + print("Proof of work: {}".format(self.blockchain.chain[-1].hash)) break else: - print('Block not mined. Invalid chain.') + print("Block not mined. Invalid chain.") self.blockchain.chain = [self.blockchain.create_genesis_block()] self.blockchain.difficulty += 1 - print('New difficulty: {}'.format(self.blockchain.difficulty)) + print("New difficulty: {}".format(self.blockchain.difficulty)) time.sleep(1) diff --git a/blockchain/test/test_blockchain.py b/blockchain/test/test_blockchain.py index bd7e51ad8..d3217a024 100644 --- a/blockchain/test/test_blockchain.py +++ b/blockchain/test/test_blockchain.py @@ -1,6 +1,8 @@ import unittest + from blockchain import Blockchain + class TestBlockchain(unittest.TestCase): def setUp(self): self.blockchain = Blockchain() @@ -8,40 +10,66 @@ def setUp(self): def test_create_genesis_block(self): genesis_block = self.blockchain.create_genesis_block() self.assertEqual(genesis_block.index, 0) - self.assertEqual(genesis_block.previous_hash, '0' * 64) + self.assertEqual(genesis_block.previous_hash, "0" * 64) self.assertIsNotNone(genesis_block.timestamp) - self.assertEqual(genesis_block.data, 'Genesis Block') - self.assertEqual(genesis_block.hash, '0' * 64) + self.assertEqual(genesis_block.data, "Genesis Block") + self.assertEqual(genesis_block.hash, "0" * 64) def test_calculate_hash(self): block = self.blockchain.create_genesis_block() - block.data = 'New Data' + block.data = "New Data" block.hash = None - calculated_hash = self.blockchain.calculate_hash(block.index, block.previous_hash, block.timestamp, block.data) - self.assertEqual(calculated_hash, '0' * 64) + calculated_hash = self.blockchain.calculate_hash( + block.index, block.previous_hash, block.timestamp, block.data + ) + self.assertEqual(calculated_hash, "0" * 64) def test_is_chain_valid(self): self.assertTrue(self.blockchain.is_chain_valid(self.blockchain.chain)) invalid_chain = self.blockchain.chain[:-1] - invalid_chain.append(Block(len(invalid_chain), invalid_chain[-1].hash, int(time.time()), 'Invalid Data', None)) + invalid_chain.append( + Block( + len(invalid_chain), + invalid_chain[-1].hash, + int(time.time()), + "Invalid Data", + None, + ) + ) self.assertFalse(self.blockchain.is_chain_valid(invalid_chain)) def test_add_block(self): - self.blockchain.add_block('Test Data') + self.blockchain.add_block("Test Data") self.assertEqual(len(self.blockchain.chain), 2) - self.assertEqual(self.blockchain.chain[-1].data, 'Test Data') + self.assertEqual(self.blockchain.chain[-1].data, "Test Data") def test_replace_chain(self): self.blockchain.replace_chain(self.blockchain.chain[:-1]) self.assertEqual(len(self.blockchain.chain), 1) invalid_chain = self.blockchain.chain[:-1] - invalid_chain.append(Block(len(invalid_chain), invalid_chain[-1].hash, int(time.time()), 'Invalid Data', None)) + invalid_chain.append( + Block( + len(invalid_chain), + invalid_chain[-1].hash, + int(time.time()), + "Invalid Data", + None, + ) + ) self.blockchain.replace_chain(invalid_chain) self.assertFalse(self.blockchain.is_chain_valid(self.blockchain.chain)) valid_chain = self.blockchain.chain[:-1] - valid_chain.append(Block(len(valid_chain), valid_chain[-1].hash, int(time.time()), 'Valid Data', None)) + valid_chain.append( + Block( + len(valid_chain), + valid_chain[-1].hash, + int(time.time()), + "Valid Data", + None, + ) + ) self.blockchain.replace_chain(valid_chain) self.assertTrue(self.blockchain.is_chain_valid(self.blockchain.chain)) diff --git a/blockchain/transaction.py b/blockchain/transaction.py index 79eb547a5..0a49fcf79 100644 --- a/blockchain/transaction.py +++ b/blockchain/transaction.py @@ -5,4 +5,11 @@ def __init__(self, sender, receiver, amount): self.amount = amount def to_string(self): - return "Sender: " + self.sender + ", Receiver: " + self.receiver + ", Amount: " + str(self.amount) + return ( + "Sender: " + + self.sender + + ", Receiver: " + + self.receiver + + ", Amount: " + + str(self.amount) + ) diff --git a/customer_service/chatbot.py b/customer_service/chatbot.py index 8ff98a9a1..786d199bd 100644 --- a/customer_service/chatbot.py +++ b/customer_service/chatbot.py @@ -1,5 +1,6 @@ import time + class Chatbot: def __init__(self, nlp_model, knowledge_base): self.nlp_model = nlp_model @@ -9,6 +10,6 @@ def chat(self, user_input): entities = self.nlp_model.extract_entities(user_input) intent = self.nlp_model.extract_intent(user_input) response = self.knowledge_base.get_response(intent) - print('User:', user_input) - print('Chatbot:', response) + print("User:", user_input) + print("Chatbot:", response) time.sleep(1) diff --git a/customer_service/knowledge_base.py b/customer_service/knowledge_base.py index eab799fb7..267a37c3a 100644 --- a/customer_service/knowledge_base.py +++ b/customer_service/knowledge_base.py @@ -1,18 +1,18 @@ class KnowledgeBase: def __init__(self): self.knowledge_base = { - 'account': { - 'open': 'To open a new account, please visit our website and fill out the account opening form.', - 'close': 'To close an existing account, please contact our customer support team.', - 'balance': 'To check your account balance, please log in to your account on our website or mobile app.' + "account": { + "open": "To open a new account, please visit our website and fill out the account opening form.", + "close": "To close an existing account, please contact our customer support team.", + "balance": "To check your account balance, please log in to your account on our website or mobile app.", }, - 'transaction': { - 'make': 'To make a new transaction, please log in to your account on our website or mobile app.', - 'history': 'To view your transaction history, please log in to your account on our website or mobile app.' + "transaction": { + "make": "To make a new transaction, please log in to your account on our website or mobile app.", + "history": "To view your transaction history, please log in to your account on our website or mobile app.", + }, + "other": { + "help": "For assistance, please contact our customer support team." }, - 'other': { - 'help': 'For assistance, please contact our customer support team.' - } } def get_response(self, intent): @@ -21,4 +21,4 @@ def get_response(self, intent): if value: return self.knowledge_base[key] else: - return self.knowledge_base['other'] + return self.knowledge_base["other"] diff --git a/customer_service/natural_language_processing.py b/customer_service/natural_language_processing.py index 07490f49a..6a1aef356 100644 --- a/customer_service/natural_language_processing.py +++ b/customer_service/natural_language_processing.py @@ -1,6 +1,7 @@ import spacy -nlp = spacy.load('en_core_web_sm') +nlp = spacy.load("en_core_web_sm") + class NLP: def __init__(self): @@ -13,14 +14,14 @@ def extract_entities(self, text): def extract_intent(self, text): doc = self.nlp(text) - intents = {'account': False, 'transaction': False, 'other': False} + intents = {"account": False, "transaction": False, "other": False} for token in doc: - if token.text.lower() in ['account', 'accounts']: - intents['account'] = True - elif token.text.lower() in ['transaction', 'transactions']: - intents['transaction'] = True + if token.text.lower() in ["account", "accounts"]: + intents["account"] = True + elif token.text.lower() in ["transaction", "transactions"]: + intents["transaction"] = True if any(intents.values()): return intents else: - intents['other'] = True + intents["other"] = True return intents diff --git a/data_analytics/data_analysis.py b/data_analytics/data_analysis.py index 20bc30414..88523a5c6 100644 --- a/data_analytics/data_analysis.py +++ b/data_analytics/data_analysis.py @@ -2,6 +2,7 @@ import pandas as pd from sklearn.cluster import KMeans + class DataAnalysis: def __init__(self, data): self.data = data @@ -9,16 +10,18 @@ def __init__(self, data): def analyze_data(self): # Perform data cleaning and preprocessing self.data = self.data.dropna() - self.data = pd.get_dummies(self.data, columns=['transaction_type']) + self.data = pd.get_dummies(self.data, columns=["transaction_type"]) # Perform data analysis - kmeans = KMeans(n_clusters=3, random_state=0).fit(self.data[['amount', 'frequency']]) - self.data['cluster'] = kmeans.labels_ + kmeans = KMeans(n_clusters=3, random_state=0).fit( + self.data[["amount", "frequency"]] + ) + self.data["cluster"] = kmeans.labels_ # Perform statistical analysis summary_stats = self.data.describe() - summary_stats.loc['count'] = len(self.data) - summary_stats.loc['mean'] = np.mean(self.data) - summary_stats.loc['std'] = np.std(self.data) + summary_stats.loc["count"] = len(self.data) + summary_stats.loc["mean"] = np.mean(self.data) + summary_stats.loc["std"] = np.std(self.data) return summary_stats diff --git a/data_analytics/data_ingestion.py b/data_analytics/data_ingestion.py index 7cf151c16..5183e94cc 100644 --- a/data_analytics/data_ingestion.py +++ b/data_analytics/data_ingestion.py @@ -1,5 +1,6 @@ import pandas as pd + class DataIngestion: def __init__(self, data_source): self.data_source = data_source diff --git a/data_analytics/data_visualization.py b/data_analytics/data_visualization.py index 96414dd96..3f711a046 100644 --- a/data_analytics/data_visualization.py +++ b/data_analytics/data_visualization.py @@ -1,16 +1,17 @@ import plotly.express as px + class DataVisualization: def __init__(self, data): self.data = data def visualize_data(self): # Create interactive dashboards - fig = px.scatter(self.data, x='amount', y='frequency', color='cluster') + fig = px.scatter(self.data, x="amount", y="frequency", color="cluster") fig.show() - fig = px.histogram(self.data, x='transaction_type', nbins=20) + fig = px.histogram(self.data, x="transaction_type", nbins=20) fig.show() - fig = px.box(self.data, x='cluster', y='amount') + fig = px.box(self.data, x="cluster", y="amount") fig.show() diff --git a/database/database.py b/database/database.py index e8b01f30d..297d7e2e7 100644 --- a/database/database.py +++ b/database/database.py @@ -1,20 +1,27 @@ import sqlite3 + class Database: def __init__(self): - self.connection = sqlite3.connect('banking.db') + self.connection = sqlite3.connect("banking.db") self.cursor = self.connection.cursor() def create_table(self): - self.cursor.execute('''CREATE TABLE IF NOT EXISTS accounts (id INTEGER PRIMARY KEY, name TEXT, balance REAL)''') + self.cursor.execute( + """CREATE TABLE IF NOT EXISTS accounts (id INTEGER PRIMARY KEY, name TEXT, balance REAL)""" + ) self.connection.commit() def insert_account(self, name, balance): - self.cursor.execute("INSERT INTO accounts (name, balance) VALUES (?, ?)", (name, balance)) + self.cursor.execute( + "INSERT INTO accounts (name, balance) VALUES (?, ?)", (name, balance) + ) self.connection.commit() def update_account(self, id, name, balance): - self.cursor.execute("UPDATE accounts SET name=?, balance=? WHERE id=?", (name, balance, id)) + self.cursor.execute( + "UPDATE accounts SET name=?, balance=? WHERE id=?", (name, balance, id) + ) self.connection.commit() def delete_account(self, id): diff --git a/fraud_detection/anomaly_detection.py b/fraud_detection/anomaly_detection.py index 290777c23..1295cd588 100644 --- a/fraud_detection/anomaly_detection.py +++ b/fraud_detection/anomaly_detection.py @@ -2,6 +2,7 @@ import pandas as pd from sklearn.ensemble import IsolationForest + class AnomalyDetection: def __init__(self, transaction_data): self.transaction_data = transaction_data diff --git a/fraud_detection/fraud_detection_model.py b/fraud_detection/fraud_detection_model.py index 7182d5030..edac1a2fa 100644 --- a/fraud_detection/fraud_detection_model.py +++ b/fraud_detection/fraud_detection_model.py @@ -3,14 +3,17 @@ from sklearn.ensemble import RandomForestClassifier from sklearn.model_selection import train_test_split + class FraudDetectionModel: def __init__(self, data): self.data = data - self.X = self.data.drop('fraudulent', axis=1) - self.y = self.data['fraudulent'] + self.X = self.data.drop("fraudulent", axis=1) + self.y = self.data["fraudulent"] def train_model(self): - X_train, X_test, y_train, y_test = train_test_split(self.X, self.y, test_size=0.2, random_state=42) + X_train, X_test, y_train, y_test = train_test_split( + self.X, self.y, test_size=0.2, random_state=42 + ) self.model = RandomForestClassifier(n_estimators=100, random_state=42) self.model.fit(X_train, y_train) diff --git a/fraud_detection/fraud_response.py b/fraud_detection/fraud_response.py index b4ee601cd..199695e3e 100644 --- a/fraud_detection/fraud_response.py +++ b/fraud_detection/fraud_response.py @@ -6,13 +6,13 @@ def __init__(self, fraud_detection_model, anomaly_detection): def respond_to_fraud(self, transactions): for transaction in transactions: fraud_level = self.fraud_detection_model.predict_fraud(transaction) - if fraud_level == 'fraudulent': + if fraud_level == "fraudulent": anomaly_score = self.anomaly_detection.detect_anomalies(transaction) if anomaly_score == -1: # Implement fraud response strategies here # For example, you could flag the transaction for review or decline it altogether - print('Fraudulent transaction detected: Flagged for review') + print("Fraudulent transaction detected: Flagged for review") else: - print('Anomalous transaction detected: Flagged for review') + print("Anomalous transaction detected: Flagged for review") else: - print('Non-fraudulent transaction detected: Transaction approved') + print("Non-fraudulent transaction detected: Transaction approved") diff --git a/logging/logging.py b/logging/logging.py index 0db567ee3..dea11d2be 100644 --- a/logging/logging.py +++ b/logging/logging.py @@ -1,12 +1,13 @@ import logging + class Logging: def __init__(self): - self.logger = logging.getLogger('banking_network') + self.logger = logging.getLogger("banking_network") self.logger.setLevel(logging.DEBUG) def setup_logging(self): - formatter = logging.Formatter('%(asctime)s - %(levelname)s - %(message)s') + formatter = logging.Formatter("%(asctime)s - %(levelname)s - %(message)s") console_handler = logging.StreamHandler() console_handler.setLevel(logging.DEBUG) console_handler.setFormatter(formatter) diff --git a/machine_learning/machine_learning.py b/machine_learning/machine_learning.py index f53b8d14a..67f4caf10 100644 --- a/machine_learning/machine_learning.py +++ b/machine_learning/machine_learning.py @@ -1,19 +1,20 @@ -import pandas as pd import numpy as np +import pandas as pd from sklearn.linear_model import LinearRegression + class MachineLearning: def __init__(self): pass def train_model(self, data): - X = data[['feature1', 'feature2']] - y = data['target'] + X = data[["feature1", "feature2"]] + y = data["target"] model = LinearRegression() model.fit(X, y) return model def predict(self, model, data): - X = data[['feature1', 'feature2']] + X = data[["feature1", "feature2"]] y = model.predict(X) return y diff --git a/network/networking.py b/network/networking.py index 0425688bf..177b46779 100644 --- a/network/networking.py +++ b/network/networking.py @@ -1,8 +1,9 @@ import socket + class Networking: def __init__(self): - self.host = '0.0.0.0' + self.host = "0.0.0.0" self.port = 8080 self.socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) self.socket.bind((self.host, self.port)) @@ -11,7 +12,7 @@ def listen_for_connections(self): self.socket.listen(5) while True: connection, address = self.socket.accept() - print(f'Connection from {address} has been established!') + print(f"Connection from {address} has been established!") self.handle_connection(connection) def handle_connection(self, connection): diff --git a/risk_management/risk_assessment.py b/risk_management/risk_assessment.py index 6bdb51647..22e91a390 100644 --- a/risk_management/risk_assessment.py +++ b/risk_management/risk_assessment.py @@ -3,14 +3,17 @@ from sklearn.ensemble import RandomForestClassifier from sklearn.model_selection import train_test_split + class RiskAssessment: def __init__(self, data): self.data = data - self.X = self.data.drop('risk_level', axis=1) - self.y = self.data['risk_level'] + self.X = self.data.drop("risk_level", axis=1) + self.y = self.data["risk_level"] def train_model(self): - X_train, X_test, y_train, y_test = train_test_split(self.X, self.y, test_size=0.2, random_state=42) + X_train, X_test, y_train, y_test = train_test_split( + self.X, self.y, test_size=0.2, random_state=42 + ) self.model = RandomForestClassifier(n_estimators=100, random_state=42) self.model.fit(X_train, y_train) diff --git a/risk_management/risk_mitigation.py b/risk_management/risk_mitigation.py index 77d3327ca..c7166831e 100644 --- a/risk_management/risk_mitigation.py +++ b/risk_management/risk_mitigation.py @@ -4,13 +4,13 @@ def __init__(self, risk_assessment_model): def mitigate_risk(self, transaction): risk_level = self.risk_assessment_model.predict_risk(transaction) - if risk_level == 'high': + if risk_level == "high": # Implement risk mitigation strategies here # For example, you could flag the transaction for review or decline it altogether - return 'Flagged for review' - elif risk_level == 'medium': + return "Flagged for review" + elif risk_level == "medium": # Implement risk mitigation strategies here # For example, you could require additional authentication or verification - return 'Additional authentication required' + return "Additional authentication required" else: - return 'Transaction approved' + return "Transaction approved" diff --git a/risk_management/risk_monitoring.py b/risk_management/risk_monitoring.py index 52eaa2db9..653c69773 100644 --- a/risk_management/risk_monitoring.py +++ b/risk_management/risk_monitoring.py @@ -1,5 +1,6 @@ import time + class RiskMonitoring: def __init__(self, risk_assessment_model, risk_mitigation_model): self.risk_assessment_model = risk_assessment_model @@ -8,12 +9,16 @@ def __init__(self, risk_assessment_model, risk_mitigation_model): def monitor_risks(self, transactions): for transaction in transactions: risk_level = self.risk_assessment_model.predict_risk(transaction) - if risk_level == 'high': - mitigation_result = self.risk_mitigation_model.mitigate_risk(transaction) - print(f'High-risk transaction detected: {mitigation_result}') - elif risk_level == 'medium': - mitigation_result = self.risk_mitigation_model.mitigate_risk(transaction) - print(f'Medium-risk transaction detected: {mitigation_result}') + if risk_level == "high": + mitigation_result = self.risk_mitigation_model.mitigate_risk( + transaction + ) + print(f"High-risk transaction detected: {mitigation_result}") + elif risk_level == "medium": + mitigation_result = self.risk_mitigation_model.mitigate_risk( + transaction + ) + print(f"Medium-risk transaction detected: {mitigation_result}") else: - print(f'Low-risk transaction detected: Transaction approved') + print(f"Low-risk transaction detected: Transaction approved") time.sleep(1) diff --git a/security/authentication.py b/security/authentication.py index f85c9eb66..54d08b6c3 100644 --- a/security/authentication.py +++ b/security/authentication.py @@ -1,14 +1,19 @@ import hashlib import secrets + class Authentication: def __init__(self): self.salt = secrets.token_hex(16) def generate_password_hash(self, password): - password_hash = hashlib.pbkdf2_hmac('sha256', password.encode(), self.salt.encode(), 100000) + password_hash = hashlib.pbkdf2_hmac( + "sha256", password.encode(), self.salt.encode(), 100000 + ) return password_hash.hex() def verify_password(self, password, password_hash): - password_hash_check = hashlib.pbkdf2_hmac('sha256', password.encode(), self.salt.encode(), 100000) + password_hash_check = hashlib.pbkdf2_hmac( + "sha256", password.encode(), self.salt.encode(), 100000 + ) return password_hash_check.hex() == password_hash diff --git a/src/app.js b/src/app.js index a2ed7cd55..286ba6401 100644 --- a/src/app.js +++ b/src/app.js @@ -1,35 +1,35 @@ -const express = require("express"); -const bodyParser = require("body-parser"); -const cors = require("cors"); -const mongoose = require("mongoose"); +const express = require('express') +const bodyParser = require('body-parser') +const cors = require('cors') +const mongoose = require('mongoose') -const { JWT_SECRET, MONGODB_URI } = process.env; +const { JWT_SECRET, MONGODB_URI } = process.env -const app = express(); +const app = express() -app.use(bodyParser.json()); -app.use(cors()); +app.use(bodyParser.json()) +app.use(cors()) -const router = require("./routes"); -app.use("/api", router); +const router = require('./routes') +app.use('/api', router) const connectToMongoDB = async () => { try { await mongoose.connect(MONGODB_URI, { useNewUrlParser: true, useUnifiedTopology: true, - useCreateIndex: true, - }); - console.log("Connected to MongoDB"); + useCreateIndex: true + }) + console.log('Connected to MongoDB') } catch (error) { - console.error("Failed to connect to MongoDB", error); - process.exit(1); + console.error('Failed to connect to MongoDB', error) + process.exit(1) } -}; +} -connectToMongoDB(); +connectToMongoDB() -const PORT = process.env.PORT || 5000; +const PORT = process.env.PORT || 5000 app.listen(PORT, () => { - console.log(`Server running on port ${PORT}`); -}); + console.log(`Server running on port ${PORT}`) +}) diff --git a/src/controllers/accounts.js b/src/controllers/accounts.js index 7a59dbdaf..8b93c643f 100644 --- a/src/controllers/accounts.js +++ b/src/controllers/accounts.js @@ -1,16 +1,16 @@ -const Account = require("../models/account"); +const Account = require('../models/account') exports.list = async (req, res) => { - const accounts = await Account.find({ owner: req.user._id }); - res.json(accounts); -}; + const accounts = await Account.find({ owner: req.user._id }) + res.json(accounts) +} exports.create = async (req, res) => { const account = new Account({ account_name: req.body.account_name, initial_balance: req.body.initial_balance, - owner: req.user._id, - }); - await account.save(); - res.json({ message: "Account created!", account }); -}; + owner: req.user._id + }) + await account.save() + res.json({ message: 'Account created!', account }) +} diff --git a/src/controllers/index.js b/src/controllers/index.js index 639a0fa07..6a19656de 100644 --- a/src/controllers/index.js +++ b/src/controllers/index.js @@ -1,10 +1,10 @@ -const express = require("express"); -const router = express.Router(); +const express = require('express') +const router = express.Router() -const AccountsController = require("./accounts"); -const TransfersController = require("./transfers"); +const AccountsController = require('./accounts') +const TransfersController = require('./transfers') -router.use("/accounts", AccountsController); -router.use("/transfers", TransfersController); +router.use('/accounts', AccountsController) +router.use('/transfers', TransfersController) -module.exports = router; +module.exports = router diff --git a/src/controllers/transfers.js b/src/controllers/transfers.js index ecc8b9e1f..b20c4b778 100644 --- a/src/controllers/transfers.js +++ b/src/controllers/transfers.js @@ -1,31 +1,31 @@ -const Transfer = require("../models/transfer"); -const Account = require("../models/account"); +const Transfer = require('../models/transfer') +const Account = require('../models/account') exports.create = async (req, res) => { - const senderAccount = await Account.findById(req.body.sender_account_id); - const receiverAccount = await Account.findById(req.body.receiver_account_id); + const senderAccount = await Account.findById(req.body.sender_account_id) + const receiverAccount = await Account.findById(req.body.receiver_account_id) if (!senderAccount || !receiverAccount) { - return res.status(400).json({ message: "Account not found" }); + return res.status(400).json({ message: 'Account not found' }) } if (senderAccount.balance < req.body.amount) { - return res.status(400).json({ message: "Insufficient balance" }); + return res.status(400).json({ message: 'Insufficient balance' }) } const transfer = new Transfer({ sender_account_id: req.body.sender_account_id, receiver_account_id: req.body.receiver_account_id, - amount: req.body.amount, - }); + amount: req.body.amount + }) - await transfer.save(); + await transfer.save() - senderAccount.balance -= req.body.amount; - receiverAccount.balance += req.body.amount; + senderAccount.balance -= req.body.amount + receiverAccount.balance += req.body.amount - await senderAccount.save(); - await receiverAccount.save(); + await senderAccount.save() + await receiverAccount.save() - res.json({ message: "Transfer successful", transfer }); -}; + res.json({ message: 'Transfer successful', transfer }) +} diff --git a/src/middleware/auth.js b/src/middleware/auth.js index 46ae82670..3c436ec5a 100644 --- a/src/middleware/auth.js +++ b/src/middleware/auth.js @@ -1,20 +1,20 @@ -const jwt = require("jsonwebtoken"); -const User = require("../models/user"); +const jwt = require('jsonwebtoken') +const User = require('../models/user') const authenticate = async (req, res, next) => { - const token = req.header("Authorization")?.split(" ")[1]; + const token = req.header('Authorization')?.split(' ')[1] if (!token) { - return res.status(401).json({ message: "Access Denied" }); + return res.status(401).json({ message: 'Access Denied' }) } try { - const decoded = jwt.verify(token, process.env.JWT_SECRET); - req.user = await User.findById(decoded.user_id); - next(); + const decoded = jwt.verify(token, process.env.JWT_SECRET) + req.user = await User.findById(decoded.user_id) + next() } catch (error) { - res.status(400).json({ message: "Invalid Token" }); + res.status(400).json({ message: 'Invalid Token' }) } -}; +} -module.exports = authenticate; +module.exports = authenticate diff --git a/src/middleware/network.js b/src/middleware/network.js index 8e58f6c1b..636f053e5 100644 --- a/src/middleware/network.js +++ b/src/middleware/network.js @@ -1,30 +1,40 @@ -const { PRIVATE_KEY, PRIVATE_KEY_PASSWORD, PRIVATE_KEY_PASSWORD_SALT, ENDPOINT, CHAIN_ID } = process.env; +const { + PRIVATE_KEY, + PRIVATE_KEY_PASSWORD, + PRIVATE_KEY_PASSWORD_SALT, + ENDPOINT, + CHAIN_ID +} = process.env -const Web3 = require("web3"); -const EthereumTx = require("ethereumjs-tx"); +const Web3 = require('web3') +const EthereumTx = require('ethereumjs-tx') -const web3 = new Web3(new Web3.providers.HttpProvider(ENDPOINT)); +const web3 = new Web3(new Web3.providers.HttpProvider(ENDPOINT)) const createTransaction = async (from, to, value) => { - const nonce = await web3.eth.getTransactionCount(from); - const gasPrice = await web3.eth.getGasPrice(); - const gasLimit = 21000; + const nonce = await web3.eth.getTransactionCount(from) + const gasPrice = await web3.eth.getGasPrice() + const gasLimit = 21000 const tx = new EthereumTx({ nonce, gasPrice, gasLimit, to, - value, - }); + value + }) - const privateKey = web3.utils.sha3(PRIVATE_KEY_PASSWORD + PRIVATE_KEY_PASSWORD_SALT); - tx.sign(privateKey); + const privateKey = web3.utils.sha3( + PRIVATE_KEY_PASSWORD + PRIVATE_KEY_PASSWORD_SALT + ) + tx.sign(privateKey) - const serializedTx = tx.serialize(); - const transactionHash = await web3.eth.sendSignedTransaction("0x" + serializedTx.toString("hex")); + const serializedTx = tx.serialize() + const transactionHash = await web3.eth.sendSignedTransaction( + '0x' + serializedTx.toString('hex') + ) - return transactionHash; -}; + return transactionHash +} -module.exports = createTransaction; +module.exports = createTransaction diff --git a/src/models/transfer.js b/src/models/transfer.js index d1cf4802b..ca77421f7 100644 --- a/src/models/transfer.js +++ b/src/models/transfer.js @@ -1,12 +1,16 @@ -const mongoose = require("mongoose"); +const mongoose = require('mongoose') const transferSchema = new mongoose.Schema({ - sender_id: { type: mongoose.Schema.Types.ObjectId, ref: "User", required: true }, + sender_id: { + type: mongoose.Schema.Types.ObjectId, + ref: 'User', + required: true + }, receiver_account_id: { type: String, required: true }, amount: { type: Number, required: true }, - transaction_hash: { type: String }, -}); + transaction_hash: { type: String } +}) -const Transfer = mongoose.model("Transfer", transferSchema); +const Transfer = mongoose.model('Transfer', transferSchema) -module.exports = Transfer; +module.exports = Transfer diff --git a/src/models/user.js b/src/models/user.js index e827eec76..2e171d0da 100644 --- a/src/models/user.js +++ b/src/models/user.js @@ -1,20 +1,20 @@ -const mongoose = require("mongoose"); -const bcrypt = require("bcrypt"); +const mongoose = require('mongoose') +const bcrypt = require('bcrypt') const UserSchema = new mongoose.Schema({ username: { type: String, required: true, unique: true }, - password: { type: String, required: true }, -}); + password: { type: String, required: true } +}) -UserSchema.pre("save", async function (next) { - if (this.isModified("password")) { - this.password = await bcrypt.hash(this.password, 10); +UserSchema.pre('save', async function (next) { + if (this.isModified('password')) { + this.password = await bcrypt.hash(this.password, 10) } - next(); -}); + next() +}) UserSchema.methods.comparePassword = function (password) { - return bcrypt.compare(password, this.password); -}; + return bcrypt.compare(password, this.password) +} -module.exports = mongoose.model("User", UserSchema); +module.exports = mongoose.model('User', UserSchema) diff --git a/src/routes/index.js b/src/routes/index.js index d1800ab91..cdb9ac6b2 100644 --- a/src/routes/index.js +++ b/src/routes/index.js @@ -1,8 +1,8 @@ -const express = require("express"); -const router = express.Router(); +const express = require('express') +const router = express.Router() -router.get("/", (req, res) => { - res.json({ message: "Welcome to the PI Nexus Autonomous Banking Network!" }); -}); +router.get('/', (req, res) => { + res.json({ message: 'Welcome to the PI Nexus Autonomous Banking Network!' }) +}) -module.exports = router; +module.exports = router diff --git a/src/routes/transfers.js b/src/routes/transfers.js index 01b221fe7..a61bd45bc 100644 --- a/src/routes/transfers.js +++ b/src/routes/transfers.js @@ -1,20 +1,24 @@ -const express = require("express"); -const router = express.Router(); +const express = require('express') +const router = express.Router() -const TransferController = require("../controllers/transfers"); -const authenticate = require("../middleware/auth"); -const createTransaction = require("../middleware/network"); +const TransferController = require('../controllers/transfers') +const authenticate = require('../middleware/auth') +const createTransaction = require('../middleware/network') -router.post("/", authenticate, async (req, res) => { +router.post('/', authenticate, async (req, res) => { try { - const transfer = await TransferController.create(req, res); - const transactionHash = await createTransaction(req.user.wallet_address, req.body.receiver_account_id, req.body.amount); - transfer.transaction_hash = transactionHash; - await transfer.save(); - res.json({ message: "Transfer successful", transfer }); + const transfer = await TransferController.create(req, res) + const transactionHash = await createTransaction( + req.user.wallet_address, + req.body.receiver_account_id, + req.body.amount + ) + transfer.transaction_hash = transactionHash + await transfer.save() + res.json({ message: 'Transfer successful', transfer }) } catch (error) { - res.status(400).json({ message: error.message }); + res.status(400).json({ message: error.message }) } -}); +}) -module.exports = router; +module.exports = router diff --git a/src/tests/integration_tests/test_ui.py b/src/tests/integration_tests/test_ui.py index 9d7546cde..60627e4e2 100644 --- a/src/tests/integration_tests/test_ui.py +++ b/src/tests/integration_tests/test_ui.py @@ -2,13 +2,15 @@ import pytest from selenium import webdriver from selenium.webdriver.common.by import By -from selenium.webdriver.support.ui import WebDriverWait from selenium.webdriver.support import expected_conditions as EC +from selenium.webdriver.support.ui import WebDriverWait + @pytest.fixture def browser(): return webdriver.Chrome() + def test_login(browser): # Test login functionality browser.get("https://pi-nexus-autonomous-banking-network.com/login") @@ -19,12 +21,14 @@ def test_login(browser): browser.find_element_by_name("login").click() WebDriverWait(browser, 10).until(EC.title_contains("Dashboard")) + def test_account_list(browser): # Test account list page browser.get("https://pi-nexus-autonomous-banking-network.com/accounts") account_list = browser.find_elements_by_css_selector(".account-list li") assert len(account_list) > 0 + def test_create_account(browser): # Test create account functionality browser.get("https://pi-nexus-autonomous-banking-network.com/accounts/new") diff --git a/system_monitoring/issue_detection.py b/system_monitoring/issue_detection.py index 8e65205c6..ee70d6e36 100644 --- a/system_monitoring/issue_detection.py +++ b/system_monitoring/issue_detection.py @@ -4,8 +4,8 @@ def __init__(self, system_monitoring): def detect_issues(self): if self.system_monitoring.cpu_percent > 80: - print('High CPU usage detected!') + print("High CPU usage detected!") if self.system_monitoring.memory_percent > 80: - print('High memory usage detected!') + print("High memory usage detected!") if self.system_monitoring.disk_percent > 80: - print('High disk usage detected!') + print("High disk usage detected!") diff --git a/system_monitoring/self_healing.py b/system_monitoring/self_healing.py index c0cc472de..f88eed8f4 100644 --- a/system_monitoring/self_healing.py +++ b/system_monitoring/self_healing.py @@ -1,10 +1,11 @@ import subprocess + class SelfHealing: def __init__(self): pass def heal_system(self): # Implement automatic healing techniques - subprocess.run(['systemctl', 'restart', 'nginx']) - subprocess.run(['systemctl', 'restart', 'postgresql']) + subprocess.run(["systemctl", "restart", "nginx"]) + subprocess.run(["systemctl", "restart", "postgresql"]) diff --git a/system_monitoring/system_monitoring.py b/system_monitoring/system_monitoring.py index 9db46a92e..2c874966e 100644 --- a/system_monitoring/system_monitoring.py +++ b/system_monitoring/system_monitoring.py @@ -1,18 +1,20 @@ -import psutil import time +import psutil + + class SystemMonitoring: def __init__(self): self.cpu_percent = psutil.cpu_percent() self.memory_percent = psutil.virtual_memory().percent - self.disk_percent = psutil.disk_usage('/').percent + self.disk_percent = psutil.disk_usage("/").percent def monitor_system(self): while True: time.sleep(60) self.cpu_percent = psutil.cpu_percent() self.memory_percent = psutil.virtual_memory().percent - self.disk_percent = psutil.disk_usage('/').percent - print(f'CPU: {self.cpu_percent}%') - print(f'Memory: {self.memory_percent}%') - print(f'Disk: {self.disk_percent}%') + self.disk_percent = psutil.disk_usage("/").percent + print(f"CPU: {self.cpu_percent}%") + print(f"Memory: {self.memory_percent}%") + print(f"Disk: {self.disk_percent}%") diff --git a/testing.py b/testing.py index c6db305da..c507e3c5d 100644 --- a/testing.py +++ b/testing.py @@ -1,8 +1,10 @@ import unittest + class Testing(unittest.TestCase): def test_something(self): self.assertTrue(True) -if __name__ == '__main__': + +if __name__ == "__main__": unittest.main() diff --git a/transaction_processing/transaction_processing.py b/transaction_processing/transaction_processing.py index df41a2cb5..133311667 100644 --- a/transaction_processing/transaction_processing.py +++ b/transaction_processing/transaction_processing.py @@ -1,5 +1,6 @@ import time + class TransactionProcessor: def __init__(self, transaction_validator, transaction_router): self.transaction_validator = transaction_validator @@ -8,11 +9,11 @@ def __init__(self, transaction_validator, transaction_router): def process_transaction(self, transaction): errors = self.transaction_validator.validate_transaction(transaction) if errors: - print('Transaction errors:', errors) + print("Transaction errors:", errors) return False else: channel = self.transaction_router.route_transaction(transaction) - print('Transaction routed to:', channel) + print("Transaction routed to:", channel) time.sleep(1) - print('Transaction processed successfully.') + print("Transaction processed successfully.") return True diff --git a/transaction_processing/transaction_routing.py b/transaction_processing/transaction_routing.py index e11ee5ea2..499748e38 100644 --- a/transaction_processing/transaction_routing.py +++ b/transaction_processing/transaction_routing.py @@ -1,18 +1,20 @@ class TransactionRouter: def __init__(self): self.routing_rules = { - 'domestic': { - 'regex': r'^\d{10}$', - 'message': 'Invalid account number format. Please enter a 10-digit account number.' + "domestic": { + "regex": r"^\d{10}$", + "message": "Invalid account number format. Please enter a 10-digit account number.", + }, + "international": { + "regex": r"^\d{12}$", + "message": "Invalid account number format. Please enter a 12-digit account number.", }, - 'international': { - 'regex': r'^\d{12}$', - 'message': 'Invalid account number format. Please enter a 12-digit account number.' - } } def route_transaction(self, transaction): for rule in self.routing_rules: - if re.match(self.routing_rules[rule]['regex'], transaction['account_number']): + if re.match( + self.routing_rules[rule]["regex"], transaction["account_number"] + ): return rule - raise ValueError('Invalid account number format.') + raise ValueError("Invalid account number format.") diff --git a/transaction_processing/transaction_validation.py b/transaction_processing/transaction_validation.py index f119832ff..23da6ba7e 100644 --- a/transaction_processing/transaction_validation.py +++ b/transaction_processing/transaction_validation.py @@ -1,17 +1,18 @@ import re + class TransactionValidation: def __init__(self): self.validation_rules = { - 'account_number': { - 'regex': r'^\d{10}$', - 'message': 'Invalid account number format. Please enter a 10-digit account number.' + "account_number": { + "regex": r"^\d{10}$", + "message": "Invalid account number format. Please enter a 10-digit account number.", + }, + "amount": { + "min": 0, + "max": 1000000, + "message": "Transaction amount must be between 0 and 1,000,000.", }, - 'amount': { - 'min': 0, - 'max': 1000000, - 'message': 'Transaction amount must be between 0 and 1,000,000.' - } } def validate_transaction(self, transaction): @@ -19,13 +20,17 @@ def validate_transaction(self, transaction): for field, rules in self.validation_rules.items(): if field in transaction: for rule in rules: - if rule == 'regex': + if rule == "regex": if not re.match(rules[rule], transaction[field]): - errors.append(rules['message']) - elif rule == 'min': + errors.append(rules["message"]) + elif rule == "min": if float(transaction[field]) < rules[rule]: - errors.append(f'Transaction amount must be at least {rules[rule]}') - elif rule == 'max': + errors.append( + f"Transaction amount must be at least {rules[rule]}" + ) + elif rule == "max": if float(transaction[field]) > rules[rule]: - errors.append(f'Transaction amount must not exceed {rules[rule]}') + errors.append( + f"Transaction amount must not exceed {rules[rule]}" + ) return errors