Skip to content
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.

Commit 293201b

Browse files
committedDec 16, 2020
first commit
0 parents  commit 293201b

38 files changed

+26613
-0
lines changed
 

‎README.md

Lines changed: 43 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,43 @@
1+
# How to integrate React.js with Spring Boot (run on same Server/Port)
2+
3+
For more detail, please visit:
4+
> [How to integrate Spring Boot with React.js](https://bezkoder.com/integrate-reactjs-spring-boot/)
5+
6+
> [React (Components) CRUD example to consume Web API](https://bezkoder.com/react-crud-web-api/)
7+
8+
> [Spring Boot JPA - Building Rest CRUD API example](https://bezkoder.com/spring-boot-jpa-crud-rest-api/)
9+
10+
Fullstack with Spring Boot:
11+
> [React.js + Spring Boot + MySQL](https://bezkoder.com/react-spring-boot-crud/)
12+
13+
> [React.js + Spring Boot + PostgreSQL](https://bezkoder.com/spring-boot-react-postgresql/)
14+
15+
> [React.js + Spring Boot + MongoDB](https://bezkoder.com/react-spring-boot-mongodb/)
16+
17+
More Practice:
18+
> [React (Hooks) CRUD example to consume Web API](https://bezkoder.com/react-hooks-crud-axios-api/)
19+
20+
> [React Material UI examples with a CRUD Application](https://bezkoder.com/react-material-ui-examples-crud/)
21+
22+
> [Spring Boot Pagination & Filter example | Spring JPA, Pageable](https://bezkoder.com/spring-boot-pagination-filter-jpa-pageable/)
23+
24+
> [Spring Data JPA Sort/Order by multiple Columns | Spring Boot](https://bezkoder.com/spring-data-sort-multiple-columns/)
25+
26+
> [Spring Boot Repository Unit Test with @DataJpaTest](https://bezkoder.com/spring-boot-unit-test-jpa-repo-datajpatest/)
27+
28+
> [Deploy Spring Boot App on AWS – Elastic Beanstalk](https://bezkoder.com/deploy-spring-boot-aws-eb/)
29+
30+
Serverless:
31+
> [React Firebase CRUD App with Realtime Database](https://bezkoder.com/react-firebase-crud/)
32+
33+
> [React Firestore CRUD App example | Firebase Cloud Firestore](https://bezkoder.com/react-firestore-crud/)
34+
35+
## Project setup
36+
```
37+
mvn clean install
38+
```
39+
40+
### Run
41+
```
42+
mvn spring-boot:run
43+
```

‎react-client/.env

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
PORT=8081

‎react-client/.gitignore

Lines changed: 23 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
# See https://help.github.com/articles/ignoring-files/ for more about ignoring files.
2+
3+
# dependencies
4+
/node_modules
5+
/.pnp
6+
.pnp.js
7+
8+
# testing
9+
/coverage
10+
11+
# production
12+
/build
13+
14+
# misc
15+
.DS_Store
16+
.env.local
17+
.env.development.local
18+
.env.test.local
19+
.env.production.local
20+
21+
npm-debug.log*
22+
yarn-debug.log*
23+
yarn-error.log*

‎react-client/README.md

Lines changed: 70 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,70 @@
1+
# React.js CRUD App with React Router & Axios
2+
3+
For more detail, please visit:
4+
> [React (Components) CRUD example to consume Web API](https://bezkoder.com/react-crud-web-api/)
5+
6+
> [React (Hooks) CRUD example to consume Web API](https://bezkoder.com/react-hooks-crud-axios-api/)
7+
8+
Using Material UI instead of Bootstrap:
9+
> [React Material UI examples with a CRUD Application](https://bezkoder.com/react-material-ui-examples-crud/)
10+
11+
Fullstack with Node.js Express:
12+
> [React.js + Node.js Express + MySQL](https://bezkoder.com/react-node-express-mysql/)
13+
14+
> [React.js + Node.js Express + PostgreSQL](https://bezkoder.com/react-node-express-postgresql/)
15+
16+
> [React.js + Node.js Express + MongoDB](https://bezkoder.com/react-node-express-mongodb-mern-stack/)
17+
18+
Fullstack with Spring Boot:
19+
> [React.js + Spring Boot + MySQL](https://bezkoder.com/react-spring-boot-crud/)
20+
21+
> [React.js + Spring Boot + PostgreSQL](https://bezkoder.com/spring-boot-react-postgresql/)
22+
23+
> [React.js + Spring Boot + MongoDB](https://bezkoder.com/react-spring-boot-mongodb/)
24+
25+
Fullstack with Django:
26+
27+
> [React.js + Django Rest Framework](https://bezkoder.com/django-react-axios-rest-framework/)
28+
29+
Serverless:
30+
> [React Firebase CRUD App with Realtime Database](https://bezkoder.com/react-firebase-crud/)
31+
32+
> [React Firestore CRUD App example | Firebase Cloud Firestore](https://bezkoder.com/react-firestore-crud/)
33+
34+
Integration (run back-end & front-end on same server/port)
35+
> [How to integrate React.js with Spring Boot](https://bezkoder.com/integrate-reactjs-spring-boot/)
36+
37+
> [Integrate React with Node.js Express on same Server/Port](https://bezkoder.com/integrate-react-express-same-server-port/)
38+
39+
40+
This project was bootstrapped with [Create React App](https://github.com/facebook/create-react-app).
41+
42+
### Set port
43+
.env
44+
```
45+
PORT=8081
46+
```
47+
48+
## Project setup
49+
50+
In the project directory, you can run:
51+
52+
```
53+
npm install
54+
# or
55+
yarn install
56+
```
57+
58+
or
59+
60+
### Compiles and hot-reloads for development
61+
62+
```
63+
npm start
64+
# or
65+
yarn start
66+
```
67+
68+
Open [http://localhost:8081](http://localhost:8081) to view it in the browser.
69+
70+
The page will reload if you make edits.

‎react-client/package-lock.json

Lines changed: 14019 additions & 0 deletions
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

‎react-client/package.json

Lines changed: 37 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
{
2+
"name": "react-crud",
3+
"version": "0.1.0",
4+
"private": true,
5+
"dependencies": {
6+
"@testing-library/jest-dom": "^4.2.4",
7+
"@testing-library/react": "^9.3.2",
8+
"@testing-library/user-event": "^7.1.2",
9+
"axios": "^0.19.2",
10+
"bootstrap": "^4.4.1",
11+
"react": "^16.13.0",
12+
"react-dom": "^16.13.0",
13+
"react-router-dom": "^5.1.2",
14+
"react-scripts": "3.4.0"
15+
},
16+
"scripts": {
17+
"start": "react-scripts start",
18+
"build": "react-scripts build",
19+
"test": "react-scripts test",
20+
"eject": "react-scripts eject"
21+
},
22+
"eslintConfig": {
23+
"extends": "react-app"
24+
},
25+
"browserslist": {
26+
"production": [
27+
">0.2%",
28+
"not dead",
29+
"not op_mini all"
30+
],
31+
"development": [
32+
"last 1 chrome version",
33+
"last 1 firefox version",
34+
"last 1 safari version"
35+
]
36+
}
37+
}

‎react-client/public/favicon.ico

3.08 KB
Binary file not shown.

‎react-client/public/index.html

Lines changed: 43 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,43 @@
1+
<!DOCTYPE html>
2+
<html lang="en">
3+
<head>
4+
<meta charset="utf-8" />
5+
<link rel="icon" href="%PUBLIC_URL%/favicon.ico" />
6+
<meta name="viewport" content="width=device-width, initial-scale=1" />
7+
<meta name="theme-color" content="#000000" />
8+
<meta
9+
name="description"
10+
content="Web site created using create-react-app"
11+
/>
12+
<link rel="apple-touch-icon" href="%PUBLIC_URL%/logo192.png" />
13+
<!--
14+
manifest.json provides metadata used when your web app is installed on a
15+
user's mobile device or desktop. See https://developers.google.com/web/fundamentals/web-app-manifest/
16+
-->
17+
<link rel="manifest" href="%PUBLIC_URL%/manifest.json" />
18+
<!--
19+
Notice the use of %PUBLIC_URL% in the tags above.
20+
It will be replaced with the URL of the `public` folder during the build.
21+
Only files inside the `public` folder can be referenced from the HTML.
22+
23+
Unlike "/favicon.ico" or "favicon.ico", "%PUBLIC_URL%/favicon.ico" will
24+
work correctly both with client-side routing and a non-root public URL.
25+
Learn how to configure a non-root public URL by running `npm run build`.
26+
-->
27+
<title>React App</title>
28+
</head>
29+
<body>
30+
<noscript>You need to enable JavaScript to run this app.</noscript>
31+
<div id="root"></div>
32+
<!--
33+
This HTML file is a template.
34+
If you open it directly in the browser, you will see an empty page.
35+
36+
You can add webfonts, meta tags, or analytics to this file.
37+
The build step will place the bundled scripts into the <body> tag.
38+
39+
To begin the development, run `npm start` or `yarn start`.
40+
To create a production bundle, use `npm run build` or `yarn build`.
41+
-->
42+
</body>
43+
</html>

‎react-client/public/logo192.png

5.22 KB
Loading

‎react-client/public/logo512.png

9.44 KB
Loading

‎react-client/public/manifest.json

Lines changed: 25 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
{
2+
"short_name": "React App",
3+
"name": "Create React App Sample",
4+
"icons": [
5+
{
6+
"src": "favicon.ico",
7+
"sizes": "64x64 32x32 24x24 16x16",
8+
"type": "image/x-icon"
9+
},
10+
{
11+
"src": "logo192.png",
12+
"type": "image/png",
13+
"sizes": "192x192"
14+
},
15+
{
16+
"src": "logo512.png",
17+
"type": "image/png",
18+
"sizes": "512x512"
19+
}
20+
],
21+
"start_url": ".",
22+
"display": "standalone",
23+
"theme_color": "#000000",
24+
"background_color": "#ffffff"
25+
}

‎react-client/public/robots.txt

Lines changed: 3 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
# https://www.robotstxt.org/robotstxt.html
2+
User-agent: *
3+
Disallow:

‎react-client/src/App.css

Lines changed: 15 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
.list {
2+
text-align: left;
3+
max-width: 750px;
4+
margin: auto;
5+
}
6+
7+
.submit-form {
8+
max-width: 300px;
9+
margin: auto;
10+
}
11+
12+
.edit-form {
13+
max-width: 300px;
14+
margin: auto;
15+
}

‎react-client/src/App.js

Lines changed: 44 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,44 @@
1+
import React, { Component } from "react";
2+
import { Switch, Route, Link } from "react-router-dom";
3+
import "bootstrap/dist/css/bootstrap.min.css";
4+
import "./App.css";
5+
6+
import AddTutorial from "./components/add-tutorial.component";
7+
import Tutorial from "./components/tutorial.component";
8+
import TutorialsList from "./components/tutorials-list.component";
9+
10+
class App extends Component {
11+
render() {
12+
return (
13+
<div>
14+
<nav className="navbar navbar-expand navbar-dark bg-dark">
15+
<Link to={"/tutorials"} className="navbar-brand">
16+
bezKoder
17+
</Link>
18+
<div className="navbar-nav mr-auto">
19+
<li className="nav-item">
20+
<Link to={"/tutorials"} className="nav-link">
21+
Tutorials
22+
</Link>
23+
</li>
24+
<li className="nav-item">
25+
<Link to={"/add"} className="nav-link">
26+
Add
27+
</Link>
28+
</li>
29+
</div>
30+
</nav>
31+
32+
<div className="container mt-3">
33+
<Switch>
34+
<Route exact path={["/", "/tutorials"]} component={TutorialsList} />
35+
<Route exact path="/add" component={AddTutorial} />
36+
<Route path="/tutorials/:id" component={Tutorial} />
37+
</Switch>
38+
</div>
39+
</div>
40+
);
41+
}
42+
}
43+
44+
export default App;
Lines changed: 114 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,114 @@
1+
import React, { Component } from "react";
2+
import TutorialDataService from "../services/tutorial.service";
3+
4+
export default class AddTutorial extends Component {
5+
constructor(props) {
6+
super(props);
7+
this.onChangeTitle = this.onChangeTitle.bind(this);
8+
this.onChangeDescription = this.onChangeDescription.bind(this);
9+
this.saveTutorial = this.saveTutorial.bind(this);
10+
this.newTutorial = this.newTutorial.bind(this);
11+
12+
this.state = {
13+
id: null,
14+
title: "",
15+
description: "",
16+
published: false,
17+
18+
submitted: false
19+
};
20+
}
21+
22+
onChangeTitle(e) {
23+
this.setState({
24+
title: e.target.value
25+
});
26+
}
27+
28+
onChangeDescription(e) {
29+
this.setState({
30+
description: e.target.value
31+
});
32+
}
33+
34+
saveTutorial() {
35+
var data = {
36+
title: this.state.title,
37+
description: this.state.description
38+
};
39+
40+
TutorialDataService.create(data)
41+
.then(response => {
42+
this.setState({
43+
id: response.data.id,
44+
title: response.data.title,
45+
description: response.data.description,
46+
published: response.data.published,
47+
48+
submitted: true
49+
});
50+
console.log(response.data);
51+
})
52+
.catch(e => {
53+
console.log(e);
54+
});
55+
}
56+
57+
newTutorial() {
58+
this.setState({
59+
id: null,
60+
title: "",
61+
description: "",
62+
published: false,
63+
64+
submitted: false
65+
});
66+
}
67+
68+
render() {
69+
return (
70+
<div className="submit-form">
71+
{this.state.submitted ? (
72+
<div>
73+
<h4>You submitted successfully!</h4>
74+
<button className="btn btn-success" onClick={this.newTutorial}>
75+
Add
76+
</button>
77+
</div>
78+
) : (
79+
<div>
80+
<div className="form-group">
81+
<label htmlFor="title">Title</label>
82+
<input
83+
type="text"
84+
className="form-control"
85+
id="title"
86+
required
87+
value={this.state.title}
88+
onChange={this.onChangeTitle}
89+
name="title"
90+
/>
91+
</div>
92+
93+
<div className="form-group">
94+
<label htmlFor="description">Description</label>
95+
<input
96+
type="text"
97+
className="form-control"
98+
id="description"
99+
required
100+
value={this.state.description}
101+
onChange={this.onChangeDescription}
102+
name="description"
103+
/>
104+
</div>
105+
106+
<button onClick={this.saveTutorial} className="btn btn-success">
107+
Submit
108+
</button>
109+
</div>
110+
)}
111+
</div>
112+
);
113+
}
114+
}
Lines changed: 195 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,195 @@
1+
import React, { Component } from "react";
2+
import TutorialDataService from "../services/tutorial.service";
3+
4+
export default class Tutorial extends Component {
5+
constructor(props) {
6+
super(props);
7+
this.onChangeTitle = this.onChangeTitle.bind(this);
8+
this.onChangeDescription = this.onChangeDescription.bind(this);
9+
this.getTutorial = this.getTutorial.bind(this);
10+
this.updatePublished = this.updatePublished.bind(this);
11+
this.updateTutorial = this.updateTutorial.bind(this);
12+
this.deleteTutorial = this.deleteTutorial.bind(this);
13+
14+
this.state = {
15+
currentTutorial: {
16+
id: null,
17+
title: "",
18+
description: "",
19+
published: false
20+
},
21+
message: ""
22+
};
23+
}
24+
25+
componentDidMount() {
26+
this.getTutorial(this.props.match.params.id);
27+
}
28+
29+
onChangeTitle(e) {
30+
const title = e.target.value;
31+
32+
this.setState(function(prevState) {
33+
return {
34+
currentTutorial: {
35+
...prevState.currentTutorial,
36+
title: title
37+
}
38+
};
39+
});
40+
}
41+
42+
onChangeDescription(e) {
43+
const description = e.target.value;
44+
45+
this.setState(prevState => ({
46+
currentTutorial: {
47+
...prevState.currentTutorial,
48+
description: description
49+
}
50+
}));
51+
}
52+
53+
getTutorial(id) {
54+
TutorialDataService.get(id)
55+
.then(response => {
56+
this.setState({
57+
currentTutorial: response.data
58+
});
59+
console.log(response.data);
60+
})
61+
.catch(e => {
62+
console.log(e);
63+
});
64+
}
65+
66+
updatePublished(status) {
67+
var data = {
68+
id: this.state.currentTutorial.id,
69+
title: this.state.currentTutorial.title,
70+
description: this.state.currentTutorial.description,
71+
published: status
72+
};
73+
74+
TutorialDataService.update(this.state.currentTutorial.id, data)
75+
.then(response => {
76+
this.setState(prevState => ({
77+
currentTutorial: {
78+
...prevState.currentTutorial,
79+
published: status
80+
}
81+
}));
82+
console.log(response.data);
83+
})
84+
.catch(e => {
85+
console.log(e);
86+
});
87+
}
88+
89+
updateTutorial() {
90+
TutorialDataService.update(
91+
this.state.currentTutorial.id,
92+
this.state.currentTutorial
93+
)
94+
.then(response => {
95+
console.log(response.data);
96+
this.setState({
97+
message: "The tutorial was updated successfully!"
98+
});
99+
})
100+
.catch(e => {
101+
console.log(e);
102+
});
103+
}
104+
105+
deleteTutorial() {
106+
TutorialDataService.delete(this.state.currentTutorial.id)
107+
.then(response => {
108+
console.log(response.data);
109+
this.props.history.push('/tutorials')
110+
})
111+
.catch(e => {
112+
console.log(e);
113+
});
114+
}
115+
116+
render() {
117+
const { currentTutorial } = this.state;
118+
119+
return (
120+
<div>
121+
{currentTutorial ? (
122+
<div className="edit-form">
123+
<h4>Tutorial</h4>
124+
<form>
125+
<div className="form-group">
126+
<label htmlFor="title">Title</label>
127+
<input
128+
type="text"
129+
className="form-control"
130+
id="title"
131+
value={currentTutorial.title}
132+
onChange={this.onChangeTitle}
133+
/>
134+
</div>
135+
<div className="form-group">
136+
<label htmlFor="description">Description</label>
137+
<input
138+
type="text"
139+
className="form-control"
140+
id="description"
141+
value={currentTutorial.description}
142+
onChange={this.onChangeDescription}
143+
/>
144+
</div>
145+
146+
<div className="form-group">
147+
<label>
148+
<strong>Status:</strong>
149+
</label>
150+
{currentTutorial.published ? "Published" : "Pending"}
151+
</div>
152+
</form>
153+
154+
{currentTutorial.published ? (
155+
<button
156+
className="badge badge-primary mr-2"
157+
onClick={() => this.updatePublished(false)}
158+
>
159+
UnPublish
160+
</button>
161+
) : (
162+
<button
163+
className="badge badge-primary mr-2"
164+
onClick={() => this.updatePublished(true)}
165+
>
166+
Publish
167+
</button>
168+
)}
169+
170+
<button
171+
className="badge badge-danger mr-2"
172+
onClick={this.deleteTutorial}
173+
>
174+
Delete
175+
</button>
176+
177+
<button
178+
type="submit"
179+
className="badge badge-success"
180+
onClick={this.updateTutorial}
181+
>
182+
Update
183+
</button>
184+
<p>{this.state.message}</p>
185+
</div>
186+
) : (
187+
<div>
188+
<br />
189+
<p>Please click on a Tutorial...</p>
190+
</div>
191+
)}
192+
</div>
193+
);
194+
}
195+
}
Lines changed: 183 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,183 @@
1+
import React, { Component } from "react";
2+
import TutorialDataService from "../services/tutorial.service";
3+
import { Link } from "react-router-dom";
4+
5+
export default class TutorialsList extends Component {
6+
constructor(props) {
7+
super(props);
8+
this.onChangeSearchTitle = this.onChangeSearchTitle.bind(this);
9+
this.retrieveTutorials = this.retrieveTutorials.bind(this);
10+
this.refreshList = this.refreshList.bind(this);
11+
this.setActiveTutorial = this.setActiveTutorial.bind(this);
12+
this.removeAllTutorials = this.removeAllTutorials.bind(this);
13+
this.searchTitle = this.searchTitle.bind(this);
14+
15+
this.state = {
16+
tutorials: [],
17+
currentTutorial: null,
18+
currentIndex: -1,
19+
searchTitle: ""
20+
};
21+
}
22+
23+
componentDidMount() {
24+
this.retrieveTutorials();
25+
}
26+
27+
onChangeSearchTitle(e) {
28+
const searchTitle = e.target.value;
29+
30+
this.setState({
31+
searchTitle: searchTitle
32+
});
33+
}
34+
35+
retrieveTutorials() {
36+
TutorialDataService.getAll()
37+
.then(response => {
38+
this.setState({
39+
tutorials: response.data
40+
});
41+
console.log(response.data);
42+
})
43+
.catch(e => {
44+
console.log(e);
45+
});
46+
}
47+
48+
refreshList() {
49+
this.retrieveTutorials();
50+
this.setState({
51+
currentTutorial: null,
52+
currentIndex: -1
53+
});
54+
}
55+
56+
setActiveTutorial(tutorial, index) {
57+
this.setState({
58+
currentTutorial: tutorial,
59+
currentIndex: index
60+
});
61+
}
62+
63+
removeAllTutorials() {
64+
TutorialDataService.deleteAll()
65+
.then(response => {
66+
console.log(response.data);
67+
this.refreshList();
68+
})
69+
.catch(e => {
70+
console.log(e);
71+
});
72+
}
73+
74+
searchTitle() {
75+
this.setState({
76+
currentTutorial: null,
77+
currentIndex: -1
78+
});
79+
80+
TutorialDataService.findByTitle(this.state.searchTitle)
81+
.then(response => {
82+
this.setState({
83+
tutorials: response.data
84+
});
85+
console.log(response.data);
86+
})
87+
.catch(e => {
88+
console.log(e);
89+
});
90+
}
91+
92+
render() {
93+
const { searchTitle, tutorials, currentTutorial, currentIndex } = this.state;
94+
95+
return (
96+
<div className="list row">
97+
<div className="col-md-8">
98+
<div className="input-group mb-3">
99+
<input
100+
type="text"
101+
className="form-control"
102+
placeholder="Search by title"
103+
value={searchTitle}
104+
onChange={this.onChangeSearchTitle}
105+
/>
106+
<div className="input-group-append">
107+
<button
108+
className="btn btn-outline-secondary"
109+
type="button"
110+
onClick={this.searchTitle}
111+
>
112+
Search
113+
</button>
114+
</div>
115+
</div>
116+
</div>
117+
<div className="col-md-6">
118+
<h4>Tutorials List</h4>
119+
120+
<ul className="list-group">
121+
{tutorials &&
122+
tutorials.map((tutorial, index) => (
123+
<li
124+
className={
125+
"list-group-item " +
126+
(index === currentIndex ? "active" : "")
127+
}
128+
onClick={() => this.setActiveTutorial(tutorial, index)}
129+
key={index}
130+
>
131+
{tutorial.title}
132+
</li>
133+
))}
134+
</ul>
135+
136+
<button
137+
className="m-3 btn btn-sm btn-danger"
138+
onClick={this.removeAllTutorials}
139+
>
140+
Remove All
141+
</button>
142+
</div>
143+
<div className="col-md-6">
144+
{currentTutorial ? (
145+
<div>
146+
<h4>Tutorial</h4>
147+
<div>
148+
<label>
149+
<strong>Title:</strong>
150+
</label>{" "}
151+
{currentTutorial.title}
152+
</div>
153+
<div>
154+
<label>
155+
<strong>Description:</strong>
156+
</label>{" "}
157+
{currentTutorial.description}
158+
</div>
159+
<div>
160+
<label>
161+
<strong>Status:</strong>
162+
</label>{" "}
163+
{currentTutorial.published ? "Published" : "Pending"}
164+
</div>
165+
166+
<Link
167+
to={"/tutorials/" + currentTutorial.id}
168+
className="badge badge-warning"
169+
>
170+
Edit
171+
</Link>
172+
</div>
173+
) : (
174+
<div>
175+
<br />
176+
<p>Please click on a Tutorial...</p>
177+
</div>
178+
)}
179+
</div>
180+
</div>
181+
);
182+
}
183+
}

‎react-client/src/http-common.js

Lines changed: 8 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
import axios from "axios";
2+
3+
export default axios.create({
4+
baseURL: "http://localhost:8080/api",
5+
headers: {
6+
"Content-type": "application/json"
7+
}
8+
});

‎react-client/src/index.js

Lines changed: 15 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
import React from "react";
2+
import ReactDOM from "react-dom";
3+
import { HashRouter } from "react-router-dom";
4+
5+
import App from "./App";
6+
import * as serviceWorker from "./serviceWorker";
7+
8+
ReactDOM.render(
9+
<HashRouter>
10+
<App />
11+
</HashRouter>,
12+
document.getElementById("root")
13+
);
14+
15+
serviceWorker.unregister();

‎react-client/src/logo.svg

Lines changed: 7 additions & 0 deletions
Loading

‎react-client/src/serviceWorker.js

Lines changed: 141 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,141 @@
1+
// This optional code is used to register a service worker.
2+
// register() is not called by default.
3+
4+
// This lets the app load faster on subsequent visits in production, and gives
5+
// it offline capabilities. However, it also means that developers (and users)
6+
// will only see deployed updates on subsequent visits to a page, after all the
7+
// existing tabs open on the page have been closed, since previously cached
8+
// resources are updated in the background.
9+
10+
// To learn more about the benefits of this model and instructions on how to
11+
// opt-in, read https://bit.ly/CRA-PWA
12+
13+
const isLocalhost = Boolean(
14+
window.location.hostname === 'localhost' ||
15+
// [::1] is the IPv6 localhost address.
16+
window.location.hostname === '[::1]' ||
17+
// 127.0.0.0/8 are considered localhost for IPv4.
18+
window.location.hostname.match(
19+
/^127(?:\.(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)){3}$/
20+
)
21+
);
22+
23+
export function register(config) {
24+
if (process.env.NODE_ENV === 'production' && 'serviceWorker' in navigator) {
25+
// The URL constructor is available in all browsers that support SW.
26+
const publicUrl = new URL(process.env.PUBLIC_URL, window.location.href);
27+
if (publicUrl.origin !== window.location.origin) {
28+
// Our service worker won't work if PUBLIC_URL is on a different origin
29+
// from what our page is served on. This might happen if a CDN is used to
30+
// serve assets; see https://github.com/facebook/create-react-app/issues/2374
31+
return;
32+
}
33+
34+
window.addEventListener('load', () => {
35+
const swUrl = `${process.env.PUBLIC_URL}/service-worker.js`;
36+
37+
if (isLocalhost) {
38+
// This is running on localhost. Let's check if a service worker still exists or not.
39+
checkValidServiceWorker(swUrl, config);
40+
41+
// Add some additional logging to localhost, pointing developers to the
42+
// service worker/PWA documentation.
43+
navigator.serviceWorker.ready.then(() => {
44+
console.log(
45+
'This web app is being served cache-first by a service ' +
46+
'worker. To learn more, visit https://bit.ly/CRA-PWA'
47+
);
48+
});
49+
} else {
50+
// Is not localhost. Just register service worker
51+
registerValidSW(swUrl, config);
52+
}
53+
});
54+
}
55+
}
56+
57+
function registerValidSW(swUrl, config) {
58+
navigator.serviceWorker
59+
.register(swUrl)
60+
.then(registration => {
61+
registration.onupdatefound = () => {
62+
const installingWorker = registration.installing;
63+
if (installingWorker == null) {
64+
return;
65+
}
66+
installingWorker.onstatechange = () => {
67+
if (installingWorker.state === 'installed') {
68+
if (navigator.serviceWorker.controller) {
69+
// At this point, the updated precached content has been fetched,
70+
// but the previous service worker will still serve the older
71+
// content until all client tabs are closed.
72+
console.log(
73+
'New content is available and will be used when all ' +
74+
'tabs for this page are closed. See https://bit.ly/CRA-PWA.'
75+
);
76+
77+
// Execute callback
78+
if (config && config.onUpdate) {
79+
config.onUpdate(registration);
80+
}
81+
} else {
82+
// At this point, everything has been precached.
83+
// It's the perfect time to display a
84+
// "Content is cached for offline use." message.
85+
console.log('Content is cached for offline use.');
86+
87+
// Execute callback
88+
if (config && config.onSuccess) {
89+
config.onSuccess(registration);
90+
}
91+
}
92+
}
93+
};
94+
};
95+
})
96+
.catch(error => {
97+
console.error('Error during service worker registration:', error);
98+
});
99+
}
100+
101+
function checkValidServiceWorker(swUrl, config) {
102+
// Check if the service worker can be found. If it can't reload the page.
103+
fetch(swUrl, {
104+
headers: { 'Service-Worker': 'script' }
105+
})
106+
.then(response => {
107+
// Ensure service worker exists, and that we really are getting a JS file.
108+
const contentType = response.headers.get('content-type');
109+
if (
110+
response.status === 404 ||
111+
(contentType != null && contentType.indexOf('javascript') === -1)
112+
) {
113+
// No service worker found. Probably a different app. Reload the page.
114+
navigator.serviceWorker.ready.then(registration => {
115+
registration.unregister().then(() => {
116+
window.location.reload();
117+
});
118+
});
119+
} else {
120+
// Service worker found. Proceed as normal.
121+
registerValidSW(swUrl, config);
122+
}
123+
})
124+
.catch(() => {
125+
console.log(
126+
'No internet connection found. App is running in offline mode.'
127+
);
128+
});
129+
}
130+
131+
export function unregister() {
132+
if ('serviceWorker' in navigator) {
133+
navigator.serviceWorker.ready
134+
.then(registration => {
135+
registration.unregister();
136+
})
137+
.catch(error => {
138+
console.error(error.message);
139+
});
140+
}
141+
}
Lines changed: 33 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,33 @@
1+
import http from "../http-common";
2+
3+
class TutorialDataService {
4+
getAll() {
5+
return http.get("/tutorials");
6+
}
7+
8+
get(id) {
9+
return http.get(`/tutorials/${id}`);
10+
}
11+
12+
create(data) {
13+
return http.post("/tutorials", data);
14+
}
15+
16+
update(id, data) {
17+
return http.put(`/tutorials/${id}`, data);
18+
}
19+
20+
delete(id) {
21+
return http.delete(`/tutorials/${id}`);
22+
}
23+
24+
deleteAll() {
25+
return http.delete(`/tutorials`);
26+
}
27+
28+
findByTitle(title) {
29+
return http.get(`/tutorials?title=${title}`);
30+
}
31+
}
32+
33+
export default new TutorialDataService();

‎react-client/src/setupTests.js

Lines changed: 5 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
// jest-dom adds custom jest matchers for asserting on DOM nodes.
2+
// allows you to do things like:
3+
// expect(element).toHaveTextContent(/react/i)
4+
// learn more: https://github.com/testing-library/jest-dom
5+
import '@testing-library/jest-dom/extend-expect';

‎react-client/yarn.lock

Lines changed: 10589 additions & 0 deletions
Large diffs are not rendered by default.

‎spring-boot-server/.gitignore

Lines changed: 33 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,33 @@
1+
HELP.md
2+
target/
3+
!.mvn/wrapper/maven-wrapper.jar
4+
!**/src/main/**/target/
5+
!**/src/test/**/target/
6+
7+
### STS ###
8+
.apt_generated
9+
.classpath
10+
.factorypath
11+
.project
12+
.settings
13+
.springBeans
14+
.sts4-cache
15+
16+
### IntelliJ IDEA ###
17+
.idea
18+
*.iws
19+
*.iml
20+
*.ipr
21+
22+
### NetBeans ###
23+
/nbproject/private/
24+
/nbbuild/
25+
/dist/
26+
/nbdist/
27+
/.nb-gradle/
28+
build/
29+
!**/src/main/**/build/
30+
!**/src/test/**/build/
31+
32+
### VS Code ###
33+
.vscode/
Lines changed: 117 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,117 @@
1+
/*
2+
* Copyright 2007-present the original author or authors.
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* https://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*/
16+
import java.net.*;
17+
import java.io.*;
18+
import java.nio.channels.*;
19+
import java.util.Properties;
20+
21+
public class MavenWrapperDownloader {
22+
23+
private static final String WRAPPER_VERSION = "0.5.6";
24+
/**
25+
* Default URL to download the maven-wrapper.jar from, if no 'downloadUrl' is provided.
26+
*/
27+
private static final String DEFAULT_DOWNLOAD_URL = "https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/"
28+
+ WRAPPER_VERSION + "/maven-wrapper-" + WRAPPER_VERSION + ".jar";
29+
30+
/**
31+
* Path to the maven-wrapper.properties file, which might contain a downloadUrl property to
32+
* use instead of the default one.
33+
*/
34+
private static final String MAVEN_WRAPPER_PROPERTIES_PATH =
35+
".mvn/wrapper/maven-wrapper.properties";
36+
37+
/**
38+
* Path where the maven-wrapper.jar will be saved to.
39+
*/
40+
private static final String MAVEN_WRAPPER_JAR_PATH =
41+
".mvn/wrapper/maven-wrapper.jar";
42+
43+
/**
44+
* Name of the property which should be used to override the default download url for the wrapper.
45+
*/
46+
private static final String PROPERTY_NAME_WRAPPER_URL = "wrapperUrl";
47+
48+
public static void main(String args[]) {
49+
System.out.println("- Downloader started");
50+
File baseDirectory = new File(args[0]);
51+
System.out.println("- Using base directory: " + baseDirectory.getAbsolutePath());
52+
53+
// If the maven-wrapper.properties exists, read it and check if it contains a custom
54+
// wrapperUrl parameter.
55+
File mavenWrapperPropertyFile = new File(baseDirectory, MAVEN_WRAPPER_PROPERTIES_PATH);
56+
String url = DEFAULT_DOWNLOAD_URL;
57+
if(mavenWrapperPropertyFile.exists()) {
58+
FileInputStream mavenWrapperPropertyFileInputStream = null;
59+
try {
60+
mavenWrapperPropertyFileInputStream = new FileInputStream(mavenWrapperPropertyFile);
61+
Properties mavenWrapperProperties = new Properties();
62+
mavenWrapperProperties.load(mavenWrapperPropertyFileInputStream);
63+
url = mavenWrapperProperties.getProperty(PROPERTY_NAME_WRAPPER_URL, url);
64+
} catch (IOException e) {
65+
System.out.println("- ERROR loading '" + MAVEN_WRAPPER_PROPERTIES_PATH + "'");
66+
} finally {
67+
try {
68+
if(mavenWrapperPropertyFileInputStream != null) {
69+
mavenWrapperPropertyFileInputStream.close();
70+
}
71+
} catch (IOException e) {
72+
// Ignore ...
73+
}
74+
}
75+
}
76+
System.out.println("- Downloading from: " + url);
77+
78+
File outputFile = new File(baseDirectory.getAbsolutePath(), MAVEN_WRAPPER_JAR_PATH);
79+
if(!outputFile.getParentFile().exists()) {
80+
if(!outputFile.getParentFile().mkdirs()) {
81+
System.out.println(
82+
"- ERROR creating output directory '" + outputFile.getParentFile().getAbsolutePath() + "'");
83+
}
84+
}
85+
System.out.println("- Downloading to: " + outputFile.getAbsolutePath());
86+
try {
87+
downloadFileFromURL(url, outputFile);
88+
System.out.println("Done");
89+
System.exit(0);
90+
} catch (Throwable e) {
91+
System.out.println("- Error downloading");
92+
e.printStackTrace();
93+
System.exit(1);
94+
}
95+
}
96+
97+
private static void downloadFileFromURL(String urlString, File destination) throws Exception {
98+
if (System.getenv("MVNW_USERNAME") != null && System.getenv("MVNW_PASSWORD") != null) {
99+
String username = System.getenv("MVNW_USERNAME");
100+
char[] password = System.getenv("MVNW_PASSWORD").toCharArray();
101+
Authenticator.setDefault(new Authenticator() {
102+
@Override
103+
protected PasswordAuthentication getPasswordAuthentication() {
104+
return new PasswordAuthentication(username, password);
105+
}
106+
});
107+
}
108+
URL website = new URL(urlString);
109+
ReadableByteChannel rbc;
110+
rbc = Channels.newChannel(website.openStream());
111+
FileOutputStream fos = new FileOutputStream(destination);
112+
fos.getChannel().transferFrom(rbc, 0, Long.MAX_VALUE);
113+
fos.close();
114+
rbc.close();
115+
}
116+
117+
}
Binary file not shown.
Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
distributionUrl=https://repo.maven.apache.org/maven2/org/apache/maven/apache-maven/3.6.3/apache-maven-3.6.3-bin.zip
2+
wrapperUrl=https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar

‎spring-boot-server/README.md

Lines changed: 41 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,41 @@
1+
# Spring Boot JPA MySQL - Building Rest CRUD API example
2+
3+
For more detail, please visit:
4+
> [Spring Boot JPA + MySQL - Building Rest CRUD API example](https://bezkoder.com/spring-boot-jpa-crud-rest-api/)
5+
6+
> [Spring Boot JPA + PostgreSQL - Building Rest CRUD API example](https://bezkoder.com/spring-boot-postgresql-example/)
7+
8+
> [Spring Boot + MongoDB - Building Rest CRUD API example](https://bezkoder.com/spring-boot-mongodb-crud/)
9+
10+
More Practice:
11+
> [Spring Boot Pagination & Filter example | Spring JPA, Pageable](https://bezkoder.com/spring-boot-pagination-filter-jpa-pageable/)
12+
13+
> [Spring Data JPA Sort/Order by multiple Columns | Spring Boot](https://bezkoder.com/spring-data-sort-multiple-columns/)
14+
15+
> [Spring Boot Repository Unit Test with @DataJpaTest](https://bezkoder.com/spring-boot-unit-test-jpa-repo-datajpatest/)
16+
17+
> [Deploy Spring Boot App on AWS – Elastic Beanstalk](https://bezkoder.com/deploy-spring-boot-aws-eb/)
18+
19+
Security:
20+
> [Spring Boot + Spring Security JWT Authentication & Authorization](https://bezkoder.com/spring-boot-jwt-authentication/)
21+
22+
Fullstack:
23+
> [Vue.js + Spring Boot + MySQL/PostgreSQL example](https://bezkoder.com/spring-boot-vue-js-crud-example/)
24+
25+
> [Angular 10 + Spring Boot + MySQL example](https://bezkoder.com/angular-10-spring-boot-crud/)
26+
27+
> [Angular 11 + Spring Boot + MySQL](https://bezkoder.com/angular-11-spring-boot-crud/)
28+
29+
> [Angular 10 + Spring Boot + PostgreSQL example](https://bezkoder.com/angular-10-spring-boot-postgresql/)
30+
31+
> [Angular 11 + Spring Boot + PostgreSQL example](https://bezkoder.com/angular-11-spring-boot-postgresql/)
32+
33+
> [React + Spring Boot + MySQL example](https://bezkoder.com/react-spring-boot-crud/)
34+
35+
> [React + Spring Boot + PostgreSQL example](https://bezkoder.com/spring-boot-react-postgresql/)
36+
37+
## Run Spring Boot application
38+
```
39+
mvn spring-boot:run
40+
```
41+

‎spring-boot-server/mvnw

Lines changed: 310 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,310 @@
1+
#!/bin/sh
2+
# ----------------------------------------------------------------------------
3+
# Licensed to the Apache Software Foundation (ASF) under one
4+
# or more contributor license agreements. See the NOTICE file
5+
# distributed with this work for additional information
6+
# regarding copyright ownership. The ASF licenses this file
7+
# to you under the Apache License, Version 2.0 (the
8+
# "License"); you may not use this file except in compliance
9+
# with the License. You may obtain a copy of the License at
10+
#
11+
# https://www.apache.org/licenses/LICENSE-2.0
12+
#
13+
# Unless required by applicable law or agreed to in writing,
14+
# software distributed under the License is distributed on an
15+
# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16+
# KIND, either express or implied. See the License for the
17+
# specific language governing permissions and limitations
18+
# under the License.
19+
# ----------------------------------------------------------------------------
20+
21+
# ----------------------------------------------------------------------------
22+
# Maven Start Up Batch script
23+
#
24+
# Required ENV vars:
25+
# ------------------
26+
# JAVA_HOME - location of a JDK home dir
27+
#
28+
# Optional ENV vars
29+
# -----------------
30+
# M2_HOME - location of maven2's installed home dir
31+
# MAVEN_OPTS - parameters passed to the Java VM when running Maven
32+
# e.g. to debug Maven itself, use
33+
# set MAVEN_OPTS=-Xdebug -Xrunjdwp:transport=dt_socket,server=y,suspend=y,address=8000
34+
# MAVEN_SKIP_RC - flag to disable loading of mavenrc files
35+
# ----------------------------------------------------------------------------
36+
37+
if [ -z "$MAVEN_SKIP_RC" ] ; then
38+
39+
if [ -f /etc/mavenrc ] ; then
40+
. /etc/mavenrc
41+
fi
42+
43+
if [ -f "$HOME/.mavenrc" ] ; then
44+
. "$HOME/.mavenrc"
45+
fi
46+
47+
fi
48+
49+
# OS specific support. $var _must_ be set to either true or false.
50+
cygwin=false;
51+
darwin=false;
52+
mingw=false
53+
case "`uname`" in
54+
CYGWIN*) cygwin=true ;;
55+
MINGW*) mingw=true;;
56+
Darwin*) darwin=true
57+
# Use /usr/libexec/java_home if available, otherwise fall back to /Library/Java/Home
58+
# See https://developer.apple.com/library/mac/qa/qa1170/_index.html
59+
if [ -z "$JAVA_HOME" ]; then
60+
if [ -x "/usr/libexec/java_home" ]; then
61+
export JAVA_HOME="`/usr/libexec/java_home`"
62+
else
63+
export JAVA_HOME="/Library/Java/Home"
64+
fi
65+
fi
66+
;;
67+
esac
68+
69+
if [ -z "$JAVA_HOME" ] ; then
70+
if [ -r /etc/gentoo-release ] ; then
71+
JAVA_HOME=`java-config --jre-home`
72+
fi
73+
fi
74+
75+
if [ -z "$M2_HOME" ] ; then
76+
## resolve links - $0 may be a link to maven's home
77+
PRG="$0"
78+
79+
# need this for relative symlinks
80+
while [ -h "$PRG" ] ; do
81+
ls=`ls -ld "$PRG"`
82+
link=`expr "$ls" : '.*-> \(.*\)$'`
83+
if expr "$link" : '/.*' > /dev/null; then
84+
PRG="$link"
85+
else
86+
PRG="`dirname "$PRG"`/$link"
87+
fi
88+
done
89+
90+
saveddir=`pwd`
91+
92+
M2_HOME=`dirname "$PRG"`/..
93+
94+
# make it fully qualified
95+
M2_HOME=`cd "$M2_HOME" && pwd`
96+
97+
cd "$saveddir"
98+
# echo Using m2 at $M2_HOME
99+
fi
100+
101+
# For Cygwin, ensure paths are in UNIX format before anything is touched
102+
if $cygwin ; then
103+
[ -n "$M2_HOME" ] &&
104+
M2_HOME=`cygpath --unix "$M2_HOME"`
105+
[ -n "$JAVA_HOME" ] &&
106+
JAVA_HOME=`cygpath --unix "$JAVA_HOME"`
107+
[ -n "$CLASSPATH" ] &&
108+
CLASSPATH=`cygpath --path --unix "$CLASSPATH"`
109+
fi
110+
111+
# For Mingw, ensure paths are in UNIX format before anything is touched
112+
if $mingw ; then
113+
[ -n "$M2_HOME" ] &&
114+
M2_HOME="`(cd "$M2_HOME"; pwd)`"
115+
[ -n "$JAVA_HOME" ] &&
116+
JAVA_HOME="`(cd "$JAVA_HOME"; pwd)`"
117+
fi
118+
119+
if [ -z "$JAVA_HOME" ]; then
120+
javaExecutable="`which javac`"
121+
if [ -n "$javaExecutable" ] && ! [ "`expr \"$javaExecutable\" : '\([^ ]*\)'`" = "no" ]; then
122+
# readlink(1) is not available as standard on Solaris 10.
123+
readLink=`which readlink`
124+
if [ ! `expr "$readLink" : '\([^ ]*\)'` = "no" ]; then
125+
if $darwin ; then
126+
javaHome="`dirname \"$javaExecutable\"`"
127+
javaExecutable="`cd \"$javaHome\" && pwd -P`/javac"
128+
else
129+
javaExecutable="`readlink -f \"$javaExecutable\"`"
130+
fi
131+
javaHome="`dirname \"$javaExecutable\"`"
132+
javaHome=`expr "$javaHome" : '\(.*\)/bin'`
133+
JAVA_HOME="$javaHome"
134+
export JAVA_HOME
135+
fi
136+
fi
137+
fi
138+
139+
if [ -z "$JAVACMD" ] ; then
140+
if [ -n "$JAVA_HOME" ] ; then
141+
if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
142+
# IBM's JDK on AIX uses strange locations for the executables
143+
JAVACMD="$JAVA_HOME/jre/sh/java"
144+
else
145+
JAVACMD="$JAVA_HOME/bin/java"
146+
fi
147+
else
148+
JAVACMD="`which java`"
149+
fi
150+
fi
151+
152+
if [ ! -x "$JAVACMD" ] ; then
153+
echo "Error: JAVA_HOME is not defined correctly." >&2
154+
echo " We cannot execute $JAVACMD" >&2
155+
exit 1
156+
fi
157+
158+
if [ -z "$JAVA_HOME" ] ; then
159+
echo "Warning: JAVA_HOME environment variable is not set."
160+
fi
161+
162+
CLASSWORLDS_LAUNCHER=org.codehaus.plexus.classworlds.launcher.Launcher
163+
164+
# traverses directory structure from process work directory to filesystem root
165+
# first directory with .mvn subdirectory is considered project base directory
166+
find_maven_basedir() {
167+
168+
if [ -z "$1" ]
169+
then
170+
echo "Path not specified to find_maven_basedir"
171+
return 1
172+
fi
173+
174+
basedir="$1"
175+
wdir="$1"
176+
while [ "$wdir" != '/' ] ; do
177+
if [ -d "$wdir"/.mvn ] ; then
178+
basedir=$wdir
179+
break
180+
fi
181+
# workaround for JBEAP-8937 (on Solaris 10/Sparc)
182+
if [ -d "${wdir}" ]; then
183+
wdir=`cd "$wdir/.."; pwd`
184+
fi
185+
# end of workaround
186+
done
187+
echo "${basedir}"
188+
}
189+
190+
# concatenates all lines of a file
191+
concat_lines() {
192+
if [ -f "$1" ]; then
193+
echo "$(tr -s '\n' ' ' < "$1")"
194+
fi
195+
}
196+
197+
BASE_DIR=`find_maven_basedir "$(pwd)"`
198+
if [ -z "$BASE_DIR" ]; then
199+
exit 1;
200+
fi
201+
202+
##########################################################################################
203+
# Extension to allow automatically downloading the maven-wrapper.jar from Maven-central
204+
# This allows using the maven wrapper in projects that prohibit checking in binary data.
205+
##########################################################################################
206+
if [ -r "$BASE_DIR/.mvn/wrapper/maven-wrapper.jar" ]; then
207+
if [ "$MVNW_VERBOSE" = true ]; then
208+
echo "Found .mvn/wrapper/maven-wrapper.jar"
209+
fi
210+
else
211+
if [ "$MVNW_VERBOSE" = true ]; then
212+
echo "Couldn't find .mvn/wrapper/maven-wrapper.jar, downloading it ..."
213+
fi
214+
if [ -n "$MVNW_REPOURL" ]; then
215+
jarUrl="$MVNW_REPOURL/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar"
216+
else
217+
jarUrl="https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar"
218+
fi
219+
while IFS="=" read key value; do
220+
case "$key" in (wrapperUrl) jarUrl="$value"; break ;;
221+
esac
222+
done < "$BASE_DIR/.mvn/wrapper/maven-wrapper.properties"
223+
if [ "$MVNW_VERBOSE" = true ]; then
224+
echo "Downloading from: $jarUrl"
225+
fi
226+
wrapperJarPath="$BASE_DIR/.mvn/wrapper/maven-wrapper.jar"
227+
if $cygwin; then
228+
wrapperJarPath=`cygpath --path --windows "$wrapperJarPath"`
229+
fi
230+
231+
if command -v wget > /dev/null; then
232+
if [ "$MVNW_VERBOSE" = true ]; then
233+
echo "Found wget ... using wget"
234+
fi
235+
if [ -z "$MVNW_USERNAME" ] || [ -z "$MVNW_PASSWORD" ]; then
236+
wget "$jarUrl" -O "$wrapperJarPath"
237+
else
238+
wget --http-user=$MVNW_USERNAME --http-password=$MVNW_PASSWORD "$jarUrl" -O "$wrapperJarPath"
239+
fi
240+
elif command -v curl > /dev/null; then
241+
if [ "$MVNW_VERBOSE" = true ]; then
242+
echo "Found curl ... using curl"
243+
fi
244+
if [ -z "$MVNW_USERNAME" ] || [ -z "$MVNW_PASSWORD" ]; then
245+
curl -o "$wrapperJarPath" "$jarUrl" -f
246+
else
247+
curl --user $MVNW_USERNAME:$MVNW_PASSWORD -o "$wrapperJarPath" "$jarUrl" -f
248+
fi
249+
250+
else
251+
if [ "$MVNW_VERBOSE" = true ]; then
252+
echo "Falling back to using Java to download"
253+
fi
254+
javaClass="$BASE_DIR/.mvn/wrapper/MavenWrapperDownloader.java"
255+
# For Cygwin, switch paths to Windows format before running javac
256+
if $cygwin; then
257+
javaClass=`cygpath --path --windows "$javaClass"`
258+
fi
259+
if [ -e "$javaClass" ]; then
260+
if [ ! -e "$BASE_DIR/.mvn/wrapper/MavenWrapperDownloader.class" ]; then
261+
if [ "$MVNW_VERBOSE" = true ]; then
262+
echo " - Compiling MavenWrapperDownloader.java ..."
263+
fi
264+
# Compiling the Java class
265+
("$JAVA_HOME/bin/javac" "$javaClass")
266+
fi
267+
if [ -e "$BASE_DIR/.mvn/wrapper/MavenWrapperDownloader.class" ]; then
268+
# Running the downloader
269+
if [ "$MVNW_VERBOSE" = true ]; then
270+
echo " - Running MavenWrapperDownloader.java ..."
271+
fi
272+
("$JAVA_HOME/bin/java" -cp .mvn/wrapper MavenWrapperDownloader "$MAVEN_PROJECTBASEDIR")
273+
fi
274+
fi
275+
fi
276+
fi
277+
##########################################################################################
278+
# End of extension
279+
##########################################################################################
280+
281+
export MAVEN_PROJECTBASEDIR=${MAVEN_BASEDIR:-"$BASE_DIR"}
282+
if [ "$MVNW_VERBOSE" = true ]; then
283+
echo $MAVEN_PROJECTBASEDIR
284+
fi
285+
MAVEN_OPTS="$(concat_lines "$MAVEN_PROJECTBASEDIR/.mvn/jvm.config") $MAVEN_OPTS"
286+
287+
# For Cygwin, switch paths to Windows format before running java
288+
if $cygwin; then
289+
[ -n "$M2_HOME" ] &&
290+
M2_HOME=`cygpath --path --windows "$M2_HOME"`
291+
[ -n "$JAVA_HOME" ] &&
292+
JAVA_HOME=`cygpath --path --windows "$JAVA_HOME"`
293+
[ -n "$CLASSPATH" ] &&
294+
CLASSPATH=`cygpath --path --windows "$CLASSPATH"`
295+
[ -n "$MAVEN_PROJECTBASEDIR" ] &&
296+
MAVEN_PROJECTBASEDIR=`cygpath --path --windows "$MAVEN_PROJECTBASEDIR"`
297+
fi
298+
299+
# Provide a "standardized" way to retrieve the CLI args that will
300+
# work with both Windows and non-Windows executions.
301+
MAVEN_CMD_LINE_ARGS="$MAVEN_CONFIG $@"
302+
export MAVEN_CMD_LINE_ARGS
303+
304+
WRAPPER_LAUNCHER=org.apache.maven.wrapper.MavenWrapperMain
305+
306+
exec "$JAVACMD" \
307+
$MAVEN_OPTS \
308+
-classpath "$MAVEN_PROJECTBASEDIR/.mvn/wrapper/maven-wrapper.jar" \
309+
"-Dmaven.home=${M2_HOME}" "-Dmaven.multiModuleProjectDirectory=${MAVEN_PROJECTBASEDIR}" \
310+
${WRAPPER_LAUNCHER} $MAVEN_CONFIG "$@"

‎spring-boot-server/mvnw.cmd

Lines changed: 182 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,182 @@
1+
@REM ----------------------------------------------------------------------------
2+
@REM Licensed to the Apache Software Foundation (ASF) under one
3+
@REM or more contributor license agreements. See the NOTICE file
4+
@REM distributed with this work for additional information
5+
@REM regarding copyright ownership. The ASF licenses this file
6+
@REM to you under the Apache License, Version 2.0 (the
7+
@REM "License"); you may not use this file except in compliance
8+
@REM with the License. You may obtain a copy of the License at
9+
@REM
10+
@REM https://www.apache.org/licenses/LICENSE-2.0
11+
@REM
12+
@REM Unless required by applicable law or agreed to in writing,
13+
@REM software distributed under the License is distributed on an
14+
@REM "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
15+
@REM KIND, either express or implied. See the License for the
16+
@REM specific language governing permissions and limitations
17+
@REM under the License.
18+
@REM ----------------------------------------------------------------------------
19+
20+
@REM ----------------------------------------------------------------------------
21+
@REM Maven Start Up Batch script
22+
@REM
23+
@REM Required ENV vars:
24+
@REM JAVA_HOME - location of a JDK home dir
25+
@REM
26+
@REM Optional ENV vars
27+
@REM M2_HOME - location of maven2's installed home dir
28+
@REM MAVEN_BATCH_ECHO - set to 'on' to enable the echoing of the batch commands
29+
@REM MAVEN_BATCH_PAUSE - set to 'on' to wait for a keystroke before ending
30+
@REM MAVEN_OPTS - parameters passed to the Java VM when running Maven
31+
@REM e.g. to debug Maven itself, use
32+
@REM set MAVEN_OPTS=-Xdebug -Xrunjdwp:transport=dt_socket,server=y,suspend=y,address=8000
33+
@REM MAVEN_SKIP_RC - flag to disable loading of mavenrc files
34+
@REM ----------------------------------------------------------------------------
35+
36+
@REM Begin all REM lines with '@' in case MAVEN_BATCH_ECHO is 'on'
37+
@echo off
38+
@REM set title of command window
39+
title %0
40+
@REM enable echoing by setting MAVEN_BATCH_ECHO to 'on'
41+
@if "%MAVEN_BATCH_ECHO%" == "on" echo %MAVEN_BATCH_ECHO%
42+
43+
@REM set %HOME% to equivalent of $HOME
44+
if "%HOME%" == "" (set "HOME=%HOMEDRIVE%%HOMEPATH%")
45+
46+
@REM Execute a user defined script before this one
47+
if not "%MAVEN_SKIP_RC%" == "" goto skipRcPre
48+
@REM check for pre script, once with legacy .bat ending and once with .cmd ending
49+
if exist "%HOME%\mavenrc_pre.bat" call "%HOME%\mavenrc_pre.bat"
50+
if exist "%HOME%\mavenrc_pre.cmd" call "%HOME%\mavenrc_pre.cmd"
51+
:skipRcPre
52+
53+
@setlocal
54+
55+
set ERROR_CODE=0
56+
57+
@REM To isolate internal variables from possible post scripts, we use another setlocal
58+
@setlocal
59+
60+
@REM ==== START VALIDATION ====
61+
if not "%JAVA_HOME%" == "" goto OkJHome
62+
63+
echo.
64+
echo Error: JAVA_HOME not found in your environment. >&2
65+
echo Please set the JAVA_HOME variable in your environment to match the >&2
66+
echo location of your Java installation. >&2
67+
echo.
68+
goto error
69+
70+
:OkJHome
71+
if exist "%JAVA_HOME%\bin\java.exe" goto init
72+
73+
echo.
74+
echo Error: JAVA_HOME is set to an invalid directory. >&2
75+
echo JAVA_HOME = "%JAVA_HOME%" >&2
76+
echo Please set the JAVA_HOME variable in your environment to match the >&2
77+
echo location of your Java installation. >&2
78+
echo.
79+
goto error
80+
81+
@REM ==== END VALIDATION ====
82+
83+
:init
84+
85+
@REM Find the project base dir, i.e. the directory that contains the folder ".mvn".
86+
@REM Fallback to current working directory if not found.
87+
88+
set MAVEN_PROJECTBASEDIR=%MAVEN_BASEDIR%
89+
IF NOT "%MAVEN_PROJECTBASEDIR%"=="" goto endDetectBaseDir
90+
91+
set EXEC_DIR=%CD%
92+
set WDIR=%EXEC_DIR%
93+
:findBaseDir
94+
IF EXIST "%WDIR%"\.mvn goto baseDirFound
95+
cd ..
96+
IF "%WDIR%"=="%CD%" goto baseDirNotFound
97+
set WDIR=%CD%
98+
goto findBaseDir
99+
100+
:baseDirFound
101+
set MAVEN_PROJECTBASEDIR=%WDIR%
102+
cd "%EXEC_DIR%"
103+
goto endDetectBaseDir
104+
105+
:baseDirNotFound
106+
set MAVEN_PROJECTBASEDIR=%EXEC_DIR%
107+
cd "%EXEC_DIR%"
108+
109+
:endDetectBaseDir
110+
111+
IF NOT EXIST "%MAVEN_PROJECTBASEDIR%\.mvn\jvm.config" goto endReadAdditionalConfig
112+
113+
@setlocal EnableExtensions EnableDelayedExpansion
114+
for /F "usebackq delims=" %%a in ("%MAVEN_PROJECTBASEDIR%\.mvn\jvm.config") do set JVM_CONFIG_MAVEN_PROPS=!JVM_CONFIG_MAVEN_PROPS! %%a
115+
@endlocal & set JVM_CONFIG_MAVEN_PROPS=%JVM_CONFIG_MAVEN_PROPS%
116+
117+
:endReadAdditionalConfig
118+
119+
SET MAVEN_JAVA_EXE="%JAVA_HOME%\bin\java.exe"
120+
set WRAPPER_JAR="%MAVEN_PROJECTBASEDIR%\.mvn\wrapper\maven-wrapper.jar"
121+
set WRAPPER_LAUNCHER=org.apache.maven.wrapper.MavenWrapperMain
122+
123+
set DOWNLOAD_URL="https://repo.maven.apache.org/maven2/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar"
124+
125+
FOR /F "tokens=1,2 delims==" %%A IN ("%MAVEN_PROJECTBASEDIR%\.mvn\wrapper\maven-wrapper.properties") DO (
126+
IF "%%A"=="wrapperUrl" SET DOWNLOAD_URL=%%B
127+
)
128+
129+
@REM Extension to allow automatically downloading the maven-wrapper.jar from Maven-central
130+
@REM This allows using the maven wrapper in projects that prohibit checking in binary data.
131+
if exist %WRAPPER_JAR% (
132+
if "%MVNW_VERBOSE%" == "true" (
133+
echo Found %WRAPPER_JAR%
134+
)
135+
) else (
136+
if not "%MVNW_REPOURL%" == "" (
137+
SET DOWNLOAD_URL="%MVNW_REPOURL%/io/takari/maven-wrapper/0.5.6/maven-wrapper-0.5.6.jar"
138+
)
139+
if "%MVNW_VERBOSE%" == "true" (
140+
echo Couldn't find %WRAPPER_JAR%, downloading it ...
141+
echo Downloading from: %DOWNLOAD_URL%
142+
)
143+
144+
powershell -Command "&{"^
145+
"$webclient = new-object System.Net.WebClient;"^
146+
"if (-not ([string]::IsNullOrEmpty('%MVNW_USERNAME%') -and [string]::IsNullOrEmpty('%MVNW_PASSWORD%'))) {"^
147+
"$webclient.Credentials = new-object System.Net.NetworkCredential('%MVNW_USERNAME%', '%MVNW_PASSWORD%');"^
148+
"}"^
149+
"[Net.ServicePointManager]::SecurityProtocol = [Net.SecurityProtocolType]::Tls12; $webclient.DownloadFile('%DOWNLOAD_URL%', '%WRAPPER_JAR%')"^
150+
"}"
151+
if "%MVNW_VERBOSE%" == "true" (
152+
echo Finished downloading %WRAPPER_JAR%
153+
)
154+
)
155+
@REM End of extension
156+
157+
@REM Provide a "standardized" way to retrieve the CLI args that will
158+
@REM work with both Windows and non-Windows executions.
159+
set MAVEN_CMD_LINE_ARGS=%*
160+
161+
%MAVEN_JAVA_EXE% %JVM_CONFIG_MAVEN_PROPS% %MAVEN_OPTS% %MAVEN_DEBUG_OPTS% -classpath %WRAPPER_JAR% "-Dmaven.multiModuleProjectDirectory=%MAVEN_PROJECTBASEDIR%" %WRAPPER_LAUNCHER% %MAVEN_CONFIG% %*
162+
if ERRORLEVEL 1 goto error
163+
goto end
164+
165+
:error
166+
set ERROR_CODE=1
167+
168+
:end
169+
@endlocal & set ERROR_CODE=%ERROR_CODE%
170+
171+
if not "%MAVEN_SKIP_RC%" == "" goto skipRcPost
172+
@REM check for post script, once with legacy .bat ending and once with .cmd ending
173+
if exist "%HOME%\mavenrc_post.bat" call "%HOME%\mavenrc_post.bat"
174+
if exist "%HOME%\mavenrc_post.cmd" call "%HOME%\mavenrc_post.cmd"
175+
:skipRcPost
176+
177+
@REM pause the script if MAVEN_BATCH_PAUSE is set to 'on'
178+
if "%MAVEN_BATCH_PAUSE%" == "on" pause
179+
180+
if "%MAVEN_TERMINATE_CMD%" == "on" exit %ERROR_CODE%
181+
182+
exit /B %ERROR_CODE%

‎spring-boot-server/pom.xml

Lines changed: 82 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,82 @@
1+
<?xml version="1.0" encoding="UTF-8"?>
2+
<project xmlns="http://maven.apache.org/POM/4.0.0"
3+
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
4+
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
5+
<modelVersion>4.0.0</modelVersion>
6+
<parent>
7+
<groupId>org.springframework.boot</groupId>
8+
<artifactId>spring-boot-starter-parent</artifactId>
9+
<version>2.3.5.RELEASE</version>
10+
<relativePath /> <!-- lookup parent from repository -->
11+
</parent>
12+
<groupId>com.bezkoder</groupId>
13+
<artifactId>spring-boot-react-mysql</artifactId>
14+
<version>0.0.1-SNAPSHOT</version>
15+
<name>spring-boot-react-mysql</name>
16+
<description>How to Integrate React.js with Spring Boot</description>
17+
18+
<properties>
19+
<java.version>1.8</java.version>
20+
</properties>
21+
22+
<dependencies>
23+
<dependency>
24+
<groupId>org.springframework.boot</groupId>
25+
<artifactId>spring-boot-starter-data-jpa</artifactId>
26+
</dependency>
27+
28+
<dependency>
29+
<groupId>org.springframework.boot</groupId>
30+
<artifactId>spring-boot-starter-web</artifactId>
31+
</dependency>
32+
33+
<dependency>
34+
<groupId>mysql</groupId>
35+
<artifactId>mysql-connector-java</artifactId>
36+
<scope>runtime</scope>
37+
</dependency>
38+
39+
<dependency>
40+
<groupId>org.springframework.boot</groupId>
41+
<artifactId>spring-boot-starter-test</artifactId>
42+
<scope>test</scope>
43+
<exclusions>
44+
<exclusion>
45+
<groupId>org.junit.vintage</groupId>
46+
<artifactId>junit-vintage-engine</artifactId>
47+
</exclusion>
48+
</exclusions>
49+
</dependency>
50+
</dependencies>
51+
52+
<build>
53+
<plugins>
54+
<plugin>
55+
<groupId>org.springframework.boot</groupId>
56+
<artifactId>spring-boot-maven-plugin</artifactId>
57+
</plugin>
58+
59+
<plugin>
60+
<artifactId>maven-resources-plugin</artifactId>
61+
<executions>
62+
<execution>
63+
<id>copy-resources</id>
64+
<phase>validate</phase>
65+
<goals>
66+
<goal>copy-resources</goal>
67+
</goals>
68+
<configuration>
69+
<outputDirectory>${basedir}/target/classes/static/</outputDirectory>
70+
<resources>
71+
<resource>
72+
<directory>${basedir}/../react-crud/build</directory>
73+
</resource>
74+
</resources>
75+
</configuration>
76+
</execution>
77+
</executions>
78+
</plugin>
79+
</plugins>
80+
</build>
81+
82+
</project>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
package com.bezkoder.integrate.spring.react;
2+
3+
import org.springframework.boot.SpringApplication;
4+
import org.springframework.boot.autoconfigure.SpringBootApplication;
5+
6+
@SpringBootApplication
7+
public class SpringBootReactMysqlApplication {
8+
9+
public static void main(String[] args) {
10+
SpringApplication.run(SpringBootReactMysqlApplication.class, args);
11+
}
12+
13+
}
Lines changed: 124 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,124 @@
1+
package com.bezkoder.integrate.spring.react.controller;
2+
3+
import java.util.ArrayList;
4+
import java.util.List;
5+
import java.util.Optional;
6+
7+
import org.springframework.beans.factory.annotation.Autowired;
8+
import org.springframework.http.HttpStatus;
9+
import org.springframework.http.ResponseEntity;
10+
import org.springframework.web.bind.annotation.CrossOrigin;
11+
import org.springframework.web.bind.annotation.DeleteMapping;
12+
import org.springframework.web.bind.annotation.GetMapping;
13+
import org.springframework.web.bind.annotation.PathVariable;
14+
import org.springframework.web.bind.annotation.PostMapping;
15+
import org.springframework.web.bind.annotation.PutMapping;
16+
import org.springframework.web.bind.annotation.RequestBody;
17+
import org.springframework.web.bind.annotation.RequestMapping;
18+
import org.springframework.web.bind.annotation.RequestParam;
19+
import org.springframework.web.bind.annotation.RestController;
20+
21+
import com.bezkoder.integrate.spring.react.model.Tutorial;
22+
import com.bezkoder.integrate.spring.react.repository.TutorialRepository;
23+
24+
@CrossOrigin(origins = "http://localhost:8081")
25+
@RestController
26+
@RequestMapping("/api")
27+
public class TutorialController {
28+
29+
@Autowired
30+
TutorialRepository tutorialRepository;
31+
32+
@GetMapping("/tutorials")
33+
public ResponseEntity<List<Tutorial>> getAllTutorials(@RequestParam(required = false) String title) {
34+
try {
35+
List<Tutorial> tutorials = new ArrayList<Tutorial>();
36+
37+
if (title == null)
38+
tutorialRepository.findAll().forEach(tutorials::add);
39+
else
40+
tutorialRepository.findByTitleContaining(title).forEach(tutorials::add);
41+
42+
if (tutorials.isEmpty()) {
43+
return new ResponseEntity<>(HttpStatus.NO_CONTENT);
44+
}
45+
46+
return new ResponseEntity<>(tutorials, HttpStatus.OK);
47+
} catch (Exception e) {
48+
return new ResponseEntity<>(null, HttpStatus.INTERNAL_SERVER_ERROR);
49+
}
50+
}
51+
52+
@GetMapping("/tutorials/{id}")
53+
public ResponseEntity<Tutorial> getTutorialById(@PathVariable("id") long id) {
54+
Optional<Tutorial> tutorialData = tutorialRepository.findById(id);
55+
56+
if (tutorialData.isPresent()) {
57+
return new ResponseEntity<>(tutorialData.get(), HttpStatus.OK);
58+
} else {
59+
return new ResponseEntity<>(HttpStatus.NOT_FOUND);
60+
}
61+
}
62+
63+
@PostMapping("/tutorials")
64+
public ResponseEntity<Tutorial> createTutorial(@RequestBody Tutorial tutorial) {
65+
try {
66+
Tutorial _tutorial = tutorialRepository
67+
.save(new Tutorial(tutorial.getTitle(), tutorial.getDescription(), false));
68+
return new ResponseEntity<>(_tutorial, HttpStatus.CREATED);
69+
} catch (Exception e) {
70+
return new ResponseEntity<>(null, HttpStatus.EXPECTATION_FAILED);
71+
}
72+
}
73+
74+
@PutMapping("/tutorials/{id}")
75+
public ResponseEntity<Tutorial> updateTutorial(@PathVariable("id") long id, @RequestBody Tutorial tutorial) {
76+
Optional<Tutorial> tutorialData = tutorialRepository.findById(id);
77+
78+
if (tutorialData.isPresent()) {
79+
Tutorial _tutorial = tutorialData.get();
80+
_tutorial.setTitle(tutorial.getTitle());
81+
_tutorial.setDescription(tutorial.getDescription());
82+
_tutorial.setPublished(tutorial.isPublished());
83+
return new ResponseEntity<>(tutorialRepository.save(_tutorial), HttpStatus.OK);
84+
} else {
85+
return new ResponseEntity<>(HttpStatus.NOT_FOUND);
86+
}
87+
}
88+
89+
@DeleteMapping("/tutorials/{id}")
90+
public ResponseEntity<HttpStatus> deleteTutorial(@PathVariable("id") long id) {
91+
try {
92+
tutorialRepository.deleteById(id);
93+
return new ResponseEntity<>(HttpStatus.NO_CONTENT);
94+
} catch (Exception e) {
95+
return new ResponseEntity<>(HttpStatus.EXPECTATION_FAILED);
96+
}
97+
}
98+
99+
@DeleteMapping("/tutorials")
100+
public ResponseEntity<HttpStatus> deleteAllTutorials() {
101+
try {
102+
tutorialRepository.deleteAll();
103+
return new ResponseEntity<>(HttpStatus.NO_CONTENT);
104+
} catch (Exception e) {
105+
return new ResponseEntity<>(HttpStatus.EXPECTATION_FAILED);
106+
}
107+
108+
}
109+
110+
@GetMapping("/tutorials/published")
111+
public ResponseEntity<List<Tutorial>> findByPublished() {
112+
try {
113+
List<Tutorial> tutorials = tutorialRepository.findByPublished(true);
114+
115+
if (tutorials.isEmpty()) {
116+
return new ResponseEntity<>(HttpStatus.NO_CONTENT);
117+
}
118+
return new ResponseEntity<>(tutorials, HttpStatus.OK);
119+
} catch (Exception e) {
120+
return new ResponseEntity<>(HttpStatus.EXPECTATION_FAILED);
121+
}
122+
}
123+
124+
}
Lines changed: 65 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,65 @@
1+
package com.bezkoder.integrate.spring.react.model;
2+
3+
import javax.persistence.*;
4+
5+
@Entity
6+
@Table(name = "tutorials")
7+
public class Tutorial {
8+
9+
@Id
10+
@GeneratedValue(strategy = GenerationType.AUTO)
11+
private long id;
12+
13+
@Column(name = "title")
14+
private String title;
15+
16+
@Column(name = "description")
17+
private String description;
18+
19+
@Column(name = "published")
20+
private boolean published;
21+
22+
public Tutorial() {
23+
24+
}
25+
26+
public Tutorial(String title, String description, boolean published) {
27+
this.title = title;
28+
this.description = description;
29+
this.published = published;
30+
}
31+
32+
public long getId() {
33+
return id;
34+
}
35+
36+
public String getTitle() {
37+
return title;
38+
}
39+
40+
public void setTitle(String title) {
41+
this.title = title;
42+
}
43+
44+
public String getDescription() {
45+
return description;
46+
}
47+
48+
public void setDescription(String description) {
49+
this.description = description;
50+
}
51+
52+
public boolean isPublished() {
53+
return published;
54+
}
55+
56+
public void setPublished(boolean isPublished) {
57+
this.published = isPublished;
58+
}
59+
60+
@Override
61+
public String toString() {
62+
return "Tutorial [id=" + id + ", title=" + title + ", desc=" + description + ", published=" + published + "]";
63+
}
64+
65+
}
Lines changed: 12 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
package com.bezkoder.integrate.spring.react.repository;
2+
3+
import java.util.List;
4+
5+
import org.springframework.data.jpa.repository.JpaRepository;
6+
7+
import com.bezkoder.integrate.spring.react.model.Tutorial;
8+
9+
public interface TutorialRepository extends JpaRepository<Tutorial, Long> {
10+
List<Tutorial> findByPublished(boolean published);
11+
List<Tutorial> findByTitleContaining(String title);
12+
}
Lines changed: 6 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
spring.datasource.url= jdbc:mysql://localhost:3306/testdb?useSSL=false
2+
spring.datasource.username= root
3+
spring.datasource.password= 123456
4+
5+
spring.jpa.properties.hibernate.dialect= org.hibernate.dialect.MySQL5InnoDBDialect
6+
spring.jpa.hibernate.ddl-auto= update
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
package com.bezkoder.integrate.spring.react;
2+
3+
import org.junit.jupiter.api.Test;
4+
import org.springframework.boot.test.context.SpringBootTest;
5+
6+
@SpringBootTest
7+
class SpringBootReactMysqlApplicationTests {
8+
9+
@Test
10+
void contextLoads() {
11+
}
12+
13+
}

0 commit comments

Comments
 (0)
Please sign in to comment.