|
@@ -20,15 +20,17 @@ use fundit::navCalculator;
|
|
|
*
|
|
|
* Example: CalEntityRankingTask('MF', 2024.09M, true);
|
|
|
*/
|
|
|
-def CalEntityRankingTask(entity_type, end_date, isFromMySQL=true) {
|
|
|
-
|
|
|
- if(!(entity_type in ['MF', 'HF'])) return NULL;
|
|
|
+def CalEntityRankingTask(entityType, endDate, isFromMySQL=true) {
|
|
|
+//entityType='PL'
|
|
|
+//endDate = 2024.10M
|
|
|
+//isFromMySQL = true
|
|
|
+ if(!(entityType in ['MF', 'HF', 'PL', 'CO'])) return NULL;
|
|
|
|
|
|
- entity_info = get_entity_info(entity_type, NULL);
|
|
|
+ entity_info = get_entity_info(entityType, NULL);
|
|
|
|
|
|
- v_ranking_tables = cal_indicator_ranking('strategy', entity_type, entity_info, end_date, isFromMySQL);
|
|
|
+ v_ranking_tables = cal_indicator_ranking('strategy', entityType, entity_info, endDate, isFromMySQL);
|
|
|
|
|
|
- save_ranking_tables(entity_type, v_ranking_tables);
|
|
|
+ save_ranking_tables(entityType, 'strategy', v_ranking_tables);
|
|
|
|
|
|
}
|
|
|
|
|
@@ -223,54 +225,55 @@ def cal_and_save_mc_nav(entity_type, entity_date, is_save_local) {
|
|
|
|
|
|
/*
|
|
|
* 计算并存储基金经理/公司的月度收益及指标
|
|
|
- *
|
|
|
- *
|
|
|
+ *
|
|
|
+ * @param entity_date <TABLE>: entity_id, curve_type, strategy, price_date
|
|
|
+ * @param monthly_returns <TABLE>: entity_id, curve_type, strategy, end_date, price_date, nav, ret
|
|
|
+ * @param indicator_type <STRING>: PBI, BFI
|
|
|
+ *
|
|
|
*/
|
|
|
-def cal_and_save_mc_indicator(entity_type, entity_date, monthly_returns, is_save_local) {
|
|
|
+def cal_and_save_mc_indicator(entity_type, entity_date, monthly_returns, indicator_type, is_save_local) {
|
|
|
|
|
|
rt = '';
|
|
|
|
|
|
- if(!(entity_type IN ['PL', 'CO'])) return rt;
|
|
|
+ if(!(entity_type IN ['PL', 'CO'] && indicator_type IN ['PBI', 'BFI'])) return rt;
|
|
|
if(entity_date.isVoid() || entity_date.size() == 0) return rt;
|
|
|
|
|
|
- i = 0;
|
|
|
- batch_size = 1000;
|
|
|
- v_entity_id = entity_date.entity_id.distinct();
|
|
|
- max_cnt = v_entity_id.size();
|
|
|
+ d_indicators = cal_mc_monthly_indicators(entity_type, indicator_type, monthly_returns);
|
|
|
|
|
|
- do {
|
|
|
+ if(d_indicators.isVoid() || d_indicators["7"].isVoid()) break;
|
|
|
|
|
|
- t_monthly_ret = SELECT * FROM monthly_returns WHERE entity_id IN v_entity_id[i : min(i+batch_size, max_cnt)];
|
|
|
+ // cal_mc_monthly_indicators 返回个两重字典,分别对应 curve_type 和不同区间的数据表,将同样区间的数据表(但不同curve_type)合并
|
|
|
+ // curve_type: 1:私募,4:公募,7:公私募综合
|
|
|
+ trailing_num = d_indicators["7"].keys().size();
|
|
|
+ for(k in d_indicators["7"].keys()) {
|
|
|
+ if(!d_indicators["1"].isVoid())
|
|
|
+ d_indicators["7"][k].append!(d_indicators["1"][k]);
|
|
|
+ if(!d_indicators["4"].isVoid())
|
|
|
+ d_indicators["7"][k].append!(d_indicators["4"][k]);
|
|
|
+ }
|
|
|
+ indicators = d_indicators["7"];
|
|
|
+ d_indicators = null;
|
|
|
|
|
|
- d_indicators = cal_mc_monthly_indicators(entity_type, 'PBI', t_monthly_ret);
|
|
|
+ //
|
|
|
+
|
|
|
|
|
|
- if(d_indicators.isVoid()) break;
|
|
|
+ // 按照 MySQL 建好各表
|
|
|
+ if(indicator_type == 'PBI') {
|
|
|
|
|
|
- // cal_mc_monthly_indicators 返回个两重字典,分别对应 curve_type 和不同区间的数据表,将同样区间的数据表(但不同curve_type)合并
|
|
|
- // curve_type: 1:私募,4:公募,7:公私募综合
|
|
|
- trailing_num = d_indicators["7"].keys().size();
|
|
|
- for(k in d_indicators["7"].keys()) {
|
|
|
- d_indicators["7"][k].append!(d_indicators["1"][k]);
|
|
|
- d_indicators["7"][k].append!(d_indicators["4"][k]);
|
|
|
- }
|
|
|
- indicators = d_indicators["7"];
|
|
|
- d_indicators = null;
|
|
|
+ entity_info = SELECT entity_id, curve_type, strategy, price_date.temporalParse('yyyy-MM') AS price_date FROM entity_date;
|
|
|
|
|
|
- // 按照 MySQL 建好各表
|
|
|
tb_mc_performance = create_mc_performance();
|
|
|
tb_mc_indicator = create_mc_indicator();
|
|
|
tb_mc_risk_stats = create_mc_risk_stats();
|
|
|
tb_mc_riskadjret_stats = create_mc_riskadjret_stats();
|
|
|
tb_mc_style_stats = create_mc_style_stats();
|
|
|
-
|
|
|
- // 仿照MySQL的表结构准备好记录 (1s)
|
|
|
- entity_info = SELECT entity_id, curve_type, strategy, price_date.temporalParse('yyyy-MM') AS price_date FROM entity_date;
|
|
|
|
|
|
+ // 写入数据
|
|
|
generate_entity_performance(entity_info, indicators, true, tb_mc_performance, ['curve_type', 'strategy']);
|
|
|
- generate_entity_indicator(entity_info, indicators, true, tb_mc_indicator, ['curve_type', 'strategy']);
|
|
|
- generate_entity_risk_stats(entity_info, indicators, true, tb_mc_risk_stats, ['curve_type', 'strategy']);
|
|
|
- generate_entity_riskadjret_stats(entity_info, indicators, true, tb_mc_riskadjret_stats, ['curve_type', 'strategy']);
|
|
|
- generate_entity_style_stats(entity_info, indicators, true, tb_mc_style_stats, ['curve_type', 'strategy']);
|
|
|
+ generate_entity_indicator(entity_info, indicators, true, tb_mc_indicator, ['curve_type', 'strategy']);
|
|
|
+ generate_entity_risk_stats(entity_info, indicators, true, tb_mc_risk_stats, ['curve_type', 'strategy']);
|
|
|
+ generate_entity_riskadjret_stats(entity_info, indicators, true, tb_mc_riskadjret_stats, ['curve_type', 'strategy']);
|
|
|
+ generate_entity_style_stats(entity_info, indicators, true, tb_mc_style_stats, ['curve_type', 'strategy']);
|
|
|
|
|
|
if(! tb_mc_performance.isVoid() && tb_mc_performance.size() > 0) {
|
|
|
|
|
@@ -279,7 +282,7 @@ def cal_and_save_mc_indicator(entity_type, entity_date, monthly_returns, is_save
|
|
|
|
|
|
chg_columns_for_mysql(tb_mc_performance, iif(entity_type == 'PL', 'manager_id', 'company_id'));
|
|
|
save_and_sync(tb_mc_performance, iif(entity_type == 'PL', 'raw_db.manager_performance', 'raw_db.company_performance'), );
|
|
|
-
|
|
|
+
|
|
|
chg_columns_for_mysql(tb_mc_indicator, iif(entity_type == 'PL', 'manager_id', 'company_id'));
|
|
|
save_and_sync(tb_mc_indicator, iif(entity_type == 'PL', 'raw_db.manager_indicator', 'raw_db.company_indicator'), );
|
|
|
|
|
@@ -308,14 +311,37 @@ def cal_and_save_mc_indicator(entity_type, entity_date, monthly_returns, is_save
|
|
|
rt += ex + '\n';
|
|
|
}
|
|
|
}
|
|
|
+ } else {
|
|
|
|
|
|
- i += batch_size;
|
|
|
+ entity_info = SELECT entity_id, curve_type, strategy, end_date.temporalParse('yyyy-MM') AS end_date, factor_id AS benchmark_id FROM entity_date;
|
|
|
+
|
|
|
+ tb_mc_bfi_indicator = create_mc_bfi_indicator();
|
|
|
|
|
|
- } while (i < max_cnt);
|
|
|
-
|
|
|
- return rt;
|
|
|
+ //tb_mc_bfi_indicator.rename!('factor_id', 'benchmark_id');
|
|
|
+
|
|
|
+ generate_entity_bfi_indicator(entity_info, indicators, true, tb_mc_bfi_indicator, ['curve_type', 'strategy']);
|
|
|
|
|
|
+ if(! tb_mc_bfi_indicator.isVoid() && tb_mc_bfi_indicator.size() > 0) {
|
|
|
+
|
|
|
+ // save data to MySQL
|
|
|
+ try {
|
|
|
|
|
|
+ chg_columns_for_mysql(tb_mc_bfi_indicator, iif(entity_type == 'PL', 'manager_id', 'company_id'));
|
|
|
+ save_and_sync(tb_mc_bfi_indicator, iif(entity_type == 'PL', 'raw_db.manager_ty_bfi_bm_indicator', 'raw_db.company_ty_bfi_bm_indicator'), );
|
|
|
+
|
|
|
+ // 数据初始化时将指标存入本地
|
|
|
+ if(is_save_local == true)
|
|
|
+ save_table(tb_mc_bfi_indicator, iif(entity_type == 'PL', 'mfdb.manager_ty_bfi_bm_indicator', 'mfdb.company_ty_bfi_bm_indicator'), false);
|
|
|
+
|
|
|
+ } catch(ex) {
|
|
|
+
|
|
|
+ //TODO: Log errors
|
|
|
+ rt = ex;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ return rt;
|
|
|
}
|
|
|
|
|
|
/*
|
|
@@ -348,13 +374,17 @@ def CalMCNavTask(entity_type, updatetime) {
|
|
|
}
|
|
|
|
|
|
/*
|
|
|
- * [定时任务]: 基金经理/公司月收益及指标计算
|
|
|
+ * [定时任务]: 基金经理/公司月收益及指标(含标准及BFI)计算
|
|
|
+ *
|
|
|
+ * @param entity_type <STRING>: PL, CO
|
|
|
*
|
|
|
* Example: CalMCIndicatorTask('CO', 2024.11.04);
|
|
|
+ * CalMCIndicatorTask('PL', 2024.11.04);
|
|
|
*/
|
|
|
def CalMCIndicatorTask(entity_type, updatetime) {
|
|
|
// entity_type = 'PL';
|
|
|
// updatetime = 2024.11.01
|
|
|
+
|
|
|
rt = '';
|
|
|
|
|
|
is_save_local = iif(updatetime <= get_ini_data_const()['updatetime'], true, false);
|
|
@@ -362,20 +392,83 @@ def CalMCIndicatorTask(entity_type, updatetime) {
|
|
|
// 3 sec
|
|
|
entity_date = get_entity_list_by_nav_updatetime(entity_type, NULL, updatetime, true);
|
|
|
|
|
|
- // 取完整月收益 1+ min
|
|
|
- tb_monthly_ret = get_monthly_ret(entity_type, entity_date.entity_id, 1900.01.01, today(), true);
|
|
|
- v_end_date = tb_monthly_ret.end_date.temporalParse('yyyy-MM');
|
|
|
- tb_monthly_ret.replaceColumn!('end_date', v_end_date);
|
|
|
- UPDATE tb_monthly_ret SET price_date = end_date.temporalFormat('yyyy-MM-dd').temporalParse('yyyy-MM-dd').businessMonthEnd() WHERE price_date IS NULL;
|
|
|
+ i = 0;
|
|
|
+ batch_size = 1000;
|
|
|
+ v_entity_id = entity_date.entity_id.distinct();
|
|
|
+ max_cnt = v_entity_id.size();
|
|
|
+ ver_old_month = 1900.01M;
|
|
|
+
|
|
|
+ do {
|
|
|
+
|
|
|
+ // 取完整月净值
|
|
|
+ tb_entity_date = SELECT entity_id, curve_type, strategy, ver_old_month AS end_date
|
|
|
+ FROM entity_date WHERE entity_id in v_entity_id[i : min(i + batch_size, max_cnt)];
|
|
|
+
|
|
|
+ if(tb_entity_date.isVoid() || tb_entity_date.size() == 0) break;
|
|
|
|
|
|
- // 40+ min
|
|
|
- cal_and_save_mc_indicator(entity_type, entity_date, tb_monthly_ret, is_save_local);
|
|
|
+ s_json = tb_entity_date.toStdJson();
|
|
|
+ tb_nav = get_mc_nav_for_return_calculation('PL', s_json, 0);
|
|
|
|
|
|
- tb_monthly_ret = null;
|
|
|
+ v_end_date = tb_nav.end_date.temporalParse('yyyy-MM');
|
|
|
+ tb_nav.replaceColumn!('end_date', v_end_date);
|
|
|
+ tb_nav.join!(v_end_date.temporalFormat('yyyy-MM-dd').temporalParse('yyyy-MM-dd').businessMonthEnd() AS price_date);
|
|
|
+ tb_nav.sortBy!(['entity_id', 'curve_type', 'strategy', 'end_date']);
|
|
|
+
|
|
|
+ // 计算月度收益
|
|
|
+ tb_monthly_ret = SELECT entity_id, curve_type, strategy, end_date, price_date, cumulative_nav AS nav, cumulative_nav.ratios()-1 AS ret
|
|
|
+ FROM tb_nav
|
|
|
+ CONTEXT BY entity_id, curve_type, strategy;
|
|
|
+
|
|
|
+ // 40+ min
|
|
|
+ cal_and_save_mc_indicator(entity_type, entity_date, tb_monthly_ret, 'PBI', is_save_local);
|
|
|
+
|
|
|
+ i += batch_size;
|
|
|
+
|
|
|
+ } while (i < max_cnt);
|
|
|
|
|
|
}
|
|
|
|
|
|
+/*
|
|
|
+ * [定时任务]: 基金经理月BFI指标计算
|
|
|
+ *
|
|
|
+ *
|
|
|
+ * Example: CalManagerBfiIndicatorTask(2024.11.04);
|
|
|
+ */
|
|
|
+def CalManagerBfiIndicatorTask(updatetime) {
|
|
|
+
|
|
|
+// updatetime = 2024.11.01
|
|
|
+
|
|
|
+ rt = '';
|
|
|
|
|
|
+ entity_type = 'PL';
|
|
|
+
|
|
|
+ is_save_local = iif(updatetime <= get_ini_data_const()['updatetime'], true, false);
|
|
|
+
|
|
|
+ // BFI indicator 计算由 bfi matching 表更新驱动
|
|
|
+ entity_date = get_mc_bfi_factors(entity_type, NULL, 1990.01M, today().month(), updatetime);
|
|
|
+ entity_date.join!(entity_date.end_date AS price_date);
|
|
|
+
|
|
|
+ i = 0;
|
|
|
+ batch_size = 1000;
|
|
|
+ v_entity_id = entity_date.entity_id.distinct();
|
|
|
+ max_cnt = v_entity_id.size();
|
|
|
+
|
|
|
+ do {
|
|
|
+
|
|
|
+ // 取完整月收益 1+ min
|
|
|
+ tb_monthly_ret = get_monthly_ret(entity_type, v_entity_id[i : min(i + batch_size, max_cnt)], 1900.01.01, today(), true);
|
|
|
+ v_end_date = tb_monthly_ret.end_date.temporalParse('yyyy-MM');
|
|
|
+ tb_monthly_ret.replaceColumn!('end_date', v_end_date);
|
|
|
+ UPDATE tb_monthly_ret SET price_date = end_date.temporalFormat('yyyy-MM-dd').temporalParse('yyyy-MM-dd').businessMonthEnd() WHERE price_date IS NULL;
|
|
|
+
|
|
|
+ // 40+ min
|
|
|
+ cal_and_save_mc_indicator(entity_type, entity_date, tb_monthly_ret, 'BFI', is_save_local);
|
|
|
+
|
|
|
+ i += batch_size;
|
|
|
+
|
|
|
+ } while (i < max_cnt);
|
|
|
+
|
|
|
+}
|
|
|
/*
|
|
|
*
|
|
|
* [定时任务]: 基金经理的BFI MATCHING
|