5.5 KiB
Ansible AWX/Tower V2
Hi there! We're excited to have you as a contributor.
Have questions about this document or anything not covered here? Feel free to reach out to any of the contributors of this repository found here: https://github.com/ansible/awx-pf/graphs/contributors
Table of contents
- Things to know prior to submitting code
- Setting up your development environment
- Build the user interface
- Accessing the AWX web interface
- Working with React
- Testing
Things to know prior to submitting code
- All code submissions are done through pull requests against the
master
branch. - If collaborating with someone else on the same branch, please use
--force-with-lease
instead of--force
when pushing up code. This will prevent you from accidentally overwriting commits pushed by someone else. For more information, see https://git-scm.com/docs/git-push#git-push---force-with-leaseltrefnamegt
Setting up your development environment
The UI is built using ReactJS and Patternfly.
Prerequisites
Node and npm
The AWX UI requires the following:
- Node 8.x LTS
- NPM 6.x LTS
Run the following to install all the dependencies:
(host) $ npm run install
Build the User Interface
Run the following to build the AWX UI:
(host) $ npm run start
Accessing the AWX web interface
You can now log into the AWX web interface at https://127.0.0.1:3001.
Working with React
Class constructors vs Class properties
It is good practice to use constructor-bound instance methods rather than methods as class properties. Methods as arrow functions provide lexical scope and are bound to the Component class instance instead of the class itself. This makes it so we cannot easily test a Component's methods without invoking an instance of the Component and calling the method directly within our tests.
BAD:
class MyComponent extends React.Component {
constructor(props) {
super(props);
}
myEventHandler = () => {
// do a thing
}
}
GOOD:
class MyComponent extends React.Component {
constructor(props) {
super(props);
this.myEventHandler = this.myEventHandler.bind(this);
}
myEventHandler() {
// do a thing
}
}
Binding
It is good practice to bind our class methods within our class constructor method for the following reasons:
- Avoid defining the method every time
render()
is called. - Performance advantages.
- Ease of testing.
Component Lifecycle
A React Component has various lifecylce methods. Understanding the basic lifecycle of a Component will help you determine which method to utilize for your specific need. Below are some general guidelines:
BAD:
Use render
method to make asynchronous calls.
render () {
const { data } = await api.get(API_CONFIG);
return(<div>`Hello, ${data.usename}!`</div>);
}
GOOD:
Use componentDidMount()
method to make asynchronous calls to retrieve data a Componenet may need.
async componentDidMount () {
try {
const { data } = await api.get(API_CONFIG);
this.setState({ data });
} catch (error) {
this.setState({ error });
}
}
render() {
return(<div>`Hello, ${this.state.data.usename}!`</div>)
}
Testing
All code, new or otherwise, should have at least 80% test coverage.
Jest
We use (Jest)[https://jestjs.io/] for our JS testing framework.
Like many other JS test frameworks (Karma, Mocha, etc), Jest includes their own spyOn
method as a way for us to test our class methods.
const spy = jest.spyOn(MyButton.prototype, 'onSubmit');
Jest also allows us to mock the data we expect from an external dependency, such as an API.
axios.get.mockImplementation((endpoint) => {
if (endpoint === '/api/v2/config') {
return new Promise((resolve, reject) => {
resolve({ data: { foo: 'bar' });
});
}
else {
return 'get results';
}
});
Enzyme
We use (Enzyme)[https://airbnb.io/enzyme/] to test our React Components.
Mounting Components wrapped with withRouter
If you are testing a Component wrapped in React Router's withRouter
class, you can mount the component by wrapping it with the <MemoryRouter>
component.
test('initially renders succesfully', () => {
mount(
<MemoryRouter>
<OrganizationAdd
match={{ path: '/organizations/add', url: '/organizations/add' }}
location={{ search: '', pathname: '/organizations/add' }}
/>
</MemoryRouter>
);
});
You can test the wrapped Component's methods like so:
const spy = jest.spyOn(OrganizationAdd.WrappedComponent.prototype, 'onCancel');