Compare commits
	
		
			367 Commits
		
	
	
		
			0.2.9
			...
			51-calcula
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | d145a852e7 | ||
|  | c090501534 | ||
|  | 6a1069780b | ||
|  | 41aa142ce0 | ||
|  | b48730137d | ||
|  | f1ec057b16 | ||
|  | f6ca82cac8 | ||
|  | 318acd5a71 | ||
|  | c8f6f5613d | ||
|  | 748d1a00bd | ||
|  | a82e4ed440 | ||
|  | c939b095de | ||
|  | ca69bddc17 | ||
|  | aa35e62a27 | ||
|  | 6dd4a44924 | ||
|  | 4291c5eabf | ||
|  | 7c7a815d1a | ||
|  | a3eb8f196c | ||
|  | 00ffbb3804 | ||
|  | 13cf0c71c5 | ||
|  | e5662d482e | ||
|  | 61181db199 | ||
|  | a1663a3f31 | ||
|  | 83b23dbdf4 | ||
|  | 4675d9acf1 | ||
|  | 6832a2816d | ||
|  | 7a8abf1823 | ||
|  | a21ce0d90e | ||
|  | a964e586d7 | ||
|  | bce42b5bb4 | ||
|  | 1313853788 | ||
|  | 697e779767 | ||
|  | 48f5ffafa1 | ||
|  | 73f7cbbe8a | ||
|  | 07a41236f8 | ||
|  | 55db97cf62 | ||
|  | d8dead1908 | ||
|  | 87dcdb9fbc | ||
|  | 67ef4b60bd | ||
|  | da4b11e5b5 | ||
|  | c0aa7ddc3c | ||
|  | 5e2ada1654 | ||
|  | 7a188586c5 | ||
|  | b768b215c5 | ||
|  | d1f1b9a15a | ||
|  | 52a0f3f4c8 | ||
|  | 55c32dcd7c | ||
|  | 0093bc34d5 | ||
|  | 67bae9a20d | ||
|  | 551a5cb176 | ||
|  | d6f4cc2dd2 | ||
|  | 4af692091a | ||
|  | ec68ff0b90 | ||
|  | 738da490db | ||
|  | d29c42fd2e | ||
|  | 23c88d0acc | ||
|  | dcaaa591b7 | ||
|  | 15ab5f4c25 | ||
|  | 92189822d8 | ||
|  | fbb418c365 | ||
|  | 081078ddd6 | ||
|  | 7c8dbf3262 | ||
|  | 41dc89b23b | ||
|  | a951696317 | ||
|  | 1087692de2 | ||
|  | 3e2b8baeb2 | ||
|  | 21a5a3f201 | ||
|  | abd401f863 | ||
|  | bfc588a915 | ||
|  | f93eed2cf5 | ||
|  | 0204e0b8e9 | ||
|  | 701f46b9f1 | ||
|  | d1eca04eeb | ||
|  | 89f3a0eca9 | ||
|  | df7efbc57d | ||
|  | aa54d1c9c8 | ||
|  | 869c00f709 | ||
|  | e329e84eef | ||
|  | 55be0e57da | ||
|  | 778746c5e8 | ||
|  | 19278d0acd | ||
|  | 694201d8d3 | ||
|  | e8413fb645 | ||
|  | 390225df45 | ||
|  | b03e03fd0a | ||
|  | 79e107bdcd | ||
|  | c6e79fa50d | ||
|  | f6bf7459a8 | ||
|  | 300f4c374a | ||
|  | 97cd443c16 | ||
|  | 49afd2cfdd | ||
|  | c1174189c6 | ||
|  | ea536c0daf | ||
|  | a67e0e45d2 | ||
|  | 5f4dc3ac5d | ||
|  | 1a3b8ee703 | ||
|  | 9c61c18220 | ||
|  | 21ff551769 | ||
|  | 3dc27f12f7 | ||
|  | 9957486f4f | ||
|  | 0dc93fc16b | ||
|  | cbef9630b4 | ||
|  | a3a9414073 | ||
|  | 14bcfd511f | ||
|  | 3d23370a59 | ||
|  | f9a75f4e21 | ||
|  | 99d4bc70bc | ||
|  | fc94b45448 | ||
|  | 53db670715 | ||
|  | 963211caf2 | ||
|  | 2ca9d36f80 | ||
|  | b1dbe432c1 | ||
|  | 921d7f23ce | ||
|  | 8bfb88c926 | ||
|  | 8ad2fae774 | ||
|  | eb09da878f | ||
|  | 407fd718a3 | ||
|  | 3bba29fc4e | ||
|  | 4344fccd57 | ||
|  | 0ccdf735e1 | ||
|  | 7444aa7ec8 | ||
|  | 5d68c0225a | ||
|  | cda9f5c4ca | ||
|  | a825e91425 | ||
|  | 473efd8dd7 | ||
|  | 6f489acdfc | ||
|  | a73f3112ab | ||
|  | 5a53ba23e2 | ||
|  | e3a9a3464c | ||
|  | 3e3f5555ff | ||
|  | 7aa91d1d60 | ||
|  | 2480ec310e | ||
|  | 9b956b2358 | ||
|  | 83ddd5e990 | ||
|  | d77d01c3e1 | ||
|  | f6271495c1 | ||
|  | 1cccd9d5cb | ||
|  | a243f68bfc | ||
|  | fca0ac00c4 | ||
|  | 9a2932b569 | ||
|  | 9acee50837 | ||
|  | cd22291dc9 | ||
|  | 85aef3d15f | ||
|  | b6f00385ab | ||
|  | 8c5f894843 | ||
|  | 9870391088 | ||
|  | ad2051307a | ||
|  | 7fd16a17fb | ||
|  | 7547fc49af | ||
|  | 7f44f9e85d | ||
|  | aaad5e8f2b | ||
|  | 9bea267f52 | ||
|  | 4d7e8e7589 | ||
|  | 8e4578dc25 | ||
|  | 24c97256e8 | ||
|  | 312e7f7f12 | ||
|  | c555b9547e | ||
|  | 991ade8f4d | ||
|  | 1104e816cb | ||
|  | c19d03b41d | ||
|  | 42c9068991 | ||
|  | 96843827bd | ||
|  | d76e4618fe | ||
|  | c9bc485535 | ||
|  | 6d7575bbcd | ||
|  | 852bcc72ba | ||
|  | bf5ed1bd7d | ||
|  | 00da75153a | ||
|  | fa082e11e7 | ||
|  | 6331d31b18 | ||
|  | 8ee324f566 | ||
|  | 188c33332a | ||
|  | 955e17eb2a | ||
|  | 3e0f55dcff | ||
|  | 2ea01aef42 | ||
|  | 147fd4a333 | ||
|  | e31bca7016 | ||
|  | 7956d54c35 | ||
|  | 5bab9a6a02 | ||
|  | 69ac95bb08 | ||
|  | 6b843a4384 | ||
|  | 65d6e47513 | ||
|  | 8d56a0b630 | ||
|  | e7ac6e66b0 | ||
|  | 0f8d1dff69 | ||
|  | 236183593c | ||
|  | 7637498517 | ||
|  | 8c70433312 | ||
|  | ce83fb3981 | ||
|  | 28f29d159a | ||
|  | c803f60fd4 | ||
|  | 12eae16e37 | ||
|  | f3372c27b6 | ||
|  | b6de72a143 | ||
|  | 0f89b92457 | ||
|  | ea91e3e4a4 | ||
|  | f76b777b9f | ||
|  | dcc965ea63 | ||
|  | 400f647b7b | ||
|  | ec1a2ff5f9 | ||
|  | e112dd55ce | ||
|  | 60ef304108 | ||
|  | 1a9dd07f7e | ||
|  | b80b0c7947 | ||
|  | 1ca6ec52fd | ||
|  | 7927cf1587 | ||
|  | 13cefbedfb | ||
|  | 4ba9535d56 | ||
|  | e582ef07d4 | ||
|  | ef40bdb545 | ||
|  | e0b4c76238 | ||
|  | 14c86ec38c | ||
|  | d3d05b3218 | ||
|  | eababcadb0 | ||
|  | 7efece0224 | ||
|  | 53138e6942 | ||
|  | 1302b0b93c | ||
|  | ad1092690b | ||
|  | e35e810ede | ||
|  | d5ddcb8d3f | ||
|  | 54c0c9c437 | ||
|  | 6e970d01f2 | ||
|  | 1d0a54ecd2 | ||
|  | 800d4a9c2c | ||
|  | 035ef98b7e | ||
|  | d7e115d7c2 | ||
|  | 548cb4c9ba | ||
|  | 7e5b55ff9c | ||
|  | 8b2c3e8d40 | ||
|  | 0c8f98d466 | ||
|  | cc298742ec | ||
|  | 250052fb99 | ||
|  | 603e086606 | ||
|  | a8614bab0c | ||
|  | 70ca74b03c | ||
|  | c9e6d78183 | ||
|  | 1a582c0843 | ||
|  | 0394bcd69c | ||
|  | cbeb3adbdb | ||
|  | efb305173e | ||
|  | 2288b04c92 | ||
|  | 7899cb4d33 | ||
|  | 62ddca79ac | ||
|  | 99403b3443 | ||
|  | a0ff528a4b | ||
|  | 97bd245dfc | ||
|  | d8b59d06a4 | ||
|  | 453b9f3257 | ||
|  | 5fb858f5fc | ||
|  | bd984a1437 | ||
|  | e741b565a1 | ||
|  | 668a803d89 | ||
|  | 9daae8dda7 | ||
|  | c72094b94b | ||
|  | 15d456d048 | ||
|  | fef06d4333 | ||
|  | 454aa61fba | ||
|  | ba2e18125c | ||
|  | 9f6a6f5ecd | ||
|  | 3cea7534ef | ||
|  | 7eaf303124 | ||
|  | b4ca5f4a7c | ||
|  | 3311af2167 | ||
|  | a4694dff2c | ||
|  | 6cb669cdb1 | ||
|  | 506feec13d | ||
|  | 2e3a6b7c84 | ||
|  | 7cc8b562f4 | ||
|  | 528bbcac35 | ||
|  | 068241fb72 | ||
|  | 39d86a2050 | ||
|  | 5371c83ab0 | ||
|  | 673992dbe8 | ||
|  | eb3a42c247 | ||
|  | 20357d2a0d | ||
|  | e9d7980e42 | ||
|  | 908090f634 | ||
|  | cb963dc438 | ||
|  | 477cb18db1 | ||
|  | fbf0384985 | ||
|  | 7a2c016cc6 | ||
|  | b072121e20 | ||
|  | ceed9b97d0 | ||
|  | 2dbdb58b06 | ||
|  | db30257373 | ||
|  | 7fd69cc690 | ||
|  | b543a4614e | ||
|  | bc1f9e4cf5 | ||
|  | d72a995fa9 | ||
|  | 40b67503ce | ||
|  | 8624562f02 | ||
|  | 4dee623ef9 | ||
|  | 2e7530d9bc | ||
|  | 07b5dd3823 | ||
|  | 0d511ad3c3 | ||
|  | 7205a0e7b2 | ||
|  | fff38bf825 | ||
|  | 5d5de0bc50 | ||
|  | 0454fb1afe | ||
|  | 5e36c71fa7 | ||
|  | c8e742f96e | ||
|  | 1e7ae13700 | ||
|  | bf30c04a52 | ||
|  | 16ce767d09 | ||
|  | 39761e0922 | ||
|  | 03eb38c12d | ||
|  | a50f026701 | ||
|  | b8339e397b | ||
|  | 407d17b2b9 | ||
|  | 56fef9e835 | ||
|  | 14a3e4103b | ||
|  | 48d7d1d02e | ||
|  | 14c9f61864 | ||
|  | a79df7a3da | ||
|  | ecc2a8312a | ||
|  | aafd6a0938 | ||
|  | b88d6c53e0 | ||
|  | 4f2aee5681 | ||
|  | a5c27bcaba | ||
|  | cefd6331e0 | ||
|  | c2bb93e86c | ||
|  | 091104bc7d | ||
|  | 81cbe5ea54 | ||
|  | ab2c1f73e4 | ||
|  | 6a84af1c5a | ||
|  | 5983493b78 | ||
|  | 61deabe13e | ||
|  | bb1b4d3266 | ||
|  | 703fb68b27 | ||
|  | 6fe68e3c40 | ||
|  | 7b9f8a8bef | ||
|  | 82496dc8e4 | ||
|  | f74ee668b6 | ||
|  | d304dec2f7 | ||
|  | 45838e7e98 | ||
|  | ff002c818a | ||
|  | 79d6b6f67f | ||
|  | b8993f7d64 | ||
|  | bd2e0f0d5c | ||
|  | 7de5b41340 | ||
|  | a63e9209fd | ||
|  | b0eb2e0628 | ||
|  | 60415f8217 | ||
|  | 724eac38d8 | ||
|  | 8fa372de15 | ||
|  | a1ffe04a30 | ||
|  | 74b0cf868e | ||
|  | 50e8e2730b | ||
|  | b484b453e0 | ||
|  | 7c2e0ddec7 | ||
|  | 384aba4654 | ||
|  | a857dd3042 | ||
|  | b1b672f66d | ||
|  | 09d9143a82 | ||
|  | c1a6b57ac5 | ||
|  | 6b78b7ccc7 | ||
|  | f0b1cfcba6 | ||
|  | 4bcd046016 | ||
|  | ae09f609c2 | ||
|  | d1006bbc92 | ||
|  | d58137e8f9 | ||
|  | 79c83e34a3 | ||
|  | 37a098109f | ||
|  | ff14925056 | ||
|  | 10f4782ad7 | ||
|  | 4351f76b60 | ||
|  | 86f45f8147 | 
							
								
								
									
										9
									
								
								.drone.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,9 @@ | |||||||
|  | build: | ||||||
|  |   image: python:$$PYTHON_VERSION | ||||||
|  |   commands: | ||||||
|  |     - python setup.py test | ||||||
|  |  | ||||||
|  | matrix: | ||||||
|  |   PYTHON_VERSION: | ||||||
|  |     - 2.7 | ||||||
|  |     - 3.4 | ||||||
							
								
								
									
										10
									
								
								.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,10 @@ | |||||||
|  | *.pyc | ||||||
|  | .* | ||||||
|  | *egg-info | ||||||
|  | dist | ||||||
|  | build | ||||||
|  | README.html | ||||||
|  | __pycache__ | ||||||
|  | VERSION | ||||||
|  | Dockerfile-* | ||||||
|  | Dockerfile | ||||||
							
								
								
									
										103
									
								
								.gitlab-ci.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,103 @@ | |||||||
|  | # Uncomment if you want to use docker-in-docker | ||||||
|  | # image: gsiupm/dockermake:latest | ||||||
|  | # services: | ||||||
|  | # - docker:dind | ||||||
|  | # When using dind, it's wise to use the overlayfs driver for | ||||||
|  | # improved performance. | ||||||
|  |  | ||||||
|  | stages: | ||||||
|  |   - test | ||||||
|  |   - push | ||||||
|  |   - deploy | ||||||
|  |   - clean | ||||||
|  |  | ||||||
|  | before_script: | ||||||
|  |   - make -e login | ||||||
|  |  | ||||||
|  | .test: &test_definition | ||||||
|  |   stage: test | ||||||
|  |   script: | ||||||
|  |     - make -e test-$PYTHON_VERSION | ||||||
|  |   except: | ||||||
|  |     - tags # Avoid unnecessary double testing | ||||||
|  |      | ||||||
|  | test-3.5: | ||||||
|  |   <<: *test_definition | ||||||
|  |   variables: | ||||||
|  |     PYTHON_VERSION: "3.5" | ||||||
|  |  | ||||||
|  | test-2.7: | ||||||
|  |   <<: *test_definition | ||||||
|  |   variables: | ||||||
|  |     PYTHON_VERSION: "2.7" | ||||||
|  |  | ||||||
|  | push: | ||||||
|  |   stage: push | ||||||
|  |   script: | ||||||
|  |     - make -e push | ||||||
|  |   only: | ||||||
|  |     - tags | ||||||
|  |     - triggers | ||||||
|  |     - fix-makefiles | ||||||
|  |  | ||||||
|  | push-latest: | ||||||
|  |   stage: push | ||||||
|  |   script: | ||||||
|  |     - make -e push-latest | ||||||
|  |   only: | ||||||
|  |     - master | ||||||
|  |     - triggers | ||||||
|  |     - fix-makefiles | ||||||
|  |  | ||||||
|  | push-github: | ||||||
|  |   stage: deploy | ||||||
|  |   script: | ||||||
|  |     - make -e push-github | ||||||
|  |   only: | ||||||
|  |     - master | ||||||
|  |     - triggers | ||||||
|  |     - fix-makefiles | ||||||
|  |  | ||||||
|  | deploy_pypi: | ||||||
|  |   stage: deploy | ||||||
|  |   script:   # Configure the PyPI credentials, then push the package, and cleanup the creds. | ||||||
|  |     - echo "[server-login]" >> ~/.pypirc | ||||||
|  |     - echo "repository=https://upload.pypi.org/legacy/" >> ~/.pypirc | ||||||
|  |     - echo "username=" ${PYPI_USER} >> ~/.pypirc | ||||||
|  |     - echo "password=" ${PYPI_PASSWORD} >> ~/.pypirc | ||||||
|  |     - make pip_upload | ||||||
|  |     - echo "" > ~/.pypirc && rm ~/.pypirc  # If the above fails, this won't run. | ||||||
|  |   only: | ||||||
|  |     - /^v?\d+\.\d+\.\d+([abc]\d*)?$/  # PEP-440 compliant version (tags) | ||||||
|  |   except: | ||||||
|  |     - branches | ||||||
|  |  | ||||||
|  | deploy: | ||||||
|  |   stage: deploy | ||||||
|  |   environment: test | ||||||
|  |   script: | ||||||
|  |     - make -e deploy | ||||||
|  |   only: | ||||||
|  |     - master | ||||||
|  |     - fix-makefiles | ||||||
|  |  | ||||||
|  | push-github: | ||||||
|  |   stage: deploy | ||||||
|  |   script: | ||||||
|  |     - make -e push-github | ||||||
|  |   only: | ||||||
|  |     - master | ||||||
|  |     - triggers | ||||||
|  |  | ||||||
|  | clean : | ||||||
|  |   stage: clean | ||||||
|  |   script: | ||||||
|  |     - make -e clean | ||||||
|  |   when: manual | ||||||
|  |  | ||||||
|  | cleanup_py: | ||||||
|  |    stage: clean | ||||||
|  |    when: always   # this is important; run even if preceding stages failed. | ||||||
|  |    script: | ||||||
|  |     - rm -vf ~/.pypirc  # we don't want to leave these around, but GitLab may clean up anyway. | ||||||
|  |     - docker logout | ||||||
							
								
								
									
										27
									
								
								.makefiles/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,27 @@ | |||||||
|  | These makefiles are recipes for several common tasks in different types of projects. | ||||||
|  | To add them to your project, simply do: | ||||||
|  |  | ||||||
|  | ``` | ||||||
|  | git remote add makefiles ssh://git@lab.cluster.gsi.dit.upm.es:2200/docs/templates/makefiles.git | ||||||
|  | git subtree add --prefix=.makefiles/ makefiles master | ||||||
|  | touch Makefile | ||||||
|  | echo "include .makefiles/base.mk" >> Makefile | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | Now you can take advantage of the recipes. | ||||||
|  | For instance, to add useful targets for a python project, just add this to your Makefile: | ||||||
|  |  | ||||||
|  | ``` | ||||||
|  | include .makefiles/python.mk | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | You may need to set special variables like the name of your project or the python versions you're targetting. | ||||||
|  | Take a look at each specific `.mk` file for more information, and the `Makefile` in the [senpy](https://lab.cluster.gsi.dit.upm.es/senpy/senpy) project for a real use case. | ||||||
|  |  | ||||||
|  | If you update the makefiles from your repository, make sure to push the changes for review in upstream (this repository): | ||||||
|  |  | ||||||
|  | ``` | ||||||
|  | make makefiles-push | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | It will automatically commit all unstaged changes in the .makefiles folder. | ||||||
							
								
								
									
										36
									
								
								.makefiles/base.mk
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,36 @@ | |||||||
|  | export | ||||||
|  | NAME ?= $(shell basename $(CURDIR)) | ||||||
|  | VERSION ?= $(shell git describe --tags --dirty 2>/dev/null) | ||||||
|  |  | ||||||
|  | ifeq ($(VERSION),) | ||||||
|  | 	VERSION:=unknown | ||||||
|  | endif | ||||||
|  |  | ||||||
|  | # Get the location of this makefile. | ||||||
|  | MK_DIR := $(dir $(abspath $(lastword $(MAKEFILE_LIST)))) | ||||||
|  |  | ||||||
|  | -include .env | ||||||
|  | -include ../.env | ||||||
|  |  | ||||||
|  | help:           ## Show this help. | ||||||
|  | 	@fgrep -h "##" $(MAKEFILE_LIST) | fgrep -v fgrep | sed -e 's/\\$$//' | sed -e 's/\(.*:\)[^#]*##\s*\(.*\)/\1\t\2/' | column -t -s "	" | ||||||
|  |  | ||||||
|  | config:  ## Load config from the environment. You should run it once in every session before other tasks. Run: eval $(make config) | ||||||
|  | 	@awk '{ print "export " $$0}' ../.env | ||||||
|  | 	@awk '{ print "export " $$0}' .env | ||||||
|  | 	@echo "# Please, run: " | ||||||
|  | 	@echo "# eval \$$(make config)" | ||||||
|  | # If you need to run a command on the key/value pairs, use this: | ||||||
|  | # @awk '{ split($$0, a, "="); "echo " a[2] " | base64 -w 0" |& getline b64; print "export " a[1] "=" a[2]; print "export " a[1] "_BASE64=" b64}' .env | ||||||
|  |  | ||||||
|  | ci:  ## Run a task using gitlab-runner. Only use to debug problems in the CI pipeline | ||||||
|  | 	gitlab-runner exec shell --builds-dir '.builds' --env CI_PROJECT_NAME=$(NAME) ${action} | ||||||
|  |  | ||||||
|  | include $(MK_DIR)/makefiles.mk | ||||||
|  | include $(MK_DIR)/docker.mk | ||||||
|  | include $(MK_DIR)/git.mk | ||||||
|  |  | ||||||
|  | info:: ## List all variables | ||||||
|  | 	env | ||||||
|  |  | ||||||
|  | .PHONY:: config help ci | ||||||
							
								
								
									
										51
									
								
								.makefiles/docker.mk
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,51 @@ | |||||||
|  | ifndef IMAGENAME | ||||||
|  | 	ifdef CI_REGISTRY_IMAGE | ||||||
|  | 		IMAGENAME=$(CI_REGISTRY_IMAGE) | ||||||
|  | 	else | ||||||
|  | 		IMAGENAME=$(NAME) | ||||||
|  | 	endif | ||||||
|  | endif | ||||||
|  |  | ||||||
|  | IMAGEWTAG?=$(IMAGENAME):$(VERSION) | ||||||
|  | DOCKER_FLAGS?=$(-ti) | ||||||
|  | DOCKER_CMD?= | ||||||
|  |  | ||||||
|  | docker-login: ## Log in to the registry. It will only be used in the server, or when running a CI task locally (if CI_BUILD_TOKEN is set). | ||||||
|  | ifeq ($(CI_BUILD_TOKEN),) | ||||||
|  | 	@echo "Not logging in to the docker registry" "$(CI_REGISTRY)" | ||||||
|  | else | ||||||
|  | 	@docker login -u gitlab-ci-token -p $(CI_BUILD_TOKEN) $(CI_REGISTRY) | ||||||
|  | endif | ||||||
|  | ifeq ($(HUB_USER),) | ||||||
|  | 	@echo "Not logging in to global the docker registry" | ||||||
|  | else | ||||||
|  | 	@docker login -u $(HUB_USER) -p $(HUB_PASSWORD) | ||||||
|  | endif | ||||||
|  |  | ||||||
|  | docker-clean: ## Remove docker credentials | ||||||
|  | ifeq ($(HUB_USER),) | ||||||
|  | else | ||||||
|  | 	@docker logout | ||||||
|  | endif | ||||||
|  |  | ||||||
|  | docker-run: ## Build a generic docker image | ||||||
|  | 	docker run $(DOCKER_FLAGS) $(IMAGEWTAG) $(DOCKER_CMD) | ||||||
|  |  | ||||||
|  | docker-build: ## Build a generic docker image | ||||||
|  | 	docker build . -t $(IMAGEWTAG) | ||||||
|  |  | ||||||
|  | docker-push: docker-login ## Push a generic docker image | ||||||
|  | 	docker push $(IMAGEWTAG) | ||||||
|  |  | ||||||
|  | docker-latest-push: docker-login ## Push the latest image | ||||||
|  | 	docker tag $(IMAGEWTAG) $(IMAGENAME) | ||||||
|  | 	docker push $(IMAGENAME) | ||||||
|  |  | ||||||
|  | login:: docker-login | ||||||
|  |  | ||||||
|  | clean:: docker-clean | ||||||
|  |  | ||||||
|  | docker-info: | ||||||
|  | 	@echo IMAGEWTAG=${IMAGEWTAG} | ||||||
|  |  | ||||||
|  | .PHONY:: docker-login docker-clean login clean | ||||||
							
								
								
									
										25
									
								
								.makefiles/git.mk
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,25 @@ | |||||||
|  | commit: | ||||||
|  | 	git commit -a | ||||||
|  |  | ||||||
|  | tag: | ||||||
|  | 	git tag ${VERSION} | ||||||
|  |  | ||||||
|  | git-push:: | ||||||
|  | 	git push --tags -u origin HEAD | ||||||
|  |  | ||||||
|  | git-pull: | ||||||
|  | 	git pull --all | ||||||
|  |  | ||||||
|  | push-github: ## Push the code to github. You need to set up GITHUB_DEPLOY_KEY | ||||||
|  | ifeq ($(GITHUB_DEPLOY_KEY),) | ||||||
|  | else | ||||||
|  | 	$(eval KEY_FILE := "$(shell mktemp)") | ||||||
|  | 	@printf '%b' '$(GITHUB_DEPLOY_KEY)' > $(KEY_FILE) | ||||||
|  | 	@git remote rm github-deploy || true | ||||||
|  | 	git remote add github-deploy $(GITHUB_REPO) | ||||||
|  | 	-@GIT_SSH_COMMAND="ssh -i $(KEY_FILE)" git fetch github-deploy $(CI_COMMIT_REF_NAME) | ||||||
|  | 	@GIT_SSH_COMMAND="ssh -i $(KEY_FILE)" git push github-deploy HEAD:$(CI_COMMIT_REF_NAME) | ||||||
|  | 	rm $(KEY_FILE) | ||||||
|  | endif | ||||||
|  |  | ||||||
|  | .PHONY:: commit tag git-push git-pull push-github  | ||||||
							
								
								
									
										51
									
								
								.makefiles/k8s.mk
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,51 @@ | |||||||
|  | # Deployment with Kubernetes | ||||||
|  |  | ||||||
|  | # KUBE_CA_PEM_FILE is the path of a certificate file. It automatically set by GitLab | ||||||
|  | # if you enable Kubernetes integration in a project. | ||||||
|  | # | ||||||
|  | # As of this writing, Kubernetes integration can not be set on a group level, so it has to | ||||||
|  | # be manually set in every project. | ||||||
|  | # Alternatively, we use a custom KUBE_CA_BUNDLE environment variable, which can be set at | ||||||
|  | # the group level. In this case, the variable contains the whole content of the certificate, | ||||||
|  | # which we dump to a temporary file | ||||||
|  | # | ||||||
|  | # Check if the KUBE_CA_PEM_FILE exists. Otherwise, create it from KUBE_CA_BUNDLE | ||||||
|  | KUBE_CA_TEMP=false | ||||||
|  | ifndef KUBE_CA_PEM_FILE | ||||||
|  | KUBE_CA_PEM_FILE:=$$PWD/.ca.crt | ||||||
|  | CREATED:=$(shell printf '%b\n' '$(KUBE_CA_BUNDLE)' > $(KUBE_CA_PEM_FILE)) | ||||||
|  | endif  | ||||||
|  | KUBE_TOKEN?="" | ||||||
|  | KUBE_NAMESPACE?=$(NAME) | ||||||
|  | KUBECTL=docker run --rm -v $(KUBE_CA_PEM_FILE):/tmp/ca.pem -i lachlanevenson/k8s-kubectl --server="$(KUBE_URL)" --token="$(KUBE_TOKEN)" --certificate-authority="/tmp/ca.pem" -n $(KUBE_NAMESPACE) | ||||||
|  | CI_COMMIT_REF_NAME?=master | ||||||
|  |  | ||||||
|  | info:: ## Print variables. Useful for debugging. | ||||||
|  | 	@echo "#KUBERNETES" | ||||||
|  | 	@echo KUBE_URL=$(KUBE_URL) | ||||||
|  | 	@echo KUBE_CA_PEM_FILE=$(KUBE_CA_PEM_FILE) | ||||||
|  | 	@echo KUBE_CA_BUNDLE=$$KUBE_CA_BUNDLE | ||||||
|  | 	@echo KUBE_TOKEN=$(KUBE_TOKEN) | ||||||
|  | 	@echo KUBE_NAMESPACE=$(KUBE_NAMESPACE) | ||||||
|  | 	@echo KUBECTL=$(KUBECTL) | ||||||
|  |  | ||||||
|  | 	@echo "#CI" | ||||||
|  | 	@echo CI_PROJECT_NAME=$(CI_PROJECT_NAME) | ||||||
|  | 	@echo CI_REGISTRY=$(CI_REGISTRY) | ||||||
|  | 	@echo CI_REGISTRY_USER=$(CI_REGISTRY_USER) | ||||||
|  | 	@echo CI_COMMIT_REF_NAME=$(CI_COMMIT_REF_NAME) | ||||||
|  | 	@echo "CREATED=$(CREATED)" | ||||||
|  |  | ||||||
|  | # | ||||||
|  | # Deployment and advanced features | ||||||
|  | #  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | deploy: ## Deploy to kubernetes using the credentials in KUBE_CA_PEM_FILE (or KUBE_CA_BUNDLE ) and TOKEN | ||||||
|  | 	@ls k8s/*.yaml k8s/*.yml k8s/*.tmpl 2>/dev/null || true | ||||||
|  | 	@cat k8s/*.yaml k8s/*.yml k8s/*.tmpl 2>/dev/null | envsubst | $(KUBECTL) apply -f - | ||||||
|  |  | ||||||
|  | deploy-check: ## Get the deployed configuration. | ||||||
|  | 	@$(KUBECTL) get deploy,pods,svc,ingress | ||||||
|  |  | ||||||
|  | .PHONY:: info deploy deploy-check | ||||||
							
								
								
									
										15
									
								
								.makefiles/makefiles.mk
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,15 @@ | |||||||
|  | makefiles-remote: | ||||||
|  | 	git ls-remote --exit-code  makefiles 2> /dev/null || git remote add makefiles ssh://git@lab.cluster.gsi.dit.upm.es:2200/docs/templates/makefiles.git | ||||||
|  |  | ||||||
|  | makefiles-commit: makefiles-remote | ||||||
|  | 	git add -f .makefiles | ||||||
|  | 	git commit -em "Updated makefiles from ${NAME}" | ||||||
|  |  | ||||||
|  | makefiles-push: | ||||||
|  | 	git fetch makefiles $(NAME) | ||||||
|  | 	git subtree push --prefix=.makefiles/ makefiles $(NAME) | ||||||
|  |  | ||||||
|  | makefiles-pull: makefiles-remote | ||||||
|  | 	git subtree pull --prefix=.makefiles/ makefiles master --squash | ||||||
|  |  | ||||||
|  | .PHONY:: makefiles-remote makefiles-commit makefiles-push makefiles-pull | ||||||
							
								
								
									
										5
									
								
								.makefiles/precommit.mk
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,5 @@ | |||||||
|  | init: ## Init pre-commit hooks (i.e. enforcing format checking before allowing a commit) | ||||||
|  | 	pip install --user pre-commit | ||||||
|  | 	pre-commit install | ||||||
|  |  | ||||||
|  | .PHONY:: init | ||||||
							
								
								
									
										101
									
								
								.makefiles/python.mk
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,101 @@ | |||||||
|  | PYVERSIONS ?= 3.5 | ||||||
|  | PYMAIN ?= $(firstword $(PYVERSIONS)) | ||||||
|  | TARNAME ?= $(NAME)-$(VERSION).tar.gz  | ||||||
|  | VERSIONFILE ?= $(NAME)/VERSION | ||||||
|  |  | ||||||
|  | DEVPORT ?= 6000 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .FORCE: | ||||||
|  |  | ||||||
|  | version: .FORCE | ||||||
|  | 	@echo $(VERSION) > $(VERSIONFILE) | ||||||
|  | 	@echo $(VERSION) | ||||||
|  |  | ||||||
|  | yapf: ## Format python code | ||||||
|  | 	yapf -i -r $(NAME) | ||||||
|  | 	yapf -i -r tests | ||||||
|  |  | ||||||
|  | dockerfiles: $(addprefix Dockerfile-,$(PYVERSIONS)) ## Generate dockerfiles for each python version | ||||||
|  | 	@unlink Dockerfile >/dev/null | ||||||
|  | 	ln -s Dockerfile-$(PYMAIN) Dockerfile | ||||||
|  |  | ||||||
|  | Dockerfile-%: Dockerfile.template  ## Generate a specific dockerfile (e.g. Dockerfile-2.7) | ||||||
|  | 	sed "s/{{PYVERSION}}/$*/" Dockerfile.template > Dockerfile-$* | ||||||
|  |  | ||||||
|  | quick_build: $(addprefix build-, $(PYMAIN)) | ||||||
|  |  | ||||||
|  | build: $(addprefix build-, $(PYVERSIONS)) ## Build all images / python versions | ||||||
|  | 	docker tag $(IMAGEWTAG)-python$(PYMAIN) $(IMAGEWTAG) | ||||||
|  |  | ||||||
|  | build-%: version Dockerfile-%  ## Build a specific version (e.g. build-2.7) | ||||||
|  | 	docker build -t '$(IMAGEWTAG)-python$*' -f Dockerfile-$* .; | ||||||
|  |  | ||||||
|  | dev-%: ## Launch a specific development environment using docker (e.g. dev-2.7) | ||||||
|  | 	@docker start $(NAME)-dev$* || (\ | ||||||
|  | 		$(MAKE) build-$*; \ | ||||||
|  | 		docker run -d -w /usr/src/app/ -p $(DEVPORT):5000 -v $$PWD:/usr/src/app --entrypoint=/bin/bash -ti --name $(NAME)-dev$* '$(IMAGEWTAG)-python$*'; \ | ||||||
|  | 	)\ | ||||||
|  |  | ||||||
|  | 	docker exec -ti $(NAME)-dev$* bash | ||||||
|  |  | ||||||
|  | dev: dev-$(PYMAIN) ## Launch a development environment using docker, using the default python version | ||||||
|  |  | ||||||
|  | quick_test: test-$(PYMAIN) | ||||||
|  |  | ||||||
|  | test-%: build-% ## Run setup.py from in an isolated container, built from the base image. (e.g. test-2.7) | ||||||
|  | # This speeds tests up because the image has most (if not all) of the dependencies already. | ||||||
|  | 	docker rm $(NAME)-test-$* || true | ||||||
|  | 	docker create -ti --name $(NAME)-test-$* --entrypoint="" -w /usr/src/app/ $(IMAGEWTAG)-python$* python setup.py test | ||||||
|  | 	docker cp . $(NAME)-test-$*:/usr/src/app | ||||||
|  | 	docker start -a $(NAME)-test-$* | ||||||
|  |  | ||||||
|  | test: $(addprefix test-,$(PYVERSIONS)) ## Run the tests with the main python version | ||||||
|  |  | ||||||
|  | run-%: build-% | ||||||
|  | 	docker run --rm -p $(DEVPORT):5000 -ti '$(IMAGEWTAG)-python$(PYMAIN)' --default-plugins | ||||||
|  |  | ||||||
|  | run: run-$(PYMAIN) | ||||||
|  |  | ||||||
|  | # Pypy - Upload a package | ||||||
|  |  | ||||||
|  | dist/$(TARNAME): version | ||||||
|  | 	python setup.py sdist; | ||||||
|  |  | ||||||
|  | sdist: dist/$(TARNAME) ## Generate the distribution file (wheel) | ||||||
|  |  | ||||||
|  | pip_test-%: sdist ## Test the distribution file using pip install and a specific python version (e.g. pip_test-2.7) | ||||||
|  | 	docker run --rm -v $$PWD/dist:/dist/ python:$* pip install /dist/$(TARNAME); | ||||||
|  |  | ||||||
|  | pip_test: $(addprefix pip_test-,$(PYVERSIONS)) ## Test pip installation with the main python version | ||||||
|  |  | ||||||
|  | pip_upload: pip_test  ## Upload package to pip | ||||||
|  | 	python setup.py sdist upload ; | ||||||
|  |  | ||||||
|  | # Pushing to docker | ||||||
|  |  | ||||||
|  | push-latest: $(addprefix push-latest-,$(PYVERSIONS)) ## Push the "latest" tag to dockerhub | ||||||
|  | 	docker tag '$(IMAGEWTAG)-python$(PYMAIN)' '$(IMAGEWTAG)' | ||||||
|  | 	docker tag '$(IMAGEWTAG)-python$(PYMAIN)' '$(IMAGENAME):latest' | ||||||
|  | 	docker push '$(IMAGENAME):latest' | ||||||
|  | 	docker push '$(IMAGEWTAG)' | ||||||
|  |  | ||||||
|  | push-latest-%: build-%  ## Push the latest image for a specific python version | ||||||
|  | 	docker tag $(IMAGENAME):$(VERSION)-python$* $(IMAGENAME):python$* | ||||||
|  | 	docker push $(IMAGENAME):$(VERSION)-python$* | ||||||
|  | 	docker push $(IMAGENAME):python$* | ||||||
|  |  | ||||||
|  | push-%: build-%  ## Push the image of the current version (tagged). e.g. push-2.7 | ||||||
|  | 	docker push $(IMAGENAME):$(VERSION)-python$* | ||||||
|  |  | ||||||
|  | push:: $(addprefix push-,$(PYVERSIONS)) ## Push an image with the current version for every python version | ||||||
|  | 	docker tag '$(IMAGEWTAG)-python$(PYMAIN)' '$(IMAGEWTAG)' | ||||||
|  | 	docker push  $(IMAGENAME):$(VERSION) | ||||||
|  |  | ||||||
|  | clean:: ## Clean older docker images and containers related to this project and dev environments | ||||||
|  | 	@docker stop $(addprefix $(NAME)-dev,$(PYVERSIONS)) 2>/dev/null || true | ||||||
|  | 	@docker rm $(addprefix $(NAME)-dev,$(PYVERSIONS)) 2>/dev/null || true | ||||||
|  | 	@docker ps -a | grep $(IMAGENAME) | awk '{ split($$2, vers, "-"); if(vers[0] != "${VERSION}"){ print $$1;}}' | xargs docker rm -v 2>/dev/null|| true | ||||||
|  | 	@docker images | grep $(IMAGENAME) | awk '{ split($$2, vers, "-"); if(vers[0] != "${VERSION}"){ print $$1":"$$2;}}' | xargs docker rmi 2>/dev/null|| true | ||||||
|  |  | ||||||
|  | .PHONY:: yapf dockerfiles Dockerfile-% quick_build build build-% dev-% quick-dev test quick_test push-latest push-latest-% push-% push version .FORCE | ||||||
							
								
								
									
										5
									
								
								.pre-commit-config.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,5 @@ | |||||||
|  | -   repo: git://github.com/pre-commit/pre-commit-hooks | ||||||
|  |     sha: e626cd57090d8df0be21e4df0f4e55cc3511d6ab | ||||||
|  |     hooks: | ||||||
|  |     -   id: flake8 | ||||||
|  |     -   id: check-json | ||||||
							
								
								
									
										12
									
								
								.travis.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,12 @@ | |||||||
|  | sudo: required | ||||||
|  |  | ||||||
|  | services: | ||||||
|  |   - docker | ||||||
|  |  | ||||||
|  | language: python | ||||||
|  |  | ||||||
|  | env: | ||||||
|  |   - PYV=2.7 | ||||||
|  |   - PYV=3.5 | ||||||
|  | # run nosetests - Tests | ||||||
|  | script: make test-$PYV | ||||||
							
								
								
									
										26
									
								
								Dockerfile.template
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,26 @@ | |||||||
|  | from python:{{PYVERSION}} | ||||||
|  |  | ||||||
|  | MAINTAINER J. Fernando Sánchez <jf.sanchez@upm.es> | ||||||
|  |  | ||||||
|  | RUN apt-get update && apt-get install -y \ | ||||||
|  | libblas-dev liblapack-dev liblapacke-dev gfortran \ | ||||||
|  |  && rm -rf /var/lib/apt/lists/* | ||||||
|  |  | ||||||
|  | RUN mkdir /cache/ /senpy-plugins /data/ | ||||||
|  |  | ||||||
|  | VOLUME /data/ | ||||||
|  |  | ||||||
|  | ENV PIP_CACHE_DIR=/cache/ SENPY_DATA=/data | ||||||
|  |  | ||||||
|  | ONBUILD COPY . /senpy-plugins/ | ||||||
|  | ONBUILD RUN python -m senpy --only-install -f /senpy-plugins | ||||||
|  | ONBUILD WORKDIR /senpy-plugins/ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | WORKDIR /usr/src/app | ||||||
|  | COPY test-requirements.txt requirements.txt extra-requirements.txt /usr/src/app/ | ||||||
|  | RUN pip install --no-cache-dir -r test-requirements.txt -r requirements.txt -r extra-requirements.txt | ||||||
|  | COPY . /usr/src/app/ | ||||||
|  | RUN pip install --no-cache-dir --no-index --no-deps --editable . | ||||||
|  |  | ||||||
|  | ENTRYPOINT ["python", "-m", "senpy", "-f", "/senpy-plugins/", "--host", "0.0.0.0"] | ||||||
							
								
								
									
										12
									
								
								MANIFEST.in
									
									
									
									
									
								
							
							
						
						| @@ -1,4 +1,10 @@ | |||||||
| include requirements.txt | include requirements.txt | ||||||
| include README.md | include test-requirements.txt | ||||||
| include senpy/context.jsonld | include extra-requirements.txt | ||||||
| recursive-include *.senpy | include README.rst | ||||||
|  | include senpy/VERSION | ||||||
|  | graft senpy/plugins | ||||||
|  | graft senpy/schemas | ||||||
|  | graft senpy/templates | ||||||
|  | graft senpy/static | ||||||
|  | graft img | ||||||
							
								
								
									
										17
									
								
								Makefile
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,17 @@ | |||||||
|  | NAME=senpy | ||||||
|  | GITHUB_REPO=git@github.com:gsi-upm/senpy.git | ||||||
|  |  | ||||||
|  | IMAGENAME=gsiupm/senpy | ||||||
|  |  | ||||||
|  | # The first version is the main one (used for quick builds) | ||||||
|  | # See .makefiles/python.mk for more info | ||||||
|  | PYVERSIONS=3.5 2.7 | ||||||
|  |  | ||||||
|  | DEVPORT=5000 | ||||||
|  |  | ||||||
|  | action="test-${PYMAIN}" | ||||||
|  | GITHUB_REPO=git@github.com:gsi-upm/senpy.git | ||||||
|  |  | ||||||
|  | include .makefiles/base.mk | ||||||
|  | include .makefiles/k8s.mk | ||||||
|  | include .makefiles/python.mk | ||||||
							
								
								
									
										2
									
								
								Procfile
									
									
									
									
									
								
							
							
						
						| @@ -1 +1 @@ | |||||||
| web: gunicorn app:app --log-file=- | web: python -m senpy --host 0.0.0.0 --port $PORT --default-plugins | ||||||
|   | |||||||
							
								
								
									
										19
									
								
								README.md
									
									
									
									
									
								
							
							
						
						| @@ -1,19 +0,0 @@ | |||||||
|  |  | ||||||
| [Senpy](http://senpy.herokuapp.com)  |  | ||||||
| ========================================= |  | ||||||
| Example endpoint that yields results compatible with the EUROSENTIMENT format and exposes the NIF API. |  | ||||||
| It can be used as a template to adapt existing services to EUROSENTIMENT or to create new services. |  | ||||||
|  |  | ||||||
| [DEMO on Heroku](http://eurosentiment-endpoint.herokuapp.com) |  | ||||||
|  |  | ||||||
| This endpoint serves as bootcampt for any developer wishing to build applications that use the EUROSENTIMENT services. |  | ||||||
|  |  | ||||||
| Acknowledgement |  | ||||||
| --------------- |  | ||||||
| EUROSENTIMENT PROJECT |  | ||||||
| Grant Agreement no: 296277 |  | ||||||
| Starting date: 01/09/2012 |  | ||||||
| Project duration: 24 months |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
							
								
								
									
										140
									
								
								README.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,140 @@ | |||||||
|  | .. image:: img/header.png | ||||||
|  |    :width: 100% | ||||||
|  |    :target: http://demos.gsi.dit.upm.es/senpy | ||||||
|  |  | ||||||
|  | .. image:: https://travis-ci.org/gsi-upm/senpy.svg?branch=master | ||||||
|  |    :target: https://travis-ci.org/gsi-upm/senpy | ||||||
|  |  | ||||||
|  | Senpy lets you create sentiment analysis web services easily, fast and using a well known API. | ||||||
|  | As a bonus, senpy services use semantic vocabularies (e.g. `NIF <http://persistence.uni-leipzig.org/nlp2rdf/>`_, `Marl <http://www.gsi.dit.upm.es/ontologies/marl>`_, `Onyx <http://www.gsi.dit.upm.es/ontologies/onyx>`_) and formats (turtle, JSON-LD, xml-rdf). | ||||||
|  |  | ||||||
|  | Have you ever wanted to turn your sentiment analysis algorithms into a service? | ||||||
|  | With senpy, now you can. | ||||||
|  | It provides all the tools so you just have to worry about improving your algorithms: | ||||||
|  |  | ||||||
|  | `See it in action. <http://senpy.cluster.gsi.dit.upm.es/>`_ | ||||||
|  |  | ||||||
|  | Installation | ||||||
|  | ------------ | ||||||
|  | The stable version can be installed in three ways. | ||||||
|  |  | ||||||
|  | Through PIP | ||||||
|  | *********** | ||||||
|  |  | ||||||
|  | .. code:: bash | ||||||
|  |  | ||||||
|  |    pip install -U --user senpy | ||||||
|  |  | ||||||
|  |     | ||||||
|  | Alternatively, you can use the development version: | ||||||
|  |   | ||||||
|  | .. code:: bash | ||||||
|  |  | ||||||
|  |    git clone http://github.com/gsi-upm/senpy | ||||||
|  |    cd senpy | ||||||
|  |    pip install --user . | ||||||
|  |  | ||||||
|  | If you want to install senpy globally, use sudo instead of the ``--user`` flag. | ||||||
|  |  | ||||||
|  | Docker Image | ||||||
|  | ************ | ||||||
|  | Build the image or use the pre-built one: ``docker run -ti -p 5000:5000 gsiupm/senpy --default-plugins``. | ||||||
|  |  | ||||||
|  | To add custom plugins, add a volume and tell senpy where to find the plugins: ``docker run -ti -p 5000:5000 -v <PATH OF PLUGINS>:/plugins gsiupm/senpy --default-plugins -f /plugins`` | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Developing | ||||||
|  | ---------- | ||||||
|  |  | ||||||
|  | Developing/debugging | ||||||
|  | ******************** | ||||||
|  | This command will run the senpy container using the latest image available, mounting your current folder so you get your latest code: | ||||||
|  |  | ||||||
|  | .. code:: bash | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     # Python 3.5 | ||||||
|  |     make dev | ||||||
|  |     # Python 2.7 | ||||||
|  |     make dev-2.7 | ||||||
|  |  | ||||||
|  | Building a docker image | ||||||
|  | *********************** | ||||||
|  |  | ||||||
|  | .. code:: bash | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     # Python 3.5 | ||||||
|  |     make build-3.5 | ||||||
|  |     # Python 2.7 | ||||||
|  |     make build-2.7 | ||||||
|  |  | ||||||
|  | Testing | ||||||
|  | ******* | ||||||
|  |  | ||||||
|  | .. code:: bash | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     make test | ||||||
|  |  | ||||||
|  | Running | ||||||
|  | ******* | ||||||
|  | This command will run the senpy server listening on localhost:5000 | ||||||
|  |  | ||||||
|  | .. code:: bash | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     # Python 3.5 | ||||||
|  |     make run-3.5 | ||||||
|  |     # Python 2.7 | ||||||
|  |     make run-2.7 | ||||||
|  |  | ||||||
|  | Usage | ||||||
|  | ----- | ||||||
|  |  | ||||||
|  | However, the easiest and recommended way is to just use the command-line tool to load your plugins and launch the server. | ||||||
|  |  | ||||||
|  | .. code:: bash | ||||||
|  |  | ||||||
|  |  | ||||||
|  |    senpy | ||||||
|  |  | ||||||
|  | or, alternatively: | ||||||
|  |  | ||||||
|  | .. code:: bash | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     python -m senpy | ||||||
|  |  | ||||||
|  |  | ||||||
|  | This will create a server with any modules found in the current path. | ||||||
|  | For more options, see the `--help` page. | ||||||
|  |  | ||||||
|  | Alternatively, you can use the modules included in senpy to build your own application. | ||||||
|  |  | ||||||
|  | Deploying on Heroku | ||||||
|  | ------------------- | ||||||
|  | Use a free heroku instance to share your service with the world. | ||||||
|  | Just use the example Procfile in this repository, or build your own. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | `DEMO on heroku <http://senpy.herokuapp.com>`_ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | For more information, check out the `documentation <http://senpy.readthedocs.org>`_. | ||||||
|  | ------------------------------------------------------------------------------------ | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Acknowledgement | ||||||
|  | --------------- | ||||||
|  | This development has been partially funded by the European Union through the MixedEmotions Project (project number H2020 655632), as part of the `RIA ICT 15 Big data and Open Data Innovation and take-up` programme. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .. image:: img/me.png | ||||||
|  |     :target: http://mixedemotions-project.eu | ||||||
|  |     :height: 100px | ||||||
|  |     :alt: MixedEmotions Logo | ||||||
|  |  | ||||||
|  | .. image:: img/eu-flag.jpg | ||||||
|  |     :height: 100px | ||||||
|  |     :target: http://ec.europa.eu/research/participants/portal/desktop/en/opportunities/index.html | ||||||
							
								
								
									
										35
									
								
								app.py
									
									
									
									
									
								
							
							
						
						| @@ -1,35 +0,0 @@ | |||||||
| #!/usr/bin/python |  | ||||||
| # -*- coding: utf-8 -*- |  | ||||||
| #    Copyright 2014 J. Fernando Sánchez Rada - Grupo de Sistemas Inteligentes |  | ||||||
| #                                                       DIT, UPM |  | ||||||
| # |  | ||||||
| #    Licensed under the Apache License, Version 2.0 (the "License"); |  | ||||||
| #    you may not use this file except in compliance with the License. |  | ||||||
| #    You may obtain a copy of the License at |  | ||||||
| # |  | ||||||
| #        http://www.apache.org/licenses/LICENSE-2.0 |  | ||||||
| # |  | ||||||
| #    Unless required by applicable law or agreed to in writing, software |  | ||||||
| #    distributed under the License is distributed on an "AS IS" BASIS, |  | ||||||
| #    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |  | ||||||
| #    See the License for the specific language governing permissions and |  | ||||||
| #    limitations under the License. |  | ||||||
| """ |  | ||||||
| Simple Sentiment Analysis server for EUROSENTIMENT |  | ||||||
|  |  | ||||||
| This class shows how to use the nif_server module to create custom services. |  | ||||||
| """ |  | ||||||
| import config |  | ||||||
| from flask import Flask |  | ||||||
| from senpy.extensions import Senpy |  | ||||||
| import logging |  | ||||||
| logging.basicConfig(level=logging.DEBUG) |  | ||||||
|  |  | ||||||
| app = Flask(__name__) |  | ||||||
|  |  | ||||||
| sp = Senpy() |  | ||||||
| sp.init_app(app) |  | ||||||
|  |  | ||||||
| if __name__ == '__main__': |  | ||||||
|     app.debug = config.DEBUG |  | ||||||
|     app.run(host="0.0.0.0", use_reloader=False) |  | ||||||
							
								
								
									
										10
									
								
								docker-compose.dev.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,10 @@ | |||||||
|  | version: '3' | ||||||
|  | services: | ||||||
|  |     senpy: | ||||||
|  |         image: "${IMAGENAME-gsiupm/senpy}:${VERSION-latest}" | ||||||
|  |         entrypoint: ["/bin/bash"] | ||||||
|  |         working_dir: "/senpy-plugins" | ||||||
|  |         ports: | ||||||
|  |             - 5000:5000 | ||||||
|  |         volumes: | ||||||
|  |             - ".:/usr/src/app/" | ||||||
							
								
								
									
										9
									
								
								docker-compose.test.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,9 @@ | |||||||
|  | version: '3' | ||||||
|  | services: | ||||||
|  |     test: | ||||||
|  |         image: "${IMAGENAME-gsiupm/senpy}:${VERSION-dev}" | ||||||
|  |         entrypoint: ["py.test"] | ||||||
|  |         volumes: | ||||||
|  |             - ".:/usr/src/app/" | ||||||
|  |         command: | ||||||
|  |           [] | ||||||
							
								
								
									
										11
									
								
								docker-compose.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,11 @@ | |||||||
|  | version: '3' | ||||||
|  | services: | ||||||
|  |     senpy: | ||||||
|  |         image: "${IMAGENAME-gsiupm/senpy}:${VERSION-dev}" | ||||||
|  |         build: | ||||||
|  |           context: . | ||||||
|  |           dockerfile: Dockerfile${PYVERSION--2.7} | ||||||
|  |         ports: | ||||||
|  |             - 5001:5000 | ||||||
|  |         volumes: | ||||||
|  |             - "./data:/data" | ||||||
							
								
								
									
										1
									
								
								docs/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | |||||||
|  | _build | ||||||
							
								
								
									
										177
									
								
								docs/Makefile
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,177 @@ | |||||||
|  | # Makefile for Sphinx documentation | ||||||
|  | # | ||||||
|  |  | ||||||
|  | # You can set these variables from the command line. | ||||||
|  | SPHINXOPTS    = | ||||||
|  | SPHINXBUILD   = sphinx-build | ||||||
|  | PAPER         = | ||||||
|  | BUILDDIR      = _build | ||||||
|  |  | ||||||
|  | # User-friendly check for sphinx-build | ||||||
|  | ifeq ($(shell which $(SPHINXBUILD) >/dev/null 2>&1; echo $$?), 1) | ||||||
|  | $(error The '$(SPHINXBUILD)' command was not found. Make sure you have Sphinx installed, then set the SPHINXBUILD environment variable to point to the full path of the '$(SPHINXBUILD)' executable. Alternatively you can add the directory with the executable to your PATH. If you don't have Sphinx installed, grab it from http://sphinx-doc.org/) | ||||||
|  | endif | ||||||
|  |  | ||||||
|  | # Internal variables. | ||||||
|  | PAPEROPT_a4     = -D latex_paper_size=a4 | ||||||
|  | PAPEROPT_letter = -D latex_paper_size=letter | ||||||
|  | ALLSPHINXOPTS   = -d $(BUILDDIR)/doctrees $(PAPEROPT_$(PAPER)) $(SPHINXOPTS) . | ||||||
|  | # the i18n builder cannot share the environment and doctrees with the others | ||||||
|  | I18NSPHINXOPTS  = $(PAPEROPT_$(PAPER)) $(SPHINXOPTS) . | ||||||
|  |  | ||||||
|  | .PHONY: help clean html dirhtml singlehtml pickle json htmlhelp qthelp devhelp epub latex latexpdf text man changes linkcheck doctest gettext | ||||||
|  |  | ||||||
|  | help: | ||||||
|  | 	@echo "Please use \`make <target>' where <target> is one of" | ||||||
|  | 	@echo "  html       to make standalone HTML files" | ||||||
|  | 	@echo "  dirhtml    to make HTML files named index.html in directories" | ||||||
|  | 	@echo "  singlehtml to make a single large HTML file" | ||||||
|  | 	@echo "  pickle     to make pickle files" | ||||||
|  | 	@echo "  json       to make JSON files" | ||||||
|  | 	@echo "  htmlhelp   to make HTML files and a HTML help project" | ||||||
|  | 	@echo "  qthelp     to make HTML files and a qthelp project" | ||||||
|  | 	@echo "  devhelp    to make HTML files and a Devhelp project" | ||||||
|  | 	@echo "  epub       to make an epub" | ||||||
|  | 	@echo "  latex      to make LaTeX files, you can set PAPER=a4 or PAPER=letter" | ||||||
|  | 	@echo "  latexpdf   to make LaTeX files and run them through pdflatex" | ||||||
|  | 	@echo "  latexpdfja to make LaTeX files and run them through platex/dvipdfmx" | ||||||
|  | 	@echo "  text       to make text files" | ||||||
|  | 	@echo "  man        to make manual pages" | ||||||
|  | 	@echo "  texinfo    to make Texinfo files" | ||||||
|  | 	@echo "  info       to make Texinfo files and run them through makeinfo" | ||||||
|  | 	@echo "  gettext    to make PO message catalogs" | ||||||
|  | 	@echo "  changes    to make an overview of all changed/added/deprecated items" | ||||||
|  | 	@echo "  xml        to make Docutils-native XML files" | ||||||
|  | 	@echo "  pseudoxml  to make pseudoxml-XML files for display purposes" | ||||||
|  | 	@echo "  linkcheck  to check all external links for integrity" | ||||||
|  | 	@echo "  doctest    to run all doctests embedded in the documentation (if enabled)" | ||||||
|  |  | ||||||
|  | clean: | ||||||
|  | 	rm -rf $(BUILDDIR)/* | ||||||
|  |  | ||||||
|  | html: | ||||||
|  | 	$(SPHINXBUILD) -b html $(ALLSPHINXOPTS) $(BUILDDIR)/html | ||||||
|  | 	@echo | ||||||
|  | 	@echo "Build finished. The HTML pages are in $(BUILDDIR)/html." | ||||||
|  |  | ||||||
|  | dirhtml: | ||||||
|  | 	$(SPHINXBUILD) -b dirhtml $(ALLSPHINXOPTS) $(BUILDDIR)/dirhtml | ||||||
|  | 	@echo | ||||||
|  | 	@echo "Build finished. The HTML pages are in $(BUILDDIR)/dirhtml." | ||||||
|  |  | ||||||
|  | singlehtml: | ||||||
|  | 	$(SPHINXBUILD) -b singlehtml $(ALLSPHINXOPTS) $(BUILDDIR)/singlehtml | ||||||
|  | 	@echo | ||||||
|  | 	@echo "Build finished. The HTML page is in $(BUILDDIR)/singlehtml." | ||||||
|  |  | ||||||
|  | pickle: | ||||||
|  | 	$(SPHINXBUILD) -b pickle $(ALLSPHINXOPTS) $(BUILDDIR)/pickle | ||||||
|  | 	@echo | ||||||
|  | 	@echo "Build finished; now you can process the pickle files." | ||||||
|  |  | ||||||
|  | json: | ||||||
|  | 	$(SPHINXBUILD) -b json $(ALLSPHINXOPTS) $(BUILDDIR)/json | ||||||
|  | 	@echo | ||||||
|  | 	@echo "Build finished; now you can process the JSON files." | ||||||
|  |  | ||||||
|  | htmlhelp: | ||||||
|  | 	$(SPHINXBUILD) -b htmlhelp $(ALLSPHINXOPTS) $(BUILDDIR)/htmlhelp | ||||||
|  | 	@echo | ||||||
|  | 	@echo "Build finished; now you can run HTML Help Workshop with the" \ | ||||||
|  | 	      ".hhp project file in $(BUILDDIR)/htmlhelp." | ||||||
|  |  | ||||||
|  | qthelp: | ||||||
|  | 	$(SPHINXBUILD) -b qthelp $(ALLSPHINXOPTS) $(BUILDDIR)/qthelp | ||||||
|  | 	@echo | ||||||
|  | 	@echo "Build finished; now you can run "qcollectiongenerator" with the" \ | ||||||
|  | 	      ".qhcp project file in $(BUILDDIR)/qthelp, like this:" | ||||||
|  | 	@echo "# qcollectiongenerator $(BUILDDIR)/qthelp/Senpy.qhcp" | ||||||
|  | 	@echo "To view the help file:" | ||||||
|  | 	@echo "# assistant -collectionFile $(BUILDDIR)/qthelp/Senpy.qhc" | ||||||
|  |  | ||||||
|  | devhelp: | ||||||
|  | 	$(SPHINXBUILD) -b devhelp $(ALLSPHINXOPTS) $(BUILDDIR)/devhelp | ||||||
|  | 	@echo | ||||||
|  | 	@echo "Build finished." | ||||||
|  | 	@echo "To view the help file:" | ||||||
|  | 	@echo "# mkdir -p $$HOME/.local/share/devhelp/Senpy" | ||||||
|  | 	@echo "# ln -s $(BUILDDIR)/devhelp $$HOME/.local/share/devhelp/Senpy" | ||||||
|  | 	@echo "# devhelp" | ||||||
|  |  | ||||||
|  | epub: | ||||||
|  | 	$(SPHINXBUILD) -b epub $(ALLSPHINXOPTS) $(BUILDDIR)/epub | ||||||
|  | 	@echo | ||||||
|  | 	@echo "Build finished. The epub file is in $(BUILDDIR)/epub." | ||||||
|  |  | ||||||
|  | latex: | ||||||
|  | 	$(SPHINXBUILD) -b latex $(ALLSPHINXOPTS) $(BUILDDIR)/latex | ||||||
|  | 	@echo | ||||||
|  | 	@echo "Build finished; the LaTeX files are in $(BUILDDIR)/latex." | ||||||
|  | 	@echo "Run \`make' in that directory to run these through (pdf)latex" \ | ||||||
|  | 	      "(use \`make latexpdf' here to do that automatically)." | ||||||
|  |  | ||||||
|  | latexpdf: | ||||||
|  | 	$(SPHINXBUILD) -b latex $(ALLSPHINXOPTS) $(BUILDDIR)/latex | ||||||
|  | 	@echo "Running LaTeX files through pdflatex..." | ||||||
|  | 	$(MAKE) -C $(BUILDDIR)/latex all-pdf | ||||||
|  | 	@echo "pdflatex finished; the PDF files are in $(BUILDDIR)/latex." | ||||||
|  |  | ||||||
|  | latexpdfja: | ||||||
|  | 	$(SPHINXBUILD) -b latex $(ALLSPHINXOPTS) $(BUILDDIR)/latex | ||||||
|  | 	@echo "Running LaTeX files through platex and dvipdfmx..." | ||||||
|  | 	$(MAKE) -C $(BUILDDIR)/latex all-pdf-ja | ||||||
|  | 	@echo "pdflatex finished; the PDF files are in $(BUILDDIR)/latex." | ||||||
|  |  | ||||||
|  | text: | ||||||
|  | 	$(SPHINXBUILD) -b text $(ALLSPHINXOPTS) $(BUILDDIR)/text | ||||||
|  | 	@echo | ||||||
|  | 	@echo "Build finished. The text files are in $(BUILDDIR)/text." | ||||||
|  |  | ||||||
|  | man: | ||||||
|  | 	$(SPHINXBUILD) -b man $(ALLSPHINXOPTS) $(BUILDDIR)/man | ||||||
|  | 	@echo | ||||||
|  | 	@echo "Build finished. The manual pages are in $(BUILDDIR)/man." | ||||||
|  |  | ||||||
|  | texinfo: | ||||||
|  | 	$(SPHINXBUILD) -b texinfo $(ALLSPHINXOPTS) $(BUILDDIR)/texinfo | ||||||
|  | 	@echo | ||||||
|  | 	@echo "Build finished. The Texinfo files are in $(BUILDDIR)/texinfo." | ||||||
|  | 	@echo "Run \`make' in that directory to run these through makeinfo" \ | ||||||
|  | 	      "(use \`make info' here to do that automatically)." | ||||||
|  |  | ||||||
|  | info: | ||||||
|  | 	$(SPHINXBUILD) -b texinfo $(ALLSPHINXOPTS) $(BUILDDIR)/texinfo | ||||||
|  | 	@echo "Running Texinfo files through makeinfo..." | ||||||
|  | 	make -C $(BUILDDIR)/texinfo info | ||||||
|  | 	@echo "makeinfo finished; the Info files are in $(BUILDDIR)/texinfo." | ||||||
|  |  | ||||||
|  | gettext: | ||||||
|  | 	$(SPHINXBUILD) -b gettext $(I18NSPHINXOPTS) $(BUILDDIR)/locale | ||||||
|  | 	@echo | ||||||
|  | 	@echo "Build finished. The message catalogs are in $(BUILDDIR)/locale." | ||||||
|  |  | ||||||
|  | changes: | ||||||
|  | 	$(SPHINXBUILD) -b changes $(ALLSPHINXOPTS) $(BUILDDIR)/changes | ||||||
|  | 	@echo | ||||||
|  | 	@echo "The overview file is in $(BUILDDIR)/changes." | ||||||
|  |  | ||||||
|  | linkcheck: | ||||||
|  | 	$(SPHINXBUILD) -b linkcheck $(ALLSPHINXOPTS) $(BUILDDIR)/linkcheck | ||||||
|  | 	@echo | ||||||
|  | 	@echo "Link check complete; look for any errors in the above output " \ | ||||||
|  | 	      "or in $(BUILDDIR)/linkcheck/output.txt." | ||||||
|  |  | ||||||
|  | doctest: | ||||||
|  | 	$(SPHINXBUILD) -b doctest $(ALLSPHINXOPTS) $(BUILDDIR)/doctest | ||||||
|  | 	@echo "Testing of doctests in the sources finished, look at the " \ | ||||||
|  | 	      "results in $(BUILDDIR)/doctest/output.txt." | ||||||
|  |  | ||||||
|  | xml: | ||||||
|  | 	$(SPHINXBUILD) -b xml $(ALLSPHINXOPTS) $(BUILDDIR)/xml | ||||||
|  | 	@echo | ||||||
|  | 	@echo "Build finished. The XML files are in $(BUILDDIR)/xml." | ||||||
|  |  | ||||||
|  | pseudoxml: | ||||||
|  | 	$(SPHINXBUILD) -b pseudoxml $(ALLSPHINXOPTS) $(BUILDDIR)/pseudoxml | ||||||
|  | 	@echo | ||||||
|  | 	@echo "Build finished. The pseudo-XML files are in $(BUILDDIR)/pseudoxml." | ||||||
							
								
								
									
										317
									
								
								docs/SenpyClientUse.ipynb
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,317 @@ | |||||||
|  | { | ||||||
|  |  "cells": [ | ||||||
|  |   { | ||||||
|  |    "cell_type": "markdown", | ||||||
|  |    "metadata": { | ||||||
|  |     "ExecuteTime": { | ||||||
|  |      "end_time": "2017-04-10T17:05:31.465571Z", | ||||||
|  |      "start_time": "2017-04-10T19:05:31.458282+02:00" | ||||||
|  |     }, | ||||||
|  |     "deletable": true, | ||||||
|  |     "editable": true | ||||||
|  |    }, | ||||||
|  |    "source": [ | ||||||
|  |     "# Client" | ||||||
|  |    ] | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "cell_type": "markdown", | ||||||
|  |    "metadata": { | ||||||
|  |     "collapsed": true, | ||||||
|  |     "deletable": true, | ||||||
|  |     "editable": true | ||||||
|  |    }, | ||||||
|  |    "source": [ | ||||||
|  |     "The built-in senpy client allows you to query any Senpy endpoint. We will illustrate how to use it with the public demo endpoint, and then show you how to spin up your own endpoint using docker." | ||||||
|  |    ] | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "cell_type": "markdown", | ||||||
|  |    "metadata": { | ||||||
|  |     "deletable": true, | ||||||
|  |     "editable": true | ||||||
|  |    }, | ||||||
|  |    "source": [ | ||||||
|  |     "Demo Endpoint\n", | ||||||
|  |     "-------------" | ||||||
|  |    ] | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "cell_type": "markdown", | ||||||
|  |    "metadata": { | ||||||
|  |     "deletable": true, | ||||||
|  |     "editable": true | ||||||
|  |    }, | ||||||
|  |    "source": [ | ||||||
|  |     "To start using senpy, simply create a new Client and point it to your endpoint. In this case, the latest version of Senpy at GSI." | ||||||
|  |    ] | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "cell_type": "code", | ||||||
|  |    "execution_count": null, | ||||||
|  |    "metadata": { | ||||||
|  |     "ExecuteTime": { | ||||||
|  |      "end_time": "2017-04-10T17:29:12.827640Z", | ||||||
|  |      "start_time": "2017-04-10T19:29:12.818617+02:00" | ||||||
|  |     }, | ||||||
|  |     "collapsed": false, | ||||||
|  |     "deletable": true, | ||||||
|  |     "editable": true | ||||||
|  |    }, | ||||||
|  |    "outputs": [], | ||||||
|  |    "source": [ | ||||||
|  |     "from senpy.client import Client\n", | ||||||
|  |     "\n", | ||||||
|  |     "c = Client('http://latest.senpy.cluster.gsi.dit.upm.es/api')\n" | ||||||
|  |    ] | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "cell_type": "markdown", | ||||||
|  |    "metadata": { | ||||||
|  |     "deletable": true, | ||||||
|  |     "editable": true | ||||||
|  |    }, | ||||||
|  |    "source": [ | ||||||
|  |     "Now, let's use that client analyse some queries:" | ||||||
|  |    ] | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "cell_type": "code", | ||||||
|  |    "execution_count": null, | ||||||
|  |    "metadata": { | ||||||
|  |     "ExecuteTime": { | ||||||
|  |      "end_time": "2017-04-10T17:29:14.011657Z", | ||||||
|  |      "start_time": "2017-04-10T19:29:13.701808+02:00" | ||||||
|  |     }, | ||||||
|  |     "collapsed": false, | ||||||
|  |     "deletable": true, | ||||||
|  |     "editable": true | ||||||
|  |    }, | ||||||
|  |    "outputs": [], | ||||||
|  |    "source": [ | ||||||
|  |     "r = c.analyse('I like sugar!!', algorithm='sentiment140')\n", | ||||||
|  |     "r" | ||||||
|  |    ] | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "cell_type": "markdown", | ||||||
|  |    "metadata": { | ||||||
|  |     "ExecuteTime": { | ||||||
|  |      "end_time": "2017-04-10T17:08:19.616754Z", | ||||||
|  |      "start_time": "2017-04-10T19:08:19.610767+02:00" | ||||||
|  |     }, | ||||||
|  |     "deletable": true, | ||||||
|  |     "editable": true | ||||||
|  |    }, | ||||||
|  |    "source": [ | ||||||
|  |     "As you can see, that gave us the full JSON result. A more concise way to print it would be:" | ||||||
|  |    ] | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "cell_type": "code", | ||||||
|  |    "execution_count": null, | ||||||
|  |    "metadata": { | ||||||
|  |     "ExecuteTime": { | ||||||
|  |      "end_time": "2017-04-10T17:29:14.854213Z", | ||||||
|  |      "start_time": "2017-04-10T19:29:14.842068+02:00" | ||||||
|  |     }, | ||||||
|  |     "collapsed": false, | ||||||
|  |     "deletable": true, | ||||||
|  |     "editable": true | ||||||
|  |    }, | ||||||
|  |    "outputs": [], | ||||||
|  |    "source": [ | ||||||
|  |     "for entry in r.entries:\n", | ||||||
|  |     "      print('{} -> {}'.format(entry['text'], entry['sentiments'][0]['marl:hasPolarity']))" | ||||||
|  |    ] | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "cell_type": "markdown", | ||||||
|  |    "metadata": { | ||||||
|  |     "deletable": true, | ||||||
|  |     "editable": true | ||||||
|  |    }, | ||||||
|  |    "source": [ | ||||||
|  |     "We can also obtain a list of available plugins with the client:" | ||||||
|  |    ] | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "cell_type": "code", | ||||||
|  |    "execution_count": null, | ||||||
|  |    "metadata": { | ||||||
|  |     "ExecuteTime": { | ||||||
|  |      "end_time": "2017-04-10T17:29:16.245198Z", | ||||||
|  |      "start_time": "2017-04-10T19:29:16.056545+02:00" | ||||||
|  |     }, | ||||||
|  |     "collapsed": false, | ||||||
|  |     "deletable": true, | ||||||
|  |     "editable": true | ||||||
|  |    }, | ||||||
|  |    "outputs": [], | ||||||
|  |    "source": [ | ||||||
|  |     "c.plugins()" | ||||||
|  |    ] | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "cell_type": "markdown", | ||||||
|  |    "metadata": { | ||||||
|  |     "deletable": true, | ||||||
|  |     "editable": true | ||||||
|  |    }, | ||||||
|  |    "source": [ | ||||||
|  |     "Or, more concisely:" | ||||||
|  |    ] | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "cell_type": "code", | ||||||
|  |    "execution_count": null, | ||||||
|  |    "metadata": { | ||||||
|  |     "ExecuteTime": { | ||||||
|  |      "end_time": "2017-04-10T17:29:17.663275Z", | ||||||
|  |      "start_time": "2017-04-10T19:29:17.484623+02:00" | ||||||
|  |     }, | ||||||
|  |     "collapsed": false, | ||||||
|  |     "deletable": true, | ||||||
|  |     "editable": true | ||||||
|  |    }, | ||||||
|  |    "outputs": [], | ||||||
|  |    "source": [ | ||||||
|  |     "c.plugins().keys()" | ||||||
|  |    ] | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "cell_type": "markdown", | ||||||
|  |    "metadata": { | ||||||
|  |     "deletable": true, | ||||||
|  |     "editable": true | ||||||
|  |    }, | ||||||
|  |    "source": [ | ||||||
|  |     "Local Endpoint\n", | ||||||
|  |     "--------------" | ||||||
|  |    ] | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "cell_type": "markdown", | ||||||
|  |    "metadata": { | ||||||
|  |     "deletable": true, | ||||||
|  |     "editable": true | ||||||
|  |    }, | ||||||
|  |    "source": [ | ||||||
|  |     "To run your own instance of senpy, just create a docker container with the latest Senpy image. Using `--default-plugins` you will get some extra plugins to start playing with the API." | ||||||
|  |    ] | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "cell_type": "code", | ||||||
|  |    "execution_count": null, | ||||||
|  |    "metadata": { | ||||||
|  |     "ExecuteTime": { | ||||||
|  |      "end_time": "2017-04-10T17:29:20.637539Z", | ||||||
|  |      "start_time": "2017-04-10T19:29:19.938322+02:00" | ||||||
|  |     }, | ||||||
|  |     "collapsed": false, | ||||||
|  |     "deletable": true, | ||||||
|  |     "editable": true | ||||||
|  |    }, | ||||||
|  |    "outputs": [], | ||||||
|  |    "source": [ | ||||||
|  |     "!docker run -ti --name 'SenpyEndpoint' -d -p 6000:5000 gsiupm/senpy:0.8.6 --host 0.0.0.0 --default-plugins" | ||||||
|  |    ] | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "cell_type": "markdown", | ||||||
|  |    "metadata": { | ||||||
|  |     "deletable": true, | ||||||
|  |     "editable": true | ||||||
|  |    }, | ||||||
|  |    "source": [ | ||||||
|  |     "To use this endpoint:" | ||||||
|  |    ] | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "cell_type": "code", | ||||||
|  |    "execution_count": null, | ||||||
|  |    "metadata": { | ||||||
|  |     "ExecuteTime": { | ||||||
|  |      "end_time": "2017-04-10T17:29:21.263976Z", | ||||||
|  |      "start_time": "2017-04-10T19:29:21.260595+02:00" | ||||||
|  |     }, | ||||||
|  |     "collapsed": false, | ||||||
|  |     "deletable": true, | ||||||
|  |     "editable": true | ||||||
|  |    }, | ||||||
|  |    "outputs": [], | ||||||
|  |    "source": [ | ||||||
|  |     "c_local = Client('http://127.0.0.1:6000/api')" | ||||||
|  |    ] | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "cell_type": "markdown", | ||||||
|  |    "metadata": { | ||||||
|  |     "deletable": true, | ||||||
|  |     "editable": true | ||||||
|  |    }, | ||||||
|  |    "source": [ | ||||||
|  |     "That's all! After you are done with your analysis, stop the docker container:" | ||||||
|  |    ] | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |    "cell_type": "code", | ||||||
|  |    "execution_count": null, | ||||||
|  |    "metadata": { | ||||||
|  |     "ExecuteTime": { | ||||||
|  |      "end_time": "2017-04-10T17:29:33.226686Z", | ||||||
|  |      "start_time": "2017-04-10T19:29:22.392121+02:00" | ||||||
|  |     }, | ||||||
|  |     "collapsed": false, | ||||||
|  |     "deletable": true, | ||||||
|  |     "editable": true | ||||||
|  |    }, | ||||||
|  |    "outputs": [], | ||||||
|  |    "source": [ | ||||||
|  |     "!docker stop SenpyEndpoint\n", | ||||||
|  |     "!docker rm SenpyEndpoint" | ||||||
|  |    ] | ||||||
|  |   } | ||||||
|  |  ], | ||||||
|  |  "metadata": { | ||||||
|  |   "anaconda-cloud": {}, | ||||||
|  |   "kernelspec": { | ||||||
|  |    "display_name": "Python 3", | ||||||
|  |    "language": "python", | ||||||
|  |    "name": "python3" | ||||||
|  |   }, | ||||||
|  |   "language_info": { | ||||||
|  |    "codemirror_mode": { | ||||||
|  |     "name": "ipython", | ||||||
|  |     "version": 3 | ||||||
|  |    }, | ||||||
|  |    "file_extension": ".py", | ||||||
|  |    "mimetype": "text/x-python", | ||||||
|  |    "name": "python", | ||||||
|  |    "nbconvert_exporter": "python", | ||||||
|  |    "pygments_lexer": "ipython3", | ||||||
|  |    "version": "3.6.0" | ||||||
|  |   }, | ||||||
|  |   "toc": { | ||||||
|  |    "colors": { | ||||||
|  |     "hover_highlight": "#DAA520", | ||||||
|  |     "running_highlight": "#FF0000", | ||||||
|  |     "selected_highlight": "#FFD700" | ||||||
|  |    }, | ||||||
|  |    "moveMenuLeft": true, | ||||||
|  |    "nav_menu": { | ||||||
|  |     "height": "68px", | ||||||
|  |     "width": "252px" | ||||||
|  |    }, | ||||||
|  |    "navigate_menu": true, | ||||||
|  |    "number_sections": true, | ||||||
|  |    "sideBar": true, | ||||||
|  |    "threshold": 4, | ||||||
|  |    "toc_cell": false, | ||||||
|  |    "toc_section_display": "block", | ||||||
|  |    "toc_window_display": false | ||||||
|  |   } | ||||||
|  |  }, | ||||||
|  |  "nbformat": 4, | ||||||
|  |  "nbformat_minor": 1 | ||||||
|  | } | ||||||
							
								
								
									
										106
									
								
								docs/SenpyClientUse.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,106 @@ | |||||||
|  |  | ||||||
|  | Client | ||||||
|  | ====== | ||||||
|  |  | ||||||
|  | Demo Endpoint | ||||||
|  | ------------- | ||||||
|  |  | ||||||
|  | Import Client and send a request | ||||||
|  |  | ||||||
|  | .. code:: python | ||||||
|  |  | ||||||
|  |     from senpy.client import Client | ||||||
|  |      | ||||||
|  |     c = Client('http://latest.senpy.cluster.gsi.dit.upm.es/api') | ||||||
|  |     r = c.analyse('I like Pizza', algorithm='sentiment140') | ||||||
|  |  | ||||||
|  | Print response | ||||||
|  |  | ||||||
|  | .. code:: python | ||||||
|  |  | ||||||
|  |     for entry in r.entries: | ||||||
|  |           print('{} -> {}'.format(entry['text'], entry['sentiments'][0]['marl:hasPolarity'])) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .. parsed-literal:: | ||||||
|  |  | ||||||
|  |     I like Pizza -> marl:Positive | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Obtain a list of available plugins | ||||||
|  |  | ||||||
|  | .. code:: python | ||||||
|  |  | ||||||
|  |     for plugin in c.request('/plugins')['plugins']: | ||||||
|  |         print(plugin['name']) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .. parsed-literal:: | ||||||
|  |  | ||||||
|  |     emoRand | ||||||
|  |     rand | ||||||
|  |     sentiment140 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Local Endpoint | ||||||
|  | -------------- | ||||||
|  |  | ||||||
|  | Run a docker container with Senpy image and default plugins | ||||||
|  |  | ||||||
|  | .. code:: | ||||||
|  |  | ||||||
|  |     docker run -ti --name 'SenpyEndpoint' -d -p 5000:5000 gsiupm/senpy:0.8.6 --host 0.0.0.0 --default-plugins | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .. parsed-literal:: | ||||||
|  |  | ||||||
|  |     a0157cd98057072388bfebeed78a830da7cf0a796f4f1a3fd9188f9f2e5fe562 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Import client and send a request to localhost | ||||||
|  |  | ||||||
|  | .. code:: python | ||||||
|  |  | ||||||
|  |     c_local = Client('http://127.0.0.1:5000/api') | ||||||
|  |     r = c_local.analyse('Hello world', algorithm='sentiment140') | ||||||
|  |  | ||||||
|  | Print response | ||||||
|  |  | ||||||
|  | .. code:: python | ||||||
|  |  | ||||||
|  |     for entry in r.entries: | ||||||
|  |           print('{} -> {}'.format(entry['text'], entry['sentiments'][0]['marl:hasPolarity'])) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .. parsed-literal:: | ||||||
|  |  | ||||||
|  |     Hello world -> marl:Neutral | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Obtain a list of available plugins deployed locally | ||||||
|  |  | ||||||
|  | .. code:: python | ||||||
|  |  | ||||||
|  |     c_local.plugins().keys() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .. parsed-literal:: | ||||||
|  |  | ||||||
|  |     rand | ||||||
|  |     sentiment140 | ||||||
|  |     emoRand | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Stop the docker container | ||||||
|  |  | ||||||
|  | .. code:: python | ||||||
|  |  | ||||||
|  |     !docker stop SenpyEndpoint | ||||||
|  |     !docker rm SenpyEndpoint | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .. parsed-literal:: | ||||||
|  |  | ||||||
|  |     SenpyEndpoint | ||||||
|  |     SenpyEndpoint | ||||||
|  |  | ||||||
							
								
								
									
										
											BIN
										
									
								
								docs/_static/header.png
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 208 KiB | 
							
								
								
									
										1
									
								
								docs/_static/schemas
									
									
									
									
										vendored
									
									
										Symbolic link
									
								
							
							
						
						| @@ -0,0 +1 @@ | |||||||
|  | ../../senpy/schemas/ | ||||||
							
								
								
									
										11
									
								
								docs/about.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,11 @@ | |||||||
|  | About | ||||||
|  | -------- | ||||||
|  |  | ||||||
|  | If you use Senpy in your research, please cite `Senpy: A Pragmatic Linked Sentiment Analysis Framework <http://gsi.dit.upm.es/index.php/es/investigacion/publicaciones?view=publication&task=show&id=417>`__ (`BibTex <http://gsi.dit.upm.es/index.php/es/investigacion/publicaciones?controller=publications&task=export&format=bibtex&id=417>`__): | ||||||
|  |  | ||||||
|  | .. code-block:: text | ||||||
|  |  | ||||||
|  | 	Sánchez-Rada, J. F., Iglesias, C. A., Corcuera, I., & Araque, Ó. (2016, October). | ||||||
|  | 	Senpy: A Pragmatic Linked Sentiment Analysis Framework. | ||||||
|  | 	In Data Science and Advanced Analytics (DSAA), | ||||||
|  | 	2016 IEEE International Conference on (pp. 735-742). IEEE. | ||||||
							
								
								
									
										217
									
								
								docs/api.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,217 @@ | |||||||
|  | NIF API | ||||||
|  | ------- | ||||||
|  | .. http:get:: /api | ||||||
|  |  | ||||||
|  |    Basic endpoint for sentiment/emotion analysis. | ||||||
|  |  | ||||||
|  |    **Example request**: | ||||||
|  |  | ||||||
|  |    .. sourcecode:: http | ||||||
|  |  | ||||||
|  |       GET /api?input=I%20love%20GSI HTTP/1.1 | ||||||
|  |       Host: localhost | ||||||
|  |       Accept: application/json, text/javascript | ||||||
|  |  | ||||||
|  |  | ||||||
|  |    **Example response**: | ||||||
|  |  | ||||||
|  |    .. sourcecode:: http | ||||||
|  |  | ||||||
|  |       HTTP/1.1 200 OK | ||||||
|  |       Vary: Accept | ||||||
|  |       Content-Type: text/javascript | ||||||
|  |  | ||||||
|  |       { | ||||||
|  |         "@context":"http://127.0.0.1/api/contexts/Results.jsonld", | ||||||
|  |         "@id":"_:Results_11241245.22", | ||||||
|  |         "@type":"results" | ||||||
|  |         "analysis": [ | ||||||
|  |           "plugins/sentiment-140_0.1"  | ||||||
|  |         ], | ||||||
|  |         "entries": [ | ||||||
|  |           {   | ||||||
|  |             "@id": "_:Entry_11241245.22" | ||||||
|  |             "@type":"entry", | ||||||
|  |             "emotions": [], | ||||||
|  |             "entities": [], | ||||||
|  |             "sentiments": [ | ||||||
|  |               {   | ||||||
|  |                 "@id": "Sentiment0",   | ||||||
|  |                 "@type": "sentiment",  | ||||||
|  |                 "marl:hasPolarity": "marl:Negative", | ||||||
|  |                 "marl:polarityValue": 0, | ||||||
|  |                 "prefix": "" | ||||||
|  |               } | ||||||
|  |             ], | ||||||
|  |             "suggestions": [], | ||||||
|  |             "text": "This text makes me sad.\nwhilst this text makes me happy and surprised at the same time.\nI cannot believe it!", | ||||||
|  |             "topics": [] | ||||||
|  |           } | ||||||
|  |         ] | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |    :query i input: No default. Depends on informat and intype | ||||||
|  |    :query f informat: one of `turtle` (default), `text`, `json-ld` | ||||||
|  |    :query t intype: one of `direct` (default), `url` | ||||||
|  |    :query o outformat: one of `turtle` (default), `text`, `json-ld` | ||||||
|  |    :query p prefix: prefix for the URIs | ||||||
|  |    :query algo algorithm: algorithm/plugin to use for the analysis. For a list of options, see :http:get:`/api/plugins`. If not provided, the default plugin will be used (:http:get:`/api/plugins/default`). | ||||||
|  |    :query algo emotionModel: desired emotion model in the results. If the requested algorithm does not use that emotion model, there are conversion plugins specifically for this. If none of the plugins match, an error will be returned, which includes the results *as is*. | ||||||
|  |  | ||||||
|  |    :reqheader Accept: the response content type depends on | ||||||
|  |                       :mailheader:`Accept` header | ||||||
|  |    :resheader Content-Type: this depends on :mailheader:`Accept` | ||||||
|  |                             header of request | ||||||
|  |    :statuscode 200: no error | ||||||
|  |    :statuscode 404: service not found | ||||||
|  |    :statuscode 400: error while processing the request | ||||||
|  |  | ||||||
|  | .. http:post:: /api | ||||||
|  |  | ||||||
|  |    The same as :http:get:`/api`. | ||||||
|  |  | ||||||
|  | .. http:get:: /api/plugins | ||||||
|  |  | ||||||
|  |    Returns a list of installed plugins.  | ||||||
|  |    **Example request**: | ||||||
|  |  | ||||||
|  |    .. sourcecode:: http | ||||||
|  |  | ||||||
|  |       GET /api/plugins HTTP/1.1 | ||||||
|  |       Host: localhost | ||||||
|  |       Accept: application/json, text/javascript | ||||||
|  |  | ||||||
|  |  | ||||||
|  |    **Example response**: | ||||||
|  |  | ||||||
|  |    .. sourcecode:: http | ||||||
|  |  | ||||||
|  |       { | ||||||
|  |         "@id": "plugins/sentiment-140_0.1",  | ||||||
|  |         "@type": "sentimentPlugin",  | ||||||
|  |         "author": "@balkian",  | ||||||
|  |         "description": "Sentiment classifier using rule-based classification for English and Spanish. This plugin uses sentiment140 data to perform classification. For more information: http://help.sentiment140.com/for-students/",  | ||||||
|  |         "extra_params": { | ||||||
|  |           "language": { | ||||||
|  |             "@id": "lang_sentiment140",  | ||||||
|  |             "aliases": [ | ||||||
|  |               "language",  | ||||||
|  |               "l" | ||||||
|  |             ],  | ||||||
|  |             "options": [ | ||||||
|  |               "es",  | ||||||
|  |               "en",  | ||||||
|  |               "auto" | ||||||
|  |             ],  | ||||||
|  |             "required": false | ||||||
|  |           } | ||||||
|  |         },  | ||||||
|  |         "is_activated": true,  | ||||||
|  |         "maxPolarityValue": 1.0,  | ||||||
|  |         "minPolarityValue": 0.0,  | ||||||
|  |         "module": "sentiment-140",  | ||||||
|  |         "name": "sentiment-140",  | ||||||
|  |         "requirements": {},  | ||||||
|  |         "version": "0.1" | ||||||
|  |       },  | ||||||
|  |       { | ||||||
|  |         "@id": "plugins/ExamplePlugin_0.1",  | ||||||
|  |         "@type": "sentimentPlugin",  | ||||||
|  |         "author": "@balkian",  | ||||||
|  |         "custom_attribute": "42",  | ||||||
|  |         "description": "I am just an example",  | ||||||
|  |         "extra_params": { | ||||||
|  |           "parameter": { | ||||||
|  |             "@id": "parameter",  | ||||||
|  |             "aliases": [ | ||||||
|  |               "parameter",  | ||||||
|  |               "param" | ||||||
|  |             ],  | ||||||
|  |             "default": 42,  | ||||||
|  |             "required": true | ||||||
|  |           } | ||||||
|  |         },  | ||||||
|  |         "is_activated": true,  | ||||||
|  |         "maxPolarityValue": 1.0,  | ||||||
|  |         "minPolarityValue": 0.0,  | ||||||
|  |         "module": "example",  | ||||||
|  |         "name": "ExamplePlugin",  | ||||||
|  |         "requirements": "noop",  | ||||||
|  |         "version": "0.1" | ||||||
|  |       } | ||||||
|  |  | ||||||
|  | .. http:get:: /api/plugins/<pluginname> | ||||||
|  |  | ||||||
|  |    Returns the information of a specific plugin. | ||||||
|  |    **Example request**: | ||||||
|  |  | ||||||
|  |    .. sourcecode:: http | ||||||
|  |  | ||||||
|  |       GET /api/plugins/rand/ HTTP/1.1 | ||||||
|  |       Host: localhost | ||||||
|  |       Accept: application/json, text/javascript | ||||||
|  |  | ||||||
|  |  | ||||||
|  |    **Example response**: | ||||||
|  |  | ||||||
|  |    .. sourcecode:: http | ||||||
|  |  | ||||||
|  |       { | ||||||
|  |         "@context": "http://127.0.0.1/api/contexts/ExamplePlugin.jsonld",  | ||||||
|  |         "@id": "plugins/ExamplePlugin_0.1",  | ||||||
|  |         "@type": "sentimentPlugin",  | ||||||
|  |         "author": "@balkian",  | ||||||
|  |         "custom_attribute": "42",  | ||||||
|  |         "description": "I am just an example",  | ||||||
|  |         "extra_params": { | ||||||
|  |           "parameter": { | ||||||
|  |             "@id": "parameter",  | ||||||
|  |             "aliases": [ | ||||||
|  |               "parameter",  | ||||||
|  |               "param" | ||||||
|  |             ],  | ||||||
|  |             "default": 42,  | ||||||
|  |             "required": true | ||||||
|  |           } | ||||||
|  |         },  | ||||||
|  |         "is_activated": true,  | ||||||
|  |         "maxPolarityValue": 1.0,  | ||||||
|  |         "minPolarityValue": 0.0,  | ||||||
|  |         "module": "example",  | ||||||
|  |         "name": "ExamplePlugin",  | ||||||
|  |         "requirements": "noop",  | ||||||
|  |         "version": "0.1" | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .. http:get:: /api/plugins/default | ||||||
|  |  | ||||||
|  |    Return the information about the default plugin. | ||||||
|  |  | ||||||
							
								
								
									
										7
									
								
								docs/apischema.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,7 @@ | |||||||
|  | API and Examples | ||||||
|  | ################ | ||||||
|  | .. toctree:: | ||||||
|  |  | ||||||
|  |    vocabularies.rst | ||||||
|  |    api.rst | ||||||
|  |    examples.rst | ||||||
							
								
								
									
										4
									
								
								docs/bad-examples/plugins/noplugins.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,4 @@ | |||||||
|  | { | ||||||
|  |   "@type": "plugins", | ||||||
|  |   "plugins": {} | ||||||
|  | } | ||||||
							
								
								
									
										77
									
								
								docs/bad-examples/results/example-analysis-as-id-FAIL.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,77 @@ | |||||||
|  | { | ||||||
|  |   "@context": "http://mixedemotions-project.eu/ns/context.jsonld", | ||||||
|  |   "@id": "me:Result1", | ||||||
|  |   "@type": "results", | ||||||
|  |   "analysis": [ | ||||||
|  |     "me:SAnalysis1", | ||||||
|  |     "me:SgAnalysis1", | ||||||
|  |     "me:EmotionAnalysis1", | ||||||
|  |     "me:NER1", | ||||||
|  |     { | ||||||
|  |       "description": "missing @id and @type" | ||||||
|  |     } | ||||||
|  |   ], | ||||||
|  |   "entries": [ | ||||||
|  |     { | ||||||
|  |       "@id": "http://micro.blog/status1", | ||||||
|  |       "@type": [ | ||||||
|  |         "nif:RFC5147String", | ||||||
|  |         "nif:Context" | ||||||
|  |       ], | ||||||
|  |       "nif:isString": "Dear Microsoft, put your Windows Phone on your newest #open technology program. You'll be awesome. #opensource", | ||||||
|  |       "entities": [ | ||||||
|  |         { | ||||||
|  |           "@id": "http://micro.blog/status1#char=5,13", | ||||||
|  |           "nif:beginIndex": 5, | ||||||
|  |           "nif:endIndex": 13, | ||||||
|  |           "nif:anchorOf": "Microsoft", | ||||||
|  |           "me:references": "http://dbpedia.org/page/Microsoft", | ||||||
|  |           "prov:wasGeneratedBy": "me:NER1" | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |           "@id": "http://micro.blog/status1#char=25,37", | ||||||
|  |           "nif:beginIndex": 25, | ||||||
|  |           "nif:endIndex": 37, | ||||||
|  |           "nif:anchorOf": "Windows Phone", | ||||||
|  |           "me:references": "http://dbpedia.org/page/Windows_Phone", | ||||||
|  |           "prov:wasGeneratedBy": "me:NER1" | ||||||
|  |         } | ||||||
|  |       ], | ||||||
|  |       "suggestions": [ | ||||||
|  |         { | ||||||
|  |           "@id": "http://micro.blog/status1#char=16,77", | ||||||
|  |           "nif:beginIndex": 16, | ||||||
|  |           "nif:endIndex": 77, | ||||||
|  |           "nif:anchorOf": "put your Windows Phone on your newest #open technology program", | ||||||
|  |           "prov:wasGeneratedBy": "me:SgAnalysis1" | ||||||
|  |         } | ||||||
|  |       ], | ||||||
|  |       "sentiments": [ | ||||||
|  |         { | ||||||
|  |           "@id": "http://micro.blog/status1#char=80,97", | ||||||
|  |           "nif:beginIndex": 80, | ||||||
|  |           "nif:endIndex": 97, | ||||||
|  |           "nif:anchorOf": "You'll be awesome.", | ||||||
|  |           "marl:hasPolarity": "marl:Positive", | ||||||
|  |           "marl:polarityValue": 0.9, | ||||||
|  |           "prov:wasGeneratedBy": "me:SAnalysis1" | ||||||
|  |         } | ||||||
|  |       ], | ||||||
|  |       "emotions": [ | ||||||
|  |         { | ||||||
|  |           "@id": "http://micro.blog/status1#char=0,109", | ||||||
|  |           "nif:anchorOf": "Dear Microsoft, put your Windows Phone on your newest #open technology program. You'll be awesome. #opensource", | ||||||
|  |           "prov:wasGeneratedBy": "me:EAnalysis1", | ||||||
|  |           "onyx:hasEmotion": [ | ||||||
|  |             { | ||||||
|  |               "onyx:hasEmotionCategory": "wna:liking" | ||||||
|  |             }, | ||||||
|  |             { | ||||||
|  |               "onyx:hasEmotionCategory": "wna:excitement" | ||||||
|  |             } | ||||||
|  |           ] | ||||||
|  |         } | ||||||
|  |       ] | ||||||
|  |     } | ||||||
|  |   ] | ||||||
|  | } | ||||||
							
								
								
									
										14
									
								
								docs/bad-examples/results/example-basic-FAIL.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,14 @@ | |||||||
|  | { | ||||||
|  |   "@context": "http://mixedemotions-project.eu/ns/context.jsonld", | ||||||
|  |   "@id": "http://example.com#NIFExample", | ||||||
|  |   "@type": "results", | ||||||
|  |   "analysis": [ | ||||||
|  |   ], | ||||||
|  |   "entries": [ | ||||||
|  |     { | ||||||
|  |       "nif:beginIndex": 0, | ||||||
|  |       "nif:endIndex": 40, | ||||||
|  |       "text": "An entry should have a nif:isString key" | ||||||
|  |     } | ||||||
|  |   ] | ||||||
|  | } | ||||||
							
								
								
									
										9
									
								
								docs/commandline.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,9 @@ | |||||||
|  | Command line | ||||||
|  | ============ | ||||||
|  |  | ||||||
|  | This video shows how to analyse text directly on the command line using the senpy tool. | ||||||
|  |  | ||||||
|  | .. image:: https://asciinema.org/a/9uwef1ghkjk062cw2t4mhzpyk.png | ||||||
|  |    :width: 100% | ||||||
|  |    :target: https://asciinema.org/a/9uwef1ghkjk062cw2t4mhzpyk | ||||||
|  |    :alt: CLI demo | ||||||
							
								
								
									
										288
									
								
								docs/conf.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,288 @@ | |||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  | # flake8: noqa | ||||||
|  | # | ||||||
|  | # Senpy documentation build configuration file, created by | ||||||
|  | # sphinx-quickstart on Tue Feb 24 08:57:32 2015. | ||||||
|  | # | ||||||
|  | # This file is execfile()d with the current directory set to its | ||||||
|  | # containing dir. | ||||||
|  | # | ||||||
|  | # Note that not all possible configuration values are present in this | ||||||
|  | # autogenerated file. | ||||||
|  | # | ||||||
|  | # All configuration values have a default; values that are commented out | ||||||
|  | # serve to show the default. | ||||||
|  |  | ||||||
|  | import sys | ||||||
|  | import os | ||||||
|  |  | ||||||
|  | # If extensions (or modules to document with autodoc) are in another directory, | ||||||
|  | # add these directories to sys.path here. If the directory is relative to the | ||||||
|  | # documentation root, use os.path.abspath to make it absolute, like shown here. | ||||||
|  | #sys.path.insert(0, os.path.abspath('.')) | ||||||
|  |  | ||||||
|  | # -- General configuration ------------------------------------------------ | ||||||
|  |  | ||||||
|  | on_rtd = os.environ.get('READTHEDOCS', None) == 'True' | ||||||
|  |  | ||||||
|  | # If your documentation needs a minimal Sphinx version, state it here. | ||||||
|  | #needs_sphinx = '1.0' | ||||||
|  |  | ||||||
|  | # Add any Sphinx extension module names here, as strings. They can be | ||||||
|  | # extensions coming with Sphinx (named 'sphinx.ext.*') or your custom | ||||||
|  | # ones. | ||||||
|  | extensions = [ | ||||||
|  |     'sphinx.ext.autodoc', | ||||||
|  |     'sphinx.ext.doctest', | ||||||
|  |     'sphinx.ext.todo', | ||||||
|  |     'sphinxcontrib.httpdomain', | ||||||
|  |     'sphinx.ext.coverage', | ||||||
|  |     'sphinx.ext.autosectionlabel', | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | # Add any paths that contain templates here, relative to this directory. | ||||||
|  | templates_path = ['_templates'] | ||||||
|  |  | ||||||
|  | # The suffix of source filenames. | ||||||
|  | source_suffix = '.rst' | ||||||
|  |  | ||||||
|  | # The encoding of source files. | ||||||
|  | #source_encoding = 'utf-8-sig' | ||||||
|  |  | ||||||
|  | # The master toctree document. | ||||||
|  | master_doc = 'index' | ||||||
|  |  | ||||||
|  | # General information about the project. | ||||||
|  | project = u'Senpy' | ||||||
|  | copyright = u'2016, J. Fernando Sánchez' | ||||||
|  | description = u'A framework for sentiment and emotion analysis services' | ||||||
|  |  | ||||||
|  | # The version info for the project you're documenting, acts as replacement for | ||||||
|  | # |version| and |release|, also used in various other places throughout the | ||||||
|  | # built documents. | ||||||
|  | # | ||||||
|  | # The short X.Y version. | ||||||
|  | # with open('../senpy/VERSION') as f: | ||||||
|  | #     version = f.read().strip() | ||||||
|  | # The full version, including alpha/beta/rc tags. | ||||||
|  | # release = version | ||||||
|  |  | ||||||
|  | # The language for content autogenerated by Sphinx. Refer to documentation | ||||||
|  | # for a list of supported languages. | ||||||
|  | language = None | ||||||
|  |  | ||||||
|  | # There are two options for replacing |today|: either, you set today to some | ||||||
|  | # non-false value, then it is used: | ||||||
|  | #today = '' | ||||||
|  | # Else, today_fmt is used as the format for a strftime call. | ||||||
|  | #today_fmt = '%B %d, %Y' | ||||||
|  |  | ||||||
|  | # List of patterns, relative to source directory, that match files and | ||||||
|  | # directories to ignore when looking for source files. | ||||||
|  | exclude_patterns = ['_build'] | ||||||
|  |  | ||||||
|  | # The reST default role (used for this markup: `text`) to use for all | ||||||
|  | # documents. | ||||||
|  | #default_role = None | ||||||
|  |  | ||||||
|  | # If true, '()' will be appended to :func: etc. cross-reference text. | ||||||
|  | #add_function_parentheses = True | ||||||
|  |  | ||||||
|  | # If true, the current module name will be prepended to all description | ||||||
|  | # unit titles (such as .. function::). | ||||||
|  | #add_module_names = True | ||||||
|  |  | ||||||
|  | # If true, sectionauthor and moduleauthor directives will be shown in the | ||||||
|  | # output. They are ignored by default. | ||||||
|  | #show_authors = False | ||||||
|  |  | ||||||
|  | # The name of the Pygments (syntax highlighting) style to use. | ||||||
|  | pygments_style = 'sphinx' | ||||||
|  |  | ||||||
|  | # A list of ignored prefixes for module index sorting. | ||||||
|  | #modindex_common_prefix = [] | ||||||
|  |  | ||||||
|  | # If true, keep warnings as "system message" paragraphs in the built documents. | ||||||
|  | #keep_warnings = False | ||||||
|  |  | ||||||
|  |  | ||||||
|  | html_theme = 'alabaster' | ||||||
|  | # -- Options for HTML output ---------------------------------------------- | ||||||
|  | # if not on_rtd:  # only import and set the theme if we're building docs locally | ||||||
|  | #     import sphinx_rtd_theme | ||||||
|  | #     html_theme_path = [sphinx_rtd_theme.get_html_theme_path()] | ||||||
|  |  | ||||||
|  | # else: | ||||||
|  | #     html_theme = 'default' | ||||||
|  |  | ||||||
|  | # The theme to use for HTML and HTML Help pages.  See the documentation for | ||||||
|  | # a list of builtin themes. | ||||||
|  |  | ||||||
|  | # Theme options are theme-specific and customize the look and feel of a theme | ||||||
|  | # further.  For a list of options available for each theme, see the | ||||||
|  | # documentation. | ||||||
|  | html_theme_options = { | ||||||
|  |     'logo': 'header.png', | ||||||
|  |     'github_user': 'gsi-upm', | ||||||
|  |     'github_repo': 'senpy', | ||||||
|  |     'github_banner': True, | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # Add any paths that contain custom themes here, relative to this directory. | ||||||
|  | #html_theme_path = [] | ||||||
|  |  | ||||||
|  | # The name for this set of Sphinx documents.  If None, it defaults to | ||||||
|  | # "<project> v<release> documentation". | ||||||
|  | #html_title = None | ||||||
|  |  | ||||||
|  | # A shorter title for the navigation bar.  Default is the same as html_title. | ||||||
|  | #html_short_title = None | ||||||
|  |  | ||||||
|  | # The name of an image file (relative to this directory) to place at the top | ||||||
|  | # of the sidebar. | ||||||
|  | #html_logo = None | ||||||
|  |  | ||||||
|  | # The name of an image file (within the static path) to use as favicon of the | ||||||
|  | # docs.  This file should be a Windows icon file (.ico) being 16x16 or 32x32 | ||||||
|  | # pixels large. | ||||||
|  | #html_favicon = None | ||||||
|  |  | ||||||
|  | # Add any paths that contain custom static files (such as style sheets) here, | ||||||
|  | # relative to this directory. They are copied after the builtin static files, | ||||||
|  | # so a file named "default.css" will overwrite the builtin "default.css". | ||||||
|  | html_static_path = ['_static'] | ||||||
|  |  | ||||||
|  | # Add any extra paths that contain custom files (such as robots.txt or | ||||||
|  | # .htaccess) here, relative to this directory. These files are copied | ||||||
|  | # directly to the root of the documentation. | ||||||
|  | #html_extra_path = [] | ||||||
|  |  | ||||||
|  | # If not '', a 'Last updated on:' timestamp is inserted at every page bottom, | ||||||
|  | # using the given strftime format. | ||||||
|  | #html_last_updated_fmt = '%b %d, %Y' | ||||||
|  |  | ||||||
|  | # If true, SmartyPants will be used to convert quotes and dashes to | ||||||
|  | # typographically correct entities. | ||||||
|  | #html_use_smartypants = True | ||||||
|  |  | ||||||
|  | # Custom sidebar templates, maps document names to template names. | ||||||
|  | html_sidebars = { | ||||||
|  |     '**': [ | ||||||
|  |         'about.html', | ||||||
|  |         'navigation.html', | ||||||
|  |         'searchbox.html', | ||||||
|  |         ] | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | # Additional templates that should be rendered to pages, maps page names to | ||||||
|  | # template names. | ||||||
|  | #html_additional_pages = {} | ||||||
|  |  | ||||||
|  | # If false, no module index is generated. | ||||||
|  | #html_domain_indices = True | ||||||
|  |  | ||||||
|  | # If false, no index is generated. | ||||||
|  | #html_use_index = True | ||||||
|  |  | ||||||
|  | # If true, the index is split into individual pages for each letter. | ||||||
|  | #html_split_index = False | ||||||
|  |  | ||||||
|  | # If true, links to the reST sources are added to the pages. | ||||||
|  | #html_show_sourcelink = True | ||||||
|  |  | ||||||
|  | # If true, "Created using Sphinx" is shown in the HTML footer. Default is True. | ||||||
|  | #html_show_sphinx = True | ||||||
|  |  | ||||||
|  | # If true, "(C) Copyright ..." is shown in the HTML footer. Default is True. | ||||||
|  | #html_show_copyright = True | ||||||
|  |  | ||||||
|  | # If true, an OpenSearch description file will be output, and all pages will | ||||||
|  | # contain a <link> tag referring to it.  The value of this option must be the | ||||||
|  | # base URL from which the finished HTML is served. | ||||||
|  | #html_use_opensearch = '' | ||||||
|  |  | ||||||
|  | # This is the file name suffix for HTML files (e.g. ".xhtml"). | ||||||
|  | #html_file_suffix = None | ||||||
|  |  | ||||||
|  | # Output file base name for HTML help builder. | ||||||
|  | htmlhelp_basename = 'Senpydoc' | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # -- Options for LaTeX output --------------------------------------------- | ||||||
|  |  | ||||||
|  | latex_elements = { | ||||||
|  | # The paper size ('letterpaper' or 'a4paper'). | ||||||
|  | #'papersize': 'letterpaper', | ||||||
|  |  | ||||||
|  | # The font size ('10pt', '11pt' or '12pt'). | ||||||
|  | #'pointsize': '10pt', | ||||||
|  |  | ||||||
|  | # Additional stuff for the LaTeX preamble. | ||||||
|  | #'preamble': '', | ||||||
|  | } | ||||||
|  |  | ||||||
|  | # Grouping the document tree into LaTeX files. List of tuples | ||||||
|  | # (source start file, target name, title, | ||||||
|  | #  author, documentclass [howto, manual, or own class]). | ||||||
|  | latex_documents = [ | ||||||
|  |   ('index', 'Senpy.tex', u'Senpy Documentation', | ||||||
|  |    u'J. Fernando Sánchez', 'manual'), | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | # The name of an image file (relative to this directory) to place at the top of | ||||||
|  | # the title page. | ||||||
|  | #latex_logo = None | ||||||
|  |  | ||||||
|  | # For "manual" documents, if this is true, then toplevel headings are parts, | ||||||
|  | # not chapters. | ||||||
|  | #latex_use_parts = False | ||||||
|  |  | ||||||
|  | # If true, show page references after internal links. | ||||||
|  | #latex_show_pagerefs = False | ||||||
|  |  | ||||||
|  | # If true, show URL addresses after external links. | ||||||
|  | #latex_show_urls = False | ||||||
|  |  | ||||||
|  | # Documents to append as an appendix to all manuals. | ||||||
|  | #latex_appendices = [] | ||||||
|  |  | ||||||
|  | # If false, no module index is generated. | ||||||
|  | #latex_domain_indices = True | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # -- Options for manual page output --------------------------------------- | ||||||
|  |  | ||||||
|  | # One entry per manual page. List of tuples | ||||||
|  | # (source start file, name, description, authors, manual section). | ||||||
|  | man_pages = [ | ||||||
|  |     ('index', 'senpy', u'Senpy Documentation', | ||||||
|  |      [u'J. Fernando Sánchez'], 1) | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | # If true, show URL addresses after external links. | ||||||
|  | #man_show_urls = False | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # -- Options for Texinfo output ------------------------------------------- | ||||||
|  |  | ||||||
|  | # Grouping the document tree into Texinfo files. List of tuples | ||||||
|  | # (source start file, target name, title, author, | ||||||
|  | #  dir menu entry, description, category) | ||||||
|  | texinfo_documents = [ | ||||||
|  |   ('index', 'Senpy', u'Senpy Documentation', | ||||||
|  |    u'J. Fernando Sánchez', 'Senpy', 'One line description of project.', | ||||||
|  |    'Miscellaneous'), | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | # Documents to append as an appendix to all manuals. | ||||||
|  | #texinfo_appendices = [] | ||||||
|  |  | ||||||
|  | # If false, no module index is generated. | ||||||
|  | #texinfo_domain_indices = True | ||||||
|  |  | ||||||
|  | # How to display URL addresses: 'footnote', 'no', or 'inline'. | ||||||
|  | #texinfo_show_urls = 'footnote' | ||||||
|  |  | ||||||
|  | # If true, do not generate a @detailmenu in the "Top" node's menu. | ||||||
|  | #texinfo_no_detailmenu = False | ||||||
							
								
								
									
										116
									
								
								docs/conversion.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,116 @@ | |||||||
|  | Conversion | ||||||
|  | ---------- | ||||||
|  |  | ||||||
|  | Senpy includes experimental support for emotion/sentiment conversion plugins. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Use | ||||||
|  | === | ||||||
|  |  | ||||||
|  | Consider the original query: http://127.0.0.1:5000/api/?i=hello&algo=emoRand | ||||||
|  |  | ||||||
|  | The requested plugin (emoRand) returns emotions using Ekman's model (or big6 in EmotionML): | ||||||
|  |  | ||||||
|  | .. code:: json | ||||||
|  |  | ||||||
|  |  | ||||||
|  |           ... rest of the document ... | ||||||
|  |           { | ||||||
|  |             "@type": "emotionSet", | ||||||
|  |             "onyx:hasEmotion": { | ||||||
|  |                 "@type": "emotion", | ||||||
|  |                 "onyx:hasEmotionCategory": "emoml:big6anger" | ||||||
|  |             }, | ||||||
|  |             "prov:wasGeneratedBy": "plugins/emoRand_0.1" | ||||||
|  |           } | ||||||
|  |  | ||||||
|  |            | ||||||
|  |  | ||||||
|  | To get these emotions in VAD space (FSRE dimensions in EmotionML), we'd do this: | ||||||
|  |  | ||||||
|  | http://127.0.0.1:5000/api/?i=hello&algo=emoRand&emotionModel=emoml:fsre-dimensions | ||||||
|  |  | ||||||
|  | This call, provided there is a valid conversion plugin from Ekman's to VAD, would return something like this: | ||||||
|  |  | ||||||
|  | .. code:: json | ||||||
|  |  | ||||||
|  |  | ||||||
|  |           ... rest of the document ... | ||||||
|  |           { | ||||||
|  |             "@type": "emotionSet", | ||||||
|  |             "onyx:hasEmotion": { | ||||||
|  |                 "@type": "emotion", | ||||||
|  |                 "onyx:hasEmotionCategory": "emoml:big6anger" | ||||||
|  |                 }, | ||||||
|  |             "prov:wasGeneratedBy": "plugins/emoRand_0.1" | ||||||
|  |           }, { | ||||||
|  |             "@type": "emotionSet", | ||||||
|  |             "onyx:hasEmotion": { | ||||||
|  |                 "@type": "emotion", | ||||||
|  |                 "A": 7.22, | ||||||
|  |                 "D": 6.28, | ||||||
|  |                 "V": 8.6 | ||||||
|  |             }, | ||||||
|  |             "prov:wasGeneratedBy": "plugins/Ekman2VAD_0.1" | ||||||
|  |  | ||||||
|  |           } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | That is called a *full* response, as it simply adds the converted emotion alongside. | ||||||
|  | It is also possible to get the original emotion nested within the new converted emotion, using the `conversion=nested` parameter: | ||||||
|  |  | ||||||
|  | .. code:: json | ||||||
|  |  | ||||||
|  |  | ||||||
|  |           ... rest of the document ... | ||||||
|  |           { | ||||||
|  |             "@type": "emotionSet", | ||||||
|  |             "onyx:hasEmotion": { | ||||||
|  |                 "@type": "emotion", | ||||||
|  |                 "onyx:hasEmotionCategory": "emoml:big6anger" | ||||||
|  |                 }, | ||||||
|  |             "prov:wasGeneratedBy": "plugins/emoRand_0.1" | ||||||
|  |             "onyx:wasDerivedFrom": { | ||||||
|  |                 "@type": "emotionSet", | ||||||
|  |                 "onyx:hasEmotion": { | ||||||
|  |                     "@type": "emotion", | ||||||
|  |                     "A": 7.22, | ||||||
|  |                     "D": 6.28, | ||||||
|  |                     "V": 8.6 | ||||||
|  |                 }, | ||||||
|  |                 "prov:wasGeneratedBy": "plugins/Ekman2VAD_0.1" | ||||||
|  |              } | ||||||
|  |  | ||||||
|  |           } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Lastly, `conversion=filtered` would only return the converted emotions. | ||||||
|  |  | ||||||
|  | Developing a conversion plugin | ||||||
|  | ================================ | ||||||
|  |  | ||||||
|  | Conversion plugins are discovered by the server just like any other plugin. | ||||||
|  | The difference is the slightly different API, and the need to specify the `source` and `target` of the conversion. | ||||||
|  | For instance, an emotion conversion plugin needs the following: | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .. code:: yaml | ||||||
|  |            | ||||||
|  |  | ||||||
|  |           --- | ||||||
|  |           onyx:doesConversion: | ||||||
|  |             - onyx:conversionFrom: emoml:big6 | ||||||
|  |               onyx:conversionTo: emoml:fsre-dimensions | ||||||
|  |             - onyx:conversionFrom: emoml:fsre-dimensions | ||||||
|  |               onyx:conversionTo: emoml:big6 | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .. code:: python | ||||||
|  |  | ||||||
|  |  | ||||||
|  |           class MyConversion(EmotionConversionPlugin): | ||||||
|  |  | ||||||
|  |               def convert(self, emotionSet, fromModel, toModel, params): | ||||||
|  |                   pass | ||||||
							
								
								
									
										16
									
								
								docs/demo.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,16 @@ | |||||||
|  | Demo | ||||||
|  | ---- | ||||||
|  |  | ||||||
|  | There is a demo available on http://senpy.cluster.gsi.dit.upm.es/, where you can test a serie of different plugins. | ||||||
|  | You can use the playground (a web interface) or make HTTP requests to the service API. | ||||||
|  |  | ||||||
|  | .. image:: senpy-playground.png | ||||||
|  |   :height: 400px | ||||||
|  |   :width: 800px | ||||||
|  |   :scale: 100 % | ||||||
|  |   :align: center | ||||||
|  |  | ||||||
|  | Plugins Demo | ||||||
|  | ============ | ||||||
|  |  | ||||||
|  | The source code and description of the plugins used in the demo is available here: https://lab.cluster.gsi.dit.upm.es/senpy/senpy-plugins-community/. | ||||||
							
								
								
									
										78
									
								
								docs/examples.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,78 @@ | |||||||
|  | Examples | ||||||
|  | ------ | ||||||
|  | All the examples in this page use the :download:`the main schema <_static/schemas/definitions.json>`. | ||||||
|  |  | ||||||
|  | Simple NIF annotation | ||||||
|  | ..................... | ||||||
|  | Description | ||||||
|  | ,,,,,,,,,,, | ||||||
|  | This example covers the basic example in the NIF documentation: `<http://persistence.uni-leipzig.org/nlp2rdf/ontologies/nif-core/nif-core.html>`_. | ||||||
|  |  | ||||||
|  | Representation | ||||||
|  | ,,,,,,,,,,,,,, | ||||||
|  | .. literalinclude:: examples/results/example-basic.json | ||||||
|  |    :language: json-ld | ||||||
|  |  | ||||||
|  | Sentiment Analysis | ||||||
|  | ..................... | ||||||
|  | Description | ||||||
|  | ,,,,,,,,,,, | ||||||
|  | This annotation corresponds to the sentiment analysis of an input. The example shows the sentiment represented according to Marl format. | ||||||
|  | The sentiments detected are contained in the Sentiments array with their related part of the text. | ||||||
|  |  | ||||||
|  | Representation | ||||||
|  | ,,,,,,,,,,,,,, | ||||||
|  |  | ||||||
|  | .. literalinclude:: examples/results/example-sentiment.json | ||||||
|  |    :emphasize-lines: 5-10,25-33 | ||||||
|  |    :language: json-ld | ||||||
|  |  | ||||||
|  | Suggestion Mining | ||||||
|  | ................. | ||||||
|  | Description | ||||||
|  | ,,,,,,,,,,, | ||||||
|  | The suggestions schema represented below shows the suggestions detected in the text. Within it, we can find the NIF fields highlighted that corresponds to the text of the detected suggestion.  | ||||||
|  |  | ||||||
|  | Representation | ||||||
|  | ,,,,,,,,,,,,,, | ||||||
|  |  | ||||||
|  | .. literalinclude:: examples/results/example-suggestion.json | ||||||
|  |    :emphasize-lines: 5-8,22-27 | ||||||
|  |    :language: json-ld | ||||||
|  |  | ||||||
|  | Emotion Analysis | ||||||
|  | ................ | ||||||
|  | Description | ||||||
|  | ,,,,,,,,,,, | ||||||
|  | This annotation represents the emotion analysis of an input to Senpy. The emotions are contained in the emotions section with the text that refers to following Onyx format and the emotion model defined beforehand. | ||||||
|  |  | ||||||
|  | Representation | ||||||
|  | ,,,,,,,,,,,,,, | ||||||
|  |  | ||||||
|  | .. literalinclude:: examples/results/example-emotion.json | ||||||
|  |    :language: json-ld | ||||||
|  |    :emphasize-lines: 5-8,25-37 | ||||||
|  |  | ||||||
|  | Named Entity Recognition | ||||||
|  | ........................ | ||||||
|  | Description | ||||||
|  | ,,,,,,,,,,, | ||||||
|  | The Named Entity Recognition is represented as follows. In this particular case, it can be seen within the entities array the entities recognised. For the example input, Microsoft and Windows Phone are the ones detected.  | ||||||
|  | Representation | ||||||
|  | ,,,,,,,,,,,,,, | ||||||
|  |  | ||||||
|  | .. literalinclude:: examples/results/example-ner.json | ||||||
|  |    :emphasize-lines: 5-8,19-34 | ||||||
|  |    :language: json-ld | ||||||
|  |  | ||||||
|  | Complete example | ||||||
|  | ................ | ||||||
|  | Description | ||||||
|  | ,,,,,,,,,,, | ||||||
|  | This example covers all of the above cases, integrating all the annotations in the same document. | ||||||
|  |  | ||||||
|  | Representation | ||||||
|  | ,,,,,,,,,,,,,, | ||||||
|  |  | ||||||
|  | .. literalinclude:: examples/results/example-complete.json | ||||||
|  |    :language: json-ld | ||||||
							
								
								
									
										5
									
								
								docs/examples/plugins/noplugins.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,5 @@ | |||||||
|  | { | ||||||
|  |   "@type": "plugins", | ||||||
|  |   "plugins": [ | ||||||
|  |     ] | ||||||
|  | } | ||||||
							
								
								
									
										85
									
								
								docs/examples/results/example-analysis-as-id.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,85 @@ | |||||||
|  | { | ||||||
|  |   "@context": "http://mixedemotions-project.eu/ns/context.jsonld", | ||||||
|  |   "@id": "me:Result1", | ||||||
|  |   "@type": "results", | ||||||
|  |   "analysis": [ | ||||||
|  |       { | ||||||
|  |           "@id": "_:SAnalysis1_Activity", | ||||||
|  |           "@type": "marl:SentimentAnalysis", | ||||||
|  |           "prov:wasAssociatedWith": "me:SAnalysis1" | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |           "@id": "_:EmotionAnalysis1_Activity", | ||||||
|  |           "@type": "onyx:EmotionAnalysis", | ||||||
|  |           "prov:wasAssociatedWith": "me:EmotionAnalysis1" | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |           "@id": "_:NER1_Activity", | ||||||
|  |           "@type": "me:NER", | ||||||
|  |           "prov:wasAssociatedWith": "me:NER1" | ||||||
|  |       } | ||||||
|  |   ], | ||||||
|  |   "entries": [ | ||||||
|  |     { | ||||||
|  |       "@id": "http://micro.blog/status1", | ||||||
|  |       "@type": [ | ||||||
|  |         "nif:RFC5147String", | ||||||
|  |         "nif:Context" | ||||||
|  |       ], | ||||||
|  |       "nif:isString": "Dear Microsoft, put your Windows Phone on your newest #open technology program. You'll be awesome. #opensource", | ||||||
|  |       "entities": [ | ||||||
|  |         { | ||||||
|  |           "@id": "http://micro.blog/status1#char=5,13", | ||||||
|  |           "nif:beginIndex": 5, | ||||||
|  |           "nif:endIndex": 13, | ||||||
|  |           "nif:anchorOf": "Microsoft", | ||||||
|  |           "me:references": "http://dbpedia.org/page/Microsoft", | ||||||
|  |           "prov:wasGeneratedBy": "_:NER1_Activity" | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |           "@id": "http://micro.blog/status1#char=25,37", | ||||||
|  |           "nif:beginIndex": 25, | ||||||
|  |           "nif:endIndex": 37, | ||||||
|  |           "nif:anchorOf": "Windows Phone", | ||||||
|  |           "me:references": "http://dbpedia.org/page/Windows_Phone", | ||||||
|  |           "prov:wasGeneratedBy": "_:NER1_Activity" | ||||||
|  |         } | ||||||
|  |       ], | ||||||
|  |       "suggestions": [ | ||||||
|  |         { | ||||||
|  |           "@id": "http://micro.blog/status1#char=16,77", | ||||||
|  |           "nif:beginIndex": 16, | ||||||
|  |           "nif:endIndex": 77, | ||||||
|  |           "nif:anchorOf": "put your Windows Phone on your newest #open technology program", | ||||||
|  |           "prov:wasGeneratedBy": "_:SgAnalysis1_Activity" | ||||||
|  |         } | ||||||
|  |       ], | ||||||
|  |       "sentiments": [ | ||||||
|  |         { | ||||||
|  |           "@id": "http://micro.blog/status1#char=80,97", | ||||||
|  |           "nif:beginIndex": 80, | ||||||
|  |           "nif:endIndex": 97, | ||||||
|  |           "nif:anchorOf": "You'll be awesome.", | ||||||
|  |           "marl:hasPolarity": "marl:Positive", | ||||||
|  |           "marl:polarityValue": 0.9, | ||||||
|  |           "prov:wasGeneratedBy": "_:SgAnalysis1_Activity" | ||||||
|  |         } | ||||||
|  |       ], | ||||||
|  |       "emotions": [ | ||||||
|  |         { | ||||||
|  |           "@id": "http://micro.blog/status1#char=0,109", | ||||||
|  |           "nif:anchorOf": "Dear Microsoft, put your Windows Phone on your newest #open technology program. You'll be awesome. #opensource", | ||||||
|  |           "prov:wasGeneratedBy": "_:EmotionAnalysis1_Activity", | ||||||
|  |           "onyx:hasEmotion": [ | ||||||
|  |             { | ||||||
|  |               "onyx:hasEmotionCategory": "wna:liking" | ||||||
|  |             }, | ||||||
|  |             { | ||||||
|  |               "onyx:hasEmotionCategory": "wna:excitement" | ||||||
|  |             } | ||||||
|  |           ] | ||||||
|  |         } | ||||||
|  |       ] | ||||||
|  |     } | ||||||
|  |   ] | ||||||
|  | } | ||||||
							
								
								
									
										18
									
								
								docs/examples/results/example-basic.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,18 @@ | |||||||
|  | { | ||||||
|  |     "@context": "http://mixedemotions-project.eu/ns/context.jsonld", | ||||||
|  |     "@id": "me:Result1", | ||||||
|  |     "@type": "results", | ||||||
|  |     "analysis": [ ], | ||||||
|  |     "entries": [ | ||||||
|  |         { | ||||||
|  |             "@id": "http://example.org#char=0,40", | ||||||
|  |             "@type": [ | ||||||
|  |                 "nif:RFC5147String", | ||||||
|  |                 "nif:Context" | ||||||
|  |             ], | ||||||
|  |             "nif:beginIndex": 0, | ||||||
|  |             "nif:endIndex": 40, | ||||||
|  |             "nif:isString": "My favourite actress is Natalie Portman" | ||||||
|  |         } | ||||||
|  |     ] | ||||||
|  | } | ||||||
							
								
								
									
										100
									
								
								docs/examples/results/example-complete.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,100 @@ | |||||||
|  | { | ||||||
|  |     "@context": "http://mixedemotions-project.eu/ns/context.jsonld", | ||||||
|  |     "@id": "me:Result1", | ||||||
|  |     "@type": "results", | ||||||
|  |     "analysis": [ | ||||||
|  |         { | ||||||
|  |             "@id": "_:SAnalysis1_Activity", | ||||||
|  |             "@type": "marl:SentimentAnalysis", | ||||||
|  |             "prov:wasAssociatedWith": "me:SentimentAnalysis", | ||||||
|  |             "prov:used": [ | ||||||
|  |                 { | ||||||
|  |                     "name": "marl:maxPolarityValue", | ||||||
|  |                     "prov:value": "1" | ||||||
|  |                 }, | ||||||
|  |                 { | ||||||
|  |                     "name": "marl:minPolarityValue", | ||||||
|  |                     "prov:value": "0" | ||||||
|  |                 } | ||||||
|  |             ] | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |             "@id": "_:SgAnalysis1_Activity", | ||||||
|  |             "prov:wasAssociatedWith": "me:SgAnalysis1", | ||||||
|  |             "@type": "me:SuggestionAnalysis" | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |             "@id": "_:EmotionAnalysis1_Activity", | ||||||
|  |             "@type": "me:EmotionAnalysis", | ||||||
|  |             "prov:wasAssociatedWith": "me:EmotionAnalysis1" | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |             "@id": "_:NER1_Activity", | ||||||
|  |             "@type": "me:NER", | ||||||
|  |             "prov:wasAssociatedWith": "me:EmotionNER1" | ||||||
|  |         } | ||||||
|  |     ], | ||||||
|  |     "entries": [ | ||||||
|  |         { | ||||||
|  |             "@id": "http://micro.blog/status1", | ||||||
|  |             "@type": [ | ||||||
|  |                 "nif:RFC5147String", | ||||||
|  |                 "nif:Context" | ||||||
|  |             ], | ||||||
|  |             "nif:isString": "Dear Microsoft, put your Windows Phone on your newest #open technology program. You'll be awesome. #opensource", | ||||||
|  |             "entities": [ | ||||||
|  |                 { | ||||||
|  |                     "@id": "http://micro.blog/status1#char=5,13", | ||||||
|  |                     "nif:beginIndex": 5, | ||||||
|  |                     "nif:endIndex": 13, | ||||||
|  |                     "nif:anchorOf": "Microsoft", | ||||||
|  |                     "me:references": "http://dbpedia.org/page/Microsoft", | ||||||
|  |                     "prov:wasGeneratedBy": "me:NER1" | ||||||
|  |                 }, | ||||||
|  |                 { | ||||||
|  |                     "@id": "http://micro.blog/status1#char=25,37", | ||||||
|  |                     "nif:beginIndex": 25, | ||||||
|  |                     "nif:endIndex": 37, | ||||||
|  |                     "nif:anchorOf": "Windows Phone", | ||||||
|  |                     "me:references": "http://dbpedia.org/page/Windows_Phone", | ||||||
|  |                     "prov:wasGeneratedBy": "me:NER1" | ||||||
|  |                 } | ||||||
|  |             ], | ||||||
|  |             "suggestions": [ | ||||||
|  |                 { | ||||||
|  |                     "@id": "http://micro.blog/status1#char=16,77", | ||||||
|  |                     "nif:beginIndex": 16, | ||||||
|  |                     "nif:endIndex": 77, | ||||||
|  |                     "nif:anchorOf": "put your Windows Phone on your newest #open technology program", | ||||||
|  |                     "prov:wasGeneratedBy": "me:SgAnalysis1" | ||||||
|  |                 } | ||||||
|  |             ], | ||||||
|  |             "sentiments": [ | ||||||
|  |                 { | ||||||
|  |                     "@id": "http://micro.blog/status1#char=80,97", | ||||||
|  |                     "nif:beginIndex": 80, | ||||||
|  |                     "nif:endIndex": 97, | ||||||
|  |                     "nif:anchorOf": "You'll be awesome.", | ||||||
|  |                     "marl:hasPolarity": "marl:Positive", | ||||||
|  |                     "marl:polarityValue": 0.9, | ||||||
|  |                     "prov:wasGeneratedBy": "me:SAnalysis1" | ||||||
|  |                 } | ||||||
|  |             ], | ||||||
|  |             "emotions": [ | ||||||
|  |                 { | ||||||
|  |                     "@id": "http://micro.blog/status1#char=0,109", | ||||||
|  |                     "nif:anchorOf": "Dear Microsoft, put your Windows Phone on your newest #open technology program. You'll be awesome. #opensource", | ||||||
|  |                     "prov:wasGeneratedBy": "me:EAnalysis1", | ||||||
|  |                     "onyx:hasEmotion": [ | ||||||
|  |                         { | ||||||
|  |                             "onyx:hasEmotionCategory": "wna:liking" | ||||||
|  |                         }, | ||||||
|  |                         { | ||||||
|  |                             "onyx:hasEmotionCategory": "wna:excitement" | ||||||
|  |                         } | ||||||
|  |                     ] | ||||||
|  |                 } | ||||||
|  |             ] | ||||||
|  |         } | ||||||
|  |     ] | ||||||
|  | } | ||||||
							
								
								
									
										43
									
								
								docs/examples/results/example-emotion.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,43 @@ | |||||||
|  | { | ||||||
|  |   "@context": "http://mixedemotions-project.eu/ns/context.jsonld", | ||||||
|  |   "@id": "me:Result1", | ||||||
|  |   "@type": "results", | ||||||
|  |   "analysis": [ | ||||||
|  |     { | ||||||
|  |       "@id": "me:EmotionAnalysis1_Activity", | ||||||
|  |       "@type": "me:EmotionAnalysis1", | ||||||
|  |       "prov:wasAssociatedWith": "me:EmotionAnalysis1" | ||||||
|  |     } | ||||||
|  |   ], | ||||||
|  |   "entries": [ | ||||||
|  |     { | ||||||
|  |       "@id": "http://micro.blog/status1", | ||||||
|  |       "@type": [ | ||||||
|  |         "nif:RFC5147String", | ||||||
|  |         "nif:Context" | ||||||
|  |       ], | ||||||
|  |       "nif:isString": "Dear Microsoft, put your Windows Phone on your newest #open technology program. You'll be awesome. #opensource", | ||||||
|  |       "entities": [ | ||||||
|  |       ], | ||||||
|  |       "suggestions": [ | ||||||
|  |       ], | ||||||
|  |       "sentiments": [ | ||||||
|  |       ], | ||||||
|  |       "emotions": [ | ||||||
|  |         { | ||||||
|  |           "@id": "http://micro.blog/status1#char=0,109", | ||||||
|  |           "nif:anchorOf": "Dear Microsoft, put your Windows Phone on your newest #open technology program. You'll be awesome. #opensource", | ||||||
|  |           "prov:wasGeneratedBy": "_:EmotionAnalysis1_Activity", | ||||||
|  |           "onyx:hasEmotion": [ | ||||||
|  |             { | ||||||
|  |               "onyx:hasEmotionCategory": "wna:liking" | ||||||
|  |             }, | ||||||
|  |             { | ||||||
|  |               "onyx:hasEmotionCategory": "wna:excitement" | ||||||
|  |             } | ||||||
|  |           ] | ||||||
|  |         } | ||||||
|  |       ] | ||||||
|  |     } | ||||||
|  |   ] | ||||||
|  | } | ||||||
							
								
								
									
										46
									
								
								docs/examples/results/example-ner.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,46 @@ | |||||||
|  | { | ||||||
|  |   "@context": "http://mixedemotions-project.eu/ns/context.jsonld", | ||||||
|  |   "@id": "me:Result1", | ||||||
|  |   "@type": "results", | ||||||
|  |   "analysis": [ | ||||||
|  |     { | ||||||
|  |       "@id": "_:NER1_Activity", | ||||||
|  |       "@type": "me:NERAnalysis", | ||||||
|  |       "prov:wasAssociatedWith": "me:NER1" | ||||||
|  |     } | ||||||
|  |   ], | ||||||
|  |   "entries": [ | ||||||
|  |     { | ||||||
|  |       "@id": "http://micro.blog/status1", | ||||||
|  |       "@type": [ | ||||||
|  |         "nif:RFC5147String", | ||||||
|  |         "nif:Context" | ||||||
|  |       ], | ||||||
|  |       "nif:isString": "Dear Microsoft, put your Windows Phone on your newest #open technology program. You'll be awesome. #opensource", | ||||||
|  |       "entities": [ | ||||||
|  |         { | ||||||
|  |           "@id": "http://micro.blog/status1#char=5,13", | ||||||
|  |           "nif:beginIndex": 5, | ||||||
|  |           "nif:endIndex": 13, | ||||||
|  |           "nif:anchorOf": "Microsoft", | ||||||
|  |           "me:references": "http://dbpedia.org/page/Microsoft", | ||||||
|  |           "prov:wasGeneratedBy": "me:NER1" | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |           "@id": "http://micro.blog/status1#char=25,37", | ||||||
|  |           "nif:beginIndex": 25, | ||||||
|  |           "nif:endIndex": 37, | ||||||
|  |           "nif:anchorOf": "Windows Phone", | ||||||
|  |           "me:references": "http://dbpedia.org/page/Windows_Phone", | ||||||
|  |           "prov:wasGeneratedBy": "me:NER1" | ||||||
|  |         } | ||||||
|  |       ], | ||||||
|  |       "suggestions": [ | ||||||
|  |       ], | ||||||
|  |       "sentiments": [ | ||||||
|  |       ], | ||||||
|  |       "emotionSets": [ | ||||||
|  |       ] | ||||||
|  |     } | ||||||
|  |   ] | ||||||
|  | } | ||||||
							
								
								
									
										52
									
								
								docs/examples/results/example-pad.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,52 @@ | |||||||
|  | { | ||||||
|  | "@context": [ | ||||||
|  |              "http://mixedemotions-project.eu/ns/context.jsonld", | ||||||
|  |              { | ||||||
|  |                "emovoc": "http://www.gsi.dit.upm.es/ontologies/onyx/vocabularies/emotionml/ns#" | ||||||
|  |              } | ||||||
|  |   ], | ||||||
|  |   "@id": "me:Result1", | ||||||
|  |   "@type": "results", | ||||||
|  |   "analysis": [ | ||||||
|  |     { | ||||||
|  |       "@id": "me:HesamsAnalysis_Activity", | ||||||
|  |       "@type": "onyx:EmotionAnalysis", | ||||||
|  |       "prov:wasAssociatedWith": "me:HesamsAnalysis", | ||||||
|  |       "prov:used": [ | ||||||
|  |         { | ||||||
|  |           "name": "emotion-model", | ||||||
|  |           "prov:value": "emovoc:pad-dimensions" | ||||||
|  |         } | ||||||
|  |       ] | ||||||
|  |     } | ||||||
|  |   ], | ||||||
|  |   "entries": [ | ||||||
|  |     { | ||||||
|  |       "@id": "Entry1", | ||||||
|  |       "@type": [ | ||||||
|  |         "nif:RFC5147String", | ||||||
|  |         "nif:Context" | ||||||
|  |       ], | ||||||
|  |       "nif:isString": "This is a test string", | ||||||
|  |       "entities": [ | ||||||
|  |       ], | ||||||
|  |       "suggestions": [ | ||||||
|  |       ], | ||||||
|  |       "sentiments": [ | ||||||
|  |       ], | ||||||
|  |       "emotions": [ | ||||||
|  |         { | ||||||
|  |           "@id": "Entry1#char=0,21", | ||||||
|  |           "nif:anchorOf": "This is a test string", | ||||||
|  |           "prov:wasGeneratedBy": "_:HesamAnalysis_Activity", | ||||||
|  |           "onyx:hasEmotion": [ | ||||||
|  |             { | ||||||
|  |                 "emovoc:pleasure": 0.5, | ||||||
|  |                 "emovoc:arousal": 0.7 | ||||||
|  |             } | ||||||
|  |           ] | ||||||
|  |         } | ||||||
|  |       ] | ||||||
|  |     } | ||||||
|  |   ] | ||||||
|  | } | ||||||
							
								
								
									
										39
									
								
								docs/examples/results/example-sentiment.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,39 @@ | |||||||
|  | { | ||||||
|  |   "@context": "http://mixedemotions-project.eu/ns/context.jsonld", | ||||||
|  |   "@id": "me:Result1", | ||||||
|  |   "@type": "results", | ||||||
|  |   "analysis": [ | ||||||
|  |     { | ||||||
|  |       "@id": "_:SAnalysis1_Activity", | ||||||
|  |       "@type": "marl:SentimentAnalysis", | ||||||
|  |       "prov:wasAssociatedWith": "me:SAnalysis1" | ||||||
|  |     } | ||||||
|  |   ], | ||||||
|  |   "entries": [ | ||||||
|  |     { | ||||||
|  |       "@id": "http://micro.blog/status1", | ||||||
|  |       "@type": [ | ||||||
|  |         "nif:RFC5147String", | ||||||
|  |         "nif:Context" | ||||||
|  |       ], | ||||||
|  |       "nif:isString": "Dear Microsoft, put your Windows Phone on your newest #open technology program. You'll be awesome. #opensource", | ||||||
|  |       "entities": [ | ||||||
|  |       ], | ||||||
|  |       "suggestions": [ | ||||||
|  |       ], | ||||||
|  |       "sentiments": [ | ||||||
|  |         { | ||||||
|  |           "@id": "http://micro.blog/status1#char=80,97", | ||||||
|  |           "nif:beginIndex": 80, | ||||||
|  |           "nif:endIndex": 97, | ||||||
|  |           "nif:anchorOf": "You'll be awesome.", | ||||||
|  |           "marl:hasPolarity": "marl:Positive", | ||||||
|  |           "marl:polarityValue": 0.9, | ||||||
|  |           "prov:wasGeneratedBy": "_:SAnalysis1_Activity" | ||||||
|  |         } | ||||||
|  |       ], | ||||||
|  |       "emotionSets": [ | ||||||
|  |       ] | ||||||
|  |     } | ||||||
|  |   ] | ||||||
|  | } | ||||||
							
								
								
									
										37
									
								
								docs/examples/results/example-suggestion.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,37 @@ | |||||||
|  | { | ||||||
|  |   "@context": "http://mixedemotions-project.eu/ns/context.jsonld", | ||||||
|  |   "@id": "me:Result1", | ||||||
|  |   "@type": "results", | ||||||
|  |   "analysis": [ | ||||||
|  |     { | ||||||
|  |       "@id": "_:SgAnalysis1_Activity", | ||||||
|  |       "@type": "me:SuggestionAnalysis", | ||||||
|  |       "prov:wasAssociatedWith": "me:SgAnalysis1" | ||||||
|  |     } | ||||||
|  |   ], | ||||||
|  |   "entries": [ | ||||||
|  |     { | ||||||
|  |       "@id": "http://micro.blog/status1", | ||||||
|  |       "@type": [ | ||||||
|  |         "nif:RFC5147String", | ||||||
|  |         "nif:Context" | ||||||
|  |       ], | ||||||
|  |       "nif:isString": "Dear Microsoft, put your Windows Phone on your newest #open technology program. You'll be awesome. #opensource", | ||||||
|  |       "entities": [ | ||||||
|  |       ], | ||||||
|  |       "suggestions": [ | ||||||
|  |         { | ||||||
|  |           "@id": "http://micro.blog/status1#char=16,77", | ||||||
|  |           "nif:beginIndex": 16, | ||||||
|  |           "nif:endIndex": 77, | ||||||
|  |           "nif:anchorOf": "put your Windows Phone on your newest #open technology program", | ||||||
|  |           "prov:wasGeneratedBy": "_:SgAnalysis1_Activity" | ||||||
|  |         } | ||||||
|  |       ], | ||||||
|  |       "sentiments": [ | ||||||
|  |       ], | ||||||
|  |       "emotionSets": [ | ||||||
|  |       ] | ||||||
|  |     } | ||||||
|  |   ] | ||||||
|  | } | ||||||
							
								
								
									
										35
									
								
								docs/index.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,35 @@ | |||||||
|  | Welcome to Senpy's documentation! | ||||||
|  | ================================= | ||||||
|  | .. image:: https://readthedocs.org/projects/senpy/badge/?version=latest | ||||||
|  |    :target: http://senpy.readthedocs.io/en/latest/ | ||||||
|  | .. image:: https://badge.fury.io/py/senpy.svg | ||||||
|  |    :target: https://badge.fury.io/py/senpy | ||||||
|  | .. image:: https://lab.cluster.gsi.dit.upm.es/senpy/senpy/badges/master/build.svg | ||||||
|  |    :target: https://lab.cluster.gsi.dit.upm.es/senpy/senpy/commits/master | ||||||
|  | .. image:: https://lab.cluster.gsi.dit.upm.es/senpy/senpy/badges/master/coverage.svg | ||||||
|  |    :target: https://lab.cluster.gsi.dit.upm.es/senpy/senpy/commits/master | ||||||
|  | .. image:: https://img.shields.io/pypi/l/requests.svg | ||||||
|  |    :target: https://lab.cluster.gsi.dit.upm.es/senpy/senpy/ | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Senpy is a framework for sentiment and emotion analysis services. | ||||||
|  | Services built with senpy are interchangeable and easy to use because they share a common :doc:`apischema`. | ||||||
|  | It also simplifies service development. | ||||||
|  |  | ||||||
|  | .. image:: senpy-architecture.png | ||||||
|  |   :width: 100% | ||||||
|  |   :align: center | ||||||
|  |  | ||||||
|  | .. toctree:: | ||||||
|  |    :caption: Learn more about senpy: | ||||||
|  |    :maxdepth: 2 | ||||||
|  |  | ||||||
|  |    senpy | ||||||
|  |    installation | ||||||
|  |    demo | ||||||
|  |    usage | ||||||
|  |    apischema | ||||||
|  |    plugins | ||||||
|  |    conversion | ||||||
|  |    about | ||||||
							
								
								
									
										72
									
								
								docs/installation.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,72 @@ | |||||||
|  | Installation | ||||||
|  | ------------ | ||||||
|  | The stable version can be used in two ways: as a system/user library through pip, or as a docker image. | ||||||
|  |  | ||||||
|  | The docker image is the recommended way because it is self-contained and isolated from the system, which means: | ||||||
|  |  | ||||||
|  |   * Downloading and using it is just one command | ||||||
|  |   * All dependencies are included | ||||||
|  |   * It is OS-independent (MacOS, Windows, GNU/Linux) | ||||||
|  |   * Several versions may coexist in the same machine without additional virtual environments | ||||||
|  |  | ||||||
|  | Additionally, you may create your own docker image with your custom plugins, ready to be used by others. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Through PIP | ||||||
|  | *********** | ||||||
|  |  | ||||||
|  | .. code:: bash | ||||||
|  |  | ||||||
|  |    pip install --user senpy | ||||||
|  |  | ||||||
|  |     | ||||||
|  | Alternatively, you can use the development version: | ||||||
|  |   | ||||||
|  | .. code:: bash | ||||||
|  |  | ||||||
|  |    git clone git@github.com:gsi-upm/senpy | ||||||
|  |    cd senpy | ||||||
|  |    pip install --user . | ||||||
|  |  | ||||||
|  | If you want to install senpy globally, use sudo instead of the ``--user`` flag. | ||||||
|  |  | ||||||
|  | Docker Image | ||||||
|  | ************ | ||||||
|  | Build the image or use the pre-built one:    | ||||||
|  |  | ||||||
|  | .. code:: bash | ||||||
|  |  | ||||||
|  |    docker run -ti -p 5000:5000 gsiupm/senpy --host 0.0.0.0 --default-plugins | ||||||
|  |  | ||||||
|  | To add custom plugins, use a docker volume:  | ||||||
|  |      | ||||||
|  | .. code:: bash | ||||||
|  |  | ||||||
|  |    docker run -ti -p 5000:5000 -v <PATH OF PLUGINS>:/plugins gsiupm/senpy --host 0.0.0.0 --default-plugins -f /plugins | ||||||
|  |   | ||||||
|  |  | ||||||
|  | Python 2 | ||||||
|  | ........ | ||||||
|  |  | ||||||
|  | There is a Senpy version for python2 too: | ||||||
|  |      | ||||||
|  | .. code:: bash | ||||||
|  |  | ||||||
|  |    docker run -ti -p 5000:5000 gsiupm/senpy:python2.7 --host 0.0.0.0 --default-plugins | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Alias | ||||||
|  | ..... | ||||||
|  |  | ||||||
|  | If you are using the docker approach regularly, it is advisable to use a script or an alias to simplify your executions: | ||||||
|  |  | ||||||
|  | .. code:: bash | ||||||
|  |  | ||||||
|  |    alias senpy='docker run --rm -ti -p 5000:5000 -v $PWD:/senpy-plugins gsiupm/senpy --default-plugins' | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Now, you may run senpy from any folder in your computer like so: | ||||||
|  |  | ||||||
|  | .. code:: bash | ||||||
|  |  | ||||||
|  |    senpy --version | ||||||
							
								
								
									
										113
									
								
								docs/plugins-definition.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,113 @@ | |||||||
|  | Advanced plugin definition | ||||||
|  | -------------------------- | ||||||
|  | In addition to finding plugins defined in source code files, senpy can also load a special type of definition file (`.senpy` files). | ||||||
|  | This used to be the only mechanism for loading in earlier versions of senpy. | ||||||
|  |  | ||||||
|  | The definition file contains basic information  | ||||||
|  |  | ||||||
|  | Lastly, it is also possible to add new plugins programmatically. | ||||||
|  |  | ||||||
|  | .. contents:: :local: | ||||||
|  |  | ||||||
|  | What is a plugin? | ||||||
|  | ================= | ||||||
|  |  | ||||||
|  | A plugin is a program that, given a text, will add annotations to it. | ||||||
|  | In practice, a plugin consists of at least two files: | ||||||
|  |  | ||||||
|  | - Definition file: a `.senpy` file that describes the plugin (e.g. what input parameters it accepts, what emotion model it uses). | ||||||
|  | - Python module: the actual code that will add annotations to each input. | ||||||
|  |  | ||||||
|  | This separation allows us to deploy plugins that use the same code but employ different parameters. | ||||||
|  | For instance, one could use the same classifier and processing in several plugins, but train with different datasets. | ||||||
|  | This scenario is particularly useful for evaluation purposes. | ||||||
|  |  | ||||||
|  | The only limitation is that the name of each plugin needs to be unique. | ||||||
|  |  | ||||||
|  | Definition files | ||||||
|  | ================ | ||||||
|  |  | ||||||
|  | The definition file complements and overrides the attributes provided by the plugin. | ||||||
|  | It can be written in YAML or JSON. | ||||||
|  | The most important attributes are: | ||||||
|  |  | ||||||
|  | * **name**: unique name that senpy will use internally to identify the plugin. | ||||||
|  | * **module**: indicates the module that contains the plugin code, which will be automatically loaded by senpy. | ||||||
|  | * **version** | ||||||
|  | * extra_params: to add parameters to the senpy API when this plugin is requested. Those parameters may be required, and have aliased names. For instance: | ||||||
|  |  | ||||||
|  |   .. code:: yaml | ||||||
|  |  | ||||||
|  |             extra_params: | ||||||
|  |                 hello_param: | ||||||
|  |                     aliases: # required | ||||||
|  |                         - hello_param | ||||||
|  |                         - hello | ||||||
|  |                     required: true | ||||||
|  |                     default: Hi you | ||||||
|  |                     values: | ||||||
|  |                         - Hi you | ||||||
|  |                         - Hello y'all | ||||||
|  |                         - Howdy | ||||||
|  |  | ||||||
|  | A complete example: | ||||||
|  |  | ||||||
|  | .. code:: yaml | ||||||
|  |            | ||||||
|  |           name: <Name of the plugin> | ||||||
|  |           module: <Python file> | ||||||
|  |           version: 0.1 | ||||||
|  |  | ||||||
|  | And the json equivalent: | ||||||
|  |  | ||||||
|  | .. code:: json | ||||||
|  |  | ||||||
|  |           { | ||||||
|  |             "name": "<Name of the plugin>", | ||||||
|  |             "module": "<Python file>", | ||||||
|  |             "version": "0.1" | ||||||
|  |           } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Example plugin with a definition file | ||||||
|  | ===================================== | ||||||
|  |  | ||||||
|  | In this section, we will implement a basic sentiment analysis plugin. | ||||||
|  | To determine the polarity of each entry, the plugin will compare the length of the string to a threshold. | ||||||
|  | This threshold will be included in the definition file. | ||||||
|  |  | ||||||
|  | The definition file would look like this: | ||||||
|  |  | ||||||
|  | .. code:: yaml | ||||||
|  |  | ||||||
|  |           name: helloworld | ||||||
|  |           module: helloworld | ||||||
|  |           version: 0.0 | ||||||
|  |           threshold: 10 | ||||||
|  |           description: Hello World | ||||||
|  |  | ||||||
|  | Now, in a file named ``helloworld.py``: | ||||||
|  |  | ||||||
|  | .. code:: python | ||||||
|  |  | ||||||
|  |           #!/bin/env python | ||||||
|  |           #helloworld.py | ||||||
|  |  | ||||||
|  |           from senpy import AnalysisPlugin | ||||||
|  |           from senpy import Sentiment | ||||||
|  |  | ||||||
|  |  | ||||||
|  |           class HelloWorld(AnalysisPlugin): | ||||||
|  |  | ||||||
|  |               def analyse_entry(entry, params): | ||||||
|  |                   '''Basically do nothing with each entry''' | ||||||
|  |  | ||||||
|  |                   sentiment = Sentiment() | ||||||
|  |                   if len(entry.text) < self.threshold: | ||||||
|  |                       sentiment['marl:hasPolarity'] = 'marl:Positive' | ||||||
|  |                   else: | ||||||
|  |                       sentiment['marl:hasPolarity'] = 'marl:Negative' | ||||||
|  |                   entry.sentiments.append(sentiment) | ||||||
|  |                   yield entry | ||||||
|  |  | ||||||
|  | The complete code of the example plugin is available `here <https://lab.cluster.gsi.dit.upm.es/senpy/plugin-prueba>`__. | ||||||
							
								
								
									
										314
									
								
								docs/plugins.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,314 @@ | |||||||
|  | Developing new plugins | ||||||
|  | ---------------------- | ||||||
|  | This document contains the minimum to get you started with developing new analysis plugin. | ||||||
|  | For an example of conversion plugins, see :doc:`conversion`. | ||||||
|  | For a description of definition files, see :doc:`plugins-definition`. | ||||||
|  |  | ||||||
|  | A more step-by-step tutorial with slides is available `here <https://lab.cluster.gsi.dit.upm.es/senpy/senpy-tutorial>`__  | ||||||
|  |  | ||||||
|  | .. contents:: :local: | ||||||
|  |  | ||||||
|  | What is a plugin? | ||||||
|  | ================= | ||||||
|  |  | ||||||
|  | A plugin is a python object that can process entries. Given an entry, it will modify it, add annotations to it, or generate new entries. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | What is an entry? | ||||||
|  | ================= | ||||||
|  |  | ||||||
|  | Entries are objects that can be annotated. | ||||||
|  | In general, they will be a piece of text. | ||||||
|  | By default, entries are `NIF contexts <http://persistence.uni-leipzig.org/nlp2rdf/ontologies/nif-core/nif-core.html>`_ represented in JSON-LD format. | ||||||
|  | It is a dictionary/JSON object that looks like this: | ||||||
|  |  | ||||||
|  |   .. code:: python | ||||||
|  |  | ||||||
|  |             { | ||||||
|  |                "@id": "<unique identifier or blank node name>", | ||||||
|  |                "nif:isString": "input text", | ||||||
|  |                "sentiments": [ { | ||||||
|  |                      ... | ||||||
|  |                } | ||||||
|  |                ], | ||||||
|  |                ... | ||||||
|  |             } | ||||||
|  |  | ||||||
|  | Annotations are added to the object like this: | ||||||
|  |  | ||||||
|  | .. code:: python | ||||||
|  |  | ||||||
|  |    entry = Entry() | ||||||
|  |    entry.vocabulary__annotationName = 'myvalue' | ||||||
|  |    entry['vocabulary:annotationName'] = 'myvalue' | ||||||
|  |    entry['annotationNameURI'] = 'myvalue' | ||||||
|  |  | ||||||
|  | Where vocabulary is one of the prefixes defined in the default senpy context, and annotationURI is a full URI. | ||||||
|  | The value may be any valid JSON-LD dictionary. | ||||||
|  | For simplicity, senpy includes a series of models by default in the ``senpy.models`` module. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | What are annotations? | ||||||
|  | ===================== | ||||||
|  | They are objects just like entries. | ||||||
|  | Senpy ships with several default annotations, including: ``Sentiment``, ``Emotion``, ``EmotionSet``...jk bb | ||||||
|  |  | ||||||
|  |  | ||||||
|  | What's a plugin made of? | ||||||
|  | ======================== | ||||||
|  |  | ||||||
|  | When receiving a query, senpy selects what plugin or plugins should process each entry, and in what order. | ||||||
|  | It also makes sure the every entry and the parameters provided by the user meet the plugin requirements. | ||||||
|  |  | ||||||
|  | Hence, two parts are necessary: 1) the code that will process the entry, and 2) some attributes and metadata that will tell senpy how to interact with the plugin. | ||||||
|  |  | ||||||
|  | In practice, this is what a plugin looks like, tests included: | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .. literalinclude:: ../senpy/plugins/example/rand_plugin.py | ||||||
|  |    :emphasize-lines: 5-11 | ||||||
|  |    :language: python | ||||||
|  |  | ||||||
|  |  | ||||||
|  | The lines highlighted contain some information about the plugin. | ||||||
|  | In particular, the following information is mandatory: | ||||||
|  |  | ||||||
|  | * A unique name for the class. In our example, Rand. | ||||||
|  | * The subclass/type of plugin. This is typically either `SentimentPlugin` or `EmotionPlugin`. However, new types of plugin can be created for different annotations. The only requirement is that these new types inherit from `senpy.Analysis` | ||||||
|  | * A description of the plugin. This can be done simply by adding a doc to the class. | ||||||
|  | * A version, which should get updated. | ||||||
|  | * An author name. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Plugins Code | ||||||
|  | ============ | ||||||
|  |  | ||||||
|  | The basic methods in a plugin are: | ||||||
|  |  | ||||||
|  | * analyse_entry: called in every user requests. It takes two parameters: ``Entry``, the entry object, and ``params``, the parameters supplied by the user. It should yield one or more ``Entry`` objects. | ||||||
|  | * activate: used to load memory-hungry resources. For instance, to train a classifier. | ||||||
|  | * deactivate: used to free up resources when the plugin is no longer needed. | ||||||
|  |  | ||||||
|  | Plugins are loaded asynchronously, so don't worry if the activate method takes too long. The plugin will be marked as activated once it is finished executing the method. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | How does senpy find modules? | ||||||
|  | ============================ | ||||||
|  |  | ||||||
|  | Senpy looks for files of two types: | ||||||
|  |  | ||||||
|  | * Python files of the form `senpy_<NAME>.py` or `<NAME>_plugin.py`. In these files, it will look for: 1) Instances that inherit from `senpy.Plugin`, or subclasses of `senpy.Plugin` that can be initialized without a configuration file. i.e. classes that contain all the required attributes for a plugin. | ||||||
|  | * Plugin definition files (see :doc:`advanced-plugins`) | ||||||
|  |  | ||||||
|  | Defining additional parameters | ||||||
|  | ============================== | ||||||
|  |  | ||||||
|  | Your plugin may ask for additional parameters from the users of the service by using the attribute ``extra_params`` in your plugin definition. | ||||||
|  | It takes a dictionary, where the keys are the name of the argument/parameter, and the value has the following fields: | ||||||
|  |  | ||||||
|  | * aliases: the different names which can be used in the request to use the parameter. | ||||||
|  | * required: if set to true, users need to provide this parameter unless a default is set. | ||||||
|  | * options: the different acceptable values of the parameter (i.e. an enum). If set, the value provided must match one of the options. | ||||||
|  | * default: the default value of the parameter, if none is provided in the request. | ||||||
|  |  | ||||||
|  | .. code:: python | ||||||
|  |  | ||||||
|  |           "extra_params":{ | ||||||
|  |              "language": { | ||||||
|  |                 "aliases": ["language", "lang", "l"], | ||||||
|  |                 "required": True, | ||||||
|  |                 "options": ["es", "en"], | ||||||
|  |                 "default": "es" | ||||||
|  |                 } | ||||||
|  |              } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Loading data and files | ||||||
|  | ====================== | ||||||
|  |  | ||||||
|  | Most plugins will need access to files (dictionaries, lexicons, etc.). | ||||||
|  | These files are usually heavy or under a license that does not allow redistribution. | ||||||
|  | For this reason, senpy has a `data_folder` that is separated from the source files. | ||||||
|  | The location of this folder is controlled programmatically or by setting the `SENPY_DATA` environment variable. | ||||||
|  |  | ||||||
|  | Plugins have a convenience function `self.open` which will automatically prepend the data folder to relative paths: | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .. code:: python | ||||||
|  |  | ||||||
|  |           import os | ||||||
|  |  | ||||||
|  |  | ||||||
|  |           class PluginWithResources(AnalysisPlugin): | ||||||
|  |               file_in_data = <FILE PATH> | ||||||
|  |               file_in_sources = <FILE PATH> | ||||||
|  |  | ||||||
|  |               def activate(self): | ||||||
|  |                   with self.open(self.file_in_data) as f: | ||||||
|  |                       self._classifier = train_from_file(f) | ||||||
|  |                   file_in_source = os.path.join(self.get_folder(), self.file_in_sources) | ||||||
|  |                   with self.open(file_in_source) as f: | ||||||
|  |                       pass | ||||||
|  |  | ||||||
|  |           | ||||||
|  | It is good practice to specify the paths of these files in the plugin configuration, so the same code can be reused with different resources. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Docker image | ||||||
|  | ============ | ||||||
|  |  | ||||||
|  | Add the following dockerfile to your project to generate a docker image with your plugin: | ||||||
|  |  | ||||||
|  | .. code:: dockerfile | ||||||
|  |  | ||||||
|  |    FROM gsiupm/senpy | ||||||
|  |  | ||||||
|  | Once you make sure your plugin works with a specific version of senpy, modify that file to make sure your build will work even if senpy gets updated. | ||||||
|  | e.g.: | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .. code:: dockerfile | ||||||
|  |  | ||||||
|  |    FROM gsiupm/senpy:1.0.1 | ||||||
|  |  | ||||||
|  |       | ||||||
|  | This will copy your source folder to the image, and install all dependencies. | ||||||
|  | Now, to build an image: | ||||||
|  |  | ||||||
|  | .. code:: shell | ||||||
|  |  | ||||||
|  |    docker build . -t gsiupm/exampleplugin | ||||||
|  |  | ||||||
|  | And you can run it with: | ||||||
|  |  | ||||||
|  | .. code:: shell | ||||||
|  |  | ||||||
|  |    docker run -p 5000:5000 gsiupm/exampleplugin | ||||||
|  |  | ||||||
|  |  | ||||||
|  | If the plugin uses non-source files (:ref:`loading data and files`), the recommended way is to use `SENPY_DATA` folder. | ||||||
|  | Data can then be mounted in the container or added to the image. | ||||||
|  | The former is recommended for open source plugins with licensed resources, whereas the latter is the most convenient and can be used for private images. | ||||||
|  |  | ||||||
|  | Mounting data: | ||||||
|  |  | ||||||
|  | .. code:: bash | ||||||
|  |  | ||||||
|  |    docker run -v $PWD/data:/data gsiupm/exampleplugin | ||||||
|  |  | ||||||
|  | Adding data to the image: | ||||||
|  |  | ||||||
|  | .. code:: dockerfile | ||||||
|  |  | ||||||
|  |    FROM gsiupm/senpy:1.0.1 | ||||||
|  |    COPY data / | ||||||
|  |  | ||||||
|  | F.A.Q. | ||||||
|  | ====== | ||||||
|  | What annotations can I use? | ||||||
|  | ??????????????????????????? | ||||||
|  |  | ||||||
|  | You can add almost any annotation to an entry. | ||||||
|  | The most common use cases are covered in the :doc:`apischema`. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Why does the analyse function yield instead of return? | ||||||
|  | ?????????????????????????????????????????????????????? | ||||||
|  |  | ||||||
|  | This is so that plugins may add new entries to the response or filter some of them. | ||||||
|  | For instance, a chunker may split one entry into several. | ||||||
|  | On the other hand, a conversion plugin may leave out those entries that do not contain relevant information. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | If I'm using a classifier, where should I train it? | ||||||
|  | ??????????????????????????????????????????????????? | ||||||
|  |  | ||||||
|  | Training a classifier can be time time consuming. To avoid running the training unnecessarily, you can use ShelfMixin to store the classifier. For instance: | ||||||
|  |  | ||||||
|  | .. code:: python | ||||||
|  |  | ||||||
|  |           from senpy.plugins import ShelfMixin, AnalysisPlugin | ||||||
|  |  | ||||||
|  |           class MyPlugin(ShelfMixin, AnalysisPlugin): | ||||||
|  |               def train(self): | ||||||
|  |                   ''' Code to train the classifier | ||||||
|  |                   ''' | ||||||
|  |                   # Here goes the code | ||||||
|  |                   # ... | ||||||
|  |                   return classifier | ||||||
|  |  | ||||||
|  |               def activate(self): | ||||||
|  |                   if 'classifier' not in self.sh: | ||||||
|  |                       classifier = self.train() | ||||||
|  |                       self.sh['classifier'] = classifier | ||||||
|  |                   self.classifier = self.sh['classifier'] | ||||||
|  |                | ||||||
|  |               def deactivate(self): | ||||||
|  |                   self.close() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | By default the ShelfMixin creates a file based on the plugin name and stores it in that plugin's folder. | ||||||
|  | However, you can manually specify a 'shelf_file' in your .senpy file. | ||||||
|  |  | ||||||
|  | Shelves may get corrupted if the plugin exists unexpectedly. | ||||||
|  | A corrupt shelf prevents the plugin from loading. | ||||||
|  | If you do not care about the data in the shelf, you can force your plugin to remove the corrupted file and load anyway, set the  'force_shelf' to True in your plugin and start it again. | ||||||
|  |  | ||||||
|  | How can I turn an external service into a plugin? | ||||||
|  | ????????????????????????????????????????????????? | ||||||
|  |  | ||||||
|  | This example ilustrate how to implement a plugin that accesses the Sentiment140 service. | ||||||
|  |  | ||||||
|  | .. code:: python | ||||||
|  |  | ||||||
|  |           class Sentiment140Plugin(SentimentPlugin): | ||||||
|  |               def analyse_entry(self, entry, params): | ||||||
|  |                   text = entry.text | ||||||
|  |                   lang = params.get("language", "auto") | ||||||
|  |                   res = requests.post("http://www.sentiment140.com/api/bulkClassifyJson", | ||||||
|  |                                       json.dumps({"language": lang, | ||||||
|  |                                                   "data": [{"text": text}] | ||||||
|  |                                                   } | ||||||
|  |                                                  ) | ||||||
|  |                                       ) | ||||||
|  |  | ||||||
|  |                   p = params.get("prefix", None) | ||||||
|  |                   polarity_value = self.maxPolarityValue*int(res.json()["data"][0] | ||||||
|  |                                                              ["polarity"]) * 0.25 | ||||||
|  |                   polarity = "marl:Neutral" | ||||||
|  |                   neutral_value = self.maxPolarityValue / 2.0 | ||||||
|  |                   if polarity_value > neutral_value: | ||||||
|  |                       polarity = "marl:Positive" | ||||||
|  |                   elif polarity_value < neutral_value: | ||||||
|  |                       polarity = "marl:Negative" | ||||||
|  |  | ||||||
|  |                   sentiment = Sentiment(id="Sentiment0", | ||||||
|  |                                       prefix=p, | ||||||
|  |                                       marl__hasPolarity=polarity, | ||||||
|  |                                       marl__polarityValue=polarity_value) | ||||||
|  |                   sentiment.prov(self) | ||||||
|  |                   entry.sentiments.append(sentiment) | ||||||
|  |                   yield entry | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Can I activate a DEBUG mode for my plugin? | ||||||
|  | ??????????????????????????????????????????? | ||||||
|  |  | ||||||
|  | You can activate the DEBUG mode by the command-line tool using the option -d. | ||||||
|  |  | ||||||
|  | .. code:: bash | ||||||
|  |  | ||||||
|  |    senpy -d | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Additionally, with the ``--pdb`` option you will be dropped into a pdb post mortem shell if an exception is raised. | ||||||
|  |  | ||||||
|  | .. code:: bash | ||||||
|  |  | ||||||
|  |    python -m pdb yourplugin.py | ||||||
|  |  | ||||||
|  | Where can I find more code examples? | ||||||
|  | ???????????????????????????????????? | ||||||
|  |  | ||||||
|  | See: `<http://github.com/gsi-upm/senpy-plugins-community>`_. | ||||||
							
								
								
									
										2
									
								
								docs/requirements.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,2 @@ | |||||||
|  | sphinxcontrib-httpdomain>=1.4 | ||||||
|  | nbsphinx | ||||||
							
								
								
									
										
											BIN
										
									
								
								docs/senpy-architecture.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 122 KiB | 
							
								
								
									
										
											BIN
										
									
								
								docs/senpy-framework.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 48 KiB | 
							
								
								
									
										
											BIN
										
									
								
								docs/senpy-playground.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 49 KiB | 
							
								
								
									
										54
									
								
								docs/senpy.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,54 @@ | |||||||
|  | What is Senpy? | ||||||
|  | -------------- | ||||||
|  |  | ||||||
|  | Senpy is a framework for text analysis using Linked Data. There are three main applications of Senpy so far: sentiment and emotion analysis, user profiling and entity recoginition. Annotations and Services are compliant with NIF (NLP Interchange Format). | ||||||
|  |  | ||||||
|  | Senpy aims at providing a framework where analysis modules can be integrated easily as plugins, and providing a core functionality for managing tasks such as data validation, user interaction, formatting, logging, translation to linked data, etc.  | ||||||
|  |  | ||||||
|  | The figure below summarizes the typical features in a text analysis service. | ||||||
|  | Senpy implements all the common blocks, so developers can focus on what really matters: great analysis algorithms that solve real problems. | ||||||
|  |  | ||||||
|  | .. image:: senpy-framework.png | ||||||
|  |   :width: 60% | ||||||
|  |   :align: center | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Senpy for end users | ||||||
|  | =================== | ||||||
|  |  | ||||||
|  | All services built using senpy share a common interface. | ||||||
|  | This allows users to use them (almost) interchangeably. | ||||||
|  | Senpy comes with a :ref:`built-in client`. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Senpy for service developers | ||||||
|  | ============================ | ||||||
|  |  | ||||||
|  | Senpy is a framework that turns your sentiment or emotion analysis algorithm into a full blown semantic service. | ||||||
|  | Senpy takes care of: | ||||||
|  |  | ||||||
|  |   * Interfacing with the user: parameter validation, error handling. | ||||||
|  |   * Formatting: JSON-LD, Turtle/n-triples input and output, or simple text input | ||||||
|  |   * Linked Data: senpy results are semantically annotated, using a series of well established vocabularies, and sane default URIs. | ||||||
|  |   * User interface: a web UI where users can explore your service and test different settings | ||||||
|  |   * A client to interact with the service. Currently only available in Python. | ||||||
|  |  | ||||||
|  | Sharing your sentiment analysis with the world has never been easier! | ||||||
|  |  | ||||||
|  | Check out the :doc:`plugins` if you have developed an analysis algorithm (e.g. sentiment analysis) and you want to publish it as a service. | ||||||
|  |  | ||||||
|  | Architecture | ||||||
|  | ============ | ||||||
|  |  | ||||||
|  | The main component of a sentiment analysis service is the algorithm itself. However, for the algorithm to work, it needs to get the appropriate parameters from the user, format the results according to the defined API, interact with the user whn errors occur or more information is needed, etc. | ||||||
|  |  | ||||||
|  | Senpy proposes a modular and dynamic architecture that allows: | ||||||
|  |  | ||||||
|  | * Implementing different algorithms in a extensible way, yet offering a common interface. | ||||||
|  | * Offering common services that facilitate development, so developers can focus on implementing new and better algorithms. | ||||||
|  |  | ||||||
|  | The framework consists of two main modules: Senpy core, which is the building block of the service, and Senpy plugins, which consist of the analysis algorithm. The next figure depicts a simplified version of the processes involved in an analysis with the Senpy framework. | ||||||
|  |  | ||||||
|  | .. image:: senpy-architecture.png | ||||||
|  |   :width: 100% | ||||||
|  |   :align: center | ||||||
							
								
								
									
										66
									
								
								docs/server.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,66 @@ | |||||||
|  | Server | ||||||
|  | ====== | ||||||
|  |  | ||||||
|  | The senpy server is launched via the `senpy` command: | ||||||
|  |  | ||||||
|  | .. code:: text | ||||||
|  |  | ||||||
|  |     usage: senpy [-h] [--level logging_level] [--debug] [--default-plugins] | ||||||
|  |                 [--host HOST] [--port PORT] [--plugins-folder PLUGINS_FOLDER] | ||||||
|  |                 [--only-install] [--only-list] [--data-folder DATA_FOLDER] | ||||||
|  |                 [--threaded] [--version] | ||||||
|  |  | ||||||
|  |     Run a Senpy server | ||||||
|  |  | ||||||
|  |     optional arguments: | ||||||
|  |       -h, --help            show this help message and exit | ||||||
|  |       --level logging_level, -l logging_level | ||||||
|  |                             Logging level | ||||||
|  |       --debug, -d           Run the application in debug mode | ||||||
|  |       --default-plugins     Load the default plugins | ||||||
|  |       --host HOST           Use 0.0.0.0 to accept requests from any host. | ||||||
|  |       --port PORT, -p PORT  Port to listen on. | ||||||
|  |       --plugins-folder PLUGINS_FOLDER, -f PLUGINS_FOLDER | ||||||
|  |                             Where to look for plugins. | ||||||
|  |       --only-install, -i    Do not run a server, only install plugin dependencies | ||||||
|  |       --only-list, --list   Do not run a server, only list plugins found | ||||||
|  |       --data-folder DATA_FOLDER, --data DATA_FOLDER | ||||||
|  |                             Where to look for data. It be set with the SENPY_DATA | ||||||
|  |                             environment variable as well. | ||||||
|  |       --threaded            Run a threaded server | ||||||
|  |       --version, -v         Output the senpy version and exit | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | When launched, the server will recursively look for plugins in the specified plugins folder (the current working directory by default). | ||||||
|  | For every plugin found, it will download its dependencies, and try to activate it. | ||||||
|  | The default server includes a playground and an endpoint with all plugins found. | ||||||
|  |  | ||||||
|  | Let's run senpy with the default plugins: | ||||||
|  |  | ||||||
|  | .. code:: bash | ||||||
|  |  | ||||||
|  |     senpy -f . --default-plugins | ||||||
|  |  | ||||||
|  | Now go to `http://localhost:5000 <http://localhost:5000>`_, you should be greeted by the senpy playground: | ||||||
|  |  | ||||||
|  | .. image:: senpy-playground.png | ||||||
|  |    :width: 100% | ||||||
|  |    :alt: Playground | ||||||
|  |  | ||||||
|  | The playground is a user-friendly way to test your plugins, but you can always use the service directly:  `http://localhost:5000/api?input=hello <http://localhost:5000/api?input=hello>`_. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | By default, senpy will listen only on the `127.0.0.1` address. | ||||||
|  | That means you can only access the API from your (or localhost). | ||||||
|  | You can listen on a different address using the `--host` flag (e.g., 0.0.0.0). | ||||||
|  | The default port is 5000. | ||||||
|  | You can change it with the `--port` flag.  | ||||||
|  |  | ||||||
|  | For instance, to accept connections on port 6000 on any interface: | ||||||
|  |  | ||||||
|  | .. code:: bash | ||||||
|  |  | ||||||
|  |     senpy --host 0.0.0.0 --port 6000 | ||||||
|  |  | ||||||
|  | For more options, see the `--help` page. | ||||||
							
								
								
									
										15
									
								
								docs/usage.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,15 @@ | |||||||
|  | Usage | ||||||
|  | ----- | ||||||
|  |  | ||||||
|  | First of all, you need to install the package. | ||||||
|  | See :doc:`installation` for instructions. | ||||||
|  | Once installed, the `senpy` command should be available.  | ||||||
|  |  | ||||||
|  | .. toctree:: | ||||||
|  |    :maxdepth: 1 | ||||||
|  |  | ||||||
|  |    server | ||||||
|  |    SenpyClientUse | ||||||
|  |    commandline | ||||||
|  |  | ||||||
|  |  | ||||||
							
								
								
									
										24
									
								
								docs/vocabularies.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,24 @@ | |||||||
|  | Vocabularies and model | ||||||
|  | ====================== | ||||||
|  |  | ||||||
|  | The model used in Senpy is based on NIF 2.0 [1], which defines a semantic format and API for improving interoperability among natural language processing services. | ||||||
|  |  | ||||||
|  | Senpy has been applied to sentiment and emotion analysis services using the following vocabularies: | ||||||
|  |  | ||||||
|  | * Marl [2,6], a vocabulary designed to annotate and describe subjetive opinions expressed on the web or in information systems. | ||||||
|  | * Onyx [3,5], which is built one the same principles as Marl to annotate and describe emotions, and provides interoperability with Emotion Markup Language. | ||||||
|  |  | ||||||
|  | An overview of the vocabularies and their use can be found in [4]. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | [1] Guidelines for developing NIF-based NLP services, Final Community Group Report 22 December 2015 Available at: https://www.w3.org/2015/09/bpmlod-reports/nif-based-nlp-webservices/ | ||||||
|  |  | ||||||
|  | [2] Marl Ontology Specification, available at http://www.gsi.dit.upm.es/ontologies/marl/ | ||||||
|  |  | ||||||
|  | [3] Onyx Ontology Specification, available at http://www.gsi.dit.upm.es/ontologies/onyx/ | ||||||
|  |  | ||||||
|  | [4] Iglesias, C. A., Sánchez-Rada, J. F., Vulcu, G., & Buitelaar, P. (2017). Linked Data Models for Sentiment and Emotion Analysis in Social Networks. In Sentiment Analysis in Social Networks (pp. 49-69). | ||||||
|  |  | ||||||
|  | [5] Sánchez-Rada, J. F., & Iglesias, C. A. (2016). Onyx: A linked data approach to emotion representation. Information Processing & Management, 52(1), 99-114. | ||||||
|  |  | ||||||
|  | [6] Westerski, A., Iglesias Fernandez, C. A., & Tapia Rico, F. (2011). Linked opinions: Describing sentiments on the structured web of data. | ||||||
							
								
								
									
										23
									
								
								example-plugins/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,23 @@ | |||||||
|  | This is a collection of plugins that exemplify certain aspects of plugin development with senpy. | ||||||
|  |  | ||||||
|  | The first series of plugins the `basic` ones. | ||||||
|  | Their starting point is a classification function defined in `basic.py`. | ||||||
|  | They all include testing and running them as a script will run all tests. | ||||||
|  | In ascending order of customization, the plugins are: | ||||||
|  |  | ||||||
|  | * Basic is the simplest plugin of all. It leverages the `SentimentBox` Plugin class to create a plugin out of a classification method, and `MappingMixin` to convert the labels from (`pos`, `neg`) to (`marl:Positive`, `marl:Negative` | ||||||
|  | * Basic_box is just like the previous one, but replaces the mixin with a custom function. | ||||||
|  | * Basic_configurable is a version of `basic` with a configurable map of emojis for each sentiment. | ||||||
|  | * Basic_parameterized like `basic_info`, but users set the map in each query (via `extra_parameters`). | ||||||
|  | * Basic_analyse\_entry uses the more general `analyse_entry` method and adds the annotations individually. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | In rest of the plugins show advanced topics: | ||||||
|  |  | ||||||
|  | * mynoop: shows how to add a definition file with external requirements for a plugin. Doing this with a python-only module would require moving all imports of the requirements to their functions, which is considered bad practice. | ||||||
|  | * Async: a barebones example of training a plugin and analyzing data in parallel. | ||||||
|  |  | ||||||
|  | All of the plugins in this folder include a set of test cases and they are periodically tested with the latest version of senpy. | ||||||
|  |  | ||||||
|  | Additioanlly, for an example of stand-alone plugin that can be tested and deployed with docker, take a look at: lab.cluster.gsi.dit.upm.es/senpy/plugin-example | ||||||
|  |  bbm | ||||||
							
								
								
									
										37
									
								
								example-plugins/async_plugin.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,37 @@ | |||||||
|  | from senpy import AnalysisPlugin | ||||||
|  |  | ||||||
|  | import multiprocessing | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def _train(process_number): | ||||||
|  |     return process_number | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Async(AnalysisPlugin): | ||||||
|  |     '''An example of an asynchronous module''' | ||||||
|  |     author = '@balkian' | ||||||
|  |     version = '0.2' | ||||||
|  |     sync = False | ||||||
|  |  | ||||||
|  |     def _do_async(self, num_processes): | ||||||
|  |         pool = multiprocessing.Pool(processes=num_processes) | ||||||
|  |         values = sorted(pool.map(_train, range(num_processes))) | ||||||
|  |  | ||||||
|  |         return values | ||||||
|  |  | ||||||
|  |     def activate(self): | ||||||
|  |         self.value = self._do_async(4) | ||||||
|  |  | ||||||
|  |     def analyse_entry(self, entry, params): | ||||||
|  |         values = self._do_async(2) | ||||||
|  |         entry.async_values = values | ||||||
|  |         yield entry | ||||||
|  |  | ||||||
|  |     test_cases = [ | ||||||
|  |         { | ||||||
|  |             'input': 'any', | ||||||
|  |             'expected': { | ||||||
|  |                 'async_values': [0, 1] | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     ] | ||||||
							
								
								
									
										23
									
								
								example-plugins/basic.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,23 @@ | |||||||
|  | #!/usr/local/bin/python | ||||||
|  | # coding: utf-8 | ||||||
|  |  | ||||||
|  | emoticons = { | ||||||
|  |     'pos': [':)', ':]', '=)', ':D'], | ||||||
|  |     'neg': [':(', ':[', '=('] | ||||||
|  | } | ||||||
|  |  | ||||||
|  | emojis = { | ||||||
|  |     'pos': ['😁', '😂', '😃', '😄', '😆', '😅', '😄' '😍'], | ||||||
|  |     'neg': ['😢', '😡', '😠', '😞', '😖', '😔', '😓', '😒'] | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def get_polarity(text, dictionaries=[emoticons, emojis]): | ||||||
|  |     polarity = 'marl:Neutral' | ||||||
|  |     for dictionary in dictionaries: | ||||||
|  |         for label, values in dictionary.items(): | ||||||
|  |             for emoticon in values: | ||||||
|  |                 if emoticon and emoticon in text: | ||||||
|  |                     polarity = label | ||||||
|  |                     break | ||||||
|  |     return polarity | ||||||
							
								
								
									
										47
									
								
								example-plugins/basic_analyse_entry_plugin.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,47 @@ | |||||||
|  | #!/usr/local/bin/python | ||||||
|  | # coding: utf-8 | ||||||
|  |  | ||||||
|  | from senpy import easy_test, models, plugins | ||||||
|  |  | ||||||
|  | import basic | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class BasicAnalyseEntry(plugins.SentimentPlugin): | ||||||
|  |     '''Equivalent to Basic, implementing the analyse_entry method''' | ||||||
|  |  | ||||||
|  |     author = '@balkian' | ||||||
|  |     version = '0.1' | ||||||
|  |  | ||||||
|  |     mappings = { | ||||||
|  |         'pos': 'marl:Positive', | ||||||
|  |         'neg': 'marl:Negative', | ||||||
|  |         'default': 'marl:Neutral' | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     def analyse_entry(self, entry, params): | ||||||
|  |         polarity = basic.get_polarity(entry.text) | ||||||
|  |  | ||||||
|  |         polarity = self.mappings.get(polarity, self.mappings['default']) | ||||||
|  |  | ||||||
|  |         s = models.Sentiment(marl__hasPolarity=polarity) | ||||||
|  |         s.prov(self) | ||||||
|  |         entry.sentiments.append(s) | ||||||
|  |         yield entry | ||||||
|  |  | ||||||
|  |     test_cases = [{ | ||||||
|  |         'input': 'Hello :)', | ||||||
|  |         'polarity': 'marl:Positive' | ||||||
|  |     }, { | ||||||
|  |         'input': 'So sad :(', | ||||||
|  |         'polarity': 'marl:Negative' | ||||||
|  |     }, { | ||||||
|  |         'input': 'Yay! Emojis  😁', | ||||||
|  |         'polarity': 'marl:Positive' | ||||||
|  |     }, { | ||||||
|  |         'input': 'But no emoticons 😢', | ||||||
|  |         'polarity': 'marl:Negative' | ||||||
|  |     }] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | if __name__ == '__main__': | ||||||
|  |     easy_test() | ||||||
							
								
								
									
										41
									
								
								example-plugins/basic_box_plugin.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,41 @@ | |||||||
|  | #!/usr/local/bin/python | ||||||
|  | # coding: utf-8 | ||||||
|  |  | ||||||
|  | from senpy import easy_test, SentimentBox | ||||||
|  |  | ||||||
|  | import basic | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class BasicBox(SentimentBox): | ||||||
|  |     ''' A modified version of Basic that also does converts annotations manually''' | ||||||
|  |  | ||||||
|  |     author = '@balkian' | ||||||
|  |     version = '0.1' | ||||||
|  |  | ||||||
|  |     mappings = { | ||||||
|  |         'pos': 'marl:Positive', | ||||||
|  |         'neg': 'marl:Negative', | ||||||
|  |         'default': 'marl:Neutral' | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     def predict_one(self, input): | ||||||
|  |         output = basic.get_polarity(input) | ||||||
|  |         return self.mappings.get(output, self.mappings['default']) | ||||||
|  |  | ||||||
|  |     test_cases = [{ | ||||||
|  |         'input': 'Hello :)', | ||||||
|  |         'polarity': 'marl:Positive' | ||||||
|  |     }, { | ||||||
|  |         'input': 'So sad :(', | ||||||
|  |         'polarity': 'marl:Negative' | ||||||
|  |     }, { | ||||||
|  |         'input': 'Yay! Emojis  😁', | ||||||
|  |         'polarity': 'marl:Positive' | ||||||
|  |     }, { | ||||||
|  |         'input': 'But no emoticons 😢', | ||||||
|  |         'polarity': 'marl:Negative' | ||||||
|  |     }] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | if __name__ == '__main__': | ||||||
|  |     easy_test() | ||||||
							
								
								
									
										40
									
								
								example-plugins/basic_plugin.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,40 @@ | |||||||
|  | #!/usr/local/bin/python | ||||||
|  | # coding: utf-8 | ||||||
|  |  | ||||||
|  | from senpy import easy_test, SentimentBox, MappingMixin | ||||||
|  |  | ||||||
|  | import basic | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Basic(MappingMixin, SentimentBox): | ||||||
|  |     '''Provides sentiment annotation using a lexicon''' | ||||||
|  |  | ||||||
|  |     author = '@balkian' | ||||||
|  |     version = '0.1' | ||||||
|  |  | ||||||
|  |     mappings = { | ||||||
|  |         'pos': 'marl:Positive', | ||||||
|  |         'neg': 'marl:Negative', | ||||||
|  |         'default': 'marl:Neutral' | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     def predict_one(self, input): | ||||||
|  |         return basic.get_polarity(input) | ||||||
|  |  | ||||||
|  |     test_cases = [{ | ||||||
|  |         'input': 'Hello :)', | ||||||
|  |         'polarity': 'marl:Positive' | ||||||
|  |     }, { | ||||||
|  |         'input': 'So sad :(', | ||||||
|  |         'polarity': 'marl:Negative' | ||||||
|  |     }, { | ||||||
|  |         'input': 'Yay! Emojis  😁', | ||||||
|  |         'polarity': 'marl:Positive' | ||||||
|  |     }, { | ||||||
|  |         'input': 'But no emoticons 😢', | ||||||
|  |         'polarity': 'marl:Negative' | ||||||
|  |     }] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | if __name__ == '__main__': | ||||||
|  |     easy_test() | ||||||
							
								
								
									
										105
									
								
								example-plugins/configurable_plugin.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,105 @@ | |||||||
|  | #!/usr/local/bin/python | ||||||
|  | # coding: utf-8 | ||||||
|  |  | ||||||
|  | from senpy import easy_test, models, plugins | ||||||
|  |  | ||||||
|  | import basic | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Dictionary(plugins.SentimentPlugin): | ||||||
|  |     '''Sentiment annotation using a configurable lexicon''' | ||||||
|  |  | ||||||
|  |     author = '@balkian' | ||||||
|  |     version = '0.2' | ||||||
|  |  | ||||||
|  |     dictionaries = [basic.emojis, basic.emoticons] | ||||||
|  |  | ||||||
|  |     mappings = {'pos': 'marl:Positive', 'neg': 'marl:Negative'} | ||||||
|  |  | ||||||
|  |     def analyse_entry(self, entry, params): | ||||||
|  |         polarity = basic.get_polarity(entry.text, self.dictionaries) | ||||||
|  |         if polarity in self.mappings: | ||||||
|  |             polarity = self.mappings[polarity] | ||||||
|  |  | ||||||
|  |         s = models.Sentiment(marl__hasPolarity=polarity) | ||||||
|  |         s.prov(self) | ||||||
|  |         entry.sentiments.append(s) | ||||||
|  |         yield entry | ||||||
|  |  | ||||||
|  |     test_cases = [{ | ||||||
|  |         'input': 'Hello :)', | ||||||
|  |         'polarity': 'marl:Positive' | ||||||
|  |     }, { | ||||||
|  |         'input': 'So sad :(', | ||||||
|  |         'polarity': 'marl:Negative' | ||||||
|  |     }, { | ||||||
|  |         'input': 'Yay! Emojis  😁', | ||||||
|  |         'polarity': 'marl:Positive' | ||||||
|  |     }, { | ||||||
|  |         'input': 'But no emoticons 😢', | ||||||
|  |         'polarity': 'marl:Negative' | ||||||
|  |     }] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class EmojiOnly(Dictionary): | ||||||
|  |     '''Sentiment annotation with a basic lexicon of emojis''' | ||||||
|  |     dictionaries = [basic.emojis] | ||||||
|  |  | ||||||
|  |     test_cases = [{ | ||||||
|  |         'input': 'Hello :)', | ||||||
|  |         'polarity': 'marl:Neutral' | ||||||
|  |     }, { | ||||||
|  |         'input': 'So sad :(', | ||||||
|  |         'polarity': 'marl:Neutral' | ||||||
|  |     }, { | ||||||
|  |         'input': 'Yay! Emojis  😁', | ||||||
|  |         'polarity': 'marl:Positive' | ||||||
|  |     }, { | ||||||
|  |         'input': 'But no emoticons 😢', | ||||||
|  |         'polarity': 'marl:Negative' | ||||||
|  |     }] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class EmoticonsOnly(Dictionary): | ||||||
|  |     '''Sentiment annotation with a basic lexicon of emoticons''' | ||||||
|  |     dictionaries = [basic.emoticons] | ||||||
|  |  | ||||||
|  |     test_cases = [{ | ||||||
|  |         'input': 'Hello :)', | ||||||
|  |         'polarity': 'marl:Positive' | ||||||
|  |     }, { | ||||||
|  |         'input': 'So sad :(', | ||||||
|  |         'polarity': 'marl:Negative' | ||||||
|  |     }, { | ||||||
|  |         'input': 'Yay! Emojis  😁', | ||||||
|  |         'polarity': 'marl:Neutral' | ||||||
|  |     }, { | ||||||
|  |         'input': 'But no emoticons 😢', | ||||||
|  |         'polarity': 'marl:Neutral' | ||||||
|  |     }] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Salutes(Dictionary): | ||||||
|  |     '''Sentiment annotation with a custom lexicon, for illustration purposes''' | ||||||
|  |     dictionaries = [{ | ||||||
|  |         'marl:Positive': ['Hello', '!'], | ||||||
|  |         'marl:Negative': ['Good bye', ] | ||||||
|  |     }] | ||||||
|  |  | ||||||
|  |     test_cases = [{ | ||||||
|  |         'input': 'Hello :)', | ||||||
|  |         'polarity': 'marl:Positive' | ||||||
|  |     }, { | ||||||
|  |         'input': 'Good bye :(', | ||||||
|  |         'polarity': 'marl:Negative' | ||||||
|  |     }, { | ||||||
|  |         'input': 'Yay! Emojis  😁', | ||||||
|  |         'polarity': 'marl:Positive' | ||||||
|  |     }, { | ||||||
|  |         'input': 'But no emoticons 😢', | ||||||
|  |         'polarity': 'marl:Neutral' | ||||||
|  |     }] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | if __name__ == '__main__': | ||||||
|  |     easy_test() | ||||||
							
								
								
									
										25
									
								
								example-plugins/dummy_plugin.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,25 @@ | |||||||
|  | from senpy import AnalysisPlugin, easy | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Dummy(AnalysisPlugin): | ||||||
|  |     '''This is a dummy self-contained plugin''' | ||||||
|  |     author = '@balkian' | ||||||
|  |     version = '0.1' | ||||||
|  |  | ||||||
|  |     def analyse_entry(self, entry, params): | ||||||
|  |         entry['nif:isString'] = entry['nif:isString'][::-1] | ||||||
|  |         entry.reversed = entry.get('reversed', 0) + 1 | ||||||
|  |         yield entry | ||||||
|  |  | ||||||
|  |     test_cases = [{ | ||||||
|  |         'entry': { | ||||||
|  |             'nif:isString': 'Hello', | ||||||
|  |         }, | ||||||
|  |         'expected': { | ||||||
|  |             'nif:isString': 'olleH' | ||||||
|  |         } | ||||||
|  |     }] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | if __name__ == '__main__': | ||||||
|  |     easy() | ||||||
							
								
								
									
										40
									
								
								example-plugins/dummy_required_plugin.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,40 @@ | |||||||
|  | from senpy import AnalysisPlugin, easy | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class DummyRequired(AnalysisPlugin): | ||||||
|  |     '''This is a dummy self-contained plugin''' | ||||||
|  |     author = '@balkian' | ||||||
|  |     version = '0.1' | ||||||
|  |     extra_params = { | ||||||
|  |         'example': { | ||||||
|  |             'description': 'An example parameter', | ||||||
|  |             'required': True, | ||||||
|  |             'options': ['a', 'b'] | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     def analyse_entry(self, entry, params): | ||||||
|  |         entry['nif:isString'] = entry['nif:isString'][::-1] | ||||||
|  |         entry.reversed = entry.get('reversed', 0) + 1 | ||||||
|  |         yield entry | ||||||
|  |  | ||||||
|  |     test_cases = [{ | ||||||
|  |         'entry': { | ||||||
|  |             'nif:isString': 'Hello', | ||||||
|  |         }, | ||||||
|  |         'should_fail': True | ||||||
|  |     }, { | ||||||
|  |         'entry': { | ||||||
|  |             'nif:isString': 'Hello', | ||||||
|  |         }, | ||||||
|  |         'params': { | ||||||
|  |             'example': 'a' | ||||||
|  |         }, | ||||||
|  |         'expected': { | ||||||
|  |             'nif:isString': 'olleH' | ||||||
|  |         } | ||||||
|  |     }] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | if __name__ == '__main__': | ||||||
|  |     easy() | ||||||
							
								
								
									
										24
									
								
								example-plugins/mynoop.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,24 @@ | |||||||
|  | import noop | ||||||
|  | from senpy.plugins import SentimentPlugin | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class NoOp(SentimentPlugin): | ||||||
|  |     '''This plugin does nothing. Literally nothing.''' | ||||||
|  |  | ||||||
|  |     version = 0 | ||||||
|  |  | ||||||
|  |     def analyse_entry(self, entry, *args, **kwargs): | ||||||
|  |         yield entry | ||||||
|  |  | ||||||
|  |     def test(self): | ||||||
|  |         print(dir(noop)) | ||||||
|  |         super(NoOp, self).test() | ||||||
|  |  | ||||||
|  |     test_cases = [{ | ||||||
|  |         'entry': { | ||||||
|  |             'nif:isString': 'hello' | ||||||
|  |         }, | ||||||
|  |         'expected': { | ||||||
|  |             'nif:isString': 'hello' | ||||||
|  |         } | ||||||
|  |     }] | ||||||
							
								
								
									
										3
									
								
								example-plugins/mynoop.senpy
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | |||||||
|  | module: mynoop | ||||||
|  | requirements: | ||||||
|  |  - noop | ||||||
							
								
								
									
										63
									
								
								example-plugins/parameterized_plugin.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,63 @@ | |||||||
|  | #!/usr/local/bin/python | ||||||
|  | # coding: utf-8 | ||||||
|  |  | ||||||
|  | from senpy import easy_test, models, plugins | ||||||
|  |  | ||||||
|  | import basic | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class ParameterizedDictionary(plugins.SentimentPlugin): | ||||||
|  |     '''This is a basic self-contained plugin''' | ||||||
|  |  | ||||||
|  |     author = '@balkian' | ||||||
|  |     version = '0.2' | ||||||
|  |  | ||||||
|  |     extra_params = { | ||||||
|  |         'positive-words': { | ||||||
|  |             'description': 'Comma-separated list of words that are considered positive', | ||||||
|  |             'aliases': ['positive'], | ||||||
|  |             'required': True | ||||||
|  |         }, | ||||||
|  |         'negative-words': { | ||||||
|  |             'description': 'Comma-separated list of words that are considered negative', | ||||||
|  |             'aliases': ['negative'], | ||||||
|  |             'required': False | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     def analyse_entry(self, entry, params): | ||||||
|  |         positive_words = params['positive-words'].split(',') | ||||||
|  |         negative_words = params['negative-words'].split(',') | ||||||
|  |         dictionary = { | ||||||
|  |             'marl:Positive': positive_words, | ||||||
|  |             'marl:Negative': negative_words, | ||||||
|  |         } | ||||||
|  |         polarity = basic.get_polarity(entry.text, [dictionary]) | ||||||
|  |  | ||||||
|  |         s = models.Sentiment(marl__hasPolarity=polarity) | ||||||
|  |         s.prov(self) | ||||||
|  |         entry.sentiments.append(s) | ||||||
|  |         yield entry | ||||||
|  |  | ||||||
|  |     test_cases = [ | ||||||
|  |         { | ||||||
|  |             'input': 'Hello :)', | ||||||
|  |             'polarity': 'marl:Positive', | ||||||
|  |             'parameters': { | ||||||
|  |                 'positive': "Hello,:)", | ||||||
|  |                 'negative': "sad,:()" | ||||||
|  |             } | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |             'input': 'Hello :)', | ||||||
|  |             'polarity': 'marl:Negative', | ||||||
|  |             'parameters': { | ||||||
|  |                 'positive': "", | ||||||
|  |                 'negative': "Hello" | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     ] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | if __name__ == '__main__': | ||||||
|  |     easy_test() | ||||||
							
								
								
									
										33
									
								
								example-plugins/sklearn/mydata.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,33 @@ | |||||||
|  | ''' | ||||||
|  | Create a dummy dataset. | ||||||
|  | Messages with a happy emoticon are labelled positive | ||||||
|  | Messages with a sad emoticon are labelled negative | ||||||
|  | ''' | ||||||
|  | import random | ||||||
|  |  | ||||||
|  | dataset = [] | ||||||
|  |  | ||||||
|  | vocabulary = ['hello', 'world', 'senpy', 'cool', 'goodbye', 'random', 'text'] | ||||||
|  |  | ||||||
|  | emojimap = { | ||||||
|  |     1: [':)', ], | ||||||
|  |     -1: [':(', ] | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | for tag, values in emojimap.items(): | ||||||
|  |     for i in range(1000): | ||||||
|  |         msg = '' | ||||||
|  |         for j in range(3): | ||||||
|  |             msg += random.choice(vocabulary) | ||||||
|  |             msg += " " | ||||||
|  |         msg += random.choice(values) | ||||||
|  |         dataset.append([msg, tag]) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | text = [] | ||||||
|  | labels = [] | ||||||
|  |  | ||||||
|  | for i in dataset: | ||||||
|  |     text.append(i[0]) | ||||||
|  |     labels.append(i[1]) | ||||||
							
								
								
									
										30
									
								
								example-plugins/sklearn/mypipeline.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,30 @@ | |||||||
|  | from sklearn.pipeline import Pipeline | ||||||
|  | from sklearn.feature_extraction.text import CountVectorizer | ||||||
|  | from sklearn.model_selection import train_test_split | ||||||
|  |  | ||||||
|  | from mydata import text, labels | ||||||
|  |  | ||||||
|  | X_train, X_test, y_train, y_test = train_test_split(text, labels, test_size=0.12, random_state=42) | ||||||
|  |  | ||||||
|  | from sklearn.naive_bayes import MultinomialNB | ||||||
|  |  | ||||||
|  |  | ||||||
|  | count_vec = CountVectorizer(tokenizer=lambda x: x.split()) | ||||||
|  | clf3 = MultinomialNB() | ||||||
|  | pipeline = Pipeline([('cv', count_vec), | ||||||
|  |                     ('clf', clf3)]) | ||||||
|  |  | ||||||
|  | pipeline.fit(X_train, y_train) | ||||||
|  | print('Feature names: {}'.format(count_vec.get_feature_names())) | ||||||
|  | print('Class count: {}'.format(clf3.class_count_)) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | if __name__ == '__main__': | ||||||
|  |     print('--Results--') | ||||||
|  |     tests = [ | ||||||
|  |         (['The sentiment for senpy should be positive :)', ], 1), | ||||||
|  |         (['The sentiment for anything else should be negative :()', ], -1) | ||||||
|  |     ] | ||||||
|  |     for features, expected in tests: | ||||||
|  |         result = pipeline.predict(features) | ||||||
|  |         print('Input: {}\nExpected: {}\nGot: {}'.format(features[0], expected, result)) | ||||||
							
								
								
									
										37
									
								
								example-plugins/sklearn/pipeline_plugin.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,37 @@ | |||||||
|  | from senpy import SentimentBox, MappingMixin, easy_test | ||||||
|  |  | ||||||
|  | from mypipeline import pipeline | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class PipelineSentiment(MappingMixin, SentimentBox): | ||||||
|  |     ''' | ||||||
|  |     This is a pipeline plugin that wraps a classifier defined in another module | ||||||
|  |     (mypipeline). | ||||||
|  |     ''' | ||||||
|  |     author = '@balkian' | ||||||
|  |     version = 0.1 | ||||||
|  |     maxPolarityValue = 1 | ||||||
|  |     minPolarityValue = -1 | ||||||
|  |  | ||||||
|  |     mappings = { | ||||||
|  |         1: 'marl:Positive', | ||||||
|  |         -1: 'marl:Negative' | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     def predict_one(self, input): | ||||||
|  |         return pipeline.predict([input, ])[0] | ||||||
|  |  | ||||||
|  |     test_cases = [ | ||||||
|  |         { | ||||||
|  |             'input': 'The sentiment for senpy should be positive :)', | ||||||
|  |             'polarity': 'marl:Positive' | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |             'input': 'The sentiment for senpy should be negative :(', | ||||||
|  |             'polarity': 'marl:Negative' | ||||||
|  |         } | ||||||
|  |     ] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | if __name__ == '__main__': | ||||||
|  |     easy_test() | ||||||
							
								
								
									
										27
									
								
								example-plugins/sleep_plugin.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,27 @@ | |||||||
|  | from senpy.plugins import AnalysisPlugin | ||||||
|  | from time import sleep | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Sleep(AnalysisPlugin): | ||||||
|  |     '''Dummy plugin to test async''' | ||||||
|  |     author = "@balkian" | ||||||
|  |     version = "0.2" | ||||||
|  |     timeout = 0.05 | ||||||
|  |     extra_params = { | ||||||
|  |         "timeout": { | ||||||
|  |             "@id": "timeout_sleep", | ||||||
|  |             "aliases": ["timeout", "to"], | ||||||
|  |             "required": False, | ||||||
|  |             "default": 0 | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     def activate(self, *args, **kwargs): | ||||||
|  |         sleep(self.timeout) | ||||||
|  |  | ||||||
|  |     def analyse_entry(self, entry, params): | ||||||
|  |         sleep(float(params.get("timeout", self.timeout))) | ||||||
|  |         yield entry | ||||||
|  |  | ||||||
|  |     def test(self): | ||||||
|  |         pass | ||||||
							
								
								
									
										1
									
								
								extra-requirements.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | |||||||
|  | gsitk | ||||||
							
								
								
									
										
											BIN
										
									
								
								img/eu-flag.jpg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 5.6 KiB | 
							
								
								
									
										828
									
								
								img/final-logo.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 81 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/gsi.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 5.8 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/header.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 208 KiB | 
| Before Width: | Height: | Size: 8.0 KiB After Width: | Height: | Size: 8.0 KiB | 
							
								
								
									
										2728
									
								
								img/logo.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 180 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/logo_grande.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 42 KiB | 
							
								
								
									
										
											BIN
										
									
								
								img/me.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 25 KiB | 
							
								
								
									
										7
									
								
								k8s/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,7 @@ | |||||||
|  | Deploy senpy to a kubernetes cluster. | ||||||
|  |  | ||||||
|  | Usage: | ||||||
|  |  | ||||||
|  | ``` | ||||||
|  | kubectl apply -f . -n senpy | ||||||
|  | ``` | ||||||
							
								
								
									
										26
									
								
								k8s/senpy-deployment.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,26 @@ | |||||||
|  | --- | ||||||
|  | apiVersion: extensions/v1beta1 | ||||||
|  | kind: Deployment | ||||||
|  | metadata: | ||||||
|  |   name: senpy-latest | ||||||
|  | spec: | ||||||
|  |   replicas: 1 | ||||||
|  |   template: | ||||||
|  |     metadata: | ||||||
|  |       labels: | ||||||
|  |         role: senpy-latest | ||||||
|  |         app: test | ||||||
|  |     spec: | ||||||
|  |       containers: | ||||||
|  |       - name: senpy-latest | ||||||
|  |         image: $IMAGEWTAG | ||||||
|  |         imagePullPolicy: Always | ||||||
|  |         args: | ||||||
|  |           - "--default-plugins" | ||||||
|  |         resources: | ||||||
|  |           limits: | ||||||
|  |             memory: "512Mi" | ||||||
|  |             cpu: "1000m" | ||||||
|  |         ports: | ||||||
|  |           - name: web | ||||||
|  |             containerPort: 5000 | ||||||
							
								
								
									
										14
									
								
								k8s/senpy-ingress.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,14 @@ | |||||||
|  | --- | ||||||
|  | apiVersion: extensions/v1beta1 | ||||||
|  | kind: Ingress | ||||||
|  | metadata: | ||||||
|  |   name: senpy-ingress | ||||||
|  | spec: | ||||||
|  |   rules: | ||||||
|  |   - host: latest.senpy.cluster.gsi.dit.upm.es | ||||||
|  |     http: | ||||||
|  |       paths: | ||||||
|  |       - path: / | ||||||
|  |         backend: | ||||||
|  |           serviceName: senpy-latest | ||||||
|  |           servicePort: 5000 | ||||||
							
								
								
									
										12
									
								
								k8s/senpy-svc.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,12 @@ | |||||||
|  | --- | ||||||
|  | apiVersion: v1 | ||||||
|  | kind: Service | ||||||
|  | metadata: | ||||||
|  |   name: senpy-latest | ||||||
|  | spec: | ||||||
|  |   type: ClusterIP | ||||||
|  |   ports: | ||||||
|  |     - port: 5000 | ||||||
|  |       protocol: TCP | ||||||
|  |   selector: | ||||||
|  |     role: senpy-latest | ||||||
| @@ -1,45 +0,0 @@ | |||||||
| import requests |  | ||||||
| import json |  | ||||||
|  |  | ||||||
| from senpy.plugins import SentimentPlugin |  | ||||||
| from senpy.models import Response, Opinion, Entry |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Sentiment140Plugin(SentimentPlugin): |  | ||||||
|     EXTRA_PARAMS = { |  | ||||||
|         "language": {"aliases": ["language", "l"], |  | ||||||
|                      "required": False, |  | ||||||
|                      "options": ["es", "en", "auto"], |  | ||||||
|                      } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def __init__(self, **kwargs): |  | ||||||
|         super(Sentiment140Plugin, self).__init__(name="sentiment140", |  | ||||||
|                                                  version="2.0", |  | ||||||
|                                                  extraparams=self.EXTRA_PARAMS, |  | ||||||
|                                                  **kwargs) |  | ||||||
|  |  | ||||||
|     def analyse(self, **params): |  | ||||||
|         lang = params.get("language", "auto") |  | ||||||
|         res = requests.post("http://www.sentiment140.com/api/bulkClassifyJson", |  | ||||||
|                             json.dumps({"language": lang, |  | ||||||
|                                         "data": [{"text": params["input"]}] |  | ||||||
|                                         } |  | ||||||
|                                        ) |  | ||||||
|                             ) |  | ||||||
|  |  | ||||||
|         response = Response() |  | ||||||
|         polarity_value = int(res.json()["data"][0]["polarity"]) * 25 |  | ||||||
|         polarity = "marl:Neutral" |  | ||||||
|         if polarity_value > 50: |  | ||||||
|             polarity = "marl:Positive" |  | ||||||
|         elif polarity_value < 50: |  | ||||||
|             polarity = "marl:Negative" |  | ||||||
|         entry = Entry(text=params["input"]) |  | ||||||
|         opinion = Opinion(hasPolarity=polarity, polarityValue=polarity_value) |  | ||||||
|         entry.opinions.append(opinion) |  | ||||||
|         entry.language = lang |  | ||||||
|         response.entries.append(entry) |  | ||||||
|         return response |  | ||||||
|  |  | ||||||
| plugin = Sentiment140Plugin() |  | ||||||
| @@ -1,4 +1,15 @@ | |||||||
| Flask==0.10.1 | Flask>=0.10.1 | ||||||
| gunicorn==19.0.0 | requests>=2.4.1 | ||||||
| requests==2.4.1 | tornado>=4.4.3 | ||||||
| GitPython==0.3.2.RC1 | PyLD>=0.6.5 | ||||||
|  | nltk | ||||||
|  | future | ||||||
|  | jsonschema | ||||||
|  | jsonref | ||||||
|  | PyYAML | ||||||
|  | rdflib | ||||||
|  | rdflib-jsonld | ||||||
|  | numpy | ||||||
|  | scipy | ||||||
|  | scikit-learn | ||||||
|  | responses | ||||||
|   | |||||||
| @@ -17,9 +17,18 @@ | |||||||
| """ | """ | ||||||
| Sentiment analysis server in Python | Sentiment analysis server in Python | ||||||
| """ | """ | ||||||
|  | from .version import __version__ | ||||||
|  |  | ||||||
| import extensions | import logging | ||||||
| import blueprints |  | ||||||
| import plugins |  | ||||||
|  |  | ||||||
| __version__ = "0.2.8" | logger = logging.getLogger(__name__) | ||||||
|  |  | ||||||
|  | logger.info('Using senpy version: {}'.format(__version__)) | ||||||
|  |  | ||||||
|  | from .utils import easy, easy_load, easy_test  # noqa: F401 | ||||||
|  |  | ||||||
|  | from .models import *  # noqa: F401,F403 | ||||||
|  | from .plugins import *  # noqa: F401,F403 | ||||||
|  | from .extensions import *  # noqa: F401,F403 | ||||||
|  |  | ||||||
|  | __all__ = ['api', 'blueprints', 'cli', 'extensions', 'models', 'plugins'] | ||||||
|   | |||||||
| @@ -1,7 +1,171 @@ | |||||||
|  | #!/usr/bin/python | ||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  | #    Copyright 2014 J. Fernando Sánchez Rada - Grupo de Sistemas Inteligentes | ||||||
|  | #                                                       DIT, UPM | ||||||
|  | # | ||||||
|  | #    Licensed under the Apache License, Version 2.0 (the "License"); | ||||||
|  | #    you may not use this file except in compliance with the License. | ||||||
|  | #    You may obtain a copy of the License at | ||||||
|  | # | ||||||
|  | #        http://www.apache.org/licenses/LICENSE-2.0 | ||||||
|  | # | ||||||
|  | #    Unless required by applicable law or agreed to in writing, software | ||||||
|  | #    distributed under the License is distributed on an "AS IS" BASIS, | ||||||
|  | #    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||||
|  | #    See the License for the specific language governing permissions and | ||||||
|  | #    limitations under the License. | ||||||
|  | """ | ||||||
|  | Senpy is a modular sentiment analysis server. This script runs an instance of | ||||||
|  | the server. | ||||||
|  |  | ||||||
|  | """ | ||||||
|  |  | ||||||
| from flask import Flask | from flask import Flask | ||||||
| from extensions import Senpy | from senpy.extensions import Senpy | ||||||
| app = Flask(__name__) | from senpy.utils import easy_test | ||||||
| sp = Senpy() |  | ||||||
| sp.init_app(app) | import logging | ||||||
| app.debug = True | import os | ||||||
| app.run() | import sys | ||||||
|  | import argparse | ||||||
|  | import senpy | ||||||
|  |  | ||||||
|  | SERVER_PORT = os.environ.get("PORT", 5000) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def main(): | ||||||
|  |     parser = argparse.ArgumentParser(description='Run a Senpy server') | ||||||
|  |     parser.add_argument( | ||||||
|  |         '--level', | ||||||
|  |         '-l', | ||||||
|  |         metavar='logging_level', | ||||||
|  |         type=str, | ||||||
|  |         default="WARN", | ||||||
|  |         help='Logging level') | ||||||
|  |     parser.add_argument( | ||||||
|  |         '--debug', | ||||||
|  |         '-d', | ||||||
|  |         action='store_true', | ||||||
|  |         default=False, | ||||||
|  |         help='Run the application in debug mode') | ||||||
|  |     parser.add_argument( | ||||||
|  |         '--default-plugins', | ||||||
|  |         action='store_true', | ||||||
|  |         default=False, | ||||||
|  |         help='Load the default plugins') | ||||||
|  |     parser.add_argument( | ||||||
|  |         '--host', | ||||||
|  |         type=str, | ||||||
|  |         default="0.0.0.0", | ||||||
|  |         help='Use 0.0.0.0 to accept requests from any host.') | ||||||
|  |     parser.add_argument( | ||||||
|  |         '--port', | ||||||
|  |         '-p', | ||||||
|  |         type=int, | ||||||
|  |         default=SERVER_PORT, | ||||||
|  |         help='Port to listen on.') | ||||||
|  |     parser.add_argument( | ||||||
|  |         '--plugins-folder', | ||||||
|  |         '-f', | ||||||
|  |         type=str, | ||||||
|  |         default='.', | ||||||
|  |         help='Where to look for plugins.') | ||||||
|  |     parser.add_argument( | ||||||
|  |         '--only-install', | ||||||
|  |         '-i', | ||||||
|  |         action='store_true', | ||||||
|  |         default=False, | ||||||
|  |         help='Do not run a server, only install plugin dependencies') | ||||||
|  |     parser.add_argument( | ||||||
|  |         '--only-test', | ||||||
|  |         action='store_true', | ||||||
|  |         default=False, | ||||||
|  |         help='Do not run a server, just test all plugins') | ||||||
|  |     parser.add_argument( | ||||||
|  |         '--test', | ||||||
|  |         '-t', | ||||||
|  |         action='store_true', | ||||||
|  |         default=False, | ||||||
|  |         help='Test all plugins before launching the server') | ||||||
|  |     parser.add_argument( | ||||||
|  |         '--only-list', | ||||||
|  |         '--list', | ||||||
|  |         action='store_true', | ||||||
|  |         default=False, | ||||||
|  |         help='Do not run a server, only list plugins found') | ||||||
|  |     parser.add_argument( | ||||||
|  |         '--data-folder', | ||||||
|  |         '--data', | ||||||
|  |         type=str, | ||||||
|  |         default=None, | ||||||
|  |         help='Where to look for data. It be set with the SENPY_DATA environment variable as well.') | ||||||
|  |     parser.add_argument( | ||||||
|  |         '--threaded', | ||||||
|  |         action='store_false', | ||||||
|  |         default=True, | ||||||
|  |         help='Run a threaded server') | ||||||
|  |     parser.add_argument( | ||||||
|  |         '--no-deps', | ||||||
|  |         '-n', | ||||||
|  |         action='store_true', | ||||||
|  |         default=False, | ||||||
|  |         help='Skip installing dependencies') | ||||||
|  |     parser.add_argument( | ||||||
|  |         '--version', | ||||||
|  |         '-v', | ||||||
|  |         action='store_true', | ||||||
|  |         default=False, | ||||||
|  |         help='Output the senpy version and exit') | ||||||
|  |     parser.add_argument( | ||||||
|  |         '--allow-fail', | ||||||
|  |         '--fail', | ||||||
|  |         action='store_true', | ||||||
|  |         default=False, | ||||||
|  |         help='Do not exit if some plugins fail to activate') | ||||||
|  |     args = parser.parse_args() | ||||||
|  |     if args.version: | ||||||
|  |         print('Senpy version {}'.format(senpy.__version__)) | ||||||
|  |         print(sys.version) | ||||||
|  |         exit(1) | ||||||
|  |     rl = logging.getLogger() | ||||||
|  |     rl.setLevel(getattr(logging, args.level)) | ||||||
|  |     app = Flask(__name__) | ||||||
|  |     app.debug = args.debug | ||||||
|  |     sp = Senpy(app, args.plugins_folder, | ||||||
|  |                default_plugins=args.default_plugins, | ||||||
|  |                data_folder=args.data_folder) | ||||||
|  |     if args.only_list: | ||||||
|  |         plugins = sp.plugins() | ||||||
|  |         maxname = max(len(x.name) for x in plugins) | ||||||
|  |         maxversion = max(len(x.version) for x in plugins) | ||||||
|  |         print('Found {} plugins:'.format(len(plugins))) | ||||||
|  |         for plugin in plugins: | ||||||
|  |             import inspect | ||||||
|  |             fpath = inspect.getfile(plugin.__class__) | ||||||
|  |             print('\t{: <{maxname}} @ {: <{maxversion}} -> {}'.format(plugin.name, | ||||||
|  |                                                                       plugin.version, | ||||||
|  |                                                                       fpath, | ||||||
|  |                                                                       maxname=maxname, | ||||||
|  |                                                                       maxversion=maxversion)) | ||||||
|  |         return | ||||||
|  |     if not args.no_deps: | ||||||
|  |         sp.install_deps() | ||||||
|  |     if args.only_install: | ||||||
|  |         return | ||||||
|  |     sp.activate_all(allow_fail=args.allow_fail) | ||||||
|  |     if args.test or args.only_test: | ||||||
|  |         easy_test(sp.plugins(), debug=args.debug) | ||||||
|  |         if args.only_test: | ||||||
|  |             return | ||||||
|  |     print('Senpy version {}'.format(senpy.__version__)) | ||||||
|  |     print('Server running on port %s:%d. Ctrl+C to quit' % (args.host, | ||||||
|  |                                                             args.port)) | ||||||
|  |     app.run(args.host, | ||||||
|  |             args.port, | ||||||
|  |             threaded=args.threaded, | ||||||
|  |             debug=app.debug) | ||||||
|  |     sp.deactivate_all() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | if __name__ == '__main__': | ||||||
|  |     main() | ||||||
|   | |||||||
							
								
								
									
										315
									
								
								senpy/api.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,315 @@ | |||||||
|  | from future.utils import iteritems | ||||||
|  | from .models import Analysis, Error, Results, Entry, from_string | ||||||
|  | import logging | ||||||
|  | logger = logging.getLogger(__name__) | ||||||
|  |  | ||||||
|  | boolean = [True, False] | ||||||
|  |  | ||||||
|  | API_PARAMS = { | ||||||
|  |     "algorithm": { | ||||||
|  |         "aliases": ["algorithms", "a", "algo"], | ||||||
|  |         "required": True, | ||||||
|  |         "default": 'default', | ||||||
|  |         "description": ("Algorithms that will be used to process the request." | ||||||
|  |                         "It may be a list of comma-separated names."), | ||||||
|  |     }, | ||||||
|  |     "expanded-jsonld": { | ||||||
|  |         "@id": "expanded-jsonld", | ||||||
|  |         "aliases": ["expanded"], | ||||||
|  |         "options": boolean, | ||||||
|  |         "required": True, | ||||||
|  |         "default": False | ||||||
|  |     }, | ||||||
|  |     "with_parameters": { | ||||||
|  |         "aliases": ['withparameters', | ||||||
|  |                     'with-parameters'], | ||||||
|  |         "options": boolean, | ||||||
|  |         "default": False, | ||||||
|  |         "required": True | ||||||
|  |     }, | ||||||
|  |     "outformat": { | ||||||
|  |         "@id": "outformat", | ||||||
|  |         "aliases": ["o"], | ||||||
|  |         "default": "json-ld", | ||||||
|  |         "required": True, | ||||||
|  |         "options": ["json-ld", "turtle", "ntriples"], | ||||||
|  |     }, | ||||||
|  |     "help": { | ||||||
|  |         "@id": "help", | ||||||
|  |         "description": "Show additional help to know more about the possible parameters", | ||||||
|  |         "aliases": ["h"], | ||||||
|  |         "required": True, | ||||||
|  |         "options": boolean, | ||||||
|  |         "default": False | ||||||
|  |     }, | ||||||
|  |     "verbose": { | ||||||
|  |         "@id": "verbose", | ||||||
|  |         "description": "Show all help, including the common API parameters, or only plugin-related info", | ||||||
|  |         "aliases": ["v"], | ||||||
|  |         "required": True, | ||||||
|  |         "options": boolean, | ||||||
|  |         "default": True | ||||||
|  |     }, | ||||||
|  |     "emotionModel": { | ||||||
|  |         "@id": "emotionModel", | ||||||
|  |         "aliases": ["emoModel"], | ||||||
|  |         "required": False | ||||||
|  |     }, | ||||||
|  |     "conversion": { | ||||||
|  |         "@id": "conversion", | ||||||
|  |         "description": "How to show the elements that have (not) been converted", | ||||||
|  |         "required": True, | ||||||
|  |         "options": ["filtered", "nested", "full"], | ||||||
|  |         "default": "full" | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | EVAL_PARAMS = { | ||||||
|  |     "algorithm": { | ||||||
|  |         "aliases": ["plug", "p", "plugins", "algorithms", 'algo', 'a', 'plugin'], | ||||||
|  |         "description": "Plugins to be evaluated", | ||||||
|  |         "required": True, | ||||||
|  |         "help": "See activated plugins in /plugins" | ||||||
|  |     }, | ||||||
|  |     "dataset": { | ||||||
|  |         "aliases": ["datasets", "data", "d"], | ||||||
|  |         "description": "Datasets to be evaluated", | ||||||
|  |         "required": True, | ||||||
|  |         "help": "See avalaible datasets in /datasets" | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | PLUGINS_PARAMS = { | ||||||
|  |     "plugin_type": { | ||||||
|  |         "@id": "pluginType", | ||||||
|  |         "description": 'What kind of plugins to list', | ||||||
|  |         "aliases": ["pluginType"], | ||||||
|  |         "required": True, | ||||||
|  |         "default": 'analysisPlugin' | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | WEB_PARAMS = { | ||||||
|  |     "inHeaders": { | ||||||
|  |         "aliases": ["headers"], | ||||||
|  |         "required": True, | ||||||
|  |         "default": False, | ||||||
|  |         "options": boolean | ||||||
|  |     }, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | CLI_PARAMS = { | ||||||
|  |     "plugin_folder": { | ||||||
|  |         "aliases": ["folder"], | ||||||
|  |         "required": True, | ||||||
|  |         "default": "." | ||||||
|  |     }, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | NIF_PARAMS = { | ||||||
|  |     "input": { | ||||||
|  |         "@id": "input", | ||||||
|  |         "aliases": ["i"], | ||||||
|  |         "required": True, | ||||||
|  |         "help": "Input text" | ||||||
|  |     }, | ||||||
|  |     "intype": { | ||||||
|  |         "@id": "intype", | ||||||
|  |         "aliases": ["t"], | ||||||
|  |         "required": False, | ||||||
|  |         "default": "direct", | ||||||
|  |         "options": ["direct", "url", "file"], | ||||||
|  |     }, | ||||||
|  |     "informat": { | ||||||
|  |         "@id": "informat", | ||||||
|  |         "aliases": ["f"], | ||||||
|  |         "required": False, | ||||||
|  |         "default": "text", | ||||||
|  |         "options": ["text", "json-ld"], | ||||||
|  |     }, | ||||||
|  |     "language": { | ||||||
|  |         "@id": "language", | ||||||
|  |         "aliases": ["l"], | ||||||
|  |         "required": False, | ||||||
|  |     }, | ||||||
|  |     "prefix": { | ||||||
|  |         "@id": "prefix", | ||||||
|  |         "aliases": ["p"], | ||||||
|  |         "required": True, | ||||||
|  |         "default": "", | ||||||
|  |     }, | ||||||
|  |     "urischeme": { | ||||||
|  |         "@id": "urischeme", | ||||||
|  |         "aliases": ["u"], | ||||||
|  |         "required": False, | ||||||
|  |         "default": "RFC5147String", | ||||||
|  |         "options": ["RFC5147String", ] | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | BUILTIN_PARAMS = {} | ||||||
|  |  | ||||||
|  | for d in [ | ||||||
|  |         NIF_PARAMS, CLI_PARAMS, WEB_PARAMS, PLUGINS_PARAMS, EVAL_PARAMS, | ||||||
|  |         API_PARAMS | ||||||
|  | ]: | ||||||
|  |     for k, v in d.items(): | ||||||
|  |         BUILTIN_PARAMS[k] = v | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def parse_params(indict, *specs): | ||||||
|  |     if not specs: | ||||||
|  |         specs = [NIF_PARAMS] | ||||||
|  |     logger.debug("Parsing: {}\n{}".format(indict, specs)) | ||||||
|  |     outdict = indict.copy() | ||||||
|  |     wrong_params = {} | ||||||
|  |     for spec in specs: | ||||||
|  |         for param, options in iteritems(spec): | ||||||
|  |             for alias in options.get("aliases", []): | ||||||
|  |                 # Replace each alias with the correct name of the parameter | ||||||
|  |                 if alias in indict and alias != param: | ||||||
|  |                     outdict[param] = indict[alias] | ||||||
|  |                     del outdict[alias] | ||||||
|  |                     continue | ||||||
|  |             if param not in outdict: | ||||||
|  |                 if "default" in options: | ||||||
|  |                     # We assume the default is correct | ||||||
|  |                     outdict[param] = options["default"] | ||||||
|  |                 elif options.get("required", False): | ||||||
|  |                     wrong_params[param] = spec[param] | ||||||
|  |             elif "options" in options: | ||||||
|  |                 if options["options"] == boolean: | ||||||
|  |                     outdict[param] = str(outdict[param]).lower() in ['true', '1'] | ||||||
|  |                 elif outdict[param] not in options["options"]: | ||||||
|  |                     wrong_params[param] = spec[param] | ||||||
|  |     if wrong_params: | ||||||
|  |         logger.debug("Error parsing: %s", wrong_params) | ||||||
|  |         message = Error( | ||||||
|  |             status=400, | ||||||
|  |             message='Missing or invalid parameters', | ||||||
|  |             parameters=outdict, | ||||||
|  |             errors=wrong_params) | ||||||
|  |         raise message | ||||||
|  |     return outdict | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def get_all_params(plugins, *specs): | ||||||
|  |     '''Return a list of parameters for a given set of specifications and plugins.''' | ||||||
|  |     dic = {} | ||||||
|  |     for s in specs: | ||||||
|  |         dic.update(s) | ||||||
|  |     dic.update(get_extra_params(plugins)) | ||||||
|  |     return dic | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def get_extra_params(plugins): | ||||||
|  |     '''Get a list of possible parameters given a list of plugins''' | ||||||
|  |     params = {} | ||||||
|  |     extra_params = {} | ||||||
|  |     for plugin in plugins: | ||||||
|  |         this_params = plugin.get('extra_params', {}) | ||||||
|  |         for k, v in this_params.items(): | ||||||
|  |             if k not in extra_params: | ||||||
|  |                 extra_params[k] = {} | ||||||
|  |             extra_params[k][plugin.name] = v | ||||||
|  |     for k, v in extra_params.items():  # Resolve conflicts | ||||||
|  |         if len(v) == 1:  # Add the extra options that do not collide | ||||||
|  |             params[k] = list(v.values())[0] | ||||||
|  |         else: | ||||||
|  |             required = False | ||||||
|  |             aliases = None | ||||||
|  |             options = None | ||||||
|  |             default = None | ||||||
|  |             nodefault = False  # Set when defaults are not compatible | ||||||
|  |  | ||||||
|  |             for plugin, opt in v.items(): | ||||||
|  |                 params['{}.{}'.format(plugin, k)] = opt | ||||||
|  |                 required = required or opt.get('required', False) | ||||||
|  |                 newaliases = set(opt.get('aliases', [])) | ||||||
|  |                 if aliases is None: | ||||||
|  |                     aliases = newaliases | ||||||
|  |                 else: | ||||||
|  |                     aliases = aliases & newaliases | ||||||
|  |                 if 'options' in opt: | ||||||
|  |                     newoptions = set(opt['options']) | ||||||
|  |                     options = newoptions if options is None else options & newoptions | ||||||
|  |                 if 'default' in opt: | ||||||
|  |                     newdefault = opt['default'] | ||||||
|  |                     if newdefault: | ||||||
|  |                         if default is None and not nodefault: | ||||||
|  |                             default = newdefault | ||||||
|  |                         elif newdefault != default: | ||||||
|  |                             nodefault = True | ||||||
|  |                             default = None | ||||||
|  |             # Check for incompatibilities | ||||||
|  |             if options != set(): | ||||||
|  |                 params[k] = { | ||||||
|  |                     'default': default, | ||||||
|  |                     'aliases': list(aliases), | ||||||
|  |                     'required': required, | ||||||
|  |                     'options': list(options) | ||||||
|  |                 } | ||||||
|  |     return params | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def parse_analysis(params, plugins): | ||||||
|  |     ''' | ||||||
|  |     Parse the given parameters individually for each plugin, and get a list of the parameters that | ||||||
|  |     belong to each of the plugins. Each item can then be used in the plugin.analyse_entries method. | ||||||
|  |     ''' | ||||||
|  |     analysis_list = [] | ||||||
|  |     for i, plugin in enumerate(plugins): | ||||||
|  |         if not plugin: | ||||||
|  |             continue | ||||||
|  |         this_params = filter_params(params, plugin, i) | ||||||
|  |         parsed = parse_params(this_params, plugin.get('extra_params', {})) | ||||||
|  |         analysis = plugin.activity(parsed) | ||||||
|  |         analysis_list.append(analysis) | ||||||
|  |     return analysis_list | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def filter_params(params, plugin, ith=-1): | ||||||
|  |     ''' | ||||||
|  |     Get the values within params that apply to a plugin. | ||||||
|  |     More specific names override more general names, in this order: | ||||||
|  |  | ||||||
|  |     <index_order>.parameter > <plugin.name>.parameter > parameter | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     Example: | ||||||
|  |  | ||||||
|  |     >>> filter_params({'0.hello': True, 'hello': False}, Plugin(), 0) | ||||||
|  |     { '0.hello': True, 'hello': True} | ||||||
|  |  | ||||||
|  |     ''' | ||||||
|  |     thisparams = {} | ||||||
|  |     if ith >= 0: | ||||||
|  |         ith = '{}.'.format(ith) | ||||||
|  |     else: | ||||||
|  |         ith = "" | ||||||
|  |     for k, v in params.items(): | ||||||
|  |         if ith and k.startswith(str(ith)): | ||||||
|  |             thisparams[k[len(ith):]] = v | ||||||
|  |         elif k.startswith(plugin.name): | ||||||
|  |             thisparams[k[len(plugin.name) + 1:]] = v | ||||||
|  |         elif k not in thisparams: | ||||||
|  |             thisparams[k] = v | ||||||
|  |     return thisparams | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def parse_call(params): | ||||||
|  |     ''' | ||||||
|  |     Return a results object based on the parameters used in a call/request. | ||||||
|  |     ''' | ||||||
|  |     params = parse_params(params, NIF_PARAMS) | ||||||
|  |     if params['informat'] == 'text': | ||||||
|  |         results = Results() | ||||||
|  |         entry = Entry(nif__isString=params['input'], id='#')  # Use @base | ||||||
|  |         results.entries.append(entry) | ||||||
|  |     elif params['informat'] == 'json-ld': | ||||||
|  |         results = from_string(params['input'], cls=Results) | ||||||
|  |     else:  # pragma: no cover | ||||||
|  |         raise NotImplementedError('Informat {} is not implemented'.format( | ||||||
|  |             params['informat'])) | ||||||
|  |     results.parameters = params | ||||||
|  |     return results | ||||||
| @@ -17,126 +17,237 @@ | |||||||
| """ | """ | ||||||
| Blueprints for Senpy | Blueprints for Senpy | ||||||
| """ | """ | ||||||
| import json | from flask import (Blueprint, request, current_app, render_template, url_for, | ||||||
|  |                    jsonify, redirect) | ||||||
|  | from .models import Error, Response, Help, Plugins, read_schema, dump_schema, Datasets | ||||||
|  | from . import api | ||||||
|  | from .version import __version__ | ||||||
|  | from functools import wraps | ||||||
|  |  | ||||||
| import logging | import logging | ||||||
|  | import json | ||||||
|  | import base64 | ||||||
|  |  | ||||||
| logger = logging.getLogger(__name__) | logger = logging.getLogger(__name__) | ||||||
|  |  | ||||||
| from flask import Blueprint, request, jsonify, current_app | api_blueprint = Blueprint("api", __name__) | ||||||
|  | demo_blueprint = Blueprint("demo", __name__, template_folder='templates') | ||||||
|  | ns_blueprint = Blueprint("ns", __name__) | ||||||
|  |  | ||||||
| nif_blueprint = Blueprint("NIF Sentiment Analysis Server", __name__) | _mimetypes_r = {'json-ld': ['application/ld+json'], | ||||||
|  |                 'turtle': ['text/turtle'], | ||||||
|  |                 'ntriples': ['application/n-triples'], | ||||||
|  |                 'text': ['text/plain']} | ||||||
|  |  | ||||||
| BASIC_PARAMS = { | MIMETYPES = {} | ||||||
|     "algorithm": {"aliases": ["algorithm", "a", "algo"], |  | ||||||
|                   "required": False, | for k, vs in _mimetypes_r.items(): | ||||||
|                   }, |     for v in vs: | ||||||
| } |         if v in MIMETYPES: | ||||||
|  |             raise Exception('MIMETYPE {} specified for two formats: {} and {}'.format(v, | ||||||
|  |                                                                                       v, | ||||||
|  |                                                                                       MIMETYPES[v])) | ||||||
|  |         MIMETYPES[v] = k | ||||||
|  |  | ||||||
|  | DEFAULT_MIMETYPE = 'application/ld+json' | ||||||
|  | DEFAULT_FORMAT = 'json-ld' | ||||||
|  |  | ||||||
|  |  | ||||||
| def get_params(req, params=BASIC_PARAMS): | def get_params(req): | ||||||
|     if req.method == 'POST': |     if req.method == 'POST': | ||||||
|         indict = req.form |         indict = req.form.to_dict(flat=True) | ||||||
|     elif req.method == 'GET': |     elif req.method == 'GET': | ||||||
|         indict = req.args |         indict = req.args.to_dict(flat=True) | ||||||
|     else: |     else: | ||||||
|         raise ValueError("Invalid data") |         raise Error(message="Invalid data") | ||||||
|  |     return indict | ||||||
|  |  | ||||||
|     outdict = {} |  | ||||||
|     wrong_params = {} | def encoded_url(url=None, base=None): | ||||||
|     for param, options in params.iteritems(): |     code = '' | ||||||
|         for alias in options["aliases"]: |     if not url: | ||||||
|             if alias in indict: |         if request.method == 'GET': | ||||||
|                 outdict[param] = indict[alias] |             url = request.full_path[1:]  # Remove the first slash | ||||||
|         if param not in outdict: |  | ||||||
|             if options.get("required", False): |  | ||||||
|                 wrong_params[param] = params[param] |  | ||||||
|             else: |  | ||||||
|                 if "default" in options: |  | ||||||
|                     outdict[param] = options["default"] |  | ||||||
|         else: |         else: | ||||||
|             if "options" in params[param] and outdict[param] not in params[param]["options"]: |             hash(frozenset(tuple(request.parameters.items()))) | ||||||
|                 wrong_params[param] = params[param] |             code = 'hash:{}'.format(hash) | ||||||
|     if wrong_params: |  | ||||||
|         message = {"status": "failed", |     code = code or base64.urlsafe_b64encode(url.encode()).decode() | ||||||
|                    "message": "Missing or invalid parameters", |  | ||||||
|                    "parameters": outdict, |     if base: | ||||||
|                    "errors": {param: error for param, error in wrong_params.iteritems()} |         return base + code | ||||||
|                    } |     return url_for('api.decode', code=code, _external=True) | ||||||
|         raise ValueError(json.dumps(message)) |  | ||||||
|     return outdict |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def basic_analysis(params): | def decoded_url(code, base=None): | ||||||
|     response = {"@context": ["http://demos.gsi.dit.upm.es/eurosentiment/static/context.jsonld", |     if code.startswith('hash:'): | ||||||
|                              { |         raise Exception('Can not decode a URL for a POST request') | ||||||
|                                  "@base": "{}#".format(request.url.encode('utf-8')) |     base = base or request.url_root | ||||||
|                              } |     path = base64.urlsafe_b64decode(code.encode()).decode() | ||||||
|                              ], |     return base + path | ||||||
|                 "analysis": [{"@type": "marl:SentimentAnalysis"}], |  | ||||||
|                 "entries": [] |  | ||||||
|                 } |  | ||||||
|     if "language" in params: |  | ||||||
|         response["language"] = params["language"] |  | ||||||
|     for idx, sentence in enumerate(params["input"].split(".")): |  | ||||||
|         response["entries"].append({ |  | ||||||
|             "@id": "Sentence{}".format(idx), |  | ||||||
|             "nif:isString": sentence |  | ||||||
|         }) |  | ||||||
|     return response |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @nif_blueprint.route('/', methods=['POST', 'GET']) | @demo_blueprint.route('/') | ||||||
| def home(): | def index(): | ||||||
|  |     ev = str(get_params(request).get('evaluation', False)) | ||||||
|  |     evaluation_enabled = ev.lower() not in ['false', 'no', 'none'] | ||||||
|  |  | ||||||
|  |     return render_template("index.html", | ||||||
|  |                            evaluation=evaluation_enabled, | ||||||
|  |                            version=__version__) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @api_blueprint.route('/contexts/<entity>.jsonld') | ||||||
|  | def context(entity="context"): | ||||||
|  |     context = Response._context | ||||||
|  |     context['@vocab'] = url_for('ns.index', _external=True) | ||||||
|  |     context['endpoint'] = url_for('api.api_root', _external=True) | ||||||
|  |     return jsonify({"@context": context}) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @api_blueprint.route('/d/<code>') | ||||||
|  | def decode(code): | ||||||
|     try: |     try: | ||||||
|         algo = get_params(request).get("algorithm", None) |         return redirect(decoded_url(code)) | ||||||
|         specific_params = current_app.senpy.parameters(algo) |     except Exception: | ||||||
|         params = get_params(request, specific_params) |         return Error('invalid URL').flask() | ||||||
|         response = current_app.senpy.analyse(**params) |  | ||||||
|         return jsonify(response) |  | ||||||
|     except ValueError as ex: |  | ||||||
|         return ex.message |  | ||||||
|     except Exception as ex: |  | ||||||
|         return jsonify(status="400", message=ex.message) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @nif_blueprint.route("/default") | @ns_blueprint.route('/')  # noqa: F811 | ||||||
| def default(): | def index(): | ||||||
|     return current_app.senpy.default_plugin |     context = Response._context.copy() | ||||||
|     #return plugins(action="list", plugin=current_app.senpy.default_algorithm) |     context['endpoint'] = url_for('api.api_root', _external=True) | ||||||
|  |     return jsonify({"@context": context}) | ||||||
|  |  | ||||||
|  |  | ||||||
| @nif_blueprint.route('/plugins/', methods=['POST', 'GET']) | @api_blueprint.route('/schemas/<schema>') | ||||||
| @nif_blueprint.route('/plugins/<plugin>', methods=['POST', 'GET']) | def schema(schema="definitions"): | ||||||
| @nif_blueprint.route('/plugins/<plugin>/<action>', methods=['POST', 'GET']) |     try: | ||||||
| def plugins(plugin=None, action="list"): |         return dump_schema(read_schema(schema)) | ||||||
|     filt = {} |     except Exception as ex:  # Should be FileNotFoundError, but it's missing from py2 | ||||||
|  |         return Error(message="Schema not found: {}".format(ex), status=404).flask() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def basic_api(f): | ||||||
|  |     default_params = { | ||||||
|  |         'inHeaders': False, | ||||||
|  |         'expanded-jsonld': False, | ||||||
|  |         'outformat': None, | ||||||
|  |         'with_parameters': True, | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @wraps(f) | ||||||
|  |     def decorated_function(*args, **kwargs): | ||||||
|  |         raw_params = get_params(request) | ||||||
|  |         logger.info('Getting request: {}'.format(raw_params)) | ||||||
|  |         headers = {'X-ORIGINAL-PARAMS': json.dumps(raw_params)} | ||||||
|  |         params = default_params | ||||||
|  |  | ||||||
|  |         try: | ||||||
|  |             params = api.parse_params(raw_params, api.WEB_PARAMS, api.API_PARAMS) | ||||||
|  |             if hasattr(request, 'parameters'): | ||||||
|  |                 request.parameters.update(params) | ||||||
|  |             else: | ||||||
|  |                 request.parameters = params | ||||||
|  |             response = f(*args, **kwargs) | ||||||
|  |         except (Exception) as ex: | ||||||
|  |             if current_app.debug or current_app.config['TESTING']: | ||||||
|  |                 raise | ||||||
|  |             if not isinstance(ex, Error): | ||||||
|  |                 msg = "{}".format(ex) | ||||||
|  |                 ex = Error(message=msg, status=500) | ||||||
|  |             response = ex | ||||||
|  |             response.parameters = raw_params | ||||||
|  |             logger.exception(ex) | ||||||
|  |  | ||||||
|  |         if 'parameters' in response and not params['with_parameters']: | ||||||
|  |             del response.parameters | ||||||
|  |  | ||||||
|  |         logger.info('Response: {}'.format(response)) | ||||||
|  |         mime = request.accept_mimetypes\ | ||||||
|  |                       .best_match(MIMETYPES.keys(), | ||||||
|  |                                   DEFAULT_MIMETYPE) | ||||||
|  |  | ||||||
|  |         mimeformat = MIMETYPES.get(mime, DEFAULT_FORMAT) | ||||||
|  |         outformat = params['outformat'] or mimeformat | ||||||
|  |  | ||||||
|  |         return response.flask( | ||||||
|  |             in_headers=params['inHeaders'], | ||||||
|  |             headers=headers, | ||||||
|  |             prefix=params.get('prefix', encoded_url()), | ||||||
|  |             context_uri=url_for('api.context', | ||||||
|  |                                 entity=type(response).__name__, | ||||||
|  |                                 _external=True), | ||||||
|  |             outformat=outformat, | ||||||
|  |             expanded=params['expanded-jsonld']) | ||||||
|  |  | ||||||
|  |     return decorated_function | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @api_blueprint.route('/', defaults={'plugin': None}, methods=['POST', 'GET']) | ||||||
|  | @api_blueprint.route('/<path:plugin>', methods=['POST', 'GET']) | ||||||
|  | @basic_api | ||||||
|  | def api_root(plugin): | ||||||
|     if plugin: |     if plugin: | ||||||
|         filt["name"] = plugin |         if request.parameters['algorithm'] != api.API_PARAMS['algorithm']['default']: | ||||||
|     plugs = current_app.senpy.filter_plugins(**filt) |             raise Error('You cannot specify the algorithm with a parameter and a URL variable.' | ||||||
|     if plugin and not plugs: |                         ' Please, remove one of them') | ||||||
|         return "Plugin not found", 400 |         request.parameters['algorithm'] =  tuple(plugin.replace('+', '/').split('/')) | ||||||
|     if action == "list": |  | ||||||
|         with_params = request.args.get("params", "") == "1" |     params = request.parameters | ||||||
|         dic = {plug: plugs[plug].jsonable(with_params) for plug in plugs} |     plugin = request.parameters['algorithm'] | ||||||
|         return jsonify(dic) |  | ||||||
|     if action == "disable": |     sp = current_app.senpy | ||||||
|         current_app.senpy.disable_plugin(plugin) |     plugins = sp.get_plugins(plugin) | ||||||
|         return "Ok" |  | ||||||
|     elif action == "enable": |     if request.parameters['help']: | ||||||
|         current_app.senpy.enable_plugin(plugin) |         apis = [] | ||||||
|         return "Ok" |         if request.parameters['verbose']: | ||||||
|     elif action == "reload": |             apis.append(api.BUILTIN_PARAMS) | ||||||
|         current_app.senpy.reload_plugin(plugin) |         allparameters = api.get_all_params(plugins, *apis) | ||||||
|         return "Ok" |         response = Help(valid_parameters=allparameters) | ||||||
|  |         return response | ||||||
|  |     req = api.parse_call(request.parameters) | ||||||
|  |     analysis = api.parse_analysis(req.parameters, plugins) | ||||||
|  |     results = current_app.senpy.analyse(req, analysis) | ||||||
|  |     return results | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @api_blueprint.route('/evaluate/', methods=['POST', 'GET']) | ||||||
|  | @basic_api | ||||||
|  | def evaluate(): | ||||||
|  |     if request.parameters['help']: | ||||||
|  |         dic = dict(api.EVAL_PARAMS) | ||||||
|  |         response = Help(parameters=dic) | ||||||
|  |         return response | ||||||
|     else: |     else: | ||||||
|         return "action '{}' not allowed".format(action), 400 |         params = api.parse_params(request.parameters, api.EVAL_PARAMS) | ||||||
|  |         response = current_app.senpy.evaluate(params) | ||||||
|  |         return response | ||||||
|  |  | ||||||
|  |  | ||||||
| if __name__ == '__main__': | @api_blueprint.route('/plugins/', methods=['POST', 'GET']) | ||||||
|     import config | @basic_api | ||||||
|     from flask import Flask | def plugins(): | ||||||
|  |     sp = current_app.senpy | ||||||
|  |     params = api.parse_params(request.parameters, api.PLUGINS_PARAMS) | ||||||
|  |     ptype = params.get('plugin_type') | ||||||
|  |     plugins = list(sp.plugins(plugin_type=ptype)) | ||||||
|  |     dic = Plugins(plugins=plugins) | ||||||
|  |     return dic | ||||||
|  |  | ||||||
|     app = Flask(__name__) |  | ||||||
|     app.register_blueprint(nif_blueprint) | @api_blueprint.route('/plugins/<plugin>/', methods=['POST', 'GET']) | ||||||
|     app.debug = config.DEBUG | @basic_api | ||||||
|     app.run() | def plugin(plugin): | ||||||
|  |     sp = current_app.senpy | ||||||
|  |     return sp.get_plugin(plugin) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @api_blueprint.route('/datasets/', methods=['POST', 'GET']) | ||||||
|  | @basic_api | ||||||
|  | def datasets(): | ||||||
|  |     sp = current_app.senpy | ||||||
|  |     datasets = sp.datasets | ||||||
|  |     dic = Datasets(datasets=list(datasets.values())) | ||||||
|  |     return dic | ||||||
|   | |||||||
							
								
								
									
										56
									
								
								senpy/cli.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,56 @@ | |||||||
|  | import sys | ||||||
|  | from .models import Error | ||||||
|  | from .extensions import Senpy | ||||||
|  | from . import api | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def argv_to_dict(argv): | ||||||
|  |     '''Turns parameters in the form of '--key value' into a dict {'key': 'value'} | ||||||
|  |     ''' | ||||||
|  |     cli_dict = {} | ||||||
|  |  | ||||||
|  |     for i in range(len(argv)): | ||||||
|  |         if argv[i][0] == '-': | ||||||
|  |             key = argv[i].strip('-') | ||||||
|  |             value = argv[i + 1] if len(argv) > i + 1 else None | ||||||
|  |             if not value or value[0] == '-': | ||||||
|  |                 cli_dict[key] = True | ||||||
|  |             else: | ||||||
|  |                 cli_dict[key] = value | ||||||
|  |     return cli_dict | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def main_function(argv): | ||||||
|  |     '''This is the method for unit testing | ||||||
|  |     ''' | ||||||
|  |     params = api.parse_params(argv_to_dict(argv), | ||||||
|  |                               api.CLI_PARAMS, | ||||||
|  |                               api.API_PARAMS, | ||||||
|  |                               api.NIF_PARAMS) | ||||||
|  |     plugin_folder = params['plugin_folder'] | ||||||
|  |     default_plugins = params.get('default-plugins', False) | ||||||
|  |     sp = Senpy(default_plugins=default_plugins, plugin_folder=plugin_folder) | ||||||
|  |     request = api.parse_call(params) | ||||||
|  |     algos = sp.get_plugins(request.parameters.get('algorithm', None)) | ||||||
|  |     if algos: | ||||||
|  |         for algo in algos: | ||||||
|  |             sp.activate_plugin(algo.name) | ||||||
|  |     else: | ||||||
|  |         sp.activate_all() | ||||||
|  |     res = sp.analyse(request) | ||||||
|  |     return res | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def main(): | ||||||
|  |     '''This method is the entrypoint for the CLI (as configured un setup.py) | ||||||
|  |     ''' | ||||||
|  |     try: | ||||||
|  |         res = main_function(sys.argv[1:]) | ||||||
|  |         print(res.serialize()) | ||||||
|  |     except Error as err: | ||||||
|  |         print(err.serialize()) | ||||||
|  |         sys.exit(2) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | if __name__ == '__main__': | ||||||
|  |     main() | ||||||
							
								
								
									
										51
									
								
								senpy/client.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,51 @@ | |||||||
|  | import requests | ||||||
|  | import logging | ||||||
|  | from . import models | ||||||
|  |  | ||||||
|  | logger = logging.getLogger(__name__) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Client(object): | ||||||
|  |     def __init__(self, endpoint): | ||||||
|  |         self.endpoint = endpoint | ||||||
|  |  | ||||||
|  |     def analyse(self, input, method='GET', **kwargs): | ||||||
|  |         return self.request('/', method=method, input=input, **kwargs) | ||||||
|  |  | ||||||
|  |     def evaluate(self, input, method='GET', **kwargs): | ||||||
|  |         return self.request('/evaluate', method=method, input=input, **kwargs) | ||||||
|  |  | ||||||
|  |     def plugins(self, *args, **kwargs): | ||||||
|  |         resp = self.request(path='/plugins').plugins | ||||||
|  |         return {p.name: p for p in resp} | ||||||
|  |  | ||||||
|  |     def datasets(self): | ||||||
|  |         resp = self.request(path='/datasets').datasets | ||||||
|  |         return {d.name: d for d in resp} | ||||||
|  |  | ||||||
|  |     def request(self, path=None, method='GET', **params): | ||||||
|  |         url = '{}{}'.format(self.endpoint.rstrip('/'), path) | ||||||
|  |         if method == 'POST': | ||||||
|  |             response = requests.post(url=url, data=params) | ||||||
|  |         else: | ||||||
|  |             response = requests.request(method=method, url=url, params=params) | ||||||
|  |  | ||||||
|  |         try: | ||||||
|  |             resp = models.from_dict(response.json()) | ||||||
|  |         except Exception as ex: | ||||||
|  |             logger.error(('There seems to be a problem with the response:\n' | ||||||
|  |                           '\tURL: {url}\n' | ||||||
|  |                           '\tError: {error}\n' | ||||||
|  |                           '\t\n' | ||||||
|  |                           '#### Response:\n' | ||||||
|  |                           '\tCode: {code}' | ||||||
|  |                           '\tContent: {content}' | ||||||
|  |                           '\n').format( | ||||||
|  |                               error=ex, | ||||||
|  |                               url=url, | ||||||
|  |                               code=response.status_code, | ||||||
|  |                               content=response.content)) | ||||||
|  |             raise ex | ||||||
|  |         if isinstance(resp, models.Error): | ||||||
|  |             raise resp | ||||||
|  |         return resp | ||||||
| @@ -1,39 +0,0 @@ | |||||||
| { |  | ||||||
|     "dc": "http://purl.org/dc/terms/", |  | ||||||
|     "dc:subject": { |  | ||||||
|         "@type": "@id" |  | ||||||
|     }, |  | ||||||
|     "xsd": "http://www.w3.org/2001/XMLSchema#", |  | ||||||
|     "marl": "http://www.gsi.dit.upm.es/ontologies/marl/ns#", |  | ||||||
|     "nif": "http://persistence.uni-leipzig.org/nlp2rdf/ontologies/nif-core#", |  | ||||||
|     "onyx": "http://www.gsi.dit.upm.es/ontologies/onyx/ns#", |  | ||||||
|     "emotions": { |  | ||||||
|         "@container": "@set", |  | ||||||
|         "@id": "onyx:hasEmotionSet" |  | ||||||
|     }, |  | ||||||
|     "opinions": { |  | ||||||
|         "@container": "@set", |  | ||||||
|         "@id": "marl:hasOpinion" |  | ||||||
|     }, |  | ||||||
|     "prov": "http://www.w3.org/ns/prov#", |  | ||||||
|     "rdfs": "http://www.w3.org/2000/01/rdf-schema#", |  | ||||||
|     "analysis": { |  | ||||||
|         "@container": "@set", |  | ||||||
|         "@id": "prov:wasInformedBy" |  | ||||||
|     }, |  | ||||||
|     "entries": { |  | ||||||
|         "@container": "@set", |  | ||||||
|         "@id": "prov:generated" |  | ||||||
|     }, |  | ||||||
|     "strings": { |  | ||||||
|         "@container": "@set", |  | ||||||
|         "@reverse": "nif:hasContext" |  | ||||||
|     }, |  | ||||||
|     "date": |  | ||||||
|     { |  | ||||||
|         "@id": "dc:date", |  | ||||||
|         "@type": "xsd:dateTime" |  | ||||||
|     }, |  | ||||||
|     "wnaffect": "http://www.gsi.dit.upm.es/ontologies/wnaffect#", |  | ||||||
|     "xsd": "http://www.w3.org/2001/XMLSchema#" |  | ||||||
| } |  | ||||||
| @@ -1,36 +1,57 @@ | |||||||
| """ | """ | ||||||
|  | Main class for Senpy. | ||||||
|  | It orchestrates plugin (de)activation and analysis. | ||||||
| """ | """ | ||||||
|  | from future import standard_library | ||||||
|  | standard_library.install_aliases() | ||||||
|  |  | ||||||
|  | from . import plugins, api | ||||||
|  | from .models import Error, AggregatedEvaluation | ||||||
|  | from .blueprints import api_blueprint, demo_blueprint, ns_blueprint | ||||||
|  |  | ||||||
|  | from threading import Thread | ||||||
|  | from functools import partial | ||||||
| import os | import os | ||||||
| import sys | import copy | ||||||
| import imp | import errno | ||||||
| import logging | import logging | ||||||
|  |  | ||||||
|  | from . import gsitk_compat | ||||||
|  |  | ||||||
| logger = logging.getLogger(__name__) | logger = logging.getLogger(__name__) | ||||||
|  |  | ||||||
| from .plugins import SentimentPlugin, EmotionPlugin |  | ||||||
|  |  | ||||||
| try: |  | ||||||
|     from flask import _app_ctx_stack as stack |  | ||||||
| except ImportError: |  | ||||||
|     from flask import _request_ctx_stack as stack |  | ||||||
|  |  | ||||||
| from .blueprints import nif_blueprint |  | ||||||
| from git import Repo, InvalidGitRepositoryError |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Senpy(object): | class Senpy(object): | ||||||
|     """ Default Senpy extension for Flask """ |     """ Default Senpy extension for Flask """ | ||||||
|  |  | ||||||
|     def __init__(self, app=None, plugin_folder="plugins"): |     def __init__(self, | ||||||
|  |                  app=None, | ||||||
|  |                  plugin_folder=".", | ||||||
|  |                  data_folder=None, | ||||||
|  |                  default_plugins=False): | ||||||
|  |  | ||||||
|  |         default_data = os.path.join(os.getcwd(), 'senpy_data') | ||||||
|  |         self.data_folder = data_folder or os.environ.get('SENPY_DATA', default_data) | ||||||
|  |         try: | ||||||
|  |             os.makedirs(self.data_folder) | ||||||
|  |         except OSError as e: | ||||||
|  |             if e.errno == errno.EEXIST: | ||||||
|  |                 logger.debug('Data folder exists: {}'.format(self.data_folder)) | ||||||
|  |             else:  # pragma: no cover | ||||||
|  |                 raise | ||||||
|  |  | ||||||
|  |         self._default = None | ||||||
|  |         self._plugins = {} | ||||||
|  |         if plugin_folder: | ||||||
|  |             self.add_folder(plugin_folder) | ||||||
|  |  | ||||||
|  |         if default_plugins: | ||||||
|  |             self.add_folder('plugins', from_root=True) | ||||||
|  |         else: | ||||||
|  |             # Add only conversion plugins | ||||||
|  |             self.add_folder(os.path.join('plugins', 'postprocessing'), | ||||||
|  |                             from_root=True) | ||||||
|         self.app = app |         self.app = app | ||||||
|         base_folder = os.path.join(os.path.dirname(__file__), "plugins") |  | ||||||
|  |  | ||||||
|         self._search_folders = set() |  | ||||||
|         self._outdated = True |  | ||||||
|  |  | ||||||
|         for folder in (base_folder, plugin_folder): |  | ||||||
|             self.add_folder(folder) |  | ||||||
|  |  | ||||||
|         if app is not None: |         if app is not None: | ||||||
|             self.init_app(app) |             self.init_app(app) | ||||||
|  |  | ||||||
| @@ -45,128 +66,352 @@ class Senpy(object): | |||||||
|         # otherwise fall back to the request context |         # otherwise fall back to the request context | ||||||
|         if hasattr(app, 'teardown_appcontext'): |         if hasattr(app, 'teardown_appcontext'): | ||||||
|             app.teardown_appcontext(self.teardown) |             app.teardown_appcontext(self.teardown) | ||||||
|         else: |         else:  # pragma: no cover | ||||||
|             app.teardown_request(self.teardown) |             app.teardown_request(self.teardown) | ||||||
|         app.register_blueprint(nif_blueprint) |         app.register_blueprint(api_blueprint, url_prefix="/api") | ||||||
|  |         app.register_blueprint(ns_blueprint, url_prefix="/ns") | ||||||
|  |         app.register_blueprint(demo_blueprint, url_prefix="/") | ||||||
|  |  | ||||||
|     def add_folder(self, folder): |     def add_plugin(self, plugin): | ||||||
|  |         self._plugins[plugin.name.lower()] = plugin | ||||||
|  |  | ||||||
|  |     def delete_plugin(self, plugin): | ||||||
|  |         del self._plugins[plugin.name.lower()] | ||||||
|  |  | ||||||
|  |     def plugins(self, plugin_type=None, is_activated=True, **kwargs): | ||||||
|  |         """ Return the plugins registered for a given application. Filtered by criteria  """ | ||||||
|  |         return list(plugins.pfilter(self._plugins, plugin_type=plugin_type, | ||||||
|  |                                     is_activated=is_activated, **kwargs)) | ||||||
|  |  | ||||||
|  |     def get_plugin(self, name, default=None): | ||||||
|  |         if name == 'default': | ||||||
|  |             return self.default_plugin | ||||||
|  |         elif name == 'conversion': | ||||||
|  |             return None | ||||||
|  |  | ||||||
|  |         if name.lower() in self._plugins: | ||||||
|  |             return self._plugins[name.lower()] | ||||||
|  |  | ||||||
|  |         results = self.plugins(id='endpoint:plugins/{}'.format(name.lower()), | ||||||
|  |                                plugin_type=None) | ||||||
|  |         if results: | ||||||
|  |             return results[0] | ||||||
|  |  | ||||||
|  |         results = self.plugins(id=name, | ||||||
|  |                                plugin_type=None) | ||||||
|  |         if results: | ||||||
|  |             return results[0] | ||||||
|  |  | ||||||
|  |         msg = ("Plugin not found: '{}'\n" | ||||||
|  |                "Make sure it is ACTIVATED\n" | ||||||
|  |                 "Valid algorithms: {}").format(name, | ||||||
|  |                                               self._plugins.keys()) | ||||||
|  |         raise Error(message=msg, status=404) | ||||||
|  |  | ||||||
|  |     def get_plugins(self, name): | ||||||
|  |         try: | ||||||
|  |             name = name.split(',') | ||||||
|  |         except AttributeError: | ||||||
|  |             pass  # Assume it is a tuple or a list | ||||||
|  |         return tuple(self.get_plugin(n) for n in name) | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def analysis_plugins(self): | ||||||
|  |         """ Return only the analysis plugins that are active""" | ||||||
|  |         return self.plugins(plugin_type='analysisPlugin', is_activated=True) | ||||||
|  |  | ||||||
|  |     def add_folder(self, folder, from_root=False): | ||||||
|  |         """ Find plugins in this folder and add them to this instance """ | ||||||
|  |         if from_root: | ||||||
|  |             folder = os.path.join(os.path.dirname(__file__), folder) | ||||||
|  |         logger.debug("Adding folder: %s", folder) | ||||||
|         if os.path.isdir(folder): |         if os.path.isdir(folder): | ||||||
|             self._search_folders.add(folder) |             new_plugins = plugins.from_folder([folder], | ||||||
|             self._outdated = True |                                               data_folder=self.data_folder) | ||||||
|             return True |             for plugin in new_plugins: | ||||||
|  |                 self.add_plugin(plugin) | ||||||
|         else: |         else: | ||||||
|             return False |             raise AttributeError("Not a folder or does not exist: %s", folder) | ||||||
|  |  | ||||||
|     def analyse(self, **params): |     # def check_analysis_request(self, analysis): | ||||||
|         algo = None |     #     '''Check if the analysis request can be fulfilled''' | ||||||
|         logger.debug("analysing with params: {}".format(params)) |     #     if not self.plugins(): | ||||||
|         if "algorithm" in params: |     #         raise Error( | ||||||
|             algo = params["algorithm"] |     #             status=404, | ||||||
|         elif self.plugins: |     #             message=("No plugins found." | ||||||
|             algo = self.default_plugin |     #                      " Please install one.")) | ||||||
|         if algo in self.plugins and self.plugins[algo].enabled: |     #     for a in analysis: | ||||||
|             plug = self.plugins[algo] |     #         algo = a.algorithm | ||||||
|             resp = plug.analyse(**params) |     #         if algo == 'default' and not self.default_plugin: | ||||||
|             resp.analysis.append(plug.jsonable()) |     #             raise Error( | ||||||
|  |     #                 status=404, | ||||||
|  |     #                 message="No default plugin found, and None provided") | ||||||
|  |     #         else: | ||||||
|  |     #             self.get_plugin(algo) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     def _process(self, req, pending, done=None): | ||||||
|  |         """ | ||||||
|  |         Recursively process the entries with the first plugin in the list, and pass the results | ||||||
|  |         to the rest of the plugins. | ||||||
|  |         """ | ||||||
|  |         done = done or [] | ||||||
|  |         if not pending: | ||||||
|  |             return req | ||||||
|  |  | ||||||
|  |         analysis = pending[0] | ||||||
|  |         results = analysis.run(req) | ||||||
|  |         results.analysis.append(analysis) | ||||||
|  |         done += analysis | ||||||
|  |         return self._process(results, pending[1:], done) | ||||||
|  |  | ||||||
|  |     def install_deps(self): | ||||||
|  |         plugins.install_deps(*self.plugins()) | ||||||
|  |  | ||||||
|  |     def analyse(self, request, analysis=None): | ||||||
|  |         """ | ||||||
|  |         Main method that analyses a request, either from CLI or HTTP. | ||||||
|  |         It takes a processed request, provided by the user, as returned | ||||||
|  |         by api.parse_call(). | ||||||
|  |         """ | ||||||
|  |         if not self.plugins(): | ||||||
|  |             raise Error( | ||||||
|  |                 status=404, | ||||||
|  |                 message=("No plugins found." | ||||||
|  |                          " Please install one.")) | ||||||
|  |         if analysis is None: | ||||||
|  |             params = str(request) | ||||||
|  |             plugins = self.get_plugins(request.parameters['algorithm']) | ||||||
|  |             analysis = api.parse_analysis(request.parameters, plugins) | ||||||
|  |         logger.debug("analysing request: {}".format(request)) | ||||||
|  |         results = self._process(request, analysis) | ||||||
|  |         logger.debug("Got analysis result: {}".format(results)) | ||||||
|  |         results = self.postprocess(results) | ||||||
|  |         logger.debug("Returning post-processed result: {}".format(results)) | ||||||
|  |         return results | ||||||
|  |  | ||||||
|  |     def convert_emotions(self, resp): | ||||||
|  |         """ | ||||||
|  |         Conversion of all emotions in a response **in place**. | ||||||
|  |         In addition to converting from one model to another, it has | ||||||
|  |         to include the conversion plugin to the analysis list. | ||||||
|  |         Needless to say, this is far from an elegant solution, but it works. | ||||||
|  |         @todo refactor and clean up | ||||||
|  |         """ | ||||||
|  |         plugins = resp.analysis | ||||||
|  |  | ||||||
|  |         if 'parameters' not in resp: | ||||||
|             return resp |             return resp | ||||||
|         else: |  | ||||||
|             return {"status": 400, "message": "The algorithm '{}' is not valid".format(algo)} |         params = resp['parameters'] | ||||||
|  |         toModel = params.get('emotionModel', None) | ||||||
|  |         if not toModel: | ||||||
|  |             return resp | ||||||
|  |  | ||||||
|  |         logger.debug('Asked for model: {}'.format(toModel)) | ||||||
|  |         output = params.get('conversion', None) | ||||||
|  |         candidates = {} | ||||||
|  |         for plugin in plugins: | ||||||
|  |             try: | ||||||
|  |                 fromModel = plugin.get('onyx:usesEmotionModel', None) | ||||||
|  |                 candidates[plugin.id] = next(self._conversion_candidates(fromModel, toModel)) | ||||||
|  |                 logger.debug('Analysis plugin {} uses model: {}'.format( | ||||||
|  |                     plugin.id, fromModel)) | ||||||
|  |             except StopIteration: | ||||||
|  |                 e = Error(('No conversion plugin found for: ' | ||||||
|  |                            '{} -> {}'.format(fromModel, toModel)), | ||||||
|  |                           status=404) | ||||||
|  |                 e.original_response = resp | ||||||
|  |                 e.parameters = params | ||||||
|  |                 raise e | ||||||
|  |         newentries = [] | ||||||
|  |         done = [] | ||||||
|  |         for i in resp.entries: | ||||||
|  |             if output == "full": | ||||||
|  |                 newemotions = copy.deepcopy(i.emotions) | ||||||
|  |             else: | ||||||
|  |                 newemotions = [] | ||||||
|  |             for j in i.emotions: | ||||||
|  |                 plugname = j['prov:wasGeneratedBy'] | ||||||
|  |                 candidate = candidates[plugname] | ||||||
|  |                 done.append({'plugin': candidate, 'parameters': params}) | ||||||
|  |                 for k in candidate.convert(j, fromModel, toModel, params): | ||||||
|  |                     k.prov__wasGeneratedBy = candidate.id | ||||||
|  |                     if output == 'nested': | ||||||
|  |                         k.prov__wasDerivedFrom = j | ||||||
|  |                     newemotions.append(k) | ||||||
|  |             i.emotions = newemotions | ||||||
|  |             newentries.append(i) | ||||||
|  |         resp.entries = newentries | ||||||
|  |         return resp | ||||||
|  |  | ||||||
|  |     def _conversion_candidates(self, fromModel, toModel): | ||||||
|  |         candidates = self.plugins(plugin_type=plugins.EmotionConversion) | ||||||
|  |         for candidate in candidates: | ||||||
|  |             for pair in candidate.onyx__doesConversion: | ||||||
|  |                 logging.debug(pair) | ||||||
|  |                 if candidate.can_convert(fromModel, toModel): | ||||||
|  |                     yield candidate | ||||||
|  |  | ||||||
|  |     def postprocess(self, response): | ||||||
|  |         ''' | ||||||
|  |         Transform the results from the analysis plugins. | ||||||
|  |         It has some pre-defined post-processing like emotion conversion, | ||||||
|  |         and it also allows plugins to auto-select themselves. | ||||||
|  |         ''' | ||||||
|  |  | ||||||
|  |         response = self.convert_emotions(response) | ||||||
|  |  | ||||||
|  |         for plug in self.plugins(plugin_type=plugins.PostProcessing): | ||||||
|  |             if plug.check(response, response.analysis): | ||||||
|  |                 response = plug.process(response) | ||||||
|  |         return response | ||||||
|  |  | ||||||
|  |     def _get_datasets(self, request): | ||||||
|  |         if not self.datasets: | ||||||
|  |             raise Error( | ||||||
|  |                 status=404, | ||||||
|  |                 message=("No datasets found." | ||||||
|  |                          " Please verify DatasetManager")) | ||||||
|  |         datasets_name = request.parameters.get('dataset', None).split(',') | ||||||
|  |         for dataset in datasets_name: | ||||||
|  |             if dataset not in self.datasets: | ||||||
|  |                 logger.debug(("The dataset '{}' is not valid\n" | ||||||
|  |                               "Valid datasets: {}").format( | ||||||
|  |                                   dataset, self.datasets.keys())) | ||||||
|  |                 raise Error( | ||||||
|  |                     status=404, | ||||||
|  |                     message="The dataset '{}' is not valid".format(dataset)) | ||||||
|  |         dm = gsitk_compat.DatasetManager() | ||||||
|  |         datasets = dm.prepare_datasets(datasets_name) | ||||||
|  |         return datasets | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def datasets(self): | ||||||
|  |         self._dataset_list = {} | ||||||
|  |         dm = gsitk_compat.DatasetManager() | ||||||
|  |         for item in dm.get_datasets(): | ||||||
|  |             for key in item: | ||||||
|  |                 if key in self._dataset_list: | ||||||
|  |                     continue | ||||||
|  |                 properties = item[key] | ||||||
|  |                 properties['@id'] = key | ||||||
|  |                 self._dataset_list[key] = properties | ||||||
|  |         return self._dataset_list | ||||||
|  |  | ||||||
|  |     def evaluate(self, params): | ||||||
|  |         logger.debug("evaluating request: {}".format(params)) | ||||||
|  |         results = AggregatedEvaluation() | ||||||
|  |         results.parameters = params | ||||||
|  |         datasets = self._get_datasets(results) | ||||||
|  |         plugins = [] | ||||||
|  |         for plugname in params.algorithm: | ||||||
|  |             plugins = self.get_plugin(plugname) | ||||||
|  |  | ||||||
|  |         for eval in plugins.evaluate(plugins, datasets): | ||||||
|  |             results.evaluations.append(eval) | ||||||
|  |         if 'with_parameters' not in results.parameters: | ||||||
|  |             del results.parameters | ||||||
|  |         logger.debug("Returning evaluation result: {}".format(results)) | ||||||
|  |         return results | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def default_plugin(self): |     def default_plugin(self): | ||||||
|         candidates = self.filter_plugins(enabled=True) |         if not self._default or not self._default.is_activated: | ||||||
|         if len(candidates) > 0: |             candidates = self.plugins( | ||||||
|             candidate = candidates.keys()[0] |                 plugin_type='analysisPlugin', is_activated=True) | ||||||
|             logger.debug("Default: {}".format(candidate)) |             if len(candidates) > 0: | ||||||
|             return candidate |                 self._default = candidates[0] | ||||||
|  |             else: | ||||||
|  |                 self._default = None | ||||||
|  |             logger.debug("Default: {}".format(self._default)) | ||||||
|  |         return self._default | ||||||
|  |  | ||||||
|  |     @default_plugin.setter | ||||||
|  |     def default_plugin(self, value): | ||||||
|  |         if isinstance(value, plugins.Plugin): | ||||||
|  |             if not value.is_activated: | ||||||
|  |                 raise AttributeError('The default plugin has to be activated.') | ||||||
|  |             self._default = value | ||||||
|  |  | ||||||
|         else: |         else: | ||||||
|             return None |             self._default = self._plugins[value.lower()] | ||||||
|  |  | ||||||
|     def parameters(self, algo): |     def activate_all(self, sync=True, allow_fail=False): | ||||||
|         return getattr(self.plugins.get(algo or self.default_plugin), "params", {}) |         ps = [] | ||||||
|  |         for plug in self._plugins.keys(): | ||||||
|     def enable_plugin(self, plugin): |  | ||||||
|         self.plugins[plugin].enable() |  | ||||||
|  |  | ||||||
|     def disable_plugin(self, plugin): |  | ||||||
|         self.plugins[plugin].disable() |  | ||||||
|  |  | ||||||
|     def reload_plugin(self, plugin): |  | ||||||
|         logger.debug("Reloading {}".format(plugin)) |  | ||||||
|         plug = self.plugins[plugin] |  | ||||||
|         nplug = self._load_plugin(plug.module, plug.path) |  | ||||||
|         del self.plugins[plugin] |  | ||||||
|         self.plugins[nplug.name] = nplug |  | ||||||
|  |  | ||||||
|     @staticmethod |  | ||||||
|     def _load_plugin(plugin, search_folder, enabled=True): |  | ||||||
|         logger.debug("Loading plugins") |  | ||||||
|         sys.path.append(search_folder) |  | ||||||
|         (fp, pathname, desc) = imp.find_module(plugin) |  | ||||||
|         try: |  | ||||||
|             tmp = imp.load_module(plugin, fp, pathname, desc).plugin |  | ||||||
|             sys.path.remove(search_folder) |  | ||||||
|             tmp.path = search_folder |  | ||||||
|             try: |             try: | ||||||
|                 repo_path = os.path.join(search_folder, plugin) |                 self.activate_plugin(plug, sync=sync) | ||||||
|                 tmp.repo = Repo(repo_path) |             except Exception as ex: | ||||||
|             except InvalidGitRepositoryError: |                 if not allow_fail: | ||||||
|                 tmp.repo = None |                     raise | ||||||
|             if not hasattr(tmp, "enabled"): |                 logger.error('Could not activate {}: {}'.format(plug, ex)) | ||||||
|                 tmp.enabled = enabled |         return ps | ||||||
|             tmp.module = plugin |  | ||||||
|         except Exception as ex: |  | ||||||
|             tmp = None |  | ||||||
|             logger.debug("Exception importing {}: {}".format(plugin, ex)) |  | ||||||
|         return tmp |  | ||||||
|  |  | ||||||
|     def _load_plugins(self): |     def deactivate_all(self, sync=True): | ||||||
|         plugins = {} |         ps = [] | ||||||
|         for search_folder in self._search_folders: |         for plug in self._plugins.keys(): | ||||||
|             for item in os.listdir(search_folder): |             ps.append(self.deactivate_plugin(plug, sync=sync)) | ||||||
|                 if os.path.isdir(os.path.join(search_folder, item)) \ |         return ps | ||||||
|                         and os.path.exists(os.path.join(search_folder, |  | ||||||
|                                                         item, |  | ||||||
|                                                         "__init__.py")): |  | ||||||
|                     plugin = self._load_plugin(item, search_folder) |  | ||||||
|                     if plugin: |  | ||||||
|                         plugins[plugin.name] = plugin |  | ||||||
|  |  | ||||||
|         self._outdated = False |     def _set_active(self, plugin, active=True, *args, **kwargs): | ||||||
|         return plugins |         ''' We're using a variable in the plugin itself to activate/deactivate plugins.\ | ||||||
|  |         Note that plugins may activate themselves by setting this variable. | ||||||
|  |         ''' | ||||||
|  |         plugin.is_activated = active | ||||||
|  |  | ||||||
|  |     def _activate(self, plugin): | ||||||
|  |         success = False | ||||||
|  |         with plugin._lock: | ||||||
|  |             if plugin.is_activated: | ||||||
|  |                 return | ||||||
|  |             plugin.activate() | ||||||
|  |             msg = "Plugin activated: {}".format(plugin.name) | ||||||
|  |             logger.info(msg) | ||||||
|  |             success = True | ||||||
|  |             self._set_active(plugin, success) | ||||||
|  |         return success | ||||||
|  |  | ||||||
|  |     def activate_plugin(self, plugin_name, sync=True): | ||||||
|  |         plugin_name = plugin_name.lower() | ||||||
|  |         if plugin_name not in self._plugins: | ||||||
|  |             raise Error( | ||||||
|  |                 message="Plugin not found: {}".format(plugin_name), status=404) | ||||||
|  |         plugin = self._plugins[plugin_name] | ||||||
|  |  | ||||||
|  |         logger.info("Activating plugin: {}".format(plugin.name)) | ||||||
|  |  | ||||||
|  |         if sync or not getattr(plugin, 'async', True) or getattr( | ||||||
|  |                 plugin, 'sync', False): | ||||||
|  |             return self._activate(plugin) | ||||||
|  |         else: | ||||||
|  |             th = Thread(target=partial(self._activate, plugin)) | ||||||
|  |             th.start() | ||||||
|  |             return th | ||||||
|  |  | ||||||
|  |     def _deactivate(self, plugin): | ||||||
|  |         with plugin._lock: | ||||||
|  |             if not plugin.is_activated: | ||||||
|  |                 return | ||||||
|  |             plugin.deactivate() | ||||||
|  |             logger.info("Plugin deactivated: {}".format(plugin.name)) | ||||||
|  |  | ||||||
|  |     def deactivate_plugin(self, plugin_name, sync=True): | ||||||
|  |         plugin_name = plugin_name.lower() | ||||||
|  |         if plugin_name not in self._plugins: | ||||||
|  |             raise Error( | ||||||
|  |                 message="Plugin not found: {}".format(plugin_name), status=404) | ||||||
|  |         plugin = self._plugins[plugin_name] | ||||||
|  |  | ||||||
|  |         self._set_active(plugin, False) | ||||||
|  |  | ||||||
|  |         if sync or not getattr(plugin, 'async', True) or not getattr( | ||||||
|  |                 plugin, 'sync', False): | ||||||
|  |             self._deactivate(plugin) | ||||||
|  |         else: | ||||||
|  |             th = Thread(target=partial(self._deactivate, plugin)) | ||||||
|  |             th.start() | ||||||
|  |             return th | ||||||
|  |  | ||||||
|     def teardown(self, exception): |     def teardown(self, exception): | ||||||
|         pass |         pass | ||||||
|  |  | ||||||
|     def enable_all(self): |  | ||||||
|         for plugin in self.plugins: |  | ||||||
|             self.enable_plugin(plugin) |  | ||||||
|  |  | ||||||
|     @property |  | ||||||
|     def plugins(self): |  | ||||||
|         """ Return the plugins registered for a given application.  """ |  | ||||||
|         ctx = stack.top |  | ||||||
|         if ctx is not None: |  | ||||||
|             if not hasattr(ctx, 'senpy_plugins') or self._outdated: |  | ||||||
|                 ctx.senpy_plugins = self._load_plugins() |  | ||||||
|             return ctx.senpy_plugins |  | ||||||
|  |  | ||||||
|     def filter_plugins(self, **kwargs): |  | ||||||
|         """ Filter plugins by different criteria """ |  | ||||||
|  |  | ||||||
|         def matches(plug): |  | ||||||
|             res = all(getattr(plug, k, None) == v for (k, v) in kwargs.items()) |  | ||||||
|             logger.debug("matching {} with {}: {}".format(plug.name, |  | ||||||
|                                                           kwargs, |  | ||||||
|                                                           res)) |  | ||||||
|             return res |  | ||||||
|  |  | ||||||
|         if not kwargs: |  | ||||||
|             return self.plugins |  | ||||||
|         else: |  | ||||||
|             return {n: p for n, p in self.plugins.items() if matches(p)} |  | ||||||
|  |  | ||||||
|     def sentiment_plugins(self): |  | ||||||
|         """ Return only the sentiment plugins """ |  | ||||||
|         return {p: plugin for p, plugin in self.plugins.items() if |  | ||||||
|                 isinstance(plugin, SentimentPlugin)} |  | ||||||
|   | |||||||