import 'dart:async'; import 'dart:convert'; import 'dart:io'; import 'package:flutter/services.dart'; import 'package:http/http.dart' as http; import 'package:url_launcher/url_launcher.dart'; import 'githost.dart'; class GitHub implements GitHost { static const _clientID = "aa3072cbfb02b1db14ed"; static const _clientSecret = "010d303ea99f82330f2b228977cef9ddbf7af2cd"; var _platform = const MethodChannel('gitjournal.io/git'); var _accessCode = ""; @override void init(OAuthCallback callback) { Future _handleMessages(MethodCall call) async { if (call.method != "onURL") { print("GitHub Unknown Call: " + call.method); return; } print("GitHub: Called onUrl with " + call.arguments.toString()); String url = call.arguments["URL"]; var uri = Uri.parse(url); var authCode = uri.queryParameters['code']; if (authCode == null) { print("GitHub: Missing auth code. Now what?"); callback(GitHostException.OAuthFailed); } _accessCode = await _getAccessCode(authCode); if (_accessCode == null || _accessCode.isEmpty) { print("GitHub: AccessCode is invalid: " + _accessCode); callback(GitHostException.OAuthFailed); } callback(null); } _platform.setMethodCallHandler(_handleMessages); print("GitHub: Installed Handler"); } Future _getAccessCode(String authCode) async { var url = "https://github.com/login/oauth/access_token?client_id=$_clientID&client_secret=$_clientSecret&code=$authCode"; var response = await http.post(url); if (response.statusCode != 200) { print("Github getAccessCode: Invalid response " + response.statusCode.toString() + ": " + response.body); throw GitHostException.OAuthFailed; } print("GithubResponse: " + response.body); var map = Uri.splitQueryString(response.body); return map["access_token"]; } @override Future launchOAuthScreen() async { // FIXME: Add some 'state' over here! var url = "https://github.com/login/oauth/authorize?client_id=" + _clientID + "&scope=repo"; return launch(url); } @override Future> listRepos() async { if (_accessCode.isEmpty) { throw GitHostException.MissingAccessCode; } var url = "https://api.github.com/user/repos?page=1&per_page=100&access_token=$_accessCode"; var response = await http.get(url); if (response.statusCode != 200) { print("Github listRepos: Invalid response " + response.statusCode.toString() + ": " + response.body); return null; } List list = jsonDecode(response.body); var repos = []; list.forEach((dynamic d) { var map = Map.from(d); var repo = _repoFromJson(map); repos.add(repo); }); // FIXME: Sort these based on some criteria return repos; } @override Future createRepo(String name) async { if (_accessCode.isEmpty) { throw GitHostException.MissingAccessCode; } var url = "https://api.github.com/user/repos?access_token=$_accessCode"; var data = { 'name': name, 'private': true, }; var headers = { HttpHeaders.contentTypeHeader: "application/json", }; var response = await http.post(url, headers: headers, body: json.encode(data)); if (response.statusCode != 201) { print("Github createRepo: Invalid response " + response.statusCode.toString() + ": " + response.body); if (response.statusCode == 422) { if (response.body.contains("name already exists")) { throw GitHostException.RepoExists; } } throw GitHostException.CreateRepoFailed; } print("GitHub createRepo: " + response.body); Map map = json.decode(response.body); return _repoFromJson(map); } @override Future getRepo(String name) async { if (_accessCode.isEmpty) { throw GitHostException.MissingAccessCode; } var userInfo = await getUserInfo(); var owner = userInfo.username; var url = "https://api.github.com/repos/$owner/$name?access_token=$_accessCode"; var response = await http.get(url); if (response.statusCode != 200) { print("Github getRepo: Invalid response " + response.statusCode.toString() + ": " + response.body); throw GitHostException.GetRepoFailed; } print("GitHub getRepo: " + response.body); Map map = json.decode(response.body); return _repoFromJson(map); } @override Future addDeployKey(String sshPublicKey, String repo) async { if (_accessCode.isEmpty) { throw GitHostException.MissingAccessCode; } var url = "https://api.github.com/repos/$repo/keys?access_token=$_accessCode"; var data = { 'title': "GitJournal", 'key': sshPublicKey, 'read_only': false, }; var headers = { HttpHeaders.contentTypeHeader: "application/json", }; var response = await http.post(url, headers: headers, body: json.encode(data)); if (response.statusCode != 201) { print("Github addDeployKey: Invalid response " + response.statusCode.toString() + ": " + response.body); throw GitHostException.DeployKeyFailed; } print("GitHub addDeployKey: " + response.body); return json.decode(response.body); } GitHostRepo _repoFromJson(Map parsedJson) { return GitHostRepo( fullName: parsedJson['full_name'], cloneUrl: parsedJson['ssh_url'], ); } @override Future getUserInfo() async { if (_accessCode.isEmpty) { throw GitHostException.MissingAccessCode; } var url = "https://api.github.com/user?access_token=$_accessCode"; var response = await http.get(url); if (response.statusCode != 200) { print("Github getUserInfo: Invalid response " + response.statusCode.toString() + ": " + response.body); return null; } Map map = jsonDecode(response.body); if (map == null || map.isEmpty) { print("Github getUserInfo: jsonDecode Failed " + response.statusCode.toString() + ": " + response.body); return null; } return UserInfo( name: map['name'], email: map['email'], username: map['login'], ); } }