타입스크립트 제가 한번 써봤습니다.
김병진 x Flitto
자바스크립트로 개발한다는 것
userService.insertUser(user, function(res) {
...
});
insertUser(user, callback) {
return $http.post('/api/users', user);
}
front-end code
user-controller.js
user-service.js
back-end code
user-controller.js
user-service.js
router.route('/users')
.post(checkPermission, function(req, res, next) {
userService.insertUser(req.body.user, function(err, result) {
return res.send(result);
});
})
insertUserService(options, callback) {
userModel.insert(options.users, callback);
}
내 삶이 힘든 이유
테스트 케이스
문서화
welcome to hell
userService.insertUser(user, function(res) {
debugger
});
insertUser(user, callback) {
debugger
return $http.post('/api/users', user);
}
router.route('/users')
.post(checkPermission, function(req, res, next) {
userService.insertUser(req.user, function(err, result) {
debugger
return res.send(result);
});
})
insertUserService(options, callback) {
debugger
userModel.insert(options.users, callback);
}
userService.insertUser()
INPUT
??????
OUTPUT
??????????
문제점
해결책 1
if ( typeof data.username === 'string') ....
if ( typeof data.username === 'string') ....
if ( typeof data.username === 'string') ....
if ( typeof data.username === 'string') ....
if ( typeof data.username === 'string') ....
if ( typeof data.username === 'string') ....
if ( typeof data.username === 'string') ....
if ( typeof data.username === 'string') ....
if ( typeof data.username === 'string') ....
if ( typeof data.username === 'string') ....
if ( typeof data.username === 'string') ....
저 어드민 개발 안 할래요
해결 2
개혁
typescript
function greeter(person: string) {
return "Hello, " + person;
}
let user = [1,2,3];
greeter(user)
index.ts
$ tsc index.ts
error TS2345: Argument of type 'number[]' is not assignable to parameter of type 'string'.
$ npm install -g typescript
front-end
INPUT
userService.insertUser()
INPUT
??????
OUTPUT
??????????
interface
interface User {
id?: string;
username: string;
email: string;
}
interface combination
interface User {
id?: string;
username: string;
email: string;
}
interface Settings {
allow_email: string;
allow_push: string;
language: string;
}
interface UserWithSettings {
user: User;
settings: Settings;
}
interface extends
interface Account {
account_name: string;
rrn: string;
}
interface Settings {
allow_email: string;
allow_push: string;
language: string;
}
interface User extends Settings, Account {
id?: string;
username: string;
email: string;
}
import {User} from '../interfaces/User';
...
userService.insertUser(user: User)
.subscribe(res => {
this.user = res;
});
import {User} from '../interfaces/User';
...
insertUser(user: User) {
return this.http.post('/api/users', {user});
}
front-end
user-controller.ts
user-service.ts
input type check
output
userService.insertUser()
INPUT
Interface User
OUTPUT
??????????
output
userService.insertUser(user: User)
.subscribe(res => {
this.user = res;
})
userService.insertUser(user: User): Observable<User> {
return this.http.post<User>('/api/users', {user});
}
user-controller
user-service
output check
아하!
userService.insertUser()
INPUT
userInterface
OUTPUT
Observable<userInterface>
node.js
with typescript
back-end
user-controller.ts
user-service.ts
import {User} from '../interfaces/User';
...
router.route('/users')
.post(checkPermission, (req, res, next) => {
const user: User = {
username: req.body.user.username,
email: req.body.user.email
}
userService.insertUser(user) {
return res.send(result);
});
});
import {User} from '../interfaces/User';
...
insertUserService(user: User): Promise<User> {
return userModel.insert(options.users);
}
DUPLICATED type
Front-end
Back-end
UserInterface
UserInterface
shared type
Front-end
Back-end
UserInterface
interface User {
id?: string;
username: string;
email: string;
}
import {User} from "../../../../shared/interfaces/user";
export class UserService {
insertUser(user: User) {
...
}
}
import {User} from "../../shared/interfaces/user";
export class UserService {
insertUser(user: User) {
...
}
}
back-end
front-end
shared type
shared
shared type
일관된 형태의 데이터로 프론트엔드와 백엔드 통신
more typescript
decorator
function Logger(target) {
console.log(`${target.name} is declared`)
}
@Logger
class Greeter {
say() {
console.log('hi');
}
}
$ tsc index.ts --experimentalDecorators
Greeter is declared
index.ts
ts-express-decorator
framework for Express
with TypeScript and Decorators
server setting
@ServerSettings({
rootDir,
mount: {
'': `${rootDir}/**/**Controller.js`
},
port: config.port,
acceptMimes: ['application/json'],
componentsScan: [
`${rootDir}/**/**Service.js`
],
serveStatic: {
'/': path.join(rootDir, '../public')
}
})
export default class Server extends ServerLoader {
...
}
cross cutting concern
user-controller.ts
router.route('/users')
.post(checkPermission, (req, res, next) => {
const user: User = {
username: req.user.username,
email: req.user.email
}
userService.insertUser(user) {
return res.send(result);
});
})
order-controller.ts
router.route('/orders')
.post(checkPermission, (req, res, next) => {
const order: Order = {
order_num: req.body.orders.order_num,
type: req.body.orders.type
}
orderService.insertOrder(order) {
return res.send(result);
});
})
Before
router.route('/users')
.post(checkPermission, (req, res, next) => {
const user: User = {
username: req.body.user.username,
email: req.body.user.email
}
userService.insertUser(user) {
return res.send(result);
});
})
@Controller('/users')
export default class UserController {
@Post('')
@Authenticated('users')
async insertUser(
@Required() @BodyParams('user') user: User
) {
return this.userService(user);
}
}
after
cross cutting concern
@Service()
export default class UserService {
constructor(private sequelize: SequelizeService) {
}
async insertUser(user: User): Promise<User> {
...
}
}
dependency injection
user-service.ts
@Controller('/users')
export default class UserController {
construct(private userService: UserService)
}
user-controller.ts
auto wired
힘들었던 점
아 내가 커미터다
why typescript?
oh my mistake
class UserService() {
insertUser(user: User) {
,
}
}
server/userService.ts(9,1): error TS1128: Declaration or statement expected.
less doc & test case
export class AuthService {
constructor(user: User);
private _checkToken(): boolean;
private _getToken(logger: Logger): Promise<Token>;
signUp(user: User): Promise<User>;
...
}
auth.service.d.ts
4061 packages
https://www.npmjs.com/~types#packages
more readable & structuring code
@Controller('/users')
export default class UserController implements OnServiceReady {
@Post('')
@Authenticated('users')
async insertUser(
@Required() @BodyParams('user') user: User
) {
return this.userService(user);
}
$onServiceReady() {
...
}
}
export abstract OnServiceReady {
$onServiceReady (): void;
}
efficient development
newest specs
- decorators
- async, await
.
.
.
no typescript
-
외주로 개발해주고 도망갈 생각이다.
-
혼자 개발한다.
-
리팩토링 할 일 없다.
-
이미 문서화와 테스트 케이스가 너무 잘 되어있다.
-
나만 쓰는 모듈 개발 중이다.
-
일정에 쫓기고 있다.
함께 타입스크립트를 전 서비스에 도입할 개발자를 찾고 있습니다.
jobs@flitto.com
end
q & A
사실 저도 잘 몰라요..
bjkim.dev@gmail.com
우리 소통해요
typescript in front-end, back-end
By BJ Kim
typescript in front-end, back-end
- 2,507