ZReviewTender 0.0.7 → 1.0.5

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: 556b1175c3cb37d22d19f7ad2f8efdcd029cebb30cee4dcec2f98bff1f6b9a5b
4
- data.tar.gz: 17c67831cb8bbafb89c51c1c551f778a18f8d5f595ec5ed6fcff3c029657fd0c
3
+ metadata.gz: d7424669fd10f388ab510a8e2dd5883fcf1341d008d3eb373935354f401bb02a
4
+ data.tar.gz: c71871531e5e011ae609b6fd42220a25350e7791882862f2c52329f8e3c9169e
5
5
  SHA512:
6
- metadata.gz: a42d994b153e45fae773118eb29320339d2e9c49d52d865dd60bb62fb23c0aa5c566386cbfeebd0608bc56ab6b5affcf4782de89451f29b7dcb61276208cfc37
7
- data.tar.gz: 1d1b1186d77d05e9de46dfeceb38adb9d538a297c3e2a1c0803099b9266ed414c2e9b3854eaf44ec49c3685ad4d03f6222622a18b94e764a71e8cfa79a88dd12
6
+ metadata.gz: 7306096441bcf8f3bdd47f4b00a5449ff3a1baaaf045159699eff5bfcde01d9876b333ca55894b976e0919061c678df3453e463f9e7a5d6fe7242f242bdaa561
7
+ data.tar.gz: 2f0544786b27029a3a745553a729e1f65aa7758f69cf8a917e0da26e6c7f15d96fd22039e195c09f765a4093f54f2bbd193b997a6d37af2d2ca80bac1fa080a4
data/bin/ZReviewTender CHANGED
@@ -27,6 +27,8 @@ class Main
27
27
  if configYMLFilePath.nil?
28
28
  configYMLFilePath = "#{basePath}/config/apple.yml"
29
29
  end
30
+ configFileCheck(configYMLFilePath, "--apple=CONFIG_YML_FILE_PATH")
31
+
30
32
  fetcher = parseConfigYMLFile(configYMLFilePath)
31
33
  fetcher.execute()
32
34
  end
@@ -35,22 +37,39 @@ class Main
35
37
  if configYMLFilePath.nil?
36
38
  configYMLFilePath = "#{basePath}/config/android.yml"
37
39
  end
40
+ configFileCheck(configYMLFilePath, "--googleAndroid=CONFIG_YML_FILE_PATH")
41
+
38
42
  fetcher = parseConfigYMLFile(configYMLFilePath)
39
43
  fetcher.execute()
40
44
  end
41
45
 
42
46
  opts.on('-r', '--run', 'execute with config yml file') do
43
- appleConfigFilePath = "#{basePath}/config/android.yml"
44
- fetcher = parseConfigYMLFile(appleConfigFilePath)
47
+ androidConfigFilePath = "#{basePath}/config/android.yml"
48
+ configFileCheck(androidConfigFilePath, "--googleAndroid=CONFIG_YML_FILE_PATH")
49
+
50
+ fetcher = parseConfigYMLFile(androidConfigFilePath)
45
51
  fetcher.execute()
46
-
52
+
53
+ #
54
+
47
55
  appleConfigFilePath = "#{basePath}/config/apple.yml"
56
+ configFileCheck(appleConfigFilePath, "--apple=CONFIG_YML_FILE_PATH")
57
+
48
58
  fetcher = parseConfigYMLFile(appleConfigFilePath)
49
59
  fetcher.execute()
50
60
  end
51
61
  end.parse!
52
62
  end
53
63
 
64
+ private
65
+ def configFileCheck(path, command)
66
+ if !File.exists? path
67
+ puts "Make sure you have vaild config file at #{path}"
68
+ puts "Or use ZReviewTender #{command} specify config.yml file path"
69
+ raise "Config file not found: #{path}"
70
+ end
71
+ end
72
+
54
73
  private
55
74
  def parseConfigYMLFile(configFilePath)
56
75
  configYMLObj = YAML.load_file(configFilePath)
@@ -26,7 +26,7 @@ class AndroidFetcher < ReviewFetcher
26
26
  # init first time, send welcome message
27
27
  if latestCheckTimestamp == 0
28
28
  sendWelcomMessage()
29
- setPlatformLatestCheckTimestamp()
29
+ setPlatformLatestCheckTimestamp(Time.now().to_i)
30
30
  return
31
31
  end
32
32
 
@@ -52,7 +52,7 @@ class AndroidFetcher < ReviewFetcher
52
52
  end
53
53
  customerReviewPlatform = "Android #{customerReview.comments[0].user_comment.android_os_version}"
54
54
 
55
- if latestCheckTimestamp > customerReviewCreatedDateTimestamp
55
+ if latestCheckTimestamp >= customerReviewCreatedDateTimestamp
56
56
  break
57
57
  else
58
58
  url = "https://play.google.com/store/apps/details?id=#{config.packageName}&reviewId=#{customerReviewID}"
@@ -66,9 +66,9 @@ class AndroidFetcher < ReviewFetcher
66
66
 
67
67
  if reviews.length > 0
68
68
  reviews.sort! { |a, b| a.createdDateTimestamp <=> b.createdDateTimestamp }
69
+ setPlatformLatestCheckTimestamp(reviews.last.createdDateTimestamp)
70
+
69
71
  processReviews(reviews, platform)
70
72
  end
71
-
72
- setPlatformLatestCheckTimestamp()
73
73
  end
74
74
  end
data/lib/AppleFetcher.rb CHANGED
@@ -2,6 +2,7 @@ $lib = File.expand_path('../lib', File.dirname(__FILE__))
2
2
 
3
3
  require "Models/Review"
4
4
  require "Helper"
5
+ require "ZLogger"
5
6
  require "Models/ReviewFetcher"
6
7
  require "jwt"
7
8
  require "time"
@@ -14,6 +15,7 @@ class AppleFetcher < ReviewFetcher
14
15
  def initialize(config)
15
16
  @processors = []
16
17
  @config = config
18
+ @logger = ZLogger.new(config.baseExecutePath)
17
19
  @platform = 'Apple'
18
20
  @token = generateJWT()
19
21
  end
@@ -26,7 +28,7 @@ class AppleFetcher < ReviewFetcher
26
28
  # init first time, send welcome message
27
29
  if latestCheckTimestamp == 0
28
30
  sendWelcomMessage()
29
- setPlatformLatestCheckTimestamp()
31
+ setPlatformLatestCheckTimestamp(Time.now().to_i)
30
32
  return;
31
33
  end
32
34
 
@@ -34,12 +36,11 @@ class AppleFetcher < ReviewFetcher
34
36
 
35
37
  if reviews.length > 0
36
38
  reviews.sort! { |a, b| a.createdDateTimestamp <=> b.createdDateTimestamp }
37
- reviews = fullfillAppInfo(reviews)
39
+ setPlatformLatestCheckTimestamp(reviews.last.createdDateTimestamp)
38
40
 
41
+ reviews = fullfillAppInfo(reviews)
39
42
  processReviews(reviews, platform)
40
43
  end
41
-
42
- setPlatformLatestCheckTimestamp()
43
44
  end
44
45
 
45
46
  private
@@ -66,7 +67,7 @@ class AppleFetcher < ReviewFetcher
66
67
  customerReviewCreatedDateTimestamp = Time.parse(customerReviewCreatedDate).to_i
67
68
  end
68
69
 
69
- if latestCheckTimestamp > customerReviewCreatedDateTimestamp
70
+ if latestCheckTimestamp >= customerReviewCreatedDateTimestamp
70
71
  customerReviewsLink = nil
71
72
  break
72
73
  else
@@ -167,7 +168,7 @@ class AppleFetcher < ReviewFetcher
167
168
  raise "Could not connect to api.appstoreconnect.apple.com, error message: #{response}"
168
169
  else
169
170
  @token = generateJWT()
170
- Helper.logWarn("JWT Expired, refresh a new one. (#{retryCount + 1})")
171
+ logger.logWarn("JWT Expired, refresh a new one. (#{retryCount + 1})")
171
172
  return request(url, retryCount + 1)
172
173
  end
173
174
  else
data/lib/Helper.rb CHANGED
@@ -1,7 +1,5 @@
1
1
  $lib = File.expand_path('../', File.dirname(__FILE__))
2
2
 
3
- require "logger"
4
-
5
3
  class Helper
6
4
  def self.unwrapRequiredParameter(obj, key)
7
5
  if obj[key].nil?
@@ -24,19 +22,4 @@ class Helper
24
22
  Dir.mkdir(currentDir) unless File.exists?(currentDir)
25
23
  end while dirs.length > 0
26
24
  end
27
-
28
- def self.logError(message)
29
- logger = Logger.new(STDOUT)
30
- logger.error("#{caller[0]}: #{message}")
31
- end
32
-
33
- def self.logWarn(message)
34
- logger = Logger.new(STDOUT)
35
- logger.warning("#{caller[0]}: #{message}")
36
- end
37
-
38
- def self.logInfo(message)
39
- logger = Logger.new(STDOUT)
40
- logger.info("#{caller[0]}: #{message}")
41
- end
42
25
  end
@@ -8,7 +8,7 @@ require "time"
8
8
 
9
9
  class ReviewFetcher
10
10
 
11
- attr_accessor :config, :platform, :processors
11
+ attr_accessor :config, :platform, :processors, :logger
12
12
 
13
13
  def execute()
14
14
 
@@ -31,14 +31,14 @@ class ReviewFetcher
31
31
  end
32
32
  end
33
33
 
34
- def setPlatformLatestCheckTimestamp()
35
- basePath = "#{config.baseExecutePath}/.cache"
34
+ def setPlatformLatestCheckTimestamp(timestamp)
35
+ basePath = "#{config.baseExecutePath}/latestCheckTimestamp/"
36
36
  Helper.createDirIfNotExist(basePath)
37
- File.open("#{basePath}/#{platform}-latestCheckTimestamp", 'w') { |file| file.write(Time.now().to_i) }
37
+ File.open("#{basePath}/#{platform}", 'w') { |file| file.write(timestamp) }
38
38
  end
39
39
 
40
40
  def getPlatformLatestCheckTimestamp()
41
- filePath = "#{config.baseExecutePath}/.cache/#{platform}-latestCheckTimestamp"
41
+ filePath = "#{config.baseExecutePath}/latestCheckTimestamp/#{platform}"
42
42
  if File.exists?(filePath)
43
43
  return File.read(filePath).to_i
44
44
  else
@@ -0,0 +1,54 @@
1
+ $lib = File.expand_path('../lib', File.dirname(__FILE__))
2
+
3
+ require "Models/Review"
4
+ require "Models/Processor"
5
+ require "Helper"
6
+
7
+ class FilterProcessor < Processor
8
+
9
+ attr_accessor :keywordsInclude, :ratingsInclude, :territoriesInclude
10
+
11
+ def initialize(config, configFilePath, baseExecutePath)
12
+ @config = config
13
+ @configFilePath = configFilePath
14
+ @baseExecutePath = baseExecutePath
15
+
16
+ @keywordsInclude = []
17
+ if !config["keywordsInclude"].nil?
18
+ @keywordsInclude = config["keywordsInclude"]
19
+ end
20
+
21
+ @ratingsInclude = []
22
+ if !config["ratingsInclude"].nil?
23
+ @ratingsInclude = config["ratingsInclude"]
24
+ end
25
+
26
+ @territoriesInclude = []
27
+ if !config["territoriesInclude"].nil?
28
+ @territoriesInclude = config["territoriesInclude"]
29
+ end
30
+ end
31
+
32
+ def processReviews(reviews, platform)
33
+ if reviews.length < 1
34
+ return reviews
35
+ end
36
+
37
+ if ratingsInclude.length > 0
38
+ reviews = reviews.select{ |review| ratingsInclude.map{ |rating| rating.to_i }.include? review.rating }
39
+ end
40
+
41
+ if territoriesInclude.length > 0
42
+ reviews = reviews.select{ |review| territoriesInclude.map{ |territory| territory.upcase }.include? review.territory.upcase }
43
+ end
44
+
45
+ if keywordsInclude.length > 0
46
+ keywordsInclude.select{ |keywordsInclude| keywordsInclude != "" }.each do |keywordInclude|
47
+ reviews = reviews.select{ |review| review.body.include? keywordInclude }
48
+ end
49
+ end
50
+
51
+ return reviews
52
+ end
53
+
54
+ end
@@ -9,7 +9,7 @@ require "google/cloud/translate/v2"
9
9
 
10
10
  class GoogleTranslateProcessor < Processor
11
11
 
12
- attr_accessor :client, :targetLang, :whiteListTerritories
12
+ attr_accessor :client, :targetLang, :territoriesExclude
13
13
 
14
14
  def initialize(config, configFilePath, baseExecutePath)
15
15
  @config = config
@@ -26,15 +26,19 @@ class GoogleTranslateProcessor < Processor
26
26
  ENV["TRANSLATE_CREDENTIALS"] = keyFilePath
27
27
  @client = Google::Cloud::Translate::V2.new
28
28
  @targetLang = Helper.unwrapRequiredParameter(config, "googleTranslateTargetLang")
29
- @whiteListTerritories = []
30
- if !config['googleTranslateWhiteListTerritories'].nil? && config['googleTranslateWhiteListTerritories'].length > 0
31
- @whiteListTerritories = config['googleTranslateWhiteListTerritories']
29
+ @territoriesExclude = []
30
+ if !config['googleTranslateTerritoriesExclude'].nil? && config['googleTranslateTerritoriesExclude'].length > 0
31
+ @territoriesExclude = config['googleTranslateTerritoriesExclude']
32
32
  end
33
33
  end
34
34
 
35
35
  def processReviews(reviews, platform)
36
+ if reviews.length < 1
37
+ return reviews
38
+ end
39
+
36
40
  reviews.each_index do |index|
37
- if whiteListTerritories.include? reviews[index].territory
41
+ if territoriesExclude.include? reviews[index].territory
38
42
  next
39
43
  end
40
44
 
@@ -0,0 +1,38 @@
1
+ $lib = File.expand_path('../lib', File.dirname(__FILE__))
2
+
3
+ require "Models/Review"
4
+ require "Models/Processor"
5
+ require "Helper"
6
+ require "ZLogger"
7
+
8
+ # Add to config.yml:
9
+ #
10
+ # processors:
11
+ # - ProcessorTemplate:
12
+ # class: "ProcessorTemplate"
13
+ # parameter1: "value"
14
+ # parameter2: "value"
15
+ # parameter3: "value"
16
+ # ...
17
+ #
18
+
19
+ class ProcessorTemplate < Processor
20
+
21
+ def initialize(config, configFilePath, baseExecutePath)
22
+ # init Processor
23
+ # get paraemter from config e.g. config["parameter1"]
24
+ # configFilePath: file path of config file (apple.yml/android.yml)
25
+ # baseExecutePath: user excute path
26
+ end
27
+
28
+ def processReviews(reviews, platform)
29
+ if reviews.length < 1
30
+ return reviews
31
+ end
32
+
33
+ ## do what your want to do with reviews...
34
+
35
+ ## return result reviews
36
+ return reviews
37
+ end
38
+ end
@@ -3,18 +3,20 @@ $lib = File.expand_path('../lib', File.dirname(__FILE__))
3
3
  require "Models/Review"
4
4
  require "Models/Processor"
5
5
  require "Helper"
6
+ require "ZLogger"
6
7
  require "net/http"
7
8
  require "json"
8
9
  require "time"
9
10
 
10
11
  class SlackProcessor < Processor
11
12
 
12
- attr_accessor :botToken, :inCommingWebHookURL, :targetChannel, :timeZoneOffset, :attachmentGroupByNumber
13
+ attr_accessor :botToken, :inCommingWebHookURL, :targetChannel, :timeZoneOffset, :attachmentGroupByNumber, :logger
13
14
 
14
15
  def initialize(config, configFilePath, baseExecutePath)
15
16
  @config = config
16
17
  @configFilePath = configFilePath
17
18
  @baseExecutePath = baseExecutePath
19
+ @logger = ZLogger.new(baseExecutePath)
18
20
 
19
21
  @botToken = config["slackBotToken"]
20
22
  @inCommingWebHookURL = config["slackInCommingWebHookURL"]
@@ -34,6 +36,9 @@ class SlackProcessor < Processor
34
36
  end
35
37
 
36
38
  def processReviews(reviews, platform)
39
+ if reviews.length < 1
40
+ return reviews
41
+ end
37
42
 
38
43
  pendingPayloads = []
39
44
 
@@ -72,7 +77,7 @@ class SlackProcessor < Processor
72
77
 
73
78
  result = request(payload)
74
79
  if !result[:ok]
75
- Helper.logError(result)
80
+ logger.logError(result)
76
81
  if result[:message] == "ratelimited"
77
82
  sleep(1)
78
83
  pendingPayloads.insert(0, payload)
data/lib/ZLogger.rb ADDED
@@ -0,0 +1,33 @@
1
+ $lib = File.expand_path('../', File.dirname(__FILE__))
2
+
3
+ require "logger"
4
+
5
+ class ZLogger
6
+
7
+ attr_accessor :logger
8
+
9
+ def initialize(baseExecutePath)
10
+ @logger = Logger.new("#{baseExecutePath}/execute.log")
11
+ end
12
+
13
+ def logError(message)
14
+ result = "#{caller[0]}: #{message}"
15
+
16
+ puts "Error: #{result}"
17
+ logger.error(result)
18
+ end
19
+
20
+ def logWarn(message)
21
+ result = "#{caller[0]}: #{message}"
22
+
23
+ puts "Warning: #{result}"
24
+ logger.warn(result)
25
+ end
26
+
27
+ def logInfo(message)
28
+ result = "#{caller[0]}: #{message}"
29
+
30
+ puts "Info: #{result}"
31
+ logger.info(result)
32
+ end
33
+ 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: 0.0.7
4
+ version: 1.0.5
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-07 00:00:00.000000000 Z
11
+ date: 2022-08-08 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: net-http
@@ -66,7 +66,7 @@ dependencies:
66
66
  - - "~>"
67
67
  - !ruby/object:Gem::Version
68
68
  version: 0.25.0
69
- description: ZReviewTender - Monitor your App reviews in your Slack channel
69
+ description: ZReviewTender - App Reviews Automatic Bot
70
70
  email:
71
71
  executables:
72
72
  - ZReviewTender
@@ -82,8 +82,11 @@ files:
82
82
  - lib/Models/Processor.rb
83
83
  - lib/Models/Review.rb
84
84
  - lib/Models/ReviewFetcher.rb
85
+ - lib/Processors/FilterProcessor.rb
85
86
  - lib/Processors/GoogleTranslateProcessor.rb
87
+ - lib/Processors/ProcessorTemplate.rb
86
88
  - lib/Processors/SlackProcessor.rb
89
+ - lib/ZLogger.rb
87
90
  homepage: https://github.com/ZhgChgLi/ZReviewTender
88
91
  licenses:
89
92
  - MIT