A concept called "ProductTree" defines what a product does. The ProductTree comprises a set of nested Capabilities in a tree-like structure. ProductTree helps to ensure that people understand what a product does, which sounds obvious however to date frameworks have only focussed on the work to be done and not what the product does.
The work to be done on a product is organised separately to the ProductTree but in a related way. The units of work are defined as Tasks, and these are grouped into Initiatives.
Whilst we deliberately separate what the product does (ProductTree) from the work (Initiatives/Tasks) to be done, these are related:
- Initiatives belong to a Capability in the ProductTree
- An Initiative's Tasks are by default related to the same Capability as that Initiative
- The Capability-Task relationship can be overriden, Tasks are allowed to relate to a different Capability than their Initiative if desired
Capability is a way of breaking down the functional and non-functional areas of the product. The nested set of Capabilities related to a Product is referred to as the ProductMap. An example ProductMap from Mailchimp:
- Audience Management
- Marketing CRM
- Audience dashboard
- Import contacts
- Export data
- Segmentation tool
- Signup Forms
- Segmentation
- Behavioral Targeting
- Predicted Demographics
- Marketing CRM
- Creative Tools
- Content Studio
- Creative Assistant (Beta)
- Dynamic Content
- Subject Line Helper
- Campaign Templates
- Marketing Automation
- Customer Journeys
- Integration
- Transactional Email
- Insight & Analytics
- Reports
- Smart Recommentations
- A/B Testing
- Serverys
Tasks are units of work to be done. Tasks are grouped into Initiatives and both Tasks and Initiatives relate to a capability in the ProductMap.
For example, a Task might be: As a user, I want to be able to tag contacts. Implement the ability for contacts to be tagged in the UI and backend.
Initiatives are a group of Tasks that relate to each other. Each initiative belongs to a Capability and can have multiple Tasks. The stories by default have their initiative's Category.
For example, an Initiative might be: Create new Admin Console
- Initialize project submodules
git submodule init
git submodule update
- Create backend/.env file from backend/.env.template and update the file variables with your information.
cp backend/.env.template backend/.env
- Run backend and frontend:
docker-compose up
You need to have docker engine and docker-compose installed.
- Import dummy data If you want to set initial default data for testing please run this command to fill the database with some information.
docker-compose exec backend python manage.py dummy_data
- Create a superuser
docker-compose exec backend python manage.py createsuperuser
-
Open the application http://0.0.0.0:8080
-
The admin panel is available http://0.0.0.0:8080/admin
-
The application works in development mode (max 2 products & 5 users/persons). Set license value in
LICENSE_TRIAL_KEY
variable and run thepython manage.py create_trial_license
command to create a trial license.
if after starting the project in the console the error is: "Invalid HTTP_HOST header: 'localhost:8000'. You may need to add 'localhost' to ALLOWED_HOSTS", and frontend crashes after first render.
- Create a local_settings.py under product-factory-composer/backend/backend folder.
- Add the following codes.
ALLOWED_HOSTS = [
"localhost",
"localhost:3000"
]
The default application is in development mode, where fake user authorization is available in the system /switch-test-user
.
To prevent this page from being accessed and authorizing using AM:
- Create the
frontend/.env
file. - Define
APPLICATION_MODE=production
variable.
- Backend tests:
docker-compose exec backend pytest
- Frontend tests:
docker-compose exec frontend npm test
- Create a local_settings.py under backend/backend folder.
- Add the following codes.
GIT_ACCESS_TOKEN = "wKcW1n/8JhergkePGMe2dZfkoyWi6dPmZCsRCHiYY5Q"
GIT_OWNER = "devsugun"
DOMAIN = "<lt domain>"
ALLOWED_HOSTS = ["<lt domain>", "localhost:8000", "localhost:3000", "localhost", "127.0.0.1"]
SECURE_PROXY_SSL_HEADER = ("HTTP_X_FORWARDED_PROTO", "https")
CORS_ORIGIN_WHITELIST = [
'http://localhost:3000'
]
- How to get lt domain? Please check how to use localtunnel.
- Replace it with domain created by localtunnel.