ZReviewTender 1.2.7 → 1.3.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: d2943cc5872e966ed140f7300a31570056b8d11a08c24ef3f3e7af07612ebfa1
4
- data.tar.gz: 7f6c610e0f1277d71f89d7678e754e9ae543865dd0942f1a2b77d40606f35538
3
+ metadata.gz: c8aa61d9df145edfd0173d98b25bb14d808cbdfc245d0e98b6bbbd8a93252d40
4
+ data.tar.gz: e9179552c2ee27f447dffd6bad8719e74d5d4081943a6c77e8a95f0347afbdd9
5
5
  SHA512:
6
- metadata.gz: 5ef30f02a3719a92277a2b4014f0cc074927aad1e3fe0466a4e1e2352baed8244a160d1800a7e188740357219e566389befb8f088d9deae6d02a933914ee7871
7
- data.tar.gz: 5605793e165c0fc01c8ff1aa635eb7dda2c5894f75e590b03e44e6194d4ce5b237e50506925bc1a100dea887adc3369e95bc36f920b02d1c315e530acb507c8e
6
+ metadata.gz: 2470ca04e3802dcd49092a5faf86a606655cfe69dd9077baa05bacc07688d3ca8ddb33fd69aa97f377a46c177be7a86ad4e2cac92184e3510c1500882472785a
7
+ data.tar.gz: acbb5ea1d6c87f91bdd7de77160c97c3da9e6733ccc89bee64510f9c66500e476113a1f89886e29eb9e5e42ec49b0a9aa366a8ab2cad120995a4b4887d7136b4
data/.version CHANGED
@@ -1 +1 @@
1
- 1.2.7
1
+ 1.3.1
@@ -1,6 +1,6 @@
1
1
  platform: 'android'
2
2
  packageName: '' # Android App Package Name
3
- keyFilePath: '' # Google Android Publisher API Credential .json File Path
3
+ keyFilePath: '' # Google Android Publisher API Service Account Credential .json File Path
4
4
  playConsoleDeveloperAccountID: '' # Google Console Developer Account ID
5
5
  playConsoleAppID: '' # Google Console App ID
6
6
  processors:
@@ -13,7 +13,7 @@ processors:
13
13
  - GoogleTranslateProcessor: # Google Translate Processor, will translate review text to your language, you can remove whole block if you don't needed it.
14
14
  class: "GoogleTranslateProcessor"
15
15
  enable: false # enable
16
- googleTranslateAPIKeyFilePath: '' # Google Translate API Credential .json File Path
16
+ googleTranslateAPIKeyFilePath: '' # Google Translate API Service Account Credential .json File Path
17
17
  googleTranslateTargetLang: 'zh-TW' # Translate to what Language
18
18
  googleTranslateTerritoriesExclude: ["zh-Hant","zh-Hans"] # Review origin Territory (language) that you don't want to translate. (language for android e.g. zh-Hant, en)
19
19
  - SlackProcessor: # Slack Processor, resend App Review to Slack.
@@ -24,3 +24,25 @@ processors:
24
24
  slackBotToken: "" # Slack Bot Token, send slack message throught Slack Bot.
25
25
  slackBotTargetChannel: "" # Slack Bot Token, send slack message throught Slack Bot. (recommended, first priority)
26
26
  slackInCommingWebHookURL: "" # Slack In-Comming WebHook URL, Send slack message throught In-Comming WebHook, not recommended, deprecated.
27
+ - GoogleSheetProcessor: # Google Sheet Processor, log review to google sheet
28
+ class: "GoogleSheetProcessor"
29
+ enable: false # enable
30
+ googleSheetAPIKeyFilePath: "" # Google Translate API Service Account Credential .json File Path
31
+ googleSheetTimeZoneOffset: "+08:00" # Review Created Date TimeZone
32
+ googleSheetID: "" # Google Sheet ID, you can get it on google sheet url: e.g. https://docs.google.com/spreadsheets/d/googleSheetID/
33
+ googleSheetName: "Sheet1" # Sheet Name
34
+ values: ["%RATING%","%TITLE%\n%BODY%","%APPVERSION%","%CREATEDDATE%"] # Columns Data, you can uses magic variable below to compose string.
35
+ # %TITLE% for review's title
36
+ # %BODY% for review's content
37
+ # %RATING% for review's rating 1~5
38
+ # %PLATFORM% for review's platform Apple or Android
39
+ # %ID% for review's ID
40
+ # %USERNAME% for review's reviewer username
41
+ # %URL% for link to review
42
+ # %TERRITORY% for review's territory (language for android e.g. zh-Hant, en)
43
+ # %APPVERSION% for review's reviewer app version
44
+ # %CREATEDDATE% for review's created date
45
+
46
+ keywordsInclude: [] # keywords you want to filter out
47
+ ratingsInclude: [] # ratings you want to filter out
48
+ territoriesInclude: [] # territories you want to filter out(language for android e.g. zh-Hant, en)
@@ -9,13 +9,13 @@ processors:
9
9
  enable: true # enable
10
10
  keywordsInclude: [] # keywords you want to filter out
11
11
  ratingsInclude: [] # ratings you want to filter out
12
- territoriesInclude: [] # territories you want to filter out
12
+ territoriesInclude: [] # territories you want to filter out (territory for Apple e.g. TWN)
13
13
  - GoogleTranslateProcessor: # Google Translate Processor, will translate review text to your language, you can remove whole block if you don't needed it.
14
14
  class: "GoogleTranslateProcessor"
15
15
  enable: false # enable
16
- googleTranslateAPIKeyFilePath: '' # Google Translate API Credential .json File Path
16
+ googleTranslateAPIKeyFilePath: '' # Google Translate API Service Account Credential .json File Path
17
17
  googleTranslateTargetLang: 'zh-TW' # Translate to what Language
18
- googleTranslateTerritoriesExclude: ["TWN","CHN"] # Review origin Territory that you don't want to translate.
18
+ googleTranslateTerritoriesExclude: ["TWN","CHN"] # Review origin Territory that you don't want to translate. (territory for Apple e.g. TWN)
19
19
  - SlackProcessor: # Slack Processor, resend App Review to Slack.
20
20
  class: "SlackProcessor"
21
21
  enable: true # enable
@@ -24,3 +24,25 @@ processors:
24
24
  slackBotToken: "" # Slack Bot Token, send slack message throught Slack Bot.
25
25
  slackBotTargetChannel: "" # Slack Bot Token, send slack message throught Slack Bot. (recommended, first priority)
26
26
  slackInCommingWebHookURL: "" # Slack In-Comming WebHook URL, Send slack message throught In-Comming WebHook, not recommended, deprecated.
27
+ - GoogleSheetProcessor: # Google Sheet Processor, log review to google sheet
28
+ class: "GoogleSheetProcessor"
29
+ enable: false # enable
30
+ googleSheetAPIKeyFilePath: "" # Google Translate API Service Account Credential .json File Path
31
+ googleSheetTimeZoneOffset: "+08:00" # Review Created Date TimeZone
32
+ googleSheetID: "" # Google Sheet ID, you can get it on google sheet url: e.g. https://docs.google.com/spreadsheets/d/googleSheetID/
33
+ googleSheetName: "Sheet1" # Sheet Name
34
+ values: ["%RATING%","%TITLE%\n%BODY%","%APPVERSION%","%CREATEDDATE%"] # Columns Data, you can uses magic variable below to compose string.
35
+ # %TITLE% for review's title
36
+ # %BODY% for review's content
37
+ # %RATING% for review's rating 1~5
38
+ # %PLATFORM% for review's platform Apple or Android
39
+ # %ID% for review's ID
40
+ # %USERNAME% for review's reviewer username
41
+ # %URL% for link to review
42
+ # %TERRITORY% for review's territory (territory for Apple e.g. TWN)
43
+ # %APPVERSION% for review's reviewer app version
44
+ # %CREATEDDATE% for review's created date
45
+
46
+ keywordsInclude: [] # keywords you want to filter out
47
+ ratingsInclude: [] # ratings you want to filter out
48
+ territoriesInclude: [] # territories you want to filter out (territory for Apple e.g. TWN)
@@ -1,22 +1,20 @@
1
1
  $lib = File.expand_path('../lib', File.dirname(__FILE__))
2
2
 
3
- require "Models/Review"
4
3
  require "Helper"
4
+ require "GoogleAPI"
5
+ require "Models/Review"
5
6
  require "Models/ReviewFetcher"
6
- require "time"
7
- require "google/apis/androidpublisher_v3"
8
7
 
9
8
  class AndroidFetcher < ReviewFetcher
10
9
 
11
- attr_accessor :token, :client
10
+ attr_accessor :token, :googleAPI
12
11
 
13
12
  def initialize(config)
14
13
  @processors = []
15
14
  @config = config
16
15
  @platform = 'Android'
17
-
18
- @client = Google::Apis::AndroidpublisherV3::AndroidPublisherService.new
19
- @client.authorization = Google::Auth::ServiceAccountCredentials.make_creds(json_key_io: config.keyContent, scope: 'https://www.googleapis.com/auth/androidpublisher')
16
+ @logger = ZLogger.new(config.baseExecutePath)
17
+ @googleAPI = GoogleAPI.new(config.keyFilePath, config.baseExecutePath, ["https://www.googleapis.com/auth/androidpublisher"])
20
18
 
21
19
  puts "[AndroidFetcher] Init Success."
22
20
  end
@@ -28,43 +26,106 @@ class AndroidFetcher < ReviewFetcher
28
26
  puts "[AndroidFetcher] Start execute(), latestCheckTimestamp: #{latestCheckTimestamp}"
29
27
 
30
28
  reviews = []
31
-
32
- # Google API Bug, couldn't specify limit/offse/pagination, google only return a few recent reviews.
33
- customerReviews = client.list_reviews(config.packageName).reviews
34
- puts "[AndroidFetcher] Fetch reviews in #{config.packageName}, count: #{customerReviews.length}"
35
- customerReviews.each do |customerReview|
36
-
37
- customerReviewID = customerReview.review_id
38
- customerReviewTitle = nil
39
- customerReviewBody = customerReview.comments[0].user_comment.text.strip
40
- customerReviewRating = customerReview.comments[0].user_comment.star_rating.to_i
41
- customerReviewReviewerNickname = customerReview.author_name
42
- customerReviewCreatedDateTimestamp = customerReview.comments[0].user_comment.last_modified.seconds.to_i
43
- customerReviewTerritory = customerReview.comments[0].user_comment.reviewer_language
44
- customerReviewVersionString = "unknown"
45
- if !customerReview.comments[0].user_comment.app_version_name.nil?
46
- customerReviewVersionString = "#{customerReview.comments[0].user_comment.app_version_name}"
47
- if !customerReview.comments[0].user_comment.app_version_code.nil?
48
- customerReviewVersionString = "#{customerReviewVersionString}(#{customerReview.comments[0].user_comment.app_version_code})"
29
+
30
+ reviewsInfoLink = "https://androidpublisher.googleapis.com/androidpublisher/v3/applications/#{config.packageName}/reviews"
31
+
32
+ # Note: You can retrieve only the reviews that users have created or modified within the last week.
33
+ # https://developers.google.com/android-publisher/reply-to-reviews#retrieving_a_set_of_reviews
34
+
35
+ puts "[AndroidFetcher] Fetch reviews in #{config.packageName}"
36
+
37
+ loop do
38
+ reviewsInfo = googleAPI.request(reviewsInfoLink)
39
+ reviewsInfoLink = reviewsInfo&.dig("tokenPagination", "nextPageToken")
40
+
41
+ customerReviews = reviewsInfo["reviews"]
42
+ puts "[AndroidFetcher] Fetch reviews in #{config.packageName}, count: #{customerReviews.length}"
43
+ customerReviews.each do |customerReview|
44
+
45
+ customerReviewID = customerReview&.dig("reviewId")
46
+ customerReviewTitle = nil
47
+ customerReviewReviewerNickname = customerReview&.dig("authorName")
48
+ customerReviewVersionString = "unknown"
49
+ customerReviewPlatform = platform
50
+ customerReviewCreatedDateTimestamp = 0
51
+
52
+ comment = customerReview&.dig("comments", 0, "userComment")
53
+ if !comment.nil?
54
+ customerReviewBody = comment&.dig("text")
55
+ customerReviewTerritory = comment&.dig("reviewerLanguage")
56
+
57
+ if !customerReviewBody.nil?
58
+ customerReviewBody = customerReviewBody.strip
59
+ end
60
+
61
+ customerReviewRating = comment&.dig("starRating")
62
+ if !customerReviewRating.nil?
63
+ customerReviewRating = customerReviewRating.to_i
64
+ end
65
+
66
+ customerReviewCreatedDateTimestamp = comment&.dig("lastModified", "seconds")
67
+ if !customerReviewCreatedDateTimestamp.nil?
68
+ customerReviewCreatedDateTimestamp = customerReviewCreatedDateTimestamp.to_i
69
+ end
70
+
71
+ androidOSVersion = comment&.dig("androidOsVersion")
72
+ if !androidOSVersion.nil?
73
+ customerReviewPlatform = "#{platform} #{androidOSVersion}"
74
+ end
75
+
76
+ versionString = comment&.dig("appVersionName")
77
+ if !versionString.nil?
78
+ customerReviewVersionString = "#{versionString}"
79
+
80
+ versionCode = comment&.dig("appVersionCode")
81
+ if !versionCode.nil?
82
+ customerReviewVersionString = "#{customerReviewVersionString}(#{versionCode})"
83
+ end
84
+ end
85
+
86
+ deviceInfo = []
87
+ manufacturer = comment&.dig("deviceMetadata", "manufacturer")
88
+ if !manufacturer.nil?
89
+ deviceInfo.append(manufacturer)
90
+ end
91
+
92
+ productName = comment&.dig("deviceMetadata", "productName")
93
+ if !productName.nil?
94
+ deviceInfo.append(productName)
95
+ end
96
+
97
+ if deviceInfo.length > 0
98
+ customerReviewReviewerNickname = "#{customerReviewReviewerNickname} - #{deviceInfo.join("/")}"
99
+ end
100
+ end
101
+
102
+ if latestCheckTimestamp >= customerReviewCreatedDateTimestamp
103
+ reviewsInfoLink = nil
104
+ break
105
+ else
106
+ url = "https://play.google.com/store/apps/details?id=#{config.packageName}&reviewId=#{customerReviewID}"
107
+ if !config.accountID.nil? && !config.appID.nil?
108
+ url = "https://play.google.com/console/developers/#{config.accountID}/app/#{config.appID}/user-feedback/review-details?reviewId=#{customerReviewID}"
109
+ end
110
+ reviews.append(Review.new(customerReviewPlatform, customerReviewID, customerReviewReviewerNickname, customerReviewRating, customerReviewTitle, customerReviewBody, customerReviewCreatedDateTimestamp, url, customerReviewVersionString, customerReviewTerritory))
111
+
112
+ # init first time, need first review to set as latestCheckTimestamp
113
+ if latestCheckTimestamp == 0
114
+ reviewsInfoLink = nil
115
+ break
116
+ end
49
117
  end
50
118
  end
51
- customerReviewPlatform = "Android #{customerReview.comments[0].user_comment.android_os_version}"
52
119
 
53
- url = "https://play.google.com/store/apps/details?id=#{config.packageName}&reviewId=#{customerReviewID}"
54
- if !config.accountID.nil? && !config.appID.nil?
55
- url = "https://play.google.com/console/developers/#{config.accountID}/app/#{config.appID}/user-feedback/review-details?reviewId=#{customerReviewID}"
56
- end
57
- reviews.append(Review.new(customerReviewPlatform, customerReviewID, customerReviewReviewerNickname, customerReviewRating, customerReviewTitle, customerReviewBody, customerReviewCreatedDateTimestamp, url, customerReviewVersionString, customerReviewTerritory))
120
+ break if reviewsInfoLink.nil?
58
121
  end
59
122
 
60
- reviews = reviews.reject{ |review| latestCheckTimestamp >= review.createdDateTimestamp }.sort! { |a, b| a.createdDateTimestamp <=> b.createdDateTimestamp }
61
-
62
- puts "[AndroidFetcher] latest reviews count: #{reviews.length}"
123
+ puts "[AndroidFetcher] Fetch reviews in #{config.packageName}, total reviews count: #{reviews.length}"
63
124
 
64
125
  if reviews.length > 0
126
+ reviews = reviews.sort! { |a, b| a.createdDateTimestamp <=> b.createdDateTimestamp }
65
127
 
66
128
  puts "[AndroidFetcher] latest review: #{reviews.last.body}, #{reviews.last.createdDateTimestamp}"
67
-
68
129
  setPlatformLatestCheckTimestamp(reviews.last.createdDateTimestamp)
69
130
 
70
131
  # init first time, send welcome message
data/lib/AppleFetcher.rb CHANGED
@@ -175,7 +175,8 @@ class AppleFetcher < ReviewFetcher
175
175
  exp: Time.now.to_i + 60*20,
176
176
  aud: 'appstoreconnect-v1'
177
177
  }
178
- token = JWT.encode payload, config.keyContent, 'ES256', header_fields={kid:config.keyID, typ:"JWT"}
178
+
179
+ return JWT.encode payload, config.keyContent, 'ES256', header_fields={kid:config.keyID, typ:"JWT"}
179
180
  end
180
181
 
181
182
  private
data/lib/GoogleAPI.rb ADDED
@@ -0,0 +1,99 @@
1
+ $lib = File.expand_path('../lib', File.dirname(__FILE__))
2
+
3
+ require "jwt"
4
+ require "time"
5
+
6
+ class GoogleAPI < Processor
7
+
8
+ attr_accessor :keyContent, :keyID, :tokenURI, :clientEmail, :logger, :scopes, :token
9
+
10
+ def initialize(keyFilePath, baseExecutePath, scopes = [])
11
+
12
+ @logger = ZLogger.new(baseExecutePath)
13
+
14
+ keyFileContent = JSON.parse(File.read(keyFilePath))
15
+
16
+ @keyContent = Helper.unwrapRequiredParameter(keyFileContent,"private_key")
17
+ @keyID = Helper.unwrapRequiredParameter(keyFileContent,"private_key_id")
18
+ @clientEmail = Helper.unwrapRequiredParameter(keyFileContent,"client_email")
19
+ @tokenURI = Helper.unwrapRequiredParameter(keyFileContent,"token_uri")
20
+
21
+ @scopes = scopes
22
+
23
+ @token = generateJWT()
24
+
25
+ puts "[GoogleAPI] Init Success."
26
+ end
27
+
28
+ def request(url, method = "GET", data = nil, retryCount = 0)
29
+ uri = URI(url)
30
+ https = Net::HTTP.new(uri.host, uri.port)
31
+ https.use_ssl = true
32
+
33
+ request = Net::HTTP::Get.new(uri)
34
+ if method.upcase == "POST"
35
+ request = Net::HTTP::Post.new(uri)
36
+ if !data.nil?
37
+ request['Content-Type'] = 'application/json'
38
+ request.body = JSON.dump(data)
39
+ end
40
+ end
41
+
42
+ request['Authorization'] = "Bearer #{token}";
43
+
44
+ response = https.request(request).read_body
45
+ result = JSON.parse(response)
46
+
47
+ if !result["error"].nil?
48
+ if retryCount >= 10
49
+ raise "Could not connect to #{tokenURI}, key id: #{keyID}, error message: #{response}"
50
+ else
51
+ @token = generateJWT()
52
+ message = "JWT Invalid, retry. (#{retryCount + 1})"
53
+ logger.logWarn(message)
54
+ puts "[GoogleAPI] #{message}"
55
+ return request(url, method, data, retryCount + 1)
56
+ end
57
+ else
58
+ return result
59
+ end
60
+
61
+ end
62
+
63
+ private
64
+ def generateJWT(retryCount = 0)
65
+ payload = {
66
+ iss: clientEmail,
67
+ sub: clientEmail,
68
+ scope: scopes.join(' '),
69
+ aud: tokenURI,
70
+ iat: Time.now.to_i,
71
+ exp: Time.now.to_i + 60*20
72
+ }
73
+
74
+ rsa_private = OpenSSL::PKey::RSA.new(keyContent)
75
+ token = JWT.encode payload, rsa_private, 'RS256', header_fields = {kid:keyID, typ:"JWT"}
76
+
77
+ uri = URI(tokenURI)
78
+ https = Net::HTTP.new(uri.host, uri.port)
79
+ https.use_ssl = true
80
+ request = Net::HTTP::Post.new(uri)
81
+ request.body = "grant_type=urn%3Aietf%3Aparams%3Aoauth%3Agrant-type%3Ajwt-bearer&assertion=#{token}"
82
+
83
+ response = https.request(request).read_body
84
+ result = JSON.parse(response)
85
+
86
+ if result["access_token"].nil?
87
+ if retryCount >= 10
88
+ raise "Could not generate google api JWT, key id: #{keyID}, error message: #{response}"
89
+ else
90
+ message = "Could not generate google api JWT, retry. (#{retryCount + 1})"
91
+ logger.logWarn(message)
92
+ puts "[GoogleAPI] #{message}"
93
+ return generateJWT(retryCount + 1)
94
+ end
95
+ else
96
+ return result["access_token"]
97
+ end
98
+ end
99
+ end
@@ -5,18 +5,17 @@ require "Helper"
5
5
  require "time"
6
6
 
7
7
  class AndroidConfig
8
- attr_accessor :keyContent, :packageName, :accountID, :appID, :baseExecutePath
8
+ attr_accessor :keyFilePath, :packageName, :accountID, :appID, :baseExecutePath
9
9
  def initialize(configYMLObj, configFilePath, baseExecutePath)
10
- keyFilePath = Helper.unwrapRequiredParameter(configYMLObj,"keyFilePath")
10
+ @keyFilePath = Helper.unwrapRequiredParameter(configYMLObj,"keyFilePath")
11
11
 
12
12
  if Pathname.new(keyFilePath).absolute?
13
13
  configDir = File.dirname(configFilePath)
14
- keyFilePath = "#{configDir}#{keyFilePath}"
14
+ @keyFilePath = "#{configDir}#{keyFilePath}"
15
15
  end
16
16
 
17
17
  @accountID = configYMLObj["playConsoleDeveloperAccountID"]
18
18
  @appID = configYMLObj["playConsoleAppID"]
19
- @keyContent = File.open(keyFilePath)
20
19
  @baseExecutePath = baseExecutePath
21
20
  @packageName = Helper.unwrapRequiredParameter(configYMLObj,"packageName")
22
21
  end
@@ -0,0 +1,100 @@
1
+ $lib = File.expand_path('../lib', File.dirname(__FILE__))
2
+
3
+ require "Models/Review"
4
+ require "Models/Processor"
5
+ require "Helper"
6
+ require "pathname"
7
+ require "GoogleAPI"
8
+
9
+ class GoogleSheetProcessor < Processor
10
+
11
+ attr_accessor :keywordsInclude, :ratingsInclude, :territoriesInclude, :logger, :googleAPI, :sheetID, :sheetName, :formatValues, :timeZoneOffset
12
+
13
+ def initialize(config, configFilePath, baseExecutePath)
14
+ @config = config
15
+ @configFilePath = configFilePath
16
+ @baseExecutePath = baseExecutePath
17
+ @logger = ZLogger.new(baseExecutePath)
18
+
19
+ keyFilePath = Helper.unwrapRequiredParameter(config, "googleSheetAPIKeyFilePath")
20
+
21
+ if Pathname.new(keyFilePath).absolute?
22
+ configDir = File.dirname(configFilePath)
23
+ keyFilePath = "#{configDir}#{keyFilePath}"
24
+ end
25
+
26
+ @googleAPI = GoogleAPI.new(keyFilePath, baseExecutePath, ["https://www.googleapis.com/auth/drive", "https://www.googleapis.com/auth/drive.file", "https://www.googleapis.com/auth/spreadsheets"])
27
+
28
+ @keywordsInclude = []
29
+ if !config["keywordsInclude"].nil?
30
+ @keywordsInclude = config["keywordsInclude"]
31
+ end
32
+
33
+ @ratingsInclude = []
34
+ if !config["ratingsInclude"].nil?
35
+ @ratingsInclude = config["ratingsInclude"]
36
+ end
37
+
38
+ @territoriesInclude = []
39
+ if !config["territoriesInclude"].nil?
40
+ @territoriesInclude = config["territoriesInclude"]
41
+ end
42
+
43
+ @timeZoneOffset = Helper.unwrapRequiredParameter(config, "googleSheetTimeZoneOffset")
44
+ @sheetID = Helper.unwrapRequiredParameter(config, "googleSheetID")
45
+ @sheetName = Helper.unwrapRequiredParameter(config, "googleSheetName")
46
+ @formatValues = []
47
+ if !config["values"].nil?
48
+ @formatValues = config["values"]
49
+ end
50
+
51
+ puts "[GoogleSheetProcessor] Init Success."
52
+ end
53
+
54
+ def processReviews(reviews, platform)
55
+
56
+ if reviews.length < 1
57
+ return reviews
58
+ end
59
+
60
+ filterReviews = reviews
61
+
62
+ if ratingsInclude.length > 0
63
+ filterReviews = filterReviews.select{ |review| ratingsInclude.map{ |rating| rating.to_i }.include? review.rating }
64
+ end
65
+
66
+ if territoriesInclude.length > 0
67
+ filterReviews = filterReviews.select{ |review| territoriesInclude.map{ |territory| territory.upcase }.include? review.territory.upcase }
68
+ end
69
+
70
+ if keywordsInclude.length > 0
71
+ keywordsInclude.select{ |keywordsInclude| keywordsInclude != "" }.each do |keywordInclude|
72
+ filterReviews = filterReviews.select{ |review| review.body.include? keywordInclude }
73
+ end
74
+ end
75
+
76
+ filterReviews.each do |review|
77
+ cols = []
78
+ formatValues.each do |formatValue|
79
+ formatValue = formatValue.gsub("%TITLE%", review.title || "")
80
+ formatValue = formatValue.gsub("%BODY%", review.body || "")
81
+ formatValue = formatValue.gsub("%RATING%", review.rating.nil? ? "" :review.rating.to_s)
82
+ formatValue = formatValue.gsub("%PLATFORM%", review.platform || "")
83
+ formatValue = formatValue.gsub("%ID%", review.id || "")
84
+ formatValue = formatValue.gsub("%USERNAME%", review.userName || "")
85
+ formatValue = formatValue.gsub("%URL%", review.url || "")
86
+ formatValue = formatValue.gsub("%TERRITORY%", review.territory || "")
87
+ formatValue = formatValue.gsub("%APPVERSION%", review.appVersion || "")
88
+ formatValue = formatValue.gsub("%CREATEDDATE%", review.createdDateTimestamp.nil? ? "" : Time.at(review.createdDateTimestamp).getlocal(timeZoneOffset).to_s)
89
+
90
+ cols.append(formatValue)
91
+ end
92
+
93
+ puts "[GoogleSheetProcessor] insert #{cols} to #{sheetID}-#{sheetName}"
94
+
95
+ googleAPI.request("https://sheets.googleapis.com/v4/spreadsheets/#{sheetID}/values/#{sheetName}:append?valueInputOption=RAW", "POST", {:values => [cols]})
96
+ end
97
+
98
+ return reviews
99
+ end
100
+ end
@@ -3,18 +3,18 @@ $lib = File.expand_path('../lib', File.dirname(__FILE__))
3
3
  require "Models/Review"
4
4
  require "Models/Processor"
5
5
  require "Helper"
6
-
7
6
  require "pathname"
8
- require "google/cloud/translate/v2"
7
+ require "GoogleAPI"
9
8
 
10
9
  class GoogleTranslateProcessor < Processor
11
10
 
12
- attr_accessor :client, :targetLang, :territoriesExclude
11
+ attr_accessor :targetLang, :territoriesExclude, :token, :logger, :googleAPI
13
12
 
14
13
  def initialize(config, configFilePath, baseExecutePath)
15
14
  @config = config
16
15
  @configFilePath = configFilePath
17
16
  @baseExecutePath = baseExecutePath
17
+ @logger = ZLogger.new(baseExecutePath)
18
18
 
19
19
  keyFilePath = Helper.unwrapRequiredParameter(config, "googleTranslateAPIKeyFilePath")
20
20
 
@@ -23,8 +23,8 @@ class GoogleTranslateProcessor < Processor
23
23
  keyFilePath = "#{configDir}#{keyFilePath}"
24
24
  end
25
25
 
26
- ENV["TRANSLATE_CREDENTIALS"] = keyFilePath
27
- @client = Google::Cloud::Translate::V2.new
26
+ @googleAPI = GoogleAPI.new(keyFilePath, baseExecutePath, ["https://www.googleapis.com/auth/cloud-translation","https://www.googleapis.com/auth/cloud-platform"])
27
+
28
28
  @targetLang = Helper.unwrapRequiredParameter(config, "googleTranslateTargetLang")
29
29
  @territoriesExclude = []
30
30
  if !config['googleTranslateTerritoriesExclude'].nil? && config['googleTranslateTerritoriesExclude'].length > 0
@@ -47,15 +47,25 @@ class GoogleTranslateProcessor < Processor
47
47
  puts "[GoogleTranslateProcessor] translate #{reviews[index].body} from #{reviews[index].territory} to #{targetLang}"
48
48
 
49
49
  if !reviews[index].title.nil?
50
- reviews[index].title = "#{client.translate reviews[index].title, to: targetLang} (#{reviews[index].title})"
50
+ translateTitle = googleAPI.request("https://translation.googleapis.com/language/translate/v2", "POST", {:q => reviews[index].title, :target => targetLang})
51
+ translateTitle = translateTitle&.dig("data", "translations", 0, "translatedText")
52
+ if !translateTitle.nil? && translateTitle != reviews[index].title
53
+ reviews[index].title = "#{translateTitle} (#{reviews[index].title})"
54
+ end
55
+ end
56
+
57
+ if !reviews[index].body.nil?
58
+ translateBody = googleAPI.request("https://translation.googleapis.com/language/translate/v2", "POST", {:q => reviews[index].body, :target => targetLang})
59
+ translateBody = translateBody&.dig("data", "translations", 0, "translatedText")
60
+ if !translateBody.nil? && translateBody != reviews[index].body
61
+ body = "#{translateBody}"
62
+ body += "\r\n===== Translate by Google =====\r\n"
63
+ body += reviews[index].body
64
+ reviews[index].body = body
65
+ end
51
66
  end
52
- body = "#{client.translate reviews[index].body, to: targetLang}"
53
- body += "\r\n===== Translate by Google =====\r\n"
54
- body += reviews[index].body
55
- reviews[index].body = body
56
67
  end
57
68
 
58
69
  return reviews
59
70
  end
60
-
61
71
  end
metadata CHANGED
@@ -1,14 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: ZReviewTender
3
3
  version: !ruby/object:Gem::Version
4
- version: 1.2.7
4
+ version: 1.3.1
5
5
  platform: ruby
6
6
  authors:
7
7
  - ZhgChgLi
8
8
  autorequire:
9
9
  bindir: bin
10
10
  cert_chain: []
11
- date: 2022-08-16 00:00:00.000000000 Z
11
+ date: 2022-08-20 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: net-http
@@ -79,6 +79,7 @@ files:
79
79
  - config/apple.example.yml
80
80
  - lib/AndroidFetcher.rb
81
81
  - lib/AppleFetcher.rb
82
+ - lib/GoogleAPI.rb
82
83
  - lib/Helper.rb
83
84
  - lib/Models/AndroidConfig.rb
84
85
  - lib/Models/AppleConfig.rb
@@ -87,6 +88,7 @@ files:
87
88
  - lib/Models/ReviewFetcher.rb
88
89
  - lib/Models/Version.rb
89
90
  - lib/Processors/FilterProcessor.rb
91
+ - lib/Processors/GoogleSheetProcessor.rb
90
92
  - lib/Processors/GoogleTranslateProcessor.rb
91
93
  - lib/Processors/ProcessorTemplate.rb
92
94
  - lib/Processors/SlackProcessor.rb