2021-02-08 11:16:37 +09:00
![Build Status ](https://storage.googleapis.com/tensorflow-kokoro-build-badges/haskell/github.png )
2016-10-28 18:08:32 -07:00
2016-10-24 19:26:42 +00:00
The tensorflow-haskell package provides Haskell bindings to
[TensorFlow ](https://www.tensorflow.org/ ).
This is not an official Google product.
2017-01-16 20:44:45 -08:00
# Documentation
https://tensorflow.github.io/haskell/haddock/
2018-08-14 20:06:46 -07:00
[TensorFlow.Core ](https://tensorflow.github.io/haskell/haddock/tensorflow/TensorFlow-Core.html )
2017-01-16 20:44:45 -08:00
is a good place to start.
# Examples
Neural network model for the MNIST dataset: [code ](tensorflow-mnist/app/Main.hs )
Toy example of a linear regression model
([full code ](tensorflow-ops/tests/RegressionTest.hs )):
```haskell
2017-05-25 19:19:22 -07:00
import Control.Monad (replicateM, replicateM_)
2017-01-16 20:44:45 -08:00
import System.Random (randomIO)
import Test.HUnit (assertBool)
import qualified TensorFlow.Core as TF
import qualified TensorFlow.GenOps.Core as TF
2017-05-25 19:19:22 -07:00
import qualified TensorFlow.Minimize as TF
import qualified TensorFlow.Ops as TF hiding (initializedVariable)
import qualified TensorFlow.Variable as TF
2017-01-16 20:44:45 -08:00
main :: IO ()
main = do
-- Generate data where `y = x*3 + 8` .
xData < - replicateM 100 randomIO
let yData = [x*3 + 8 | x < - xData ]
-- Fit linear regression model.
(w, b) < - fit xData yData
assertBool "w == 3" (abs (3 - w) < 0.001 )
assertBool "b == 8" (abs (8 - b) < 0.001 )
fit :: [Float] -> [Float] -> IO (Float, Float)
fit xData yData = TF.runSession $ do
-- Create tensorflow constants for x and y.
let x = TF.vector xData
y = TF.vector yData
-- Create scalar variables for slope and intercept.
2017-03-18 12:08:53 -07:00
w < - TF . initializedVariable 0
b < - TF . initializedVariable 0
2017-01-16 20:44:45 -08:00
-- Define the loss function.
2017-05-25 19:19:22 -07:00
let yHat = (x `TF.mul` TF.readValue w) `TF.add` TF.readValue b
2017-01-16 20:44:45 -08:00
loss = TF.square (yHat `TF.sub` y)
-- Optimize with gradient descent.
2017-05-25 19:19:22 -07:00
trainStep < - TF . minimizeWith ( TF . gradientDescent 0 . 001 ) loss [ w , b ]
2017-01-16 20:44:45 -08:00
replicateM_ 1000 (TF.run trainStep)
-- Return the learned parameters.
2017-05-25 19:19:22 -07:00
(TF.Scalar w', TF.Scalar b') < - TF . run ( TF . readValue w , TF . readValue b )
2017-01-16 20:44:45 -08:00
return (w', b')
```
# Installation Instructions
2016-10-24 19:26:42 +00:00
2020-11-11 09:21:35 -08:00
Note: building this repository with `stack` requires version `2.3.1` or newer.
2017-07-18 00:04:15 +08:00
Check your stack version with `stack --version` in a terminal.
2017-06-02 16:02:30 -07:00
2016-10-26 11:13:42 -07:00
## Build with Docker on Linux
2016-10-24 19:26:42 +00:00
2016-10-25 09:53:35 -07:00
As an expedient we use [docker ](https://www.docker.com/ ) for building. Once you have docker
2016-10-24 19:26:42 +00:00
working, the following commands will compile and run the tests.
2018-08-17 22:48:13 +02:00
```
git clone --recursive https://github.com/tensorflow/haskell.git tensorflow-haskell
cd tensorflow-haskell
2023-01-17 12:43:09 +01:00
docker build -t tensorflow/haskell:2.3.0 docker
2018-08-17 22:48:13 +02:00
# TODO: move the setup step to the docker script.
2023-01-17 12:43:09 +01:00
stack --docker setup
stack --docker test
2018-08-17 22:48:13 +02:00
```
2016-10-24 19:26:42 +00:00
There is also a demo application:
2018-08-17 22:48:13 +02:00
```
cd tensorflow-mnist
2023-01-17 12:43:09 +01:00
stack --docker build --exec Main
2018-08-17 22:48:13 +02:00
```
2016-10-26 11:13:42 -07:00
2018-08-17 22:48:13 +02:00
### Stack + Docker + GPU
2017-10-22 02:32:19 +02:00
If you want to use GPU you can do:
2018-08-17 22:48:13 +02:00
```
2020-11-13 12:21:27 -08:00
IMAGE_NAME=tensorflow/haskell:2.3.0-gpu
2018-08-17 22:48:13 +02:00
docker build -t $IMAGE_NAME docker/gpu
2023-01-17 12:43:09 +01:00
# TODO: move the setup step to the docker script.
stack --docker --docker-image=$IMAGE_NAME setup
stack --docker --docker-image=$IMAGE_NAME test
2018-08-17 22:48:13 +02:00
```
2017-10-22 02:32:19 +02:00
2018-08-17 22:48:13 +02:00
### Using nvidia-docker version 2
See [Nvidia docker 2 install instructions ](https://github.com/nvidia/nvidia-docker/wiki/Installation-(version-2.0 ))
2017-10-22 02:32:19 +02:00
2018-08-17 22:48:13 +02:00
```
stack --docker --docker-image=$IMAGE_NAME setup
stack --docker --docker-run-args "--runtime=nvidia" --docker-image=$IMAGE_NAME test
```
### Using nvidia-docker classic
Stack needs to use `nvidia-docker` instead of the normal `docker` for GPU support. We must wrap 'docker' with a script. This script will shadow the normal `docker` command.
```
ln -s `pwd` /tools/nvidia-docker-wrapper.sh < somewhere in your path > /docker
stack --docker --docker-image=$IMAGE_NAME setup
stack --docker --docker-image=$IMAGE_NAME test
```
2017-10-22 02:32:19 +02:00
2017-07-20 13:17:50 -07:00
## Build on macOS
2016-10-26 11:13:42 -07:00
2017-07-20 13:17:50 -07:00
Run the [install_macos_dependencies.sh ](./tools/install_macos_dependencies.sh )
2017-03-09 19:54:24 -05:00
script in the `tools/` directory. The script installs dependencies
2017-08-21 12:59:22 -07:00
via [Homebrew ](https://brew.sh/ ) and then downloads and installs the TensorFlow
2017-03-09 19:54:24 -05:00
library on your machine under `/usr/local` .
2016-10-26 11:13:42 -07:00
2017-07-18 00:04:15 +08:00
After running the script to install system dependencies, build the project with stack:
2016-10-26 11:13:42 -07:00
2017-03-09 19:54:24 -05:00
stack test
2017-06-11 08:24:54 +03:00
## Build on NixOS
2023-01-17 10:53:52 +01:00
The `stack.yaml` file describes a NixOS environment containing the necessary
2018-05-25 04:33:15 +02:00
dependencies. To build, run:
2017-06-11 08:24:54 +03:00
2018-05-25 04:33:15 +02:00
$ stack --nix build
2019-02-28 19:55:34 -08:00
## Installation on CentOS
[Xiaokui Shu (@subbyte) ](https://github.com/subbyte ) maintains [separate instructions for installation on CentOS ](https://github.com/subbyte/haskell-learn/blob/master/tensorflow_setup.md ).
2017-12-21 20:24:19 -05:00
# Related Projects
2018-01-17 13:31:23 -05:00
## Statically validated tensor shapes
2017-12-21 20:24:19 -05:00
https://github.com/helq/tensorflow-haskell-deptyped is experimenting with using dependent types to statically validate tensor shapes. May be merged with this repository in the future.
2018-01-17 13:31:23 -05:00
Example:
```haskell
{-# LANGUAGE DataKinds, ScopedTypeVariables #-}
import Data.Maybe (fromJust)
import Data.Vector.Sized (Vector, fromList)
import TensorFlow.DepTyped
test :: IO (Vector 8 Float)
test = runSession $ do
(x :: Placeholder "x" '[4,3] Float) < - placeholder
let elems1 = fromJust $ fromList [1,2,3,4,1,2]
elems2 = fromJust $ fromList [5,6,7,8]
(w :: Tensor '[3,2] '[] Build Float) = constant elems1
(b :: Tensor '[4,1] '[] Build Float) = constant elems2
y = (x `matMul` w) `add` b -- y shape: [4,2] (b shape is [4.1] but `add` broadcasts it to [4,2])
let (inputX :: TensorData "x" [4,3] Float) =
encodeTensorData . fromJust $ fromList [1,2,3,4,1,0,7,9,5,3,5,4]
runWithFeeds (feed x inputX :~~ NilFeedList) y
main :: IO ()
main = test >>= print
```
2017-12-21 20:24:19 -05:00
# License
2017-08-21 12:59:22 -07:00
This project is licensed under the terms of the [Apache 2.0 license ](LICENSE ).