Compare commits
133 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 000c459d76 | |||
| 307f65c898 | |||
| 5db54adfb7 | |||
| 19fb9a85fc | |||
| 9d48ebbad9 | |||
| 077b5257e8 | |||
| b84602a5ed | |||
| 43541c1af2 | |||
| 8c4d0fa1a1 | |||
| 16fed7a8ca | |||
| c1009a19fb | |||
| be189cf899 | |||
| 8116d1d4de | |||
| ab698c8903 | |||
| 74b16f2165 | |||
| fafbbe9b3a | |||
| 04b32c3f33 | |||
| 0babcbfac4 | |||
| 2da57d95b7 | |||
| d317c03819 | |||
| 2e187b5e08 | |||
| 064343c1ff | |||
| efc67c63d8 | |||
| a978377501 | |||
| 1bf05a1b87 | |||
| 5e3c993a15 | |||
| 4220bb629f | |||
| bdd30c238c | |||
| 778d799113 | |||
| 47cd1b6a91 | |||
| 118232e906 | |||
| 64c84fe708 | |||
| cff3c02851 | |||
| 60577d02c7 | |||
| 99749bd9c9 | |||
| b3ca2e5927 | |||
| b71e9e2e80 | |||
| 72a8aacabe | |||
| a0e9cfb40d | |||
| 46707c1149 | |||
| 497efcfa76 | |||
| 1201c248ee | |||
| 395eb31801 | |||
| b27edd9818 | |||
| 51c43e9893 | |||
| ec2019430e | |||
| 05174e93ad | |||
| e8ec94bfa8 | |||
| c6642e028c | |||
| 6d5a5f46b9 | |||
| e651eb86ca | |||
| 84171da29b | |||
| d463ec689b | |||
| 416a82058b | |||
| 6f2324ad1b | |||
| c19f13edc1 | |||
| 390b137e0b | |||
| 0c7d4a83f1 | |||
| 25112cb03a | |||
| 5ade4b35a0 | |||
| 00067c56d4 | |||
| 620566490b | |||
| 7245f4cc69 | |||
| 575fecb163 | |||
| 4120b1abfa | |||
| 801d3739fc | |||
| 92bdf14508 | |||
| fa25a82693 | |||
| 1684f3e0cb | |||
| a31f807da8 | |||
| 6d92b49f3d | |||
| 11cdf975bc | |||
| 7bacc28e3b | |||
| 4bbb71d434 | |||
| 8da153a476 | |||
| 1189325638 | |||
| e93459ae55 | |||
| b1fcf51546 | |||
| 75716368bb | |||
| ec15e2bb63 | |||
| 9a3e030ce7 | |||
| 4f5894ef4a | |||
| e0b5610d90 | |||
| bc34519a26 | |||
| dc69bfa8c7 | |||
| cf7c5fc23a | |||
| 16d5b80657 | |||
| 8dd153fb53 | |||
| 89bfb28019 | |||
| 1215e47297 | |||
| 4016899179 | |||
| 1cad9b83fb | |||
| 780ee76dc3 | |||
| 4d8e17f59f | |||
| 21c27e22da | |||
| 2e978089b5 | |||
| 803fe7147e | |||
| 6490364a5d | |||
| 2ad773952e | |||
| 138e71107e | |||
| bde399f589 | |||
| 8a43602363 | |||
| 5a56790fd4 | |||
| 892f681174 | |||
| 997b5420ee | |||
| 643bbe3af2 | |||
| f85f0f19b9 | |||
| 3f9a1bafa0 | |||
| 6f72a03ecf | |||
| 5b8e4c634e | |||
| 70c3f7162e | |||
| cb9199431a | |||
| cba9fe1e7b | |||
| baa49e77eb | |||
| b015462e50 | |||
| c9f1fc1bea | |||
| 1177886271 | |||
| 0e1c56dd18 | |||
| eefe237dff | |||
| 8d4e004177 | |||
| 1c63e2b856 | |||
| 3040cbf49a | |||
| 1a124a2571 | |||
| 26c8c3f3b9 | |||
| 50d814ebf6 | |||
| 7fc20876dd | |||
| 183108400e | |||
| 3055d34979 | |||
| 747f5f5f42 | |||
| 4db9409b94 | |||
| 8693bb29ca | |||
| 524d8ca41d | |||
| 3c77eca689 |
@@ -0,0 +1,16 @@
|
||||
.git
|
||||
.env
|
||||
node_modules
|
||||
packages
|
||||
vendor
|
||||
tests
|
||||
.DS_Store
|
||||
vapor.yml
|
||||
.vapor
|
||||
storage/app/livewire-tmp/*
|
||||
storage/app/public/profile-photos/*
|
||||
storage/framework/cache/*
|
||||
storage/framework/sessions/*
|
||||
storage/framework/testing/*
|
||||
storage/framework/views/*
|
||||
storage/framework/logs/*
|
||||
+25
-10
@@ -1,24 +1,35 @@
|
||||
APP_NAME=Investbrain
|
||||
APP_ENV=production
|
||||
# Generate a secure key using `openssl rand -base64 32`
|
||||
APP_KEY=
|
||||
APP_DEBUG=false
|
||||
APP_TIMEZONE=UTC
|
||||
|
||||
# Port for NGINX to listen on
|
||||
APP_PORT=8000
|
||||
|
||||
# Used internally to generate absolute links
|
||||
APP_URL="http://localhost:${APP_PORT}"
|
||||
SELF_HOSTED=true
|
||||
|
||||
# Webroot for static assets (css, js, images, etc)
|
||||
ASSET_URL="${APP_URL}"
|
||||
|
||||
# Enables or disables new user registration
|
||||
REGISTRATION_ENABLED=true
|
||||
|
||||
# ASSET_URL="http://localhost:8000" # (optional) webroot for static assets (css, js, images, etc)
|
||||
|
||||
# Enable or disable AI chat feature
|
||||
AI_CHAT_ENABLED=false
|
||||
|
||||
# API key for OpenAI (for Llama support, see docs)
|
||||
OPENAI_API_KEY=
|
||||
OPENAI_ORGANIZATION=
|
||||
|
||||
# Market data provider to use (comma separated list)
|
||||
MARKET_DATA_PROVIDER=yahoo
|
||||
MARKET_DATA_REFRESH=30
|
||||
ALPHAVANTAGE_API_KEY=
|
||||
FINNHUB_API_KEY=
|
||||
|
||||
# Cadence to refresh market data (in minutes)
|
||||
MARKET_DATA_REFRESH=30
|
||||
DAILY_CHANGE_TIME=
|
||||
|
||||
#### Advanced configurations ####
|
||||
ENABLED_LOGIN_PROVIDERS=
|
||||
GITHUB_CLIENT_ID=
|
||||
GITHUB_CLIENT_SECRET=
|
||||
@@ -29,9 +40,13 @@ LINKEDIN_CLIENT_SECRET=
|
||||
FACEBOOK_CLIENT_ID=
|
||||
FACEBOOK_CLIENT_SECRET=
|
||||
|
||||
APP_NAME=Investbrain
|
||||
APP_TIMEZONE=UTC
|
||||
APP_ENV=production
|
||||
APP_DEBUG=true
|
||||
APP_LOCALE=en
|
||||
APP_FALLBACK_LOCALE=en
|
||||
APP_FAKER_LOCALE=en_US
|
||||
SELF_HOSTED=true
|
||||
|
||||
DB_CONNECTION=mysql
|
||||
DB_HOST=investbrain-mysql
|
||||
@@ -52,7 +67,7 @@ QUEUE_CONNECTION=redis
|
||||
CACHE_STORE=redis
|
||||
|
||||
REDIS_CLIENT=predis
|
||||
REDIS_HOST=127.0.0.1
|
||||
REDIS_HOST=investbrain-redis
|
||||
REDIS_PATH=/tmp/database_server.sock
|
||||
REDIS_PASSWORD=null
|
||||
REDIS_PORT=6379
|
||||
|
||||
@@ -0,0 +1,58 @@
|
||||
name: Build and push Docker images
|
||||
|
||||
on:
|
||||
push:
|
||||
tags:
|
||||
- "v*"
|
||||
workflow_dispatch:
|
||||
|
||||
jobs:
|
||||
build:
|
||||
runs-on: ubuntu-22.04 #ubuntu-latest
|
||||
steps:
|
||||
- name: Increase swap space
|
||||
run: sudo /bin/dd if=/dev/zero of=/var/swap.1 bs=1M count=5120 && sudo chmod 600 /var/swap.1 && sudo /sbin/mkswap /var/swap.1 && sudo /sbin/swapon /var/swap.1
|
||||
|
||||
- name: Login to Docker Hub
|
||||
uses: docker/login-action@v3
|
||||
with:
|
||||
username: ${{ vars.DOCKERHUB_USERNAME }}
|
||||
password: ${{ secrets.DOCKERHUB_TOKEN }}
|
||||
|
||||
- name: Login to GitHub Container Registry
|
||||
uses: docker/login-action@v3
|
||||
with:
|
||||
registry: ghcr.io
|
||||
username: ${{ github.actor }}
|
||||
password: ${{ secrets.GIT_HUB_TOKEN }}
|
||||
|
||||
- name: Set up QEMU
|
||||
uses: docker/setup-qemu-action@v3
|
||||
|
||||
- name: Set up Docker Buildx
|
||||
uses: docker/setup-buildx-action@v3
|
||||
|
||||
- name: Cache Docker layers
|
||||
uses: actions/cache@v4
|
||||
with:
|
||||
path: /tmp/.buildx-cache
|
||||
key: ${{ runner.os }}-buildx-${{ github.sha }}
|
||||
restore-keys: |
|
||||
${{ runner.os }}-buildx-
|
||||
|
||||
- name: Extract version from tag
|
||||
id: extract-version
|
||||
run: |
|
||||
echo "version=${GITHUB_REF_NAME#v}" >> $GITHUB_ENV
|
||||
|
||||
- name: Build and push
|
||||
uses: docker/build-push-action@v6
|
||||
with:
|
||||
platforms: linux/amd64,linux/arm64
|
||||
file: ./docker/Dockerfile
|
||||
push: true
|
||||
tags: |
|
||||
investbrainapp/investbrain:latest
|
||||
investbrainapp/investbrain:${{ env.version }}
|
||||
ghcr.io/investbrainapp/investbrain:latest
|
||||
ghcr.io/investbrainapp/investbrain:${{ env.version }}
|
||||
@@ -1,5 +1,13 @@
|
||||
|
||||
|
||||
<p align="center"><a href="https://investbra.in" target="_blank"><img src="https://raw.githubusercontent.com/investbrainapp/investbrain/main/investbrain-logo.png" width="400" alt="Investbrain Logo"></a></p>
|
||||
|
||||
[](https://github.com/investbrainapp/investbrain/)
|
||||
[](https://github.com/investbrainapp/investbrain/)
|
||||
[](https://github.com/investbrainapp/investbrain/issues)
|
||||
[](https://hub.docker.com/r/investbrainapp/investbrain/)
|
||||
|
||||
|
||||
## About Investbrain
|
||||
|
||||
Investbrain is a smart open-source investment tracker that helps you manage, track, and make informed decisions about your investments.
|
||||
@@ -11,6 +19,7 @@ Investbrain is a smart open-source investment tracker that helps you manage, tra
|
||||
- [Install (self hosting)](#self-hosting)
|
||||
- [Chat with your holdings](#chat-with-your-holdings)
|
||||
- [Market data providers](#market-data-providers)
|
||||
- [Import / Export](#import--export)
|
||||
- [Configuration](#configuration)
|
||||
- [Updating](#updating)
|
||||
- [Command line utilities](#command-line-utilities)
|
||||
@@ -19,31 +28,31 @@ Investbrain is a smart open-source investment tracker that helps you manage, tra
|
||||
|
||||
## Under the hood
|
||||
|
||||
Investbrain is a Laravel PHP web application that leverages Livewire and Tailwind for its frontend. Most databases should work, including MySQL and SQLite. Out of the box, we feature three market data providers: [Yahoo Finance](https://finance.yahoo.com/), [Finnhub](https://finnhub.io/pricing-stock-api-market-data), and [Alpha Vantage](https://www.alphavantage.co/support/). But we also offer an extensible market data provider interface for intrepid developers to create their own! We also offer an integration with OpenAI's LLMs for our ["chat with your holdings"](#chat-with-your-holdings) capability. Finally, of course we have robust support for i18n, a11y, and dark mode.
|
||||
Investbrain is a Laravel PHP web application that leverages Livewire and Tailwind for its frontend. Most databases should work, including MySQL and SQLite. Out of the box, we feature three market data providers: [Yahoo Finance](https://finance.yahoo.com/), [Finnhub](https://finnhub.io/pricing-stock-api-market-data), and [Alpha Vantage](https://www.alphavantage.co/support/). But we also offer an extensible market data provider interface for intrepid developers to create their own! We also offer integrations with OpenAI and Ollama for our ["chat with your holdings"](#chat-with-your-holdings) capability. Finally, of course we have robust support for i18n, a11y, and dark mode.
|
||||
|
||||
## Self hosting
|
||||
|
||||
For ease of installation, we _highly recommend_ installing Investbrain using the provided [Docker Compose](https://github.com/investbrainapp/investbrain/blob/main/docker-compose.yml) file, which downloads all the necessary dependencies and seamlessly builds everything you need to get started quickly!
|
||||
For ease of installation, we _highly recommend_ installing Investbrain using the provided [Docker Compose](https://github.com/investbrainapp/investbrain/blob/main/docker-compose.yml) file, which uses the official Investbrain Docker image and includes all the necessary dependencies to seamlessly build everything you need to get started quickly!
|
||||
|
||||
Before getting started, you should already have the following installed on your machine: [Docker Engine](https://docs.docker.com/engine/install/), [Git](https://git-scm.com/book/en/v2/Getting-Started-Installing-Git), and a wild sense of adventure.
|
||||
Before getting started, you should already have [Docker Engine](https://docs.docker.com/engine/install/) installed on your machine.
|
||||
|
||||
Ready? Let's get started!
|
||||
|
||||
First, you can clone this repository:
|
||||
**1. Copy Docker Compose file**
|
||||
|
||||
```bash
|
||||
git clone https://github.com/investbrainapp/investbrain.git && cd investbrain
|
||||
```
|
||||
Grab a copy of the [docker-compose.yml](https://github.com/investbrainapp/investbrain/blob/main/docker-compose.yml)** file and paste the contents into the directory where you plan to install Investbrain.
|
||||
|
||||
Then, build the Docker image and bring up the container (this will take a few minutes):
|
||||
**2. Set your environment**
|
||||
|
||||
```bash
|
||||
docker compose up
|
||||
```
|
||||
Adjust the `environment` properties in the Docker Compose file to your preferences. Alternatively, create a .env file in the same directory as your compose file, then reference the .env file using the `env_file` property.
|
||||
|
||||
In the previous step, all of the default configurations are set automatically. This includes creating a .env file and setting the required Laravel `APP_KEY`.
|
||||
_Importantly_, you need to set the `APP_KEY` value to a complex random value. If you're unsure, Investbrain will generate an `APP_KEY` for you on first run - but you must **manually** update your environment configuration with this generated value!
|
||||
|
||||
If everything worked as expected, you should now be able to access Investbrain in the browser at. You should create an account by visiting:
|
||||
> Tip: Want to know what other configuration options are available? You can reference the [.env.example](https://github.com/investbrainapp/investbrain/blob/main/.env.example) file in this respository for available environment configurations.
|
||||
|
||||
**3. Run `docker compose up`**
|
||||
|
||||
This might take a few minutes to pull the Docker images. But assuming everything worked as expected, you should now be able to access Investbrain in the browser by visiting:
|
||||
|
||||
```bash
|
||||
http://localhost:8000/register
|
||||
@@ -57,7 +66,9 @@ Investbrain offers an AI powered chat assistant that is grounded on *your* inves
|
||||
|
||||
When self-hosting, you can enable the chat assistant by configuring your OpenAI Secret Key and Organization ID in your [.env](https://github.com/investbrainapp/investbrain/blob/main/.env.example) file. Navigate to OpenAI to [create your keys](https://platform.openai.com/api-keys).
|
||||
|
||||
Always keep in mind the limitations of large language models. When in doubt, consult a licensed investment advisor.
|
||||
If you are self-hosting your own large language models ("LLMs") that expose an OpenAI compatible API (e.g. [Ollama](https://ollama.com/blog/openai-compatibility)), you can update the `OPENAI_BASE_URI` configuration to your self-hosted instance. Ensure you also update the `OPENAI_MODEL` to an available model.
|
||||
|
||||
Always keep in mind the limitations of LLMs. When in doubt, consult a licensed investment advisor.
|
||||
|
||||
## Market data providers
|
||||
|
||||
@@ -65,7 +76,7 @@ Investbrain includes an extensible market data provider interface that allows yo
|
||||
|
||||
### Configuration
|
||||
|
||||
You can specify the market data provider you want to use in your .env file:
|
||||
You can specify the market data provider you want to use in your environment variables:
|
||||
|
||||
```bash
|
||||
MARKET_DATA_PROVIDER=yahoo
|
||||
@@ -73,7 +84,7 @@ MARKET_DATA_PROVIDER=yahoo
|
||||
|
||||
You can also use Investbrain's built-in fallback mechanism to ensure reliable data access. If any provider fails, Investbrain will automatically attempt to retrieve data from the next available provider, continuing through your configured providers until one returns successfully.
|
||||
|
||||
Your selected providers should be listed in your .env file. Each should be separated by a comma:
|
||||
Your selected providers should be listed in your environment variables. Each should be separated by a comma:
|
||||
|
||||
```bash
|
||||
MARKET_DATA_PROVIDER=yahoo,alphavantage
|
||||
@@ -104,14 +115,27 @@ MARKET_DATA_PROVIDER=yahoo,alphavantage,custom_provider
|
||||
|
||||
Feel free to submit a PR with any custom providers you create.
|
||||
|
||||
## Import / Export
|
||||
|
||||
Investbrain includes a convenient feature which allows you to maintain the portability of your portfolios and transaction data.
|
||||
|
||||
### Import
|
||||
|
||||
Imports are "upserted" to the database. If the record does not already exist in the database, the record will be created. However, when a portfolio or transaction exists (i.e. the record's ID matches an existing record), the record will be updated. This way, you can simultaneously create new records, but also bulk update records.
|
||||
|
||||
### Export
|
||||
|
||||
Exporting your portfolios and transactions is a convenient way to back-up your Investbrain data. It is also a convenient way to maintain portability of *your* data.
|
||||
|
||||
## Configuration
|
||||
|
||||
There are several optional configurations available when installing using the recommended [Docker method](#self-hosting). These options are configurable using an environment file. Changes can be made in your [.env](https://github.com/investbrainapp/investbrain/blob/main/.env.example) file before installation.
|
||||
There are several optional configurations available when installing using the recommended [Docker method](#self-hosting). These options are configurable using an environment file. Configurations can be added to your [.env](https://github.com/investbrainapp/investbrain/blob/main/.env.example) file or to the `environment` property in the [docker-compose.yml](https://github.com/investbrainapp/investbrain/blob/main/docker-compose.yml) file.
|
||||
|
||||
| Option | Description | Default |
|
||||
| ------------- | ------------- | ------------- |
|
||||
| APP_URL | The URL where your Investbrain installation will be accessible | http://localhost |
|
||||
| APP_PORT | The HTTP port exposed by the NGINX container | 8000 |
|
||||
| APP_KEY | Must be set during install - encryption key for various security-related functions | `null` |
|
||||
| MARKET_DATA_PROVIDER | The market data provider to use (either `yahoo`, `alphavantage`, or `finnhub`) | yahoo |
|
||||
| ALPHAVANTAGE_API_KEY | If using the Alpha Vantage provider | `null` |
|
||||
| FINNHUB_API_KEY | If using the Finnhub provider | `null` |
|
||||
@@ -120,12 +144,13 @@ There are several optional configurations available when installing using the re
|
||||
| AI_CHAT_ENABLED | Whether to enable AI chat features | `false` |
|
||||
| OPENAI_API_KEY | OpenAI secret key (required for AI chat) | `null` |
|
||||
| OPENAI_ORGANIZATION | OpenAI org id (required for AI chat) | `null` |
|
||||
| OPENAI_MODEL | The selected LLM used for AI chat | gpt-4o |
|
||||
| OPENAI_BASE_URI | The URI for your self-hosted LLM | api.openai.com/v1 |
|
||||
| DAILY_CHANGE_TIME | The time of day to capture daily change | 23:00 |
|
||||
| REGISTRATION_ENABLED | Whether to enable registration of new users | `true` |
|
||||
|
||||
|
||||
|
||||
> Note: These options affect the [docker-compose.yml](https://github.com/investbrainapp/investbrain/blob/main/docker-compose.yml) file, so if you decide to make any changes to these default configurations, you'll have to restart the Docker containers before your changes take effect.
|
||||
> Note: These options affect the [docker-compose.yml](https://github.com/investbrainapp/investbrain/blob/main/docker-compose.yml) file and are cached during run-time. If change any environment configurations, you'll have to restart the container before your changes take effect.
|
||||
|
||||
## Updating
|
||||
|
||||
@@ -135,10 +160,10 @@ To update Investbrain using the recommended [Docker installation](#self-hosting)
|
||||
docker compose stop
|
||||
```
|
||||
|
||||
Then pull the latest updates from this repository using git:
|
||||
Then pull the latest Docker image:
|
||||
|
||||
```bash
|
||||
git pull
|
||||
docker image pull investbrainapp/investbrain:latest
|
||||
```
|
||||
|
||||
Finally bring the containers back up!
|
||||
@@ -221,7 +246,7 @@ We ask that you be kind and polite when interacting with the Investbrain communi
|
||||
|
||||
## Security Vulnerabilities
|
||||
|
||||
If you discover a security vulnerability within Investbrain, please create an issue in the [Github repository](https://github.com/investbrainapp/investbrain). All security vulnerabilities will be promptly addressed.
|
||||
If you discover a security vulnerability within Investbrain, please submit your report via [Github](https://github.com/investbrainapp/investbrain/security/advisories/new). All security vulnerabilities will be promptly addressed. We ask that you keep any suspected vulnerabilities private and confidential until they have been appropriately addressed.
|
||||
|
||||
## License
|
||||
|
||||
|
||||
+12
@@ -0,0 +1,12 @@
|
||||
# Security Policy
|
||||
|
||||
## Supported Versions
|
||||
|
||||
| Version | Supported |
|
||||
| ------- | ------------------ |
|
||||
| 1.0.x | :white_check_mark: |
|
||||
| < 1.0.0 | :x: |
|
||||
|
||||
## Reporting a Vulnerability
|
||||
|
||||
If you discover a security vulnerability within Investbrain, please submit your report via [Github](https://github.com/investbrainapp/investbrain/security/advisories/new). All security vulnerabilities will be promptly addressed. We ask that you keep any suspected vulnerabilities private and confidential until they have been appropriately addressed.
|
||||
@@ -27,7 +27,7 @@ class Dividend extends Model
|
||||
|
||||
protected $casts = [
|
||||
'date' => 'datetime',
|
||||
'last_date' => 'datetime',
|
||||
'last_dividend_update' => 'datetime',
|
||||
];
|
||||
|
||||
public function marketData() {
|
||||
@@ -55,22 +55,22 @@ class Dividend extends Model
|
||||
{
|
||||
$dividends_meta = self::where(['symbol' => $symbol])
|
||||
->selectRaw('COUNT(symbol) as total_dividends')
|
||||
->selectRaw('MAX(date) as last_date')
|
||||
->selectRaw('MAX(created_at) as last_dividend_update')
|
||||
->get()
|
||||
->first();
|
||||
|
||||
// assume we need to populate ALL dividend data
|
||||
$start_date = new \DateTime('@0');
|
||||
$start_date = new Carbon('@0');
|
||||
$end_date = now();
|
||||
|
||||
// nope, refresh forward looking only
|
||||
if ( $dividends_meta->total_dividends ) {
|
||||
|
||||
$start_date = $dividends_meta->last_date->addHours(24);
|
||||
|
||||
$start_date = $dividends_meta->last_dividend_update->addHours(24);
|
||||
}
|
||||
|
||||
|
||||
// skip refresh if there's already recent data
|
||||
if ($start_date >= $end_date) {
|
||||
if ($start_date->greaterThan($end_date)) {
|
||||
|
||||
return;
|
||||
}
|
||||
|
||||
@@ -2,7 +2,10 @@
|
||||
|
||||
namespace App\Providers;
|
||||
|
||||
use Illuminate\Support\Arr;
|
||||
use Laravel\Jetstream\Features;
|
||||
use App\Actions\Jetstream\DeleteUser;
|
||||
use Illuminate\Support\Facades\Config;
|
||||
use Illuminate\Support\ServiceProvider;
|
||||
use Laravel\Jetstream\Jetstream;
|
||||
|
||||
@@ -26,6 +29,13 @@ class JetstreamServiceProvider extends ServiceProvider
|
||||
|
||||
Jetstream::deleteUsersUsing(DeleteUser::class);
|
||||
|
||||
if ( config('investbrain.self_hosted', false) ) {
|
||||
|
||||
Config::set(
|
||||
'jetstream.features',
|
||||
array_keys(Arr::except(array_values(config('jetstream.features')), Features::termsAndPrivacyPolicy()))
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
|
||||
+6
-3
@@ -5,9 +5,12 @@
|
||||
"keywords": ["stocks", "dividends", "investments", "tracking"],
|
||||
"license": "CC-BY-NC 4.0",
|
||||
"require": {
|
||||
"php": "^8.2",
|
||||
"php": "^8.3",
|
||||
"ext-gd": "*",
|
||||
"ext-mbstring": "*",
|
||||
"ext-zip": "*",
|
||||
"finnhub/client": "master@dev",
|
||||
"laravel/framework": "^11.9",
|
||||
"laravel/framework": "^11.35",
|
||||
"laravel/jetstream": "^5.1",
|
||||
"laravel/sanctum": "^4.0",
|
||||
"laravel/socialite": "^5.16",
|
||||
@@ -16,7 +19,7 @@
|
||||
"livewire/livewire": "^3.5",
|
||||
"livewire/volt": "^1.6",
|
||||
"maatwebsite/excel": "^3.1",
|
||||
"openai-php/laravel": "^0.10.2",
|
||||
"openai-php/client": "^0.10.3",
|
||||
"predis/predis": "^2.2",
|
||||
"robsontenorio/mary": "^1.35",
|
||||
"scheb/yahoo-finance-api": "^4.11",
|
||||
|
||||
Generated
+881
-673
File diff suppressed because it is too large
Load Diff
+1
-1
@@ -13,7 +13,7 @@ return [
|
||||
|
|
||||
*/
|
||||
|
||||
'name' => env('APP_NAME', 'Laravel'),
|
||||
'name' => env('APP_NAME', 'Investbrain'),
|
||||
|
||||
/*
|
||||
|--------------------------------------------------------------------------
|
||||
|
||||
+1
-1
@@ -143,7 +143,7 @@ return [
|
||||
|
||||
'redis' => [
|
||||
|
||||
'client' => env('REDIS_CLIENT', 'phpredis'),
|
||||
'client' => env('REDIS_CLIENT', 'predis'),
|
||||
|
||||
'options' => [
|
||||
'cluster' => env('REDIS_CLUSTER', 'redis'),
|
||||
|
||||
+1
-1
@@ -143,7 +143,7 @@ return [
|
||||
|
|
||||
*/
|
||||
|
||||
'inject_morph_markers' => true,
|
||||
'inject_morph_markers' => false,
|
||||
|
||||
/*
|
||||
|---------------------------------------------------------------------------
|
||||
|
||||
+1
-1
@@ -110,7 +110,7 @@ return [
|
||||
|
||||
'from' => [
|
||||
'address' => env('MAIL_FROM_ADDRESS', 'hello@example.com'),
|
||||
'name' => env('MAIL_FROM_NAME', 'Example'),
|
||||
'name' => env('MAIL_FROM_NAME', 'Investbrain'),
|
||||
],
|
||||
|
||||
];
|
||||
|
||||
@@ -27,5 +27,6 @@ return [
|
||||
'request_timeout' => env('OPENAI_REQUEST_TIMEOUT', 30),
|
||||
|
||||
//
|
||||
'base_uri' => env('OPENAI_BASE_URI', 'api.openai.com/v1'),
|
||||
'model' => env('OPENAI_MODEL', 'gpt-4o'),
|
||||
];
|
||||
|
||||
+30
-20
@@ -3,34 +3,42 @@ networks:
|
||||
driver: bridge
|
||||
services:
|
||||
app:
|
||||
build:
|
||||
context: .
|
||||
dockerfile: docker/Dockerfile
|
||||
image: investbrainapp/investbrain:latest
|
||||
container_name: investbrain-app
|
||||
restart: unless-stopped
|
||||
tty: true
|
||||
expose:
|
||||
- "9000"
|
||||
volumes:
|
||||
- .:/var/www/app:delegated
|
||||
depends_on:
|
||||
- mysql
|
||||
networks:
|
||||
- investbrain-network
|
||||
nginx:
|
||||
image: nginx:alpine
|
||||
container_name: investbrain-nginx
|
||||
restart: unless-stopped
|
||||
tty: true
|
||||
ports:
|
||||
- "${APP_PORT:-8000}:80"
|
||||
environment:
|
||||
APP_KEY: "" # Generate a key using `openssl rand -base64 32`
|
||||
APP_URL: "http://localhost:8000"
|
||||
ASSET_URL: "http://localhost:8000"
|
||||
DB_CONNECTION: mysql
|
||||
DB_HOST: investbrain-mysql
|
||||
DB_PORT: 3306
|
||||
DB_DATABASE: ${DB_DATABASE:-investbrain}
|
||||
DB_USERNAME: ${DB_USERNAME:-investbrain}
|
||||
DB_PASSWORD: ${DB_PASSWORD:-investbrain}
|
||||
SESSION_DRIVER: redis
|
||||
QUEUE_CONNECTION: redis
|
||||
CACHE_STORE: redis
|
||||
REDIS_HOST: investbrain-redis
|
||||
volumes:
|
||||
- ./docker/nginx.conf:/etc/nginx/conf.d/default.conf
|
||||
- .:/var/www/app:delegated
|
||||
- ./storage:/var/www/app/storage:delegated
|
||||
depends_on:
|
||||
- app
|
||||
- mysql
|
||||
- redis
|
||||
networks:
|
||||
- investbrain-network
|
||||
redis:
|
||||
image: redis:alpine
|
||||
container_name: investbrain-redis
|
||||
restart: unless-stopped
|
||||
tty: true
|
||||
networks:
|
||||
- investbrain-network
|
||||
volumes:
|
||||
- investbrain-redis:/data
|
||||
mysql:
|
||||
image: mysql:8.0
|
||||
container_name: investbrain-mysql
|
||||
@@ -40,10 +48,12 @@ services:
|
||||
MYSQL_USER: ${DB_USERNAME:-investbrain}
|
||||
MYSQL_PASSWORD: ${DB_PASSWORD:-investbrain}
|
||||
MYSQL_ROOT_PASSWORD: ${DB_PASSWORD:-investbrain}
|
||||
command:
|
||||
- --cte-max-recursion-depth=25000
|
||||
volumes:
|
||||
- ./docker/mysql.conf:/etc/mysql/conf.d/my.cnf
|
||||
- investbrain-mysql:/var/lib/mysql
|
||||
networks:
|
||||
- investbrain-network
|
||||
volumes:
|
||||
investbrain-redis:
|
||||
investbrain-mysql:
|
||||
|
||||
@@ -1 +0,0 @@
|
||||
dump.rdb
|
||||
+50
-31
@@ -1,47 +1,66 @@
|
||||
FROM php:8.3-fpm
|
||||
|
||||
ENV DEBIAN_FRONTEND noninteractive
|
||||
ENV DEBIAN_FRONTEND=noninteractive
|
||||
ENV APP_NAME=Investbrain
|
||||
ENV VITE_APP_NAME=Investbrain
|
||||
ENV APP_DEBUG=true
|
||||
ENV SELF_HOSTED=true
|
||||
|
||||
# Set the working directory
|
||||
COPY . /var/www/app
|
||||
WORKDIR /var/www/app
|
||||
|
||||
# Install common php extension dependencies
|
||||
RUN apt-get update && apt-get install -y \
|
||||
libfreetype-dev \
|
||||
libjpeg62-turbo-dev \
|
||||
libpng-dev \
|
||||
zlib1g-dev \
|
||||
libzip-dev \
|
||||
unzip \
|
||||
libicu-dev \
|
||||
git \
|
||||
curl \
|
||||
redis \
|
||||
supervisor \
|
||||
# Allow PHP installs to be built cross-platform
|
||||
#ENV CFLAGS="-fstack-protector-strong -fpic -fPIC -fpie -O2 -D_LARGEFILE_SOURCE -D_FILE_OFFSET_BITS=64"
|
||||
|
||||
# Install required packages
|
||||
RUN apt-get update && apt-get upgrade -y \
|
||||
&& apt-get upgrade -y \
|
||||
nginx \
|
||||
libfreetype-dev \
|
||||
libjpeg62-turbo-dev \
|
||||
libpng-dev \
|
||||
zlib1g-dev \
|
||||
libzip-dev \
|
||||
libicu-dev \
|
||||
libpq-dev \
|
||||
binutils libc6-dev \
|
||||
supervisor \
|
||||
unzip curl git \
|
||||
nodejs npm \
|
||||
# Clean up APT
|
||||
&& apt-get -y autoremove \
|
||||
&& apt-get clean \
|
||||
&& rm -rf /var/lib/apt/lists/* /tmp/* /var/tmp/* \
|
||||
# Install PHP extensions
|
||||
&& docker-php-ext-configure gd --with-freetype --with-jpeg \
|
||||
&& docker-php-ext-install -j$(nproc) \
|
||||
gd \
|
||||
zip \
|
||||
pdo_mysql \
|
||||
mysqli \
|
||||
intl
|
||||
gd pgsql zip pdo_mysql mysqli intl
|
||||
|
||||
# Install Node.js and npm
|
||||
RUN curl -fsSL https://deb.nodesource.com/setup_20.x | bash - \
|
||||
&& apt-get install -y nodejs \
|
||||
&& npm install -g npm@latest
|
||||
# Set permissions
|
||||
RUN chown -R www-data:www-data . \
|
||||
&& chmod -R 775 ./storage \
|
||||
&& chmod +x ./docker/entrypoint.sh \
|
||||
&& usermod -s /bin/bash www-data
|
||||
|
||||
# Copy over supervisor configuration
|
||||
# Install Composer and Node.js Install PHP dependencies and build front end assets
|
||||
RUN curl -sS https://getcomposer.org/installer | php -- --install-dir=/usr/local/bin --filename=composer \
|
||||
&& composer install --no-scripts --optimize-autoloader \
|
||||
&& npm install && npm run build
|
||||
|
||||
# Remove default nginx config
|
||||
RUN rm /etc/nginx/sites-enabled/default
|
||||
|
||||
# Copy over configs
|
||||
COPY ./docker/nginx.conf /etc/nginx/conf.d/default.conf
|
||||
COPY ./docker/supervisord.conf /etc/supervisor/conf.d/supervisord.conf
|
||||
|
||||
# Update permissions
|
||||
RUN chown -R www-data:www-data . \
|
||||
&& chmod -R 775 ./storage \
|
||||
&& chmod +x ./docker/entrypoint.sh
|
||||
# Serve on port 80
|
||||
EXPOSE 80
|
||||
|
||||
# install composer
|
||||
COPY --from=composer:2.6.5 /usr/bin/composer /usr/local/bin/composer
|
||||
# Set up healthcheck
|
||||
HEALTHCHECK --interval=30s --timeout=10s --retries=3 CMD curl -f http://localhost || exit 1
|
||||
|
||||
# Run everything else
|
||||
CMD ["./docker/entrypoint.sh"]
|
||||
ENTRYPOINT ["/bin/bash", "./docker/entrypoint.sh"]
|
||||
|
||||
|
||||
Executable → Regular
+44
-31
@@ -2,43 +2,56 @@
|
||||
|
||||
cd /var/www/app
|
||||
|
||||
echo "====================== Running entrypoint script... ====================== "
|
||||
if [ ! -f ".env" ]; then
|
||||
echo " > Ope, gotta create an .env file!"
|
||||
echo -e "\n====================== Validating environment... ====================== "
|
||||
if [[ -z "$APP_KEY" ]]; then
|
||||
echo -e "\n > Oops! The required APP_KEY configuration is missing in your environment! "
|
||||
echo -e "\n > Generating a key (see below) but this will NOT be persisted between container restarts. "
|
||||
echo -e "\n > You should set this APP_KEY in your .env file! "
|
||||
|
||||
cp .env.example .env
|
||||
draw_box() {
|
||||
local text="$1"
|
||||
local length=${#text}
|
||||
local border=$(printf '%*s' "$((length + 4))" | tr ' ' '*')
|
||||
|
||||
echo -e "\n\n$border"
|
||||
echo "* $text *"
|
||||
echo "$border"
|
||||
}
|
||||
|
||||
export APP_KEY=base64:$(openssl rand -base64 32)
|
||||
draw_box $APP_KEY
|
||||
fi
|
||||
|
||||
echo "====================== Checking for updates... ====================== "
|
||||
/usr/bin/git pull
|
||||
|
||||
echo "====================== Installing Composer dependencies... ====================== "
|
||||
/usr/local/bin/composer install
|
||||
|
||||
echo "====================== Validating environment... ====================== "
|
||||
if [ $(stat -c '%U' .) != "www-data" ]; then
|
||||
echo " > Setting correct permissions for pwd..."
|
||||
chown -R www-data:www-data .
|
||||
fi
|
||||
|
||||
if ( ! grep -q "^APP_KEY=" ".env" || grep -q "^APP_KEY=$" ".env"); then
|
||||
echo " > Ah, APP_KEY is missing in .env file. Generating a new key!"
|
||||
|
||||
/usr/local/bin/php artisan key:generate --force
|
||||
fi
|
||||
for dir in storage/framework/cache storage/framework/sessions storage/framework/views; do
|
||||
if [ ! -d "$dir" ]; then
|
||||
echo -e "\n > $dir is missing. Creating scaffold for storage directory... "
|
||||
mkdir -p storage/framework/{cache,sessions,views}
|
||||
chmod -R 775 storage
|
||||
chown -R www-data:www-data storage
|
||||
fi
|
||||
done
|
||||
|
||||
if [ ! -L "public/storage" ]; then
|
||||
echo " > Creating symbolic link for app public storage..."
|
||||
echo -e "\n > Creating symbolic link for app public storage... "
|
||||
|
||||
/usr/local/bin/php artisan storage:link
|
||||
/usr/local/bin/php /var/www/app/artisan storage:link
|
||||
fi
|
||||
|
||||
echo "====================== Installing NPM dependencies and building frontend... ====================== "
|
||||
/usr/bin/npm install
|
||||
/usr/bin/npm run build
|
||||
echo -e "\n====================== Running migrations... ====================== "
|
||||
run_migrations() {
|
||||
/usr/local/bin/php /var/www/app/artisan migrate --force
|
||||
}
|
||||
RETRIES=10
|
||||
DELAY=5
|
||||
until run_migrations; do
|
||||
RETRIES=$((RETRIES-1))
|
||||
if [ $RETRIES -le 0 ]; then
|
||||
echo -e "\n > Database is not ready after $RETRIES attempts. Exiting... "
|
||||
exit 1
|
||||
fi
|
||||
echo -e "\n > Waiting for database to be ready... retrying in $DELAY seconds. "
|
||||
sleep $DELAY
|
||||
done
|
||||
|
||||
echo "====================== Running migrations... ====================== "
|
||||
/usr/local/bin/php artisan migrate --force
|
||||
|
||||
echo "====================== Spinning up Supervisor daemon... ====================== "
|
||||
exec /usr/bin/supervisord -c /etc/supervisor/conf.d/supervisord.conf
|
||||
echo -e "\n====================== Spinning up Supervisor daemon... ====================== \n"
|
||||
exec /usr/bin/supervisord -c /etc/supervisor/conf.d/supervisord.conf
|
||||
@@ -1,2 +0,0 @@
|
||||
[mysqld]
|
||||
cte_max_recursion_depth = 25000
|
||||
+1
-1
@@ -14,7 +14,7 @@ server {
|
||||
fastcgi_param HTTPS $http_x_forwarded_proto;
|
||||
fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name;
|
||||
fastcgi_param PATH_INFO $fastcgi_path_info;
|
||||
fastcgi_pass investbrain-app:9000;
|
||||
fastcgi_pass 127.0.0.1:9000;
|
||||
}
|
||||
|
||||
location ~ /\.ht {
|
||||
|
||||
@@ -1,2 +0,0 @@
|
||||
# Redis RDB and AOF file location
|
||||
dir /var/www/app/docker
|
||||
+11
-18
@@ -1,41 +1,34 @@
|
||||
[supervisord]
|
||||
nodaemon=true
|
||||
user=root
|
||||
logfile=/var/log/supervisor/supervisord.log
|
||||
pidfile=/var/run/supervisord.pid
|
||||
|
||||
[program:nginx]
|
||||
command=nginx -g 'daemon off;'
|
||||
autostart=true
|
||||
autorestart=true
|
||||
redirect_stderr=true
|
||||
redirect_stdout=true
|
||||
|
||||
[program:php]
|
||||
command=php-fpm -F
|
||||
autostart=true
|
||||
autorestart=true
|
||||
stdout_logfile=/var/log/supervisor/php.log
|
||||
stderr_logfile=/var/log/supervisor/php_error.log
|
||||
|
||||
[program:redis]
|
||||
command=redis-server /var/www/app/docker/redis.conf
|
||||
autostart=true
|
||||
autorestart=true
|
||||
stdout_logfile=/var/log/supervisor/redis.log
|
||||
stderr_logfile=/var/log/supervisor/redis_error.log
|
||||
redirect_stderr=true
|
||||
redirect_stdout=true
|
||||
|
||||
[program:scheduler]
|
||||
command=php artisan schedule:work
|
||||
autorestart=true
|
||||
redirect_stderr=true
|
||||
stdout_logfile=/dev/stdout
|
||||
stdout_logfile_maxbytes=0
|
||||
stderr_logfile=/dev/stderr
|
||||
stderr_logfile_maxbytes=0
|
||||
redirect_stdout=true
|
||||
|
||||
[program:queue-worker]
|
||||
command=php artisan queue:work --sleep=3 --tries=1 --memory=256 --timeout=3600
|
||||
process_name=%(program_name)s_%(process_num)02d
|
||||
autorestart=true
|
||||
redirect_stderr=true
|
||||
stdout_logfile=/dev/stdout
|
||||
stdout_logfile_maxbytes=0
|
||||
stderr_logfile=/dev/stderr
|
||||
stderr_logfile_maxbytes=0
|
||||
redirect_stdout=true
|
||||
numprocs=2
|
||||
|
||||
[supervisorctl]
|
||||
Generated
+494
-253
File diff suppressed because it is too large
Load Diff
+2
-2
@@ -9,12 +9,12 @@
|
||||
"@tailwindcss/forms": "^0.5.7",
|
||||
"@tailwindcss/typography": "^0.5.10",
|
||||
"autoprefixer": "^10.4.19",
|
||||
"axios": "^1.6.4",
|
||||
"axios": "^1.7.4",
|
||||
"daisyui": "^4.12.10",
|
||||
"laravel-vite-plugin": "^1.0",
|
||||
"postcss": "^8.4.40",
|
||||
"tailwindcss": "^3.4.7",
|
||||
"vite": "^5.0"
|
||||
"vite": "^5.4"
|
||||
},
|
||||
"dependencies": {
|
||||
"apexcharts": "^3.51.0"
|
||||
|
||||
@@ -63,7 +63,9 @@
|
||||
<x-ib-card title="{{ __('Recent activity') }}" class="md:col-span-3">
|
||||
|
||||
@livewire('transactions-list', [
|
||||
'transactions' => $user->transactions
|
||||
'transactions' => $user->transactions,
|
||||
'showPortfolio' => true,
|
||||
'paginate' => false
|
||||
])
|
||||
|
||||
</x-ib-card>
|
||||
|
||||
@@ -207,7 +207,7 @@
|
||||
* 52 week high: {$holding->market_data->fifty_two_week_high}
|
||||
* Dividend yield: {$holding->market_data->dividend_yield}
|
||||
|
||||
Based on this current market data, quantity owned, and average cost basis, you should determine if the {$holding->symbol} holding is making or losing money.
|
||||
This data is current as of today's date: " . now()->format('Y-m-d') . ". Based on this current market data, quantity owned, and average cost basis, you should determine if the {$holding->symbol} holding is making or losing money.
|
||||
|
||||
Below is the question from the investor. Considering these facts, provide a concise response to the following question (give a direct response). Limit your response to no more than 75 words and consider using a common decision framework. Use github style markdown for any formatting:"
|
||||
])
|
||||
|
||||
@@ -5,7 +5,7 @@ use App\Models\AiChat;
|
||||
use App\Models\Holding;
|
||||
use Illuminate\Database\Eloquent\Model;
|
||||
use Livewire\Volt\Component;
|
||||
use OpenAI\Laravel\Facades\OpenAI;
|
||||
use OpenAI\Factory;
|
||||
use OpenAI\Responses\StreamResponse;
|
||||
|
||||
new class extends Component {
|
||||
@@ -67,7 +67,9 @@ new class extends Component {
|
||||
{
|
||||
|
||||
try {
|
||||
$stream = OpenAI::chat()->createStreamed([
|
||||
$client = $this->createOpenAiClient();
|
||||
|
||||
$stream = $client->chat()->createStreamed([
|
||||
'model' => config('openai.model'),
|
||||
'messages' => [
|
||||
['role' => 'system', 'content' => "Today's date is "
|
||||
@@ -104,7 +106,9 @@ new class extends Component {
|
||||
public function generateSuggestedPrompts(): void
|
||||
{
|
||||
try {
|
||||
$suggested_prompts = OpenAI::chat()->create([
|
||||
$client = $this->createOpenAiClient();
|
||||
|
||||
$suggested_prompts = $client->chat()->create([
|
||||
'model' => config('openai.model'),
|
||||
'response_format' => [
|
||||
'type' => 'json_schema',
|
||||
@@ -192,6 +196,21 @@ new class extends Component {
|
||||
return false;
|
||||
}
|
||||
|
||||
private function createOpenAiClient()
|
||||
{
|
||||
$apiKey = config('openai.api_key');
|
||||
$organization = config('openai.organization');
|
||||
$baseUri = config('openai.base_uri');
|
||||
|
||||
return OpenAI::factory()
|
||||
->withApiKey($apiKey)
|
||||
->withOrganization($organization)
|
||||
->withHttpHeader('OpenAI-Beta', 'assistants=v2')
|
||||
->withHttpClient(new \GuzzleHttp\Client(['timeout' => config('openai.request_timeout', 30)]))
|
||||
->withBaseUri($baseUri)
|
||||
->make();
|
||||
}
|
||||
|
||||
}; ?>
|
||||
|
||||
<div
|
||||
|
||||
@@ -15,6 +15,10 @@ new class extends Component {
|
||||
public ?Portfolio $portfolio;
|
||||
public ?Transaction $editingTransaction;
|
||||
public Bool $shouldGoToHolding = true;
|
||||
public Bool $showPortfolio = false;
|
||||
public Bool $paginate = true;
|
||||
public Int $perPage = 5;
|
||||
public Int $offset = 0;
|
||||
|
||||
protected $listeners = [
|
||||
'transaction-updated' => '$refresh',
|
||||
@@ -38,17 +42,23 @@ new class extends Component {
|
||||
return $this->redirect(route('holding.show', ['portfolio' => $holding['portfolio_id'], 'symbol' => $holding['symbol']]));
|
||||
}
|
||||
|
||||
public function updateOffset($amount = 0)
|
||||
{
|
||||
$this->offset = $this->offset + $amount;
|
||||
}
|
||||
|
||||
}; ?>
|
||||
|
||||
<div class="">
|
||||
|
||||
@foreach($transactions->sortByDesc('date')->take(10) as $transaction)
|
||||
@foreach($transactions->sortByDesc('date')->slice($offset)->take($perPage) as $transaction)
|
||||
|
||||
<x-list-item
|
||||
no-separator
|
||||
:item="$transaction"
|
||||
class="cursor-pointer"
|
||||
x-data="{ loading: false, timeout: null }"
|
||||
:key="$transaction->id"
|
||||
@click="
|
||||
if ($wire.shouldGoToHolding) {
|
||||
|
||||
@@ -83,12 +93,44 @@ new class extends Component {
|
||||
<x-loading x-show="loading" x-cloak class="text-gray-400 ml-2" />
|
||||
</x-slot:value>
|
||||
<x-slot:sub-value>
|
||||
@if($showPortfolio)
|
||||
<span title="{{ __('Portfolio') }}">{{ $transaction->portfolio->title }} </span>
|
||||
·
|
||||
@endif
|
||||
<span title="{{ __('Transaction Date') }}">{{ $transaction->date->format('F j, Y') }} </span>
|
||||
</x-slot:sub-value>
|
||||
</x-list-item>
|
||||
|
||||
@endforeach
|
||||
|
||||
@if ($paginate && count($transactions) > $perPage)
|
||||
<div class="flex justify-between">
|
||||
|
||||
<span>
|
||||
@if($offset > 0)
|
||||
<x-button
|
||||
class="btn btn-sm btn-ghost text-secondary"
|
||||
wire:click="updateOffset(-{{ $perPage }})"
|
||||
>
|
||||
{!! __('pagination.previous') !!}
|
||||
</x-button>
|
||||
@endif
|
||||
</span>
|
||||
|
||||
<span>
|
||||
@if(count($transactions) - $offset > $offset)
|
||||
<x-button
|
||||
class="btn btn-sm btn-ghost text-secondary"
|
||||
wire:click="updateOffset({{ $perPage }})"
|
||||
>
|
||||
{!! __('pagination.next') !!}
|
||||
</x-button>
|
||||
@endif
|
||||
</span>
|
||||
|
||||
</div>
|
||||
@endif
|
||||
|
||||
<x-ib-alpine-modal
|
||||
key="manage-transaction"
|
||||
title="{{ __('Manage Transaction') }}"
|
||||
@@ -96,7 +138,7 @@ new class extends Component {
|
||||
@livewire('manage-transaction-form', [
|
||||
'portfolio' => $portfolio,
|
||||
'transaction' => $editingTransaction,
|
||||
], key($editingTransaction->id ?? 'new'))
|
||||
], key($editingTransaction?->id.rand()))
|
||||
|
||||
</x-ib-alpine-modal>
|
||||
</div>
|
||||
@@ -173,7 +173,7 @@
|
||||
|
||||
{$formattedHoldings}
|
||||
|
||||
Based on the current market data, quantity owned, and average cost basis, you can determine the performance of any holding.
|
||||
This data is current as of today's date: " . now()->format('Y-m-d') . ". Based on the current market data, quantity owned, and average cost basis, you can determine the performance of any holding.
|
||||
|
||||
Below is the question from the investor. Considering these facts, provide a concise response to the following question (give a direct response). Limit your response to no more than 75 words and consider using a common decision framework. Use github style markdown for any formatting:"
|
||||
])
|
||||
|
||||
+2
-2
@@ -8,7 +8,7 @@ use App\Console\Commands\{RefreshMarketData, CaptureDailyChange, RefreshDividend
|
||||
* This scheduled job refreshes market data from your selected data provider
|
||||
* Update the cadence with the MARKET_DATA_REFRESH key in your env file
|
||||
*/
|
||||
Schedule::command(RefreshMarketData::class)->everyMinute()->weekdays();
|
||||
Schedule::command(RefreshMarketData::class)->weekdays()->everyMinute();
|
||||
|
||||
/**
|
||||
*
|
||||
@@ -20,7 +20,7 @@ Schedule::command(CaptureDailyChange::class)->dailyAt(config('investbrain.daily_
|
||||
*
|
||||
* Refreshes dividend data for your holdings (and syncs new dividends to holdings)
|
||||
*/
|
||||
Schedule::command(RefreshDividendData::class)->days([1, 3, 5])->weekdays();
|
||||
Schedule::command(RefreshDividendData::class)->daily()->days([1, 3, 5]);
|
||||
|
||||
/**
|
||||
*
|
||||
|
||||
@@ -61,4 +61,26 @@ class DividendsTest extends TestCase
|
||||
$this->assertCount(3, $transactions);
|
||||
$this->assertEqualsWithDelta(4.95, $dividendsReinvested * $market_data->market_value, 0.01);
|
||||
}
|
||||
|
||||
/**
|
||||
*/
|
||||
public function test_do_not_duplicate_recent_dividends(): void
|
||||
{
|
||||
$this->actingAs($user = User::factory()->create());
|
||||
|
||||
$portfolio = Portfolio::factory()->create();
|
||||
Transaction::factory()->buy()->yearsAgo()->portfolio($portfolio->id)->symbol('ACME')->create();
|
||||
|
||||
$holding = Holding::query()->portfolio($portfolio->id)->symbol('ACME')->first();
|
||||
|
||||
Dividend::create([
|
||||
'symbol' => 'ACME',
|
||||
'date' => now()->subDay(2),
|
||||
'dividend_amount' => .01
|
||||
]);
|
||||
|
||||
Dividend::refreshDividendData('ACME');
|
||||
|
||||
$this->assertCount(1, $holding->dividends);
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user