diff --git a/.gitignore b/.gitignore index c864e82..88c9b30 100644 --- a/.gitignore +++ b/.gitignore @@ -1,5 +1,6 @@ /vendor /.idea +/.standalone /.vscode /.vagrant /database/database.sqlite diff --git a/README.md b/README.md index 136dacf..eb6e51a 100644 --- a/README.md +++ b/README.md @@ -58,9 +58,35 @@ You should seek the advice of a professional accountant before using this progra ## Installation +### Standalone application 🧪 + +> [!NOTE] +> This is an experimental feature. Please [open an issue](#issue-reporting) if the application doesn't behave as expected. + +Download the binary for your operating system: + +| OS | Binary | +| -- | ------ | +| macOS aarch64 (M1, M2, M3...) | [dime-macos-aarch64](https://github.com/osteel/dime/releases/download/v0.3/dime-macos-aarch64) | + +You can use the application straight away: + +``` +$ php dime-your-os +``` + +Or you may want to move it to a directory that is in your system's `PATH`: + +``` +$ chmod +x dime-your-os +$ mv dime-your-os /usr/local/bin/dime +``` + +You can now run `dime` from anywhere instead of `php dime-your-os`. + ### Composer -> **Note** +> [!IMPORTANT] > Requires [Composer](https://getcomposer.org/) as well as PHP 8.2 and the [BCMath](https://www.php.net/manual/en/book.bc.php) extension. You can instal Dime via [Composer](https://getcomposer.org): @@ -93,7 +119,7 @@ $ source ~/.bash_profile ### PHAR (Linux / Unix / macOS) -> **Note** +> [!IMPORTANT] > Requires PHP 8.2 and the [BCMath](https://www.php.net/manual/en/book.bc.php) extension. @@ -117,7 +143,7 @@ You can now run `dime` from anywhere instead of `php dime`. ### Docker -> **Note** +> [!IMPORTANT] > Requires [Docker](https://www.docker.com/) and sh or Bash. [Download this shell script](/.docker/dime.sh) and use it straight away: @@ -157,7 +183,7 @@ $ dime process transactions.csv It will validate and process each transaction and display the corresponding tax figures or report any errors. -> **Note** +> [!NOTE] > You will need to run this command every time you update the spreadsheet. See the [Spreadsheet format](#spreadsheet-format) section to learn how to report your transactions. @@ -239,7 +265,7 @@ The market value is at the centre of Dime's calculation rules so it is crucial t When the transaction includes a sent asset (and is not a transfer – transfers don't need a market value), you must use the sent asset's market value as the transaction's market value, _based on the reported sent quantity_ (see [Sent quantity](#sent-quantity) section below to learn how to report it correctly). -> **Note** +> [!IMPORTANT] > Be particularly vigilant when selling an asset for some fiat currency. It is tempting to use the received fiat amount as the market value, but if a fee was taken, the actual market value is the received amount plus the fee's market value. When the transaction does not include a sent asset (i.e. `receive` transactions), the transaction's market value is the received asset's market value. @@ -255,7 +281,7 @@ Keeping in mind that when a transaction includes a sent asset, its market value * If using a decentralised protocol, use the value reported by the corresponding blockchain's explorer (see the value between brackets in the _Tokens Transferred_ section [here](https://etherscan.io/tx/0x2c9310e04c01e1329973c205cc6f3d3a7be3237ed09b968faef7ed85d9dfea65), for instance); * If none of the above applies, look up the asset on price-tracking websites such as [CoinMarketCap](https://coinmarketcap.com/) or [CoinGecko](https://www.coingecko.com/). -> **Note** +> [!TIP] > If any of the above amounts are expressed in a foreign currency, use HMRC's [exchange rates](https://www.gov.uk/government/publications/hmrc-exchange-rates-for-2023-monthly "HMRC exchange rates for 2023: monthly") to convert them to pound sterling. Sometimes, while the transaction includes a sent asset, its price isn't tracked anywhere. In that case, use the received asset's market value plus the fee's market value, if you paid for that fee. @@ -280,7 +306,7 @@ The quantity sent. For NFTs, that would be `1`. Be as precise as possible and report all available decimal places. -> **Note** +> [!IMPORTANT] > When the sent asset and the [fee](#fee)'s currency are the same (e.g. GBP in both cases), _and the fee's quantity was deducted from the sent quantity_, the sent quantity must exclude the fee's quantity. > * Example #1: If you send £50 to an exchange and spend it all in a transaction that incurs a £0.50 fee, the sent quantity is 49.5. > * Example #2: If you swap 1 bitcoin for 10 ethers on an exchange and the transaction incurs a BTC 0.01 fee, the sent quantity is 0.99. @@ -322,7 +348,7 @@ Different values must be used depending on the context: The reason for the latter is that exchanges usually estimate how much the fee will be and charge that estimate. This amount is almost always different from the actual fee, but that's the amount you will be charged anyway. -> **Note** +> [!TIP] > Be sure not to report fees that you didn't pay for. For instance, if you received an airdrop that was sent by the token issuer and the transaction includes a fee, the issuer paid for that fee, not you, so you don't have to report it. Here is the detail of each fee-related column. diff --git a/builds/dime-macos-aarch64 b/builds/dime-macos-aarch64 new file mode 100755 index 0000000..fd343fb Binary files /dev/null and b/builds/dime-macos-aarch64 differ diff --git a/config/commands.php b/config/commands.php index d50bf31..1913486 100644 --- a/config/commands.php +++ b/config/commands.php @@ -111,7 +111,7 @@ LaravelZero\Framework\Components\Updater\SelfUpdateCommand::class, ] : [], Helpers::installedViaDocker() === false ? [ - \App\Commands\Delete::class, + App\Commands\Delete::class, ] : [], ), diff --git a/domain/tests/Aggregates/SharePoolingAsset/Entities/SharePoolingAssetTransactionsTest.php b/domain/tests/Aggregates/SharePoolingAsset/Entities/SharePoolingAssetTransactionsTest.php index e6aa3a5..343b3b0 100644 --- a/domain/tests/Aggregates/SharePoolingAsset/Entities/SharePoolingAssetTransactionsTest.php +++ b/domain/tests/Aggregates/SharePoolingAsset/Entities/SharePoolingAssetTransactionsTest.php @@ -149,7 +149,7 @@ SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-22')]), SharePoolingAssetAcquisition::factory()->make(['date' => LocalDate::parse('2015-10-22')]), SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-24')]), - )->madeOn(Localdate::parse($date)); + )->madeOn(LocalDate::parse($date)); expect($transactions)->toBeInstanceOf(SharePoolingAssetTransactions::class); expect($transactions->count())->toEqual($count); @@ -168,7 +168,7 @@ SharePoolingAssetAcquisition::factory()->make(['date' => LocalDate::parse('2015-10-22')]), SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-23')]), SharePoolingAssetAcquisition::factory()->make(['date' => LocalDate::parse('2015-10-24')]), - )->acquisitionsMadeOn(Localdate::parse($date)); + )->acquisitionsMadeOn(LocalDate::parse($date)); expect($transactions)->toBeInstanceOf(SharePoolingAssetAcquisitions::class); expect($transactions->count())->toEqual($count); @@ -187,7 +187,7 @@ SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-22')]), SharePoolingAssetAcquisition::factory()->make(['date' => LocalDate::parse('2015-10-23')]), SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-24')]), - )->disposalsMadeOn(Localdate::parse($date)); + )->disposalsMadeOn(LocalDate::parse($date)); expect($transactions)->toBeInstanceOf(SharePoolingAssetDisposals::class); expect($transactions->count())->toEqual($count); @@ -207,7 +207,7 @@ SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-23')]), SharePoolingAssetAcquisition::factory()->make(['date' => LocalDate::parse('2015-10-24')]), SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-25')]), - )->madeBetween(Localdate::parse($date1), Localdate::parse($date2)); + )->madeBetween(LocalDate::parse($date1), LocalDate::parse($date2)); expect($transactions)->toBeInstanceOf(SharePoolingAssetTransactions::class); expect($transactions->count())->toEqual($count); @@ -230,7 +230,7 @@ SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-23')]), SharePoolingAssetAcquisition::factory()->make(['date' => LocalDate::parse('2015-10-24')]), SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-25')]), - )->acquisitionsMadeBetween(Localdate::parse($date1), Localdate::parse($date2)); + )->acquisitionsMadeBetween(LocalDate::parse($date1), LocalDate::parse($date2)); expect($transactions)->toBeInstanceOf(SharePoolingAssetAcquisitions::class); expect($transactions->count())->toEqual($count); @@ -252,7 +252,7 @@ SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-23')]), SharePoolingAssetAcquisition::factory()->make(['date' => LocalDate::parse('2015-10-24')]), SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-25')]), - )->disposalsMadeBetween(Localdate::parse($date1), Localdate::parse($date2)); + )->disposalsMadeBetween(LocalDate::parse($date1), LocalDate::parse($date2)); expect($transactions)->toBeInstanceOf(SharePoolingAssetDisposals::class); expect($transactions->count())->toEqual($count); @@ -274,7 +274,7 @@ SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-23')]), SharePoolingAssetAcquisition::factory()->make(['date' => LocalDate::parse('2015-10-24')]), SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-25')]), - )->madeBefore(Localdate::parse($date)); + )->madeBefore(LocalDate::parse($date)); expect($transactions)->toBeInstanceOf(SharePoolingAssetTransactions::class); expect($transactions->count())->toEqual($count); @@ -293,7 +293,7 @@ SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-23')]), SharePoolingAssetAcquisition::factory()->make(['date' => LocalDate::parse('2015-10-24')]), SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-25')]), - )->acquisitionsMadeBefore(Localdate::parse($date)); + )->acquisitionsMadeBefore(LocalDate::parse($date)); expect($transactions)->toBeInstanceOf(SharePoolingAssetAcquisitions::class); expect($transactions->count())->toEqual($count); @@ -312,7 +312,7 @@ SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-23')]), SharePoolingAssetAcquisition::factory()->make(['date' => LocalDate::parse('2015-10-24')]), SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-25')]), - )->disposalsMadeBefore(Localdate::parse($date)); + )->disposalsMadeBefore(LocalDate::parse($date)); expect($transactions)->toBeInstanceOf(SharePoolingAssetDisposals::class); expect($transactions->count())->toEqual($count); @@ -331,7 +331,7 @@ SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-23')]), SharePoolingAssetAcquisition::factory()->make(['date' => LocalDate::parse('2015-10-24')]), SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-25')]), - )->madeBeforeOrOn(Localdate::parse($date)); + )->madeBeforeOrOn(LocalDate::parse($date)); expect($transactions)->toBeInstanceOf(SharePoolingAssetTransactions::class); expect($transactions->count())->toEqual($count); @@ -351,7 +351,7 @@ SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-23')]), SharePoolingAssetAcquisition::factory()->make(['date' => LocalDate::parse('2015-10-24')]), SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-25')]), - )->acquisitionsMadeBeforeOrOn(Localdate::parse($date)); + )->acquisitionsMadeBeforeOrOn(LocalDate::parse($date)); expect($transactions)->toBeInstanceOf(SharePoolingAssetAcquisitions::class); expect($transactions->count())->toEqual($count); @@ -371,7 +371,7 @@ SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-23')]), SharePoolingAssetAcquisition::factory()->make(['date' => LocalDate::parse('2015-10-24')]), SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-25')]), - )->disposalsMadeBeforeOrOn(Localdate::parse($date)); + )->disposalsMadeBeforeOrOn(LocalDate::parse($date)); expect($transactions)->toBeInstanceOf(SharePoolingAssetDisposals::class); expect($transactions->count())->toEqual($count); @@ -391,7 +391,7 @@ SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-23')]), SharePoolingAssetAcquisition::factory()->make(['date' => LocalDate::parse('2015-10-24')]), SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-25')]), - )->madeAfter(Localdate::parse($date)); + )->madeAfter(LocalDate::parse($date)); expect($transactions)->toBeInstanceOf(SharePoolingAssetTransactions::class); expect($transactions->count())->toEqual($count); @@ -410,7 +410,7 @@ SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-23')]), SharePoolingAssetAcquisition::factory()->make(['date' => LocalDate::parse('2015-10-24')]), SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-25')]), - )->acquisitionsMadeAfter(Localdate::parse($date)); + )->acquisitionsMadeAfter(LocalDate::parse($date)); expect($transactions)->toBeInstanceOf(SharePoolingAssetAcquisitions::class); expect($transactions->count())->toEqual($count); @@ -429,7 +429,7 @@ SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-23')]), SharePoolingAssetAcquisition::factory()->make(['date' => LocalDate::parse('2015-10-24')]), SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-25')]), - )->disposalsMadeAfter(Localdate::parse($date)); + )->disposalsMadeAfter(LocalDate::parse($date)); expect($transactions)->toBeInstanceOf(SharePoolingAssetDisposals::class); expect($transactions->count())->toEqual($count); @@ -448,7 +448,7 @@ SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-23')]), SharePoolingAssetAcquisition::factory()->make(['date' => LocalDate::parse('2015-10-24')]), SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-25')]), - )->madeAfterOrOn(Localdate::parse($date)); + )->madeAfterOrOn(LocalDate::parse($date)); expect($transactions)->toBeInstanceOf(SharePoolingAssetTransactions::class); expect($transactions->count())->toEqual($count); @@ -468,7 +468,7 @@ SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-23')]), SharePoolingAssetAcquisition::factory()->make(['date' => LocalDate::parse('2015-10-24')]), SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-25')]), - )->acquisitionsMadeAfterOrOn(Localdate::parse($date)); + )->acquisitionsMadeAfterOrOn(LocalDate::parse($date)); expect($transactions)->toBeInstanceOf(SharePoolingAssetAcquisitions::class); expect($transactions->count())->toEqual($count); @@ -488,7 +488,7 @@ SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-23')]), SharePoolingAssetAcquisition::factory()->make(['date' => LocalDate::parse('2015-10-24')]), SharePoolingAssetDisposal::factory()->make(['date' => LocalDate::parse('2015-10-25')]), - )->disposalsMadeAfterOrOn(Localdate::parse($date)); + )->disposalsMadeAfterOrOn(LocalDate::parse($date)); expect($transactions)->toBeInstanceOf(SharePoolingAssetDisposals::class); expect($transactions->count())->toEqual($count);